main.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603
  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 "camera_config.h"
  20. #include "quickpreview.h"
  21. #include "io_pipeline.h"
  22. enum user_control { USER_CONTROL_ISO, USER_CONTROL_SHUTTER };
  23. static bool camera_is_initialized = false;
  24. static const struct mp_camera_config *camera = NULL;
  25. static MPCameraMode mode;
  26. static int preview_width = -1;
  27. static int preview_height = -1;
  28. static bool gain_is_manual = false;
  29. static int gain;
  30. static int gain_max;
  31. static bool exposure_is_manual = false;
  32. static int exposure;
  33. static bool has_auto_focus_continuous;
  34. static bool has_auto_focus_start;
  35. static cairo_surface_t *surface = NULL;
  36. static cairo_surface_t *status_surface = NULL;
  37. static char last_path[260] = "";
  38. static int burst_length = 3;
  39. static enum user_control current_control;
  40. // Widgets
  41. GtkWidget *preview;
  42. GtkWidget *error_box;
  43. GtkWidget *error_message;
  44. GtkWidget *main_stack;
  45. GtkWidget *thumb_last;
  46. GtkWidget *control_box;
  47. GtkWidget *control_name;
  48. GtkAdjustment *control_slider;
  49. GtkWidget *control_auto;
  50. int
  51. remap(int value, int input_min, int input_max, int output_min, int output_max)
  52. {
  53. const long long factor = 1000000000;
  54. long long output_spread = output_max - output_min;
  55. long long input_spread = input_max - input_min;
  56. long long zero_value = value - input_min;
  57. zero_value *= factor;
  58. long long percentage = zero_value / input_spread;
  59. long long zero_output = percentage * output_spread / factor;
  60. long long result = output_min + zero_output;
  61. return (int)result;
  62. }
  63. static void
  64. update_io_pipeline()
  65. {
  66. struct mp_io_pipeline_state io_state = {
  67. .camera = camera,
  68. .burst_length = burst_length,
  69. .preview_width = preview_width,
  70. .preview_height = preview_height,
  71. .gain_is_manual = gain_is_manual,
  72. .gain = gain,
  73. .exposure_is_manual = exposure_is_manual,
  74. .exposure = exposure,
  75. };
  76. mp_io_pipeline_update_state(&io_state);
  77. }
  78. static bool
  79. update_state(const struct mp_main_state *state)
  80. {
  81. if (!camera_is_initialized) {
  82. camera_is_initialized = true;
  83. }
  84. if (camera == state->camera) {
  85. mode = state->mode;
  86. if (!gain_is_manual) {
  87. gain = state->gain;
  88. }
  89. gain_max = state->gain_max;
  90. if (!exposure_is_manual) {
  91. exposure = state->exposure;
  92. }
  93. has_auto_focus_continuous = state->has_auto_focus_continuous;
  94. has_auto_focus_start = state->has_auto_focus_start;
  95. }
  96. return false;
  97. }
  98. void
  99. mp_main_update_state(const struct mp_main_state *state)
  100. {
  101. struct mp_main_state *state_copy = malloc(sizeof(struct mp_main_state));
  102. *state_copy = *state;
  103. g_main_context_invoke_full(g_main_context_default(), G_PRIORITY_DEFAULT_IDLE,
  104. (GSourceFunc)update_state, state_copy, free);
  105. }
  106. static bool
  107. set_preview(cairo_surface_t *image)
  108. {
  109. if (surface) {
  110. cairo_surface_destroy(surface);
  111. }
  112. surface = image;
  113. gtk_widget_queue_draw(preview);
  114. return false;
  115. }
  116. void
  117. mp_main_set_preview(cairo_surface_t *image)
  118. {
  119. g_main_context_invoke_full(g_main_context_default(), G_PRIORITY_DEFAULT_IDLE,
  120. (GSourceFunc)set_preview, image, NULL);
  121. }
  122. static void
  123. draw_surface_scaled_centered(cairo_t *cr, uint32_t dst_width, uint32_t dst_height,
  124. cairo_surface_t *surface)
  125. {
  126. cairo_save(cr);
  127. cairo_translate(cr, dst_width / 2, dst_height / 2);
  128. int width = cairo_image_surface_get_width(surface);
  129. int height = cairo_image_surface_get_height(surface);
  130. double scale = MIN(dst_width / (double)width, dst_height / (double)height);
  131. cairo_scale(cr, scale, scale);
  132. cairo_translate(cr, -width / 2, -height / 2);
  133. cairo_set_source_surface(cr, surface, 0, 0);
  134. cairo_paint(cr);
  135. cairo_restore(cr);
  136. }
  137. static bool
  138. capture_completed(const char *fname)
  139. {
  140. strncpy(last_path, fname, 260);
  141. // Create a thumbnail from the current surface
  142. cairo_surface_t *thumb =
  143. cairo_image_surface_create(CAIRO_FORMAT_ARGB32, 24, 24);
  144. cairo_t *cr = cairo_create(thumb);
  145. draw_surface_scaled_centered(cr, 24, 24, surface);
  146. cairo_destroy(cr);
  147. gtk_image_set_from_surface(GTK_IMAGE(thumb_last), thumb);
  148. cairo_surface_destroy(thumb);
  149. return false;
  150. }
  151. void
  152. mp_main_capture_completed(const char *fname)
  153. {
  154. gchar *name = g_strdup(fname);
  155. g_main_context_invoke_full(g_main_context_default(), G_PRIORITY_DEFAULT_IDLE,
  156. (GSourceFunc)capture_completed, name, g_free);
  157. }
  158. static void
  159. draw_controls()
  160. {
  161. cairo_t *cr;
  162. char iso[6];
  163. int temp;
  164. char shutterangle[6];
  165. if (exposure_is_manual) {
  166. temp = (int)((float)exposure / (float)camera->capture_mode.height *
  167. 360);
  168. sprintf(shutterangle, "%d\u00b0", temp);
  169. } else {
  170. sprintf(shutterangle, "auto");
  171. }
  172. if (gain_is_manual) {
  173. temp = remap(gain - 1, 0, gain_max, camera->iso_min,
  174. camera->iso_max);
  175. sprintf(iso, "%d", temp);
  176. } else {
  177. sprintf(iso, "auto");
  178. }
  179. if (status_surface)
  180. cairo_surface_destroy(status_surface);
  181. // Make a service to show status of controls, 32px high
  182. if (gtk_widget_get_window(preview) == NULL) {
  183. return;
  184. }
  185. status_surface =
  186. gdk_window_create_similar_surface(gtk_widget_get_window(preview),
  187. CAIRO_CONTENT_COLOR_ALPHA,
  188. preview_width, 32);
  189. cr = cairo_create(status_surface);
  190. cairo_set_source_rgba(cr, 0, 0, 0, 0.0);
  191. cairo_paint(cr);
  192. // Draw the outlines for the headings
  193. cairo_select_font_face(cr, "sans-serif", CAIRO_FONT_SLANT_NORMAL,
  194. CAIRO_FONT_WEIGHT_BOLD);
  195. cairo_set_font_size(cr, 9);
  196. cairo_set_source_rgba(cr, 0, 0, 0, 1);
  197. cairo_move_to(cr, 16, 16);
  198. cairo_text_path(cr, "ISO");
  199. cairo_stroke(cr);
  200. cairo_move_to(cr, 60, 16);
  201. cairo_text_path(cr, "Shutter");
  202. cairo_stroke(cr);
  203. // Draw the fill for the headings
  204. cairo_set_source_rgba(cr, 1, 1, 1, 1);
  205. cairo_move_to(cr, 16, 16);
  206. cairo_show_text(cr, "ISO");
  207. cairo_move_to(cr, 60, 16);
  208. cairo_show_text(cr, "Shutter");
  209. // Draw the outlines for the values
  210. cairo_select_font_face(cr, "sans-serif", CAIRO_FONT_SLANT_NORMAL,
  211. CAIRO_FONT_WEIGHT_NORMAL);
  212. cairo_set_font_size(cr, 11);
  213. cairo_set_source_rgba(cr, 0, 0, 0, 1);
  214. cairo_move_to(cr, 16, 26);
  215. cairo_text_path(cr, iso);
  216. cairo_stroke(cr);
  217. cairo_move_to(cr, 60, 26);
  218. cairo_text_path(cr, shutterangle);
  219. cairo_stroke(cr);
  220. // Draw the fill for the values
  221. cairo_set_source_rgba(cr, 1, 1, 1, 1);
  222. cairo_move_to(cr, 16, 26);
  223. cairo_show_text(cr, iso);
  224. cairo_move_to(cr, 60, 26);
  225. cairo_show_text(cr, shutterangle);
  226. cairo_destroy(cr);
  227. gtk_widget_queue_draw_area(preview, 0, 0, preview_width, 32);
  228. }
  229. static gboolean
  230. preview_draw(GtkWidget *widget, cairo_t *cr, gpointer data)
  231. {
  232. if (!camera_is_initialized) {
  233. return FALSE;
  234. }
  235. // Clear preview area with black
  236. cairo_paint(cr);
  237. // Draw camera preview
  238. if (surface) {
  239. draw_surface_scaled_centered(cr, preview_width, preview_height,
  240. surface);
  241. }
  242. // Draw control overlay
  243. cairo_set_source_surface(cr, status_surface, 0, 0);
  244. cairo_paint(cr);
  245. return FALSE;
  246. }
  247. static gboolean
  248. preview_configure(GtkWidget *widget, GdkEventConfigure *event)
  249. {
  250. int new_preview_width = gtk_widget_get_allocated_width(widget);
  251. int new_preview_height = gtk_widget_get_allocated_height(widget);
  252. if (preview_width != new_preview_width ||
  253. preview_height != new_preview_height) {
  254. preview_width = new_preview_width;
  255. preview_height = new_preview_height;
  256. update_io_pipeline();
  257. }
  258. draw_controls();
  259. return TRUE;
  260. }
  261. void
  262. on_open_last_clicked(GtkWidget *widget, gpointer user_data)
  263. {
  264. char uri[275];
  265. GError *error = NULL;
  266. if (strlen(last_path) == 0) {
  267. return;
  268. }
  269. sprintf(uri, "file://%s.tiff", last_path);
  270. if (!g_app_info_launch_default_for_uri(uri, NULL, &error)) {
  271. g_printerr("Could not launch image viewer: %s\n", error->message);
  272. }
  273. }
  274. void
  275. on_open_directory_clicked(GtkWidget *widget, gpointer user_data)
  276. {
  277. char uri[270];
  278. GError *error = NULL;
  279. sprintf(uri, "file://%s/Pictures", getenv("HOME"));
  280. if (!g_app_info_launch_default_for_uri(uri, NULL, &error)) {
  281. g_printerr("Could not launch image viewer: %s\n", error->message);
  282. }
  283. }
  284. void
  285. on_shutter_clicked(GtkWidget *widget, gpointer user_data)
  286. {
  287. mp_io_pipeline_capture();
  288. }
  289. void
  290. on_preview_tap(GtkWidget *widget, GdkEventButton *event, gpointer user_data)
  291. {
  292. if (event->type != GDK_BUTTON_PRESS)
  293. return;
  294. // Handle taps on the controls
  295. if (event->y < 32) {
  296. if (gtk_widget_is_visible(control_box)) {
  297. gtk_widget_hide(control_box);
  298. return;
  299. } else {
  300. gtk_widget_show(control_box);
  301. }
  302. if (event->x < 60) {
  303. // ISO
  304. current_control = USER_CONTROL_ISO;
  305. gtk_label_set_text(GTK_LABEL(control_name), "ISO");
  306. gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(control_auto),
  307. !gain_is_manual);
  308. gtk_adjustment_set_lower(control_slider, 0.0);
  309. gtk_adjustment_set_upper(control_slider, (float)gain_max);
  310. gtk_adjustment_set_value(control_slider, (double)gain);
  311. } else if (event->x > 60 && event->x < 120) {
  312. // Shutter angle
  313. current_control = USER_CONTROL_SHUTTER;
  314. gtk_label_set_text(GTK_LABEL(control_name), "Shutter");
  315. gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(control_auto),
  316. !exposure_is_manual);
  317. gtk_adjustment_set_lower(control_slider, 1.0);
  318. gtk_adjustment_set_upper(control_slider, 360.0);
  319. gtk_adjustment_set_value(control_slider, (double)exposure);
  320. }
  321. return;
  322. }
  323. // Tapped preview image itself, try focussing
  324. if (has_auto_focus_start) {
  325. mp_io_pipeline_focus();
  326. }
  327. }
  328. void
  329. on_error_close_clicked(GtkWidget *widget, gpointer user_data)
  330. {
  331. gtk_widget_hide(error_box);
  332. }
  333. void
  334. on_camera_switch_clicked(GtkWidget *widget, gpointer user_data)
  335. {
  336. size_t next_index = camera->index + 1;
  337. const struct mp_camera_config *next_camera =
  338. mp_get_camera_config(next_index);
  339. if (!next_camera) {
  340. next_index = 0;
  341. next_camera = mp_get_camera_config(next_index);
  342. }
  343. camera = next_camera;
  344. update_io_pipeline();
  345. }
  346. void
  347. on_settings_btn_clicked(GtkWidget *widget, gpointer user_data)
  348. {
  349. gtk_stack_set_visible_child_name(GTK_STACK(main_stack), "settings");
  350. }
  351. void
  352. on_back_clicked(GtkWidget *widget, gpointer user_data)
  353. {
  354. gtk_stack_set_visible_child_name(GTK_STACK(main_stack), "main");
  355. }
  356. void
  357. on_control_auto_toggled(GtkToggleButton *widget, gpointer user_data)
  358. {
  359. bool is_manual = gtk_toggle_button_get_active(widget) ? false : true;
  360. bool has_changed;
  361. switch (current_control) {
  362. case USER_CONTROL_ISO:
  363. if (gain_is_manual != is_manual) {
  364. gain_is_manual = is_manual;
  365. has_changed = true;
  366. }
  367. break;
  368. case USER_CONTROL_SHUTTER:
  369. if (exposure_is_manual != is_manual) {
  370. exposure_is_manual = is_manual;
  371. has_changed = true;
  372. }
  373. break;
  374. }
  375. if (has_changed) {
  376. // The slider might have been moved while Auto mode is active. When entering
  377. // Manual mode, first read the slider value to sync with those changes.
  378. double value = gtk_adjustment_get_value(control_slider);
  379. switch (current_control) {
  380. case USER_CONTROL_ISO:
  381. if (value != gain) {
  382. gain = (int)value;
  383. }
  384. break;
  385. case USER_CONTROL_SHUTTER: {
  386. // So far all sensors use exposure time in number of sensor rows
  387. int new_exposure =
  388. (int)(value / 360.0 * camera->capture_mode.height);
  389. if (new_exposure != exposure) {
  390. exposure = new_exposure;
  391. }
  392. break;
  393. }
  394. }
  395. update_io_pipeline();
  396. draw_controls();
  397. }
  398. }
  399. void
  400. on_control_slider_changed(GtkAdjustment *widget, gpointer user_data)
  401. {
  402. double value = gtk_adjustment_get_value(widget);
  403. bool has_changed = false;
  404. switch (current_control) {
  405. case USER_CONTROL_ISO:
  406. if (value != gain) {
  407. gain = (int)value;
  408. has_changed = true;
  409. }
  410. break;
  411. case USER_CONTROL_SHUTTER: {
  412. // So far all sensors use exposure time in number of sensor rows
  413. int new_exposure =
  414. (int)(value / 360.0 * camera->capture_mode.height);
  415. if (new_exposure != exposure) {
  416. exposure = new_exposure;
  417. has_changed = true;
  418. }
  419. break;
  420. }
  421. }
  422. if (has_changed) {
  423. update_io_pipeline();
  424. draw_controls();
  425. }
  426. }
  427. int
  428. main(int argc, char *argv[])
  429. {
  430. if (!mp_load_config())
  431. return 1;
  432. setenv("LC_NUMERIC", "C", 1);
  433. gtk_init(&argc, &argv);
  434. g_object_set(gtk_settings_get_default(), "gtk-application-prefer-dark-theme",
  435. TRUE, NULL);
  436. GtkBuilder *builder = gtk_builder_new_from_resource(
  437. "/org/postmarketos/Megapixels/camera.glade");
  438. GtkWidget *window = GTK_WIDGET(gtk_builder_get_object(builder, "window"));
  439. GtkWidget *shutter = GTK_WIDGET(gtk_builder_get_object(builder, "shutter"));
  440. GtkWidget *switch_btn =
  441. GTK_WIDGET(gtk_builder_get_object(builder, "switch_camera"));
  442. GtkWidget *settings_btn =
  443. GTK_WIDGET(gtk_builder_get_object(builder, "settings"));
  444. GtkWidget *settings_back =
  445. GTK_WIDGET(gtk_builder_get_object(builder, "settings_back"));
  446. GtkWidget *error_close =
  447. GTK_WIDGET(gtk_builder_get_object(builder, "error_close"));
  448. GtkWidget *open_last =
  449. GTK_WIDGET(gtk_builder_get_object(builder, "open_last"));
  450. GtkWidget *open_directory =
  451. GTK_WIDGET(gtk_builder_get_object(builder, "open_directory"));
  452. preview = GTK_WIDGET(gtk_builder_get_object(builder, "preview"));
  453. error_box = GTK_WIDGET(gtk_builder_get_object(builder, "error_box"));
  454. error_message = GTK_WIDGET(gtk_builder_get_object(builder, "error_message"));
  455. main_stack = GTK_WIDGET(gtk_builder_get_object(builder, "main_stack"));
  456. thumb_last = GTK_WIDGET(gtk_builder_get_object(builder, "thumb_last"));
  457. control_box = GTK_WIDGET(gtk_builder_get_object(builder, "control_box"));
  458. control_name = GTK_WIDGET(gtk_builder_get_object(builder, "control_name"));
  459. control_slider =
  460. GTK_ADJUSTMENT(gtk_builder_get_object(builder, "control_adj"));
  461. control_auto = GTK_WIDGET(gtk_builder_get_object(builder, "control_auto"));
  462. g_signal_connect(window, "destroy", G_CALLBACK(gtk_main_quit), NULL);
  463. g_signal_connect(shutter, "clicked", G_CALLBACK(on_shutter_clicked), NULL);
  464. g_signal_connect(error_close, "clicked", G_CALLBACK(on_error_close_clicked),
  465. NULL);
  466. g_signal_connect(switch_btn, "clicked", G_CALLBACK(on_camera_switch_clicked),
  467. NULL);
  468. g_signal_connect(settings_btn, "clicked",
  469. G_CALLBACK(on_settings_btn_clicked), NULL);
  470. g_signal_connect(settings_back, "clicked", G_CALLBACK(on_back_clicked),
  471. NULL);
  472. g_signal_connect(open_last, "clicked", G_CALLBACK(on_open_last_clicked),
  473. NULL);
  474. g_signal_connect(open_directory, "clicked",
  475. G_CALLBACK(on_open_directory_clicked), NULL);
  476. g_signal_connect(preview, "draw", G_CALLBACK(preview_draw), NULL);
  477. g_signal_connect(preview, "configure-event", G_CALLBACK(preview_configure),
  478. NULL);
  479. gtk_widget_set_events(preview, gtk_widget_get_events(preview) |
  480. GDK_BUTTON_PRESS_MASK |
  481. GDK_POINTER_MOTION_MASK);
  482. g_signal_connect(preview, "button-press-event", G_CALLBACK(on_preview_tap),
  483. NULL);
  484. g_signal_connect(control_auto, "toggled",
  485. G_CALLBACK(on_control_auto_toggled), NULL);
  486. g_signal_connect(control_slider, "value-changed",
  487. G_CALLBACK(on_control_slider_changed), NULL);
  488. GtkCssProvider *provider = gtk_css_provider_new();
  489. if (access("camera.css", F_OK) != -1) {
  490. gtk_css_provider_load_from_path(provider, "camera.css", NULL);
  491. } else {
  492. gtk_css_provider_load_from_resource(
  493. provider, "/org/postmarketos/Megapixels/camera.css");
  494. }
  495. GtkStyleContext *context = gtk_widget_get_style_context(error_box);
  496. gtk_style_context_add_provider(context, GTK_STYLE_PROVIDER(provider),
  497. GTK_STYLE_PROVIDER_PRIORITY_USER);
  498. context = gtk_widget_get_style_context(control_box);
  499. gtk_style_context_add_provider(context, GTK_STYLE_PROVIDER(provider),
  500. GTK_STYLE_PROVIDER_PRIORITY_USER);
  501. mp_io_pipeline_start();
  502. camera = mp_get_camera_config(0);
  503. update_io_pipeline();
  504. gtk_widget_show(window);
  505. gtk_main();
  506. mp_io_pipeline_stop();
  507. return 0;
  508. }