4 * Copyright (C) 2008 - 2009 The Geeqie Team
6 * Author: Vladimir Nadvornik
8 * This software is released under the GNU General Public License (GNU GPL).
9 * Please read the included file COPYING for more information.
10 * This software comes with no warranty of any kind, use at your own risk!
20 #include "history_list.h"
27 #include "ui_utildlg.h"
34 *-------------------------------------------------------------------
36 *-------------------------------------------------------------------
39 typedef struct _ExifEntry ExifEntry;
44 GtkWidget *title_label;
45 GtkWidget *value_label;
54 typedef struct _PaneExifData PaneExifData;
60 GtkSizeGroup *size_group;
70 typedef struct _ConfDialogData ConfDialogData;
71 struct _ConfDialogData
73 GtkWidget *widget; /* pane or entry, devidet by presenceof "pane_data" or "entry_data" */
78 GtkWidget *title_entry;
82 static void bar_pane_exif_entry_dnd_init(GtkWidget *entry);
83 static void bar_pane_exif_entry_update_title(ExifEntry *ee);
84 static void bar_pane_exif_update(PaneExifData *ped);
85 static gboolean bar_pane_exif_menu_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data);
87 static void bar_pane_exif_entry_destroy(GtkWidget *widget, gpointer data)
97 static GtkWidget *bar_pane_exif_add_entry(PaneExifData *ped, const gchar *key, const gchar *title, gint if_set)
99 ExifEntry *ee = g_new0(ExifEntry, 1);
101 ee->key = g_strdup(key);
102 if (title && title[0])
104 ee->title = g_strdup(title);
108 ee->title = exif_get_description_by_key(key);
109 ee->auto_title = TRUE;
114 ee->ebox = gtk_event_box_new();
115 g_object_set_data(G_OBJECT(ee->ebox), "entry_data", ee);
116 g_signal_connect_after(G_OBJECT(ee->ebox), "destroy",
117 G_CALLBACK(bar_pane_exif_entry_destroy), ee);
119 ee->hbox = gtk_hbox_new(FALSE, 0);
120 gtk_container_add(GTK_CONTAINER(ee->ebox), ee->hbox);
121 gtk_widget_show(ee->hbox);
123 ee->title_label = gtk_label_new(NULL);
124 gtk_misc_set_alignment(GTK_MISC(ee->title_label), 1.0, 0.5);
125 gtk_size_group_add_widget(ped->size_group, ee->title_label);
126 gtk_box_pack_start(GTK_BOX(ee->hbox), ee->title_label, FALSE, TRUE, 0);
127 gtk_widget_show(ee->title_label);
129 ee->value_label = gtk_label_new(NULL);
130 // gtk_label_set_width_chars(GTK_LABEL(ee->value_label), 20);
131 gtk_label_set_ellipsize(GTK_LABEL(ee->value_label), PANGO_ELLIPSIZE_END);
132 // gtk_widget_set_size_request(ee->value_label, 100, -1);
133 gtk_misc_set_alignment(GTK_MISC(ee->value_label), 0.0, 0.5);
134 gtk_box_pack_start(GTK_BOX(ee->hbox), ee->value_label, TRUE, TRUE, 1);
135 gtk_widget_show(ee->value_label);
136 gtk_box_pack_start(GTK_BOX(ped->vbox), ee->ebox, FALSE, FALSE, 0);
138 bar_pane_exif_entry_dnd_init(ee->ebox);
139 g_signal_connect(ee->ebox, "button_press_event", G_CALLBACK(bar_pane_exif_menu_cb), ped);
141 bar_pane_exif_entry_update_title(ee);
142 bar_pane_exif_update(ped);
147 static void bar_pane_exif_reparent_entry(GtkWidget *entry, GtkWidget *pane)
149 GtkWidget *old_pane = entry->parent;
150 PaneExifData *ped = g_object_get_data(G_OBJECT(pane), "pane_data");
151 PaneExifData *old_ped = g_object_get_data(G_OBJECT(old_pane), "pane_data");
152 ExifEntry *ee = g_object_get_data(G_OBJECT(entry), "entry_data");
154 if (!ped || !old_ped || !ee) return;
158 gtk_size_group_remove_widget(old_ped->size_group, ee->title_label);
159 gtk_container_remove(GTK_CONTAINER(old_ped->vbox), entry);
161 gtk_size_group_add_widget(ped->size_group, ee->title_label);
162 gtk_box_pack_start(GTK_BOX(ped->vbox), entry, FALSE, FALSE, 0);
165 static void bar_pane_exif_entry_update_title(ExifEntry *ee)
169 markup = g_markup_printf_escaped("<span size='small'>%s:</span>", (ee->title) ? ee->title : "fixme");
170 gtk_label_set_markup(GTK_LABEL(ee->title_label), markup);
174 static void bar_pane_exif_update_entry(PaneExifData *ped, GtkWidget *entry, gboolean update_title)
177 ExifEntry *ee = g_object_get_data(G_OBJECT(entry), "entry_data");
180 text = metadata_read_string(ped->fd, ee->key, METADATA_FORMATTED);
182 if (!ped->show_all && ee->if_set && (!text || !*text))
184 gtk_label_set_text(GTK_LABEL(ee->value_label), NULL);
185 gtk_widget_hide(entry);
189 gtk_label_set_text(GTK_LABEL(ee->value_label), text);
190 #if GTK_CHECK_VERSION(2,12,0)
191 gtk_widget_set_tooltip_text(ee->hbox, text);
193 gtk_widget_show(entry);
194 ped->all_hidden = FALSE;
199 if (update_title) bar_pane_exif_entry_update_title(ee);
202 static void bar_pane_exif_update(PaneExifData *ped)
206 ped->all_hidden = TRUE;
208 list = gtk_container_get_children(GTK_CONTAINER(ped->vbox));
212 GtkWidget *entry = work->data;
215 bar_pane_exif_update_entry(ped, entry, FALSE);
219 gtk_widget_set_sensitive(ped->pane.title, !ped->all_hidden);
222 void bar_pane_exif_set_fd(GtkWidget *widget, FileData *fd)
226 ped = g_object_get_data(G_OBJECT(widget), "pane_data");
229 file_data_unref(ped->fd);
230 ped->fd = file_data_ref(fd);
232 bar_pane_exif_update(ped);
235 static void bar_pane_exif_notify_cb(FileData *fd, NotifyType type, gpointer data)
237 PaneExifData *ped = data;
238 if ((type & (NOTIFY_REREAD | NOTIFY_CHANGE | NOTIFY_METADATA)) && fd == ped->fd) bar_pane_exif_update(ped);
242 *-------------------------------------------------------------------
244 *-------------------------------------------------------------------
247 static GtkTargetEntry bar_pane_exif_drag_types[] = {
248 { TARGET_APP_EXIF_ENTRY_STRING, GTK_TARGET_SAME_APP, TARGET_APP_EXIF_ENTRY },
249 { "text/plain", 0, TARGET_TEXT_PLAIN }
251 static gint n_exif_entry_drag_types = 2;
253 static GtkTargetEntry bar_pane_exif_drop_types[] = {
254 { TARGET_APP_EXIF_ENTRY_STRING, GTK_TARGET_SAME_APP, TARGET_APP_EXIF_ENTRY },
255 { "text/plain", 0, TARGET_TEXT_PLAIN }
257 static gint n_exif_entry_drop_types = 2;
260 static void bar_pane_exif_entry_dnd_get(GtkWidget *entry, GdkDragContext *context,
261 GtkSelectionData *selection_data, guint info,
262 guint time, gpointer data)
264 ExifEntry *ee = g_object_get_data(G_OBJECT(entry), "entry_data");
268 case TARGET_APP_EXIF_ENTRY:
269 gtk_selection_data_set(selection_data, selection_data->target,
270 8, (gpointer) &entry, sizeof(entry));
273 case TARGET_TEXT_PLAIN:
275 gtk_selection_data_set_text(selection_data, ee->key, -1);
281 static void bar_pane_exif_dnd_receive(GtkWidget *pane, GdkDragContext *context,
283 GtkSelectionData *selection_data, guint info,
284 guint time, gpointer data)
289 GtkWidget *new_entry = NULL;
291 ped = g_object_get_data(G_OBJECT(pane), "pane_data");
296 case TARGET_APP_EXIF_ENTRY:
297 new_entry = *(gpointer *)selection_data->data;
299 if (new_entry->parent && new_entry->parent != ped->vbox) bar_pane_exif_reparent_entry(new_entry, pane);
303 /* FIXME: this needs a check for valid exif keys */
304 new_entry = bar_pane_exif_add_entry(ped, (gchar *)selection_data->data, NULL, TRUE);
308 list = gtk_container_get_children(GTK_CONTAINER(ped->vbox));
314 GtkWidget *entry = work->data;
317 if (entry == new_entry) continue;
319 if (GTK_WIDGET_DRAWABLE(entry) &&
320 gtk_widget_translate_coordinates(pane, entry, x, y, &nx, &ny) &&
321 ny < entry->allocation.height / 2) break;
326 gtk_box_reorder_child(GTK_BOX(ped->vbox), new_entry, pos);
329 static void bar_pane_exif_entry_dnd_begin(GtkWidget *entry, GdkDragContext *context, gpointer data)
331 ExifEntry *ee = g_object_get_data(G_OBJECT(entry), "entry_data");
334 dnd_set_drag_label(entry, context, ee->key);
337 static void bar_pane_exif_entry_dnd_end(GtkWidget *widget, GdkDragContext *context, gpointer data)
341 static void bar_pane_exif_entry_dnd_init(GtkWidget *entry)
343 ExifEntry *ee = g_object_get_data(G_OBJECT(entry), "entry_data");
345 gtk_drag_source_set(entry, GDK_BUTTON1_MASK | GDK_BUTTON2_MASK,
346 bar_pane_exif_drag_types, n_exif_entry_drag_types,
347 GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK);
348 g_signal_connect(G_OBJECT(entry), "drag_data_get",
349 G_CALLBACK(bar_pane_exif_entry_dnd_get), ee);
351 g_signal_connect(G_OBJECT(entry), "drag_begin",
352 G_CALLBACK(bar_pane_exif_entry_dnd_begin), ee);
353 g_signal_connect(G_OBJECT(entry), "drag_end",
354 G_CALLBACK(bar_pane_exif_entry_dnd_end), ee);
357 static void bar_pane_exif_dnd_init(GtkWidget *pane)
359 gtk_drag_dest_set(pane,
360 GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_HIGHLIGHT | GTK_DEST_DEFAULT_DROP,
361 bar_pane_exif_drop_types, n_exif_entry_drop_types,
362 GDK_ACTION_COPY | GDK_ACTION_MOVE);
363 g_signal_connect(G_OBJECT(pane), "drag_data_received",
364 G_CALLBACK(bar_pane_exif_dnd_receive), NULL);
367 static void bar_pane_exif_edit_close_cb(GtkWidget *widget, gpointer data)
369 GenericDialog *gd = data;
370 generic_dialog_close(gd);
373 static void bar_pane_exif_edit_destroy_cb(GtkWidget *widget, gpointer data)
375 ConfDialogData *cdd = data;
376 g_signal_handlers_disconnect_by_func(cdd->widget, G_CALLBACK(bar_pane_exif_edit_close_cb), cdd->gd);
380 static void bar_pane_exif_edit_cancel_cb(GenericDialog *gd, gpointer data)
384 static void bar_pane_exif_edit_ok_cb(GenericDialog *gd, gpointer data)
386 ConfDialogData *cdd = data;
388 /* either one or the other */
389 PaneExifData *ped = g_object_get_data(G_OBJECT(cdd->widget), "pane_data");
390 ExifEntry *ee = g_object_get_data(G_OBJECT(cdd->widget), "entry_data");
394 bar_pane_exif_add_entry(ped,
395 gtk_entry_get_text(GTK_ENTRY(cdd->key_entry)),
396 gtk_entry_get_text(GTK_ENTRY(cdd->title_entry)),
403 GtkWidget *pane = cdd->widget->parent;
407 ped = g_object_get_data(G_OBJECT(pane), "pane_data");
415 ee->key = g_strdup(gtk_entry_get_text(GTK_ENTRY(cdd->key_entry)));
416 title = gtk_entry_get_text(GTK_ENTRY(cdd->title_entry));
417 if (strcmp(ee->title, title) != 0)
420 ee->title = g_strdup(title);
421 ee->auto_title = FALSE;
424 ee->if_set = cdd->if_set;
426 bar_pane_exif_entry_update_title(ee);
427 bar_pane_exif_update(ped);
431 static void bar_pane_exif_conf_dialog(GtkWidget *widget)
437 /* the widget can be either ExifEntry (for editing) or Pane (for new entry)
438 we can decide it by the attached data */
439 ExifEntry *ee = g_object_get_data(G_OBJECT(widget), "entry_data");
441 cdd = g_new0(ConfDialogData, 1);
443 cdd->widget = widget;
446 cdd->if_set = ee ? ee->if_set : TRUE;
448 cdd->gd = gd = generic_dialog_new(ee ? _("Configure entry") : _("Add entry"), "exif_entry_edit",
450 bar_pane_exif_edit_cancel_cb, cdd);
451 g_signal_connect(G_OBJECT(gd->dialog), "destroy",
452 G_CALLBACK(bar_pane_exif_edit_destroy_cb), cdd);
454 /* in case the entry is deleted during editing */
455 g_signal_connect(G_OBJECT(widget), "destroy",
456 G_CALLBACK(bar_pane_exif_edit_close_cb), gd);
458 generic_dialog_add_message(gd, NULL, ee ? _("Configure entry") : _("Add entry"), NULL);
460 generic_dialog_add_button(gd, GTK_STOCK_OK, NULL,
461 bar_pane_exif_edit_ok_cb, TRUE);
463 table = pref_table_new(gd->vbox, 3, 2, FALSE, TRUE);
464 pref_table_label(table, 0, 0, _("Key:"), 1.0);
466 cdd->key_entry = gtk_entry_new();
467 gtk_widget_set_size_request(cdd->key_entry, 300, -1);
468 if (ee) gtk_entry_set_text(GTK_ENTRY(cdd->key_entry), ee->key);
469 gtk_table_attach_defaults(GTK_TABLE(table), cdd->key_entry, 1, 2, 0, 1);
470 generic_dialog_attach_default(gd, cdd->key_entry);
471 gtk_widget_show(cdd->key_entry);
473 pref_table_label(table, 0, 1, _("Title:"), 1.0);
475 cdd->title_entry = gtk_entry_new();
476 gtk_widget_set_size_request(cdd->title_entry, 300, -1);
477 if (ee) gtk_entry_set_text(GTK_ENTRY(cdd->title_entry), ee->title);
478 gtk_table_attach_defaults(GTK_TABLE(table), cdd->title_entry, 1, 2, 1, 2);
479 generic_dialog_attach_default(gd, cdd->title_entry);
480 gtk_widget_show(cdd->title_entry);
482 pref_checkbox_new_int(gd->vbox, _("Show only if set"), cdd->if_set, &cdd->if_set);
484 gtk_widget_show(gd->dialog);
487 static void bar_pane_exif_conf_dialog_cb(GtkWidget *menu_widget, gpointer data)
489 GtkWidget *widget = data;
490 bar_pane_exif_conf_dialog(widget);
493 static void bar_pane_exif_delete_entry_cb(GtkWidget *menu_widget, gpointer data)
495 GtkWidget *entry = data;
496 gtk_widget_destroy(entry);
499 static void bar_pane_exif_toggle_show_all_cb(GtkWidget *menu_widget, gpointer data)
501 PaneExifData *ped = data;
502 ped->show_all = !ped->show_all;
503 bar_pane_exif_update(ped);
506 static void bar_pane_exif_menu_popup(GtkWidget *widget, PaneExifData *ped)
509 /* the widget can be either ExifEntry (for editing) or Pane (for new entry)
510 we can decide it by the attached data */
511 ExifEntry *ee = g_object_get_data(G_OBJECT(widget), "entry_data");
513 menu = popup_menu_short_lived();
518 gchar *conf = g_strdup_printf(_("Configure \"%s\""), ee->title);
519 gchar *del = g_strdup_printf(_("Delete \"%s\""), ee->title);
521 menu_item_add_stock(menu, conf, GTK_STOCK_EDIT, G_CALLBACK(bar_pane_exif_conf_dialog_cb), widget);
522 menu_item_add_stock(menu, del, GTK_STOCK_DELETE, G_CALLBACK(bar_pane_exif_delete_entry_cb), widget);
523 menu_item_add_divider(menu);
530 menu_item_add_stock(menu, _("Add entry"), GTK_STOCK_ADD, G_CALLBACK(bar_pane_exif_conf_dialog_cb), ped->widget);
531 menu_item_add_check(menu, _("Show hidden entries"), ped->show_all, G_CALLBACK(bar_pane_exif_toggle_show_all_cb), ped);
533 gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, 0, GDK_CURRENT_TIME);
536 static gboolean bar_pane_exif_menu_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
538 PaneExifData *ped = data;
539 if (bevent->button == MOUSE_BUTTON_RIGHT)
541 bar_pane_exif_menu_popup(widget, ped);
547 static void bar_pane_exif_entry_write_config(GtkWidget *entry, GString *outstr, gint indent)
549 ExifEntry *ee = g_object_get_data(G_OBJECT(entry), "entry_data");
552 WRITE_STRING("<entry\n");
554 WRITE_CHAR(*ee, key);
555 if (!ee->auto_title) WRITE_CHAR(*ee, title);
556 WRITE_BOOL(*ee, if_set);
558 WRITE_STRING("/>\n");
561 static void bar_pane_exif_write_config(GtkWidget *pane, GString *outstr, gint indent)
566 ped = g_object_get_data(G_OBJECT(pane), "pane_data");
569 WRITE_STRING("<pane_exif\n");
571 write_char_option(outstr, indent, "pane.title", gtk_label_get_text(GTK_LABEL(ped->pane.title)));
572 WRITE_BOOL(*ped, pane.expanded);
577 list = gtk_container_get_children(GTK_CONTAINER(ped->vbox));
581 GtkWidget *entry = work->data;
584 bar_pane_exif_entry_write_config(entry, outstr, indent);
588 WRITE_STRING("</pane_exif>\n");
592 void bar_pane_exif_close(GtkWidget *widget)
596 ped = g_object_get_data(G_OBJECT(widget), "pane_data");
599 gtk_widget_destroy(ped->vbox);
602 static void bar_pane_exif_destroy(GtkWidget *widget, gpointer data)
604 PaneExifData *ped = data;
606 file_data_unregister_notify_func(bar_pane_exif_notify_cb, ped);
607 g_object_unref(ped->size_group);
608 file_data_unref(ped->fd);
612 static void bar_pane_exif_size_request(GtkWidget *pane, GtkRequisition *requisition, gpointer data)
614 PaneExifData *ped = data;
615 if (requisition->height < ped->min_height)
617 requisition->height = ped->min_height;
621 static void bar_pane_exif_size_allocate(GtkWidget *pane, GtkAllocation *alloc, gpointer data)
623 PaneExifData *ped = data;
624 ped->min_height = alloc->height;
627 GtkWidget *bar_pane_exif_new(const gchar *title, gboolean expanded, gboolean populate)
631 ped = g_new0(PaneExifData, 1);
633 ped->pane.pane_set_fd = bar_pane_exif_set_fd;
634 ped->pane.pane_write_config = bar_pane_exif_write_config;
635 ped->pane.title = bar_pane_expander_title(title);
636 ped->pane.expanded = expanded;
638 ped->size_group = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
639 ped->widget = gtk_event_box_new();
640 ped->vbox = gtk_vbox_new(FALSE, PREF_PAD_GAP);
641 gtk_container_add(GTK_CONTAINER(ped->widget), ped->vbox);
642 gtk_widget_show(ped->vbox);
644 ped->min_height = MIN_HEIGHT;
645 g_object_set_data(G_OBJECT(ped->widget), "pane_data", ped);
646 g_signal_connect_after(G_OBJECT(ped->widget), "destroy",
647 G_CALLBACK(bar_pane_exif_destroy), ped);
648 g_signal_connect(G_OBJECT(ped->widget), "size-request",
649 G_CALLBACK(bar_pane_exif_size_request), ped);
650 g_signal_connect(G_OBJECT(ped->widget), "size-allocate",
651 G_CALLBACK(bar_pane_exif_size_allocate), ped);
653 bar_pane_exif_dnd_init(ped->widget);
654 g_signal_connect(ped->widget, "button_press_event", G_CALLBACK(bar_pane_exif_menu_cb), ped);
656 file_data_register_notify_func(bar_pane_exif_notify_cb, ped, NOTIFY_PRIORITY_LOW);
660 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("Camera"), NULL, TRUE);
661 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("DateTime"), NULL, TRUE);
662 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("ShutterSpeed"), NULL, TRUE);
663 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("Aperture"), NULL, TRUE);
664 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("ExposureBias"), NULL, TRUE);
665 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("ISOSpeedRating"), NULL, TRUE);
666 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("FocalLength"), NULL, TRUE);
667 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("FocalLength35mmFilm"), NULL, TRUE);
668 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("Flash"), NULL, TRUE);
669 bar_pane_exif_add_entry(ped, "Exif.Photo.ExposureProgram", NULL, TRUE);
670 bar_pane_exif_add_entry(ped, "Exif.Photo.MeteringMode", NULL, TRUE);
671 bar_pane_exif_add_entry(ped, "Exif.Photo.LightSource", NULL, TRUE);
672 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("ColorProfile"), NULL, TRUE);
673 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("SubjectDistance"), NULL, TRUE);
674 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("Resolution"), NULL, TRUE);
675 bar_pane_exif_add_entry(ped, "Exif.Image.Orientation", NULL, TRUE);
676 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("GPSPosition"), NULL, TRUE);
677 bar_pane_exif_add_entry(ped, EXIF_FORMATTED("GPSAltitude"), NULL, TRUE);
678 bar_pane_exif_add_entry(ped, "Exif.Image.ImageDescription", NULL, TRUE);
679 bar_pane_exif_add_entry(ped, "Exif.Image.Copyright", NULL, TRUE);
682 gtk_widget_show(ped->widget);
687 GtkWidget *bar_pane_exif_new_from_config(const gchar **attribute_names, const gchar **attribute_values)
689 gchar *title = g_strdup(_("NoName"));
690 gboolean expanded = TRUE;
692 while (*attribute_names)
694 const gchar *option = *attribute_names++;
695 const gchar *value = *attribute_values++;
697 if (READ_CHAR_FULL("pane.title", title)) continue;
698 if (READ_BOOL_FULL("pane.expanded", expanded)) continue;
700 DEBUG_1("unknown attribute %s = %s", option, value);
703 return bar_pane_exif_new(title, expanded, FALSE);
706 void bar_pane_exif_entry_add_from_config(GtkWidget *pane, const gchar **attribute_names, const gchar **attribute_values)
711 gboolean if_set = TRUE;
713 ped = g_object_get_data(G_OBJECT(pane), "pane_data");
716 while (*attribute_names)
718 const gchar *option = *attribute_names++;
719 const gchar *value = *attribute_values++;
721 if (READ_CHAR_FULL("key", key)) continue;
722 if (READ_CHAR_FULL("title", title)) continue;
723 if (READ_BOOL_FULL("if_set", if_set)) continue;
725 DEBUG_1("unknown attribute %s = %s", option, value);
728 if (key && key[0]) bar_pane_exif_add_entry(ped, key, title, if_set);
732 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */