main.c 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792
  1. #include "main.h"
  2. #include <errno.h>
  3. #include <fcntl.h>
  4. #include <linux/videodev2.h>
  5. #include <linux/media.h>
  6. #include <linux/v4l2-subdev.h>
  7. #include <sys/ioctl.h>
  8. #include <sys/mman.h>
  9. #include <sys/stat.h>
  10. #include <time.h>
  11. #include <assert.h>
  12. #include <limits.h>
  13. #include <linux/kdev_t.h>
  14. #include <sys/sysmacros.h>
  15. #include <asm/errno.h>
  16. #include <wordexp.h>
  17. #include <gtk/gtk.h>
  18. #include <locale.h>
  19. #include <zbar.h>
  20. #include "camera_config.h"
  21. #include "quickpreview.h"
  22. #include "io_pipeline.h"
  23. enum user_control { USER_CONTROL_ISO, USER_CONTROL_SHUTTER };
  24. static bool camera_is_initialized = false;
  25. static const struct mp_camera_config *camera = NULL;
  26. static MPCameraMode mode;
  27. static int preview_width = -1;
  28. static int preview_height = -1;
  29. static bool gain_is_manual = false;
  30. static int gain;
  31. static int gain_max;
  32. static bool exposure_is_manual = false;
  33. static int exposure;
  34. static bool has_auto_focus_continuous;
  35. static bool has_auto_focus_start;
  36. static cairo_surface_t *surface = NULL;
  37. static cairo_surface_t *status_surface = NULL;
  38. static char last_path[260] = "";
  39. static MPZBarScanResult *zbar_result = NULL;
  40. static int burst_length = 3;
  41. static enum user_control current_control;
  42. // Widgets
  43. GtkWidget *preview;
  44. GtkWidget *error_box;
  45. GtkWidget *error_message;
  46. GtkWidget *main_stack;
  47. GtkWidget *open_last_stack;
  48. GtkWidget *thumb_last;
  49. GtkWidget *process_spinner;
  50. GtkWidget *control_box;
  51. GtkWidget *control_name;
  52. GtkAdjustment *control_slider;
  53. GtkWidget *control_auto;
  54. int
  55. remap(int value, int input_min, int input_max, int output_min, int output_max)
  56. {
  57. const long long factor = 1000000000;
  58. long long output_spread = output_max - output_min;
  59. long long input_spread = input_max - input_min;
  60. long long zero_value = value - input_min;
  61. zero_value *= factor;
  62. long long percentage = zero_value / input_spread;
  63. long long zero_output = percentage * output_spread / factor;
  64. long long result = output_min + zero_output;
  65. return (int)result;
  66. }
  67. static void
  68. update_io_pipeline()
  69. {
  70. struct mp_io_pipeline_state io_state = {
  71. .camera = camera,
  72. .burst_length = burst_length,
  73. .preview_width = preview_width,
  74. .preview_height = preview_height,
  75. .gain_is_manual = gain_is_manual,
  76. .gain = gain,
  77. .exposure_is_manual = exposure_is_manual,
  78. .exposure = exposure,
  79. };
  80. mp_io_pipeline_update_state(&io_state);
  81. }
  82. static bool
  83. update_state(const struct mp_main_state *state)
  84. {
  85. if (!camera_is_initialized) {
  86. camera_is_initialized = true;
  87. }
  88. if (camera == state->camera) {
  89. mode = state->mode;
  90. if (!gain_is_manual) {
  91. gain = state->gain;
  92. }
  93. gain_max = state->gain_max;
  94. if (!exposure_is_manual) {
  95. exposure = state->exposure;
  96. }
  97. has_auto_focus_continuous = state->has_auto_focus_continuous;
  98. has_auto_focus_start = state->has_auto_focus_start;
  99. }
  100. return false;
  101. }
  102. void
  103. mp_main_update_state(const struct mp_main_state *state)
  104. {
  105. struct mp_main_state *state_copy = malloc(sizeof(struct mp_main_state));
  106. *state_copy = *state;
  107. g_main_context_invoke_full(g_main_context_default(), G_PRIORITY_DEFAULT_IDLE,
  108. (GSourceFunc)update_state, state_copy, free);
  109. }
  110. static bool set_zbar_result(MPZBarScanResult *result)
  111. {
  112. if (zbar_result) {
  113. for (uint8_t i = 0; i < zbar_result->size; ++i) {
  114. free(zbar_result->codes[i].data);
  115. }
  116. free(zbar_result);
  117. }
  118. zbar_result = result;
  119. gtk_widget_queue_draw(preview);
  120. return false;
  121. }
  122. void mp_main_set_zbar_result(MPZBarScanResult *result)
  123. {
  124. g_main_context_invoke_full(g_main_context_default(), G_PRIORITY_DEFAULT_IDLE,
  125. (GSourceFunc)set_zbar_result, result, NULL);
  126. }
  127. static bool
  128. set_preview(cairo_surface_t *image)
  129. {
  130. if (surface) {
  131. cairo_surface_destroy(surface);
  132. }
  133. surface = image;
  134. gtk_widget_queue_draw(preview);
  135. return false;
  136. }
  137. void
  138. mp_main_set_preview(cairo_surface_t *image)
  139. {
  140. g_main_context_invoke_full(g_main_context_default(), G_PRIORITY_DEFAULT_IDLE,
  141. (GSourceFunc)set_preview, image, NULL);
  142. }
  143. static void transform_centered(cairo_t *cr, uint32_t dst_width, uint32_t dst_height,
  144. int src_width, int src_height)
  145. {
  146. cairo_translate(cr, dst_width / 2, dst_height / 2);
  147. double scale = MIN(dst_width / (double)src_width, dst_height / (double)src_height);
  148. cairo_scale(cr, scale, scale);
  149. cairo_translate(cr, -src_width / 2, -src_height / 2);
  150. }
  151. void
  152. draw_surface_scaled_centered(cairo_t *cr, uint32_t dst_width, uint32_t dst_height,
  153. cairo_surface_t *surface)
  154. {
  155. cairo_save(cr);
  156. int width = cairo_image_surface_get_width(surface);
  157. int height = cairo_image_surface_get_height(surface);
  158. transform_centered(cr, dst_width, dst_height, width, height);
  159. cairo_set_source_surface(cr, surface, 0, 0);
  160. cairo_paint(cr);
  161. cairo_restore(cr);
  162. }
  163. struct capture_completed_args {
  164. cairo_surface_t *thumb;
  165. char *fname;
  166. };
  167. static bool
  168. capture_completed(struct capture_completed_args *args)
  169. {
  170. strncpy(last_path, args->fname, 259);
  171. gtk_image_set_from_surface(GTK_IMAGE(thumb_last), args->thumb);
  172. gtk_spinner_stop(GTK_SPINNER(process_spinner));
  173. gtk_stack_set_visible_child(GTK_STACK(open_last_stack), thumb_last);
  174. cairo_surface_destroy(args->thumb);
  175. g_free(args->fname);
  176. return false;
  177. }
  178. void
  179. mp_main_capture_completed(cairo_surface_t *thumb, const char *fname)
  180. {
  181. struct capture_completed_args *args = malloc(sizeof(struct capture_completed_args));
  182. args->thumb = thumb;
  183. args->fname = g_strdup(fname);
  184. g_main_context_invoke_full(g_main_context_default(), G_PRIORITY_DEFAULT_IDLE,
  185. (GSourceFunc)capture_completed, args, free);
  186. }
  187. static void
  188. draw_controls()
  189. {
  190. cairo_t *cr;
  191. char iso[6];
  192. int temp;
  193. char shutterangle[6];
  194. if (exposure_is_manual) {
  195. temp = (int)((float)exposure / (float)camera->capture_mode.height *
  196. 360);
  197. sprintf(shutterangle, "%d\u00b0", temp);
  198. } else {
  199. sprintf(shutterangle, "auto");
  200. }
  201. if (gain_is_manual) {
  202. temp = remap(gain - 1, 0, gain_max, camera->iso_min,
  203. camera->iso_max);
  204. sprintf(iso, "%d", temp);
  205. } else {
  206. sprintf(iso, "auto");
  207. }
  208. if (status_surface)
  209. cairo_surface_destroy(status_surface);
  210. // Make a service to show status of controls, 32px high
  211. if (gtk_widget_get_window(preview) == NULL) {
  212. return;
  213. }
  214. status_surface =
  215. gdk_window_create_similar_surface(gtk_widget_get_window(preview),
  216. CAIRO_CONTENT_COLOR_ALPHA,
  217. preview_width, 32);
  218. cr = cairo_create(status_surface);
  219. cairo_set_source_rgba(cr, 0, 0, 0, 0.0);
  220. cairo_paint(cr);
  221. // Draw the outlines for the headings
  222. cairo_select_font_face(cr, "sans-serif", CAIRO_FONT_SLANT_NORMAL,
  223. CAIRO_FONT_WEIGHT_BOLD);
  224. cairo_set_font_size(cr, 9);
  225. cairo_set_source_rgba(cr, 0, 0, 0, 1);
  226. cairo_move_to(cr, 16, 16);
  227. cairo_text_path(cr, "ISO");
  228. cairo_stroke(cr);
  229. cairo_move_to(cr, 60, 16);
  230. cairo_text_path(cr, "Shutter");
  231. cairo_stroke(cr);
  232. // Draw the fill for the headings
  233. cairo_set_source_rgba(cr, 1, 1, 1, 1);
  234. cairo_move_to(cr, 16, 16);
  235. cairo_show_text(cr, "ISO");
  236. cairo_move_to(cr, 60, 16);
  237. cairo_show_text(cr, "Shutter");
  238. // Draw the outlines for the values
  239. cairo_select_font_face(cr, "sans-serif", CAIRO_FONT_SLANT_NORMAL,
  240. CAIRO_FONT_WEIGHT_NORMAL);
  241. cairo_set_font_size(cr, 11);
  242. cairo_set_source_rgba(cr, 0, 0, 0, 1);
  243. cairo_move_to(cr, 16, 26);
  244. cairo_text_path(cr, iso);
  245. cairo_stroke(cr);
  246. cairo_move_to(cr, 60, 26);
  247. cairo_text_path(cr, shutterangle);
  248. cairo_stroke(cr);
  249. // Draw the fill for the values
  250. cairo_set_source_rgba(cr, 1, 1, 1, 1);
  251. cairo_move_to(cr, 16, 26);
  252. cairo_show_text(cr, iso);
  253. cairo_move_to(cr, 60, 26);
  254. cairo_show_text(cr, shutterangle);
  255. cairo_destroy(cr);
  256. gtk_widget_queue_draw_area(preview, 0, 0, preview_width, 32);
  257. }
  258. static gboolean
  259. preview_draw(GtkWidget *widget, cairo_t *cr, gpointer data)
  260. {
  261. if (!camera_is_initialized) {
  262. return FALSE;
  263. }
  264. // Clear preview area with black
  265. cairo_paint(cr);
  266. if (surface) {
  267. // Draw camera preview
  268. cairo_save(cr);
  269. int width = cairo_image_surface_get_width(surface);
  270. int height = cairo_image_surface_get_height(surface);
  271. transform_centered(cr, preview_width, preview_height, width, height);
  272. cairo_set_source_surface(cr, surface, 0, 0);
  273. cairo_paint(cr);
  274. // Draw zbar image
  275. if (zbar_result) {
  276. for (uint8_t i = 0; i < zbar_result->size; ++i) {
  277. MPZBarCode *code = &zbar_result->codes[i];
  278. cairo_set_line_width(cr, 3.0);
  279. cairo_set_source_rgba(cr, 0, 0.5, 1, 0.75);
  280. cairo_new_path(cr);
  281. cairo_move_to(cr, code->bounds_x[0], code->bounds_y[0]);
  282. for (uint8_t i = 0; i < 4; ++i) {
  283. cairo_line_to(cr, code->bounds_x[i], code->bounds_y[i]);
  284. }
  285. cairo_close_path(cr);
  286. cairo_stroke(cr);
  287. cairo_save(cr);
  288. cairo_translate(cr, code->bounds_x[0], code->bounds_y[0]);
  289. cairo_show_text(cr, code->data);
  290. cairo_restore(cr);
  291. }
  292. }
  293. cairo_restore(cr);
  294. }
  295. // Draw control overlay
  296. cairo_set_source_surface(cr, status_surface, 0, 0);
  297. cairo_paint(cr);
  298. return FALSE;
  299. }
  300. static gboolean
  301. preview_configure(GtkWidget *widget, GdkEventConfigure *event)
  302. {
  303. int new_preview_width = gtk_widget_get_allocated_width(widget);
  304. int new_preview_height = gtk_widget_get_allocated_height(widget);
  305. if (preview_width != new_preview_width ||
  306. preview_height != new_preview_height) {
  307. preview_width = new_preview_width;
  308. preview_height = new_preview_height;
  309. update_io_pipeline();
  310. }
  311. draw_controls();
  312. return TRUE;
  313. }
  314. void
  315. on_open_last_clicked(GtkWidget *widget, gpointer user_data)
  316. {
  317. char uri[275];
  318. GError *error = NULL;
  319. if (strlen(last_path) == 0) {
  320. return;
  321. }
  322. sprintf(uri, "file://%s", last_path);
  323. if (!g_app_info_launch_default_for_uri(uri, NULL, &error)) {
  324. g_printerr("Could not launch image viewer for '%s': %s\n", uri, error->message);
  325. }
  326. }
  327. void
  328. on_open_directory_clicked(GtkWidget *widget, gpointer user_data)
  329. {
  330. char uri[270];
  331. GError *error = NULL;
  332. sprintf(uri, "file://%s", g_get_user_special_dir(G_USER_DIRECTORY_PICTURES));
  333. if (!g_app_info_launch_default_for_uri(uri, NULL, &error)) {
  334. g_printerr("Could not launch image viewer: %s\n", error->message);
  335. }
  336. }
  337. void
  338. on_shutter_clicked(GtkWidget *widget, gpointer user_data)
  339. {
  340. gtk_spinner_start(GTK_SPINNER(process_spinner));
  341. gtk_stack_set_visible_child(GTK_STACK(open_last_stack), process_spinner);
  342. mp_io_pipeline_capture();
  343. }
  344. void
  345. on_capture_shortcut(void)
  346. {
  347. on_shutter_clicked(NULL, NULL);
  348. }
  349. static bool
  350. check_point_inside_bounds(int x, int y, int *bounds_x, int *bounds_y)
  351. {
  352. bool right = false, left = false, top = false, bottom = false;
  353. for (int i = 0; i < 4; ++i) {
  354. if (x <= bounds_x[i])
  355. left = true;
  356. if (x >= bounds_x[i])
  357. right = true;
  358. if (y <= bounds_y[i])
  359. top = true;
  360. if (y >= bounds_y[i])
  361. bottom = true;
  362. }
  363. return right && left && top && bottom;
  364. }
  365. static void
  366. on_zbar_code_tapped(GtkWidget *widget, const MPZBarCode *code)
  367. {
  368. GtkWidget *dialog;
  369. GtkDialogFlags flags = GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT;
  370. bool data_is_url = g_uri_is_valid(
  371. code->data, G_URI_FLAGS_PARSE_RELAXED, NULL);
  372. char* data = strdup(code->data);
  373. if (data_is_url) {
  374. dialog = gtk_message_dialog_new(
  375. GTK_WINDOW(gtk_widget_get_toplevel(widget)),
  376. flags,
  377. GTK_MESSAGE_QUESTION,
  378. GTK_BUTTONS_NONE,
  379. "Found a URL '%s' encoded in a %s code.",
  380. code->data,
  381. code->type);
  382. gtk_dialog_add_buttons(
  383. GTK_DIALOG(dialog),
  384. "_Open URL",
  385. GTK_RESPONSE_YES,
  386. NULL);
  387. } else {
  388. dialog = gtk_message_dialog_new(
  389. GTK_WINDOW(gtk_widget_get_toplevel(widget)),
  390. flags,
  391. GTK_MESSAGE_QUESTION,
  392. GTK_BUTTONS_NONE,
  393. "Found '%s' encoded in a %s code.",
  394. code->data,
  395. code->type);
  396. }
  397. gtk_dialog_add_buttons(
  398. GTK_DIALOG(dialog),
  399. "_Copy",
  400. GTK_RESPONSE_ACCEPT,
  401. "_Cancel",
  402. GTK_RESPONSE_CANCEL,
  403. NULL);
  404. int result = gtk_dialog_run(GTK_DIALOG(dialog));
  405. GError *error = NULL;
  406. switch (result) {
  407. case GTK_RESPONSE_YES:
  408. if (!g_app_info_launch_default_for_uri(data,
  409. NULL, &error)) {
  410. g_printerr("Could not launch application: %s\n",
  411. error->message);
  412. }
  413. case GTK_RESPONSE_ACCEPT:
  414. gtk_clipboard_set_text(
  415. gtk_clipboard_get(GDK_SELECTION_PRIMARY),
  416. data, -1);
  417. case GTK_RESPONSE_CANCEL:
  418. break;
  419. default:
  420. g_printerr("Wrong dialog result: %d\n", result);
  421. }
  422. gtk_widget_destroy(dialog);
  423. }
  424. void
  425. on_preview_tap(GtkWidget *widget, GdkEventButton *event, gpointer user_data)
  426. {
  427. if (event->type != GDK_BUTTON_PRESS)
  428. return;
  429. // Handle taps on the controls
  430. if (event->y < 32) {
  431. if (gtk_widget_is_visible(control_box)) {
  432. gtk_widget_hide(control_box);
  433. return;
  434. } else {
  435. gtk_widget_show(control_box);
  436. }
  437. if (event->x < 60) {
  438. // ISO
  439. current_control = USER_CONTROL_ISO;
  440. gtk_label_set_text(GTK_LABEL(control_name), "ISO");
  441. gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(control_auto),
  442. !gain_is_manual);
  443. gtk_adjustment_set_lower(control_slider, 0.0);
  444. gtk_adjustment_set_upper(control_slider, (float)gain_max);
  445. gtk_adjustment_set_value(control_slider, (double)gain);
  446. } else if (event->x > 60 && event->x < 120) {
  447. // Shutter angle
  448. current_control = USER_CONTROL_SHUTTER;
  449. gtk_label_set_text(GTK_LABEL(control_name), "Shutter");
  450. gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(control_auto),
  451. !exposure_is_manual);
  452. gtk_adjustment_set_lower(control_slider, 1.0);
  453. gtk_adjustment_set_upper(control_slider, 360.0);
  454. gtk_adjustment_set_value(control_slider, (double)exposure);
  455. }
  456. return;
  457. }
  458. // Tapped zbar result
  459. if (zbar_result) {
  460. // Transform the event coordinates to the image
  461. int width = cairo_image_surface_get_width(surface);
  462. int height = cairo_image_surface_get_height(surface);
  463. double scale = MIN(preview_width / (double)width, preview_height / (double)height);
  464. int x = (event->x - preview_width / 2) / scale + width / 2;
  465. int y = (event->y - preview_height / 2) / scale + height / 2;
  466. for (uint8_t i = 0; i < zbar_result->size; ++i) {
  467. MPZBarCode *code = &zbar_result->codes[i];
  468. if (check_point_inside_bounds(x, y, code->bounds_x, code->bounds_y)) {
  469. on_zbar_code_tapped(widget, code);
  470. return;
  471. }
  472. }
  473. }
  474. // Tapped preview image itself, try focussing
  475. if (has_auto_focus_start) {
  476. mp_io_pipeline_focus();
  477. }
  478. }
  479. void
  480. on_error_close_clicked(GtkWidget *widget, gpointer user_data)
  481. {
  482. gtk_widget_hide(error_box);
  483. }
  484. void
  485. on_camera_switch_clicked(GtkWidget *widget, gpointer user_data)
  486. {
  487. size_t next_index = camera->index + 1;
  488. const struct mp_camera_config *next_camera =
  489. mp_get_camera_config(next_index);
  490. if (!next_camera) {
  491. next_index = 0;
  492. next_camera = mp_get_camera_config(next_index);
  493. }
  494. camera = next_camera;
  495. update_io_pipeline();
  496. }
  497. void
  498. on_settings_btn_clicked(GtkWidget *widget, gpointer user_data)
  499. {
  500. gtk_stack_set_visible_child_name(GTK_STACK(main_stack), "settings");
  501. }
  502. void
  503. on_back_clicked(GtkWidget *widget, gpointer user_data)
  504. {
  505. gtk_stack_set_visible_child_name(GTK_STACK(main_stack), "main");
  506. }
  507. void
  508. on_control_auto_toggled(GtkToggleButton *widget, gpointer user_data)
  509. {
  510. bool is_manual = gtk_toggle_button_get_active(widget) ? false : true;
  511. bool has_changed;
  512. switch (current_control) {
  513. case USER_CONTROL_ISO:
  514. if (gain_is_manual != is_manual) {
  515. gain_is_manual = is_manual;
  516. has_changed = true;
  517. }
  518. break;
  519. case USER_CONTROL_SHUTTER:
  520. if (exposure_is_manual != is_manual) {
  521. exposure_is_manual = is_manual;
  522. has_changed = true;
  523. }
  524. break;
  525. }
  526. if (has_changed) {
  527. // The slider might have been moved while Auto mode is active. When entering
  528. // Manual mode, first read the slider value to sync with those changes.
  529. double value = gtk_adjustment_get_value(control_slider);
  530. switch (current_control) {
  531. case USER_CONTROL_ISO:
  532. if (value != gain) {
  533. gain = (int)value;
  534. }
  535. break;
  536. case USER_CONTROL_SHUTTER: {
  537. // So far all sensors use exposure time in number of sensor rows
  538. int new_exposure =
  539. (int)(value / 360.0 * camera->capture_mode.height);
  540. if (new_exposure != exposure) {
  541. exposure = new_exposure;
  542. }
  543. break;
  544. }
  545. }
  546. update_io_pipeline();
  547. draw_controls();
  548. }
  549. }
  550. void
  551. on_control_slider_changed(GtkAdjustment *widget, gpointer user_data)
  552. {
  553. double value = gtk_adjustment_get_value(widget);
  554. bool has_changed = false;
  555. switch (current_control) {
  556. case USER_CONTROL_ISO:
  557. if (value != gain) {
  558. gain = (int)value;
  559. has_changed = true;
  560. }
  561. break;
  562. case USER_CONTROL_SHUTTER: {
  563. // So far all sensors use exposure time in number of sensor rows
  564. int new_exposure =
  565. (int)(value / 360.0 * camera->capture_mode.height);
  566. if (new_exposure != exposure) {
  567. exposure = new_exposure;
  568. has_changed = true;
  569. }
  570. break;
  571. }
  572. }
  573. if (has_changed) {
  574. update_io_pipeline();
  575. draw_controls();
  576. }
  577. }
  578. int
  579. main(int argc, char *argv[])
  580. {
  581. if (!mp_load_config())
  582. return 1;
  583. setenv("LC_NUMERIC", "C", 1);
  584. gtk_init(&argc, &argv);
  585. g_object_set(gtk_settings_get_default(), "gtk-application-prefer-dark-theme",
  586. TRUE, NULL);
  587. GtkBuilder *builder = gtk_builder_new_from_resource(
  588. "/org/postmarketos/Megapixels/camera.glade");
  589. GtkWidget *window = GTK_WIDGET(gtk_builder_get_object(builder, "window"));
  590. GtkWidget *shutter = GTK_WIDGET(gtk_builder_get_object(builder, "shutter"));
  591. GtkWidget *switch_btn =
  592. GTK_WIDGET(gtk_builder_get_object(builder, "switch_camera"));
  593. GtkWidget *settings_btn =
  594. GTK_WIDGET(gtk_builder_get_object(builder, "settings"));
  595. GtkWidget *settings_back =
  596. GTK_WIDGET(gtk_builder_get_object(builder, "settings_back"));
  597. GtkWidget *error_close =
  598. GTK_WIDGET(gtk_builder_get_object(builder, "error_close"));
  599. GtkWidget *open_last =
  600. GTK_WIDGET(gtk_builder_get_object(builder, "open_last"));
  601. GtkWidget *open_directory =
  602. GTK_WIDGET(gtk_builder_get_object(builder, "open_directory"));
  603. preview = GTK_WIDGET(gtk_builder_get_object(builder, "preview"));
  604. error_box = GTK_WIDGET(gtk_builder_get_object(builder, "error_box"));
  605. error_message = GTK_WIDGET(gtk_builder_get_object(builder, "error_message"));
  606. main_stack = GTK_WIDGET(gtk_builder_get_object(builder, "main_stack"));
  607. open_last_stack = GTK_WIDGET(gtk_builder_get_object(builder, "open_last_stack"));
  608. thumb_last = GTK_WIDGET(gtk_builder_get_object(builder, "thumb_last"));
  609. process_spinner = GTK_WIDGET(gtk_builder_get_object(builder, "process_spinner"));
  610. control_box = GTK_WIDGET(gtk_builder_get_object(builder, "control_box"));
  611. control_name = GTK_WIDGET(gtk_builder_get_object(builder, "control_name"));
  612. control_slider =
  613. GTK_ADJUSTMENT(gtk_builder_get_object(builder, "control_adj"));
  614. control_auto = GTK_WIDGET(gtk_builder_get_object(builder, "control_auto"));
  615. g_signal_connect(window, "destroy", G_CALLBACK(gtk_main_quit), NULL);
  616. g_signal_connect(shutter, "clicked", G_CALLBACK(on_shutter_clicked), NULL);
  617. g_signal_connect(error_close, "clicked", G_CALLBACK(on_error_close_clicked),
  618. NULL);
  619. g_signal_connect(switch_btn, "clicked", G_CALLBACK(on_camera_switch_clicked),
  620. NULL);
  621. g_signal_connect(settings_btn, "clicked",
  622. G_CALLBACK(on_settings_btn_clicked), NULL);
  623. g_signal_connect(settings_back, "clicked", G_CALLBACK(on_back_clicked),
  624. NULL);
  625. g_signal_connect(open_last, "clicked", G_CALLBACK(on_open_last_clicked),
  626. NULL);
  627. g_signal_connect(open_directory, "clicked",
  628. G_CALLBACK(on_open_directory_clicked), NULL);
  629. g_signal_connect(preview, "draw", G_CALLBACK(preview_draw), NULL);
  630. g_signal_connect(preview, "configure-event", G_CALLBACK(preview_configure),
  631. NULL);
  632. gtk_widget_set_events(preview, gtk_widget_get_events(preview) |
  633. GDK_BUTTON_PRESS_MASK |
  634. GDK_POINTER_MOTION_MASK);
  635. g_signal_connect(preview, "button-press-event", G_CALLBACK(on_preview_tap),
  636. NULL);
  637. g_signal_connect(control_auto, "toggled",
  638. G_CALLBACK(on_control_auto_toggled), NULL);
  639. g_signal_connect(control_slider, "value-changed",
  640. G_CALLBACK(on_control_slider_changed), NULL);
  641. GtkCssProvider *provider = gtk_css_provider_new();
  642. if (access("camera.css", F_OK) != -1) {
  643. gtk_css_provider_load_from_path(provider, "camera.css", NULL);
  644. } else {
  645. gtk_css_provider_load_from_resource(
  646. provider, "/org/postmarketos/Megapixels/camera.css");
  647. }
  648. GtkStyleContext *context = gtk_widget_get_style_context(error_box);
  649. gtk_style_context_add_provider(context, GTK_STYLE_PROVIDER(provider),
  650. GTK_STYLE_PROVIDER_PRIORITY_USER);
  651. context = gtk_widget_get_style_context(control_box);
  652. gtk_style_context_add_provider(context, GTK_STYLE_PROVIDER(provider),
  653. GTK_STYLE_PROVIDER_PRIORITY_USER);
  654. GClosure* capture_shortcut = g_cclosure_new(on_capture_shortcut, 0, 0);
  655. GtkAccelGroup* accel_group = gtk_accel_group_new();
  656. gtk_accel_group_connect(accel_group,
  657. GDK_KEY_space,
  658. 0,
  659. 0,
  660. capture_shortcut);
  661. gtk_window_add_accel_group(GTK_WINDOW(window), accel_group);
  662. mp_io_pipeline_start();
  663. camera = mp_get_camera_config(0);
  664. update_io_pipeline();
  665. gtk_widget_show(window);
  666. gtk_main();
  667. mp_io_pipeline_stop();
  668. return 0;
  669. }