2 * Copyright (C) 2004 John Ellis
3 * Copyright (C) 2008 - 2016 The Geeqie Team
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License along
18 * with this program; if not, write to the Free Software Foundation, Inc.,
19 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
23 #include "advanced-exif.h"
27 #include "history-list.h"
28 #include "layout-util.h"
34 #define ADVANCED_EXIF_DATA_COLUMN_WIDTH 200
37 *-------------------------------------------------------------------
39 *-------------------------------------------------------------------
48 GtkWidget *label_file_name;
54 EXIF_ADVCOL_ENABLED = 0,
60 EXIF_ADVCOL_DESCRIPTION,
64 gint display_order [6] = {
65 EXIF_ADVCOL_DESCRIPTION,
73 static gboolean advanced_exif_row_enabled(const gchar *name)
77 if (!name) return FALSE;
79 list = history_list_get_by_key("exif_extras");
82 if (strcmp(name, static_cast<gchar *>(list->data)) == 0) return TRUE;
89 static void advanced_exif_update(ExifWin *ew)
95 ExifData *exif_original;
98 exif = exif_read_fd(ew->fd);
100 gtk_widget_set_sensitive(ew->scrolled, !!exif);
104 exif_original = exif_get_original(exif);
106 store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(ew->listview)));
107 gtk_list_store_clear(store);
109 item = exif_get_first_item(exif_original);
120 tag = g_strdup_printf("0x%04x", exif_item_get_tag_id(item));
121 tag_name = exif_item_get_tag_name(item);
122 format = exif_item_get_format_name(item, TRUE);
123 text = exif_item_get_data_as_text(item, exif);
124 utf8_text = utf8_validate_or_convert(text);
126 elements = g_strdup_printf("%d", exif_item_get_elements(item));
127 description = exif_item_get_description(item);
128 if (!description || *description == '\0')
131 description = g_strdup(tag_name);
134 gtk_list_store_append(store, &iter);
135 gtk_list_store_set(store, &iter,
136 EXIF_ADVCOL_ENABLED, advanced_exif_row_enabled(tag_name),
137 EXIF_ADVCOL_TAG, tag,
138 EXIF_ADVCOL_NAME, tag_name,
139 EXIF_ADVCOL_VALUE, utf8_text,
140 EXIF_ADVCOL_FORMAT, format,
141 EXIF_ADVCOL_ELEMENTS, elements,
142 EXIF_ADVCOL_DESCRIPTION, description, -1);
148 item = exif_get_next_item(exif_original);
150 exif_free_fd(ew->fd, exif);
154 static void advanced_exif_clear(ExifWin *ew)
158 store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(ew->listview)));
159 gtk_list_store_clear(store);
162 void advanced_exif_set_fd(GtkWidget *window, FileData *fd)
166 ew = static_cast<ExifWin *>(g_object_get_data(G_OBJECT(window), "advanced_exif_data"));
169 /* store this, advanced view toggle needs to reload data */
170 file_data_unref(ew->fd);
171 ew->fd = file_data_ref(fd);
173 gtk_label_set_text(GTK_LABEL(ew->label_file_name), (ew->fd) ? ew->fd->path : "");
175 advanced_exif_clear(ew);
176 advanced_exif_update(ew);
179 static GtkTargetEntry advanced_exif_drag_types[] = {
180 { const_cast<gchar *>("text/plain"), 0, TARGET_TEXT_PLAIN }
182 static gint n_exif_drag_types = 1;
185 static void advanced_exif_dnd_get(GtkWidget *listview, GdkDragContext *,
186 GtkSelectionData *selection_data,
187 guint, guint, gpointer)
189 GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(listview));
192 if (gtk_tree_selection_get_selected(sel, nullptr, &iter))
194 GtkTreeModel *store = gtk_tree_view_get_model(GTK_TREE_VIEW(listview));
197 gtk_tree_model_get(store, &iter, EXIF_ADVCOL_NAME, &key, -1);
198 gtk_selection_data_set_text(selection_data, key, -1);
205 static void advanced_exif_dnd_begin(GtkWidget *listview, GdkDragContext *context, gpointer)
207 GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(listview));
210 if (gtk_tree_selection_get_selected(sel, nullptr, &iter))
212 GtkTreeModel *store = gtk_tree_view_get_model(GTK_TREE_VIEW(listview));
215 gtk_tree_model_get(store, &iter, EXIF_ADVCOL_NAME, &key, -1);
217 dnd_set_drag_label(listview, context, key);
224 static void advanced_exif_add_column(GtkWidget *listview, const gchar *title, gint n, gboolean sizable)
226 GtkTreeViewColumn *column;
227 GtkCellRenderer *renderer;
229 column = gtk_tree_view_column_new();
230 gtk_tree_view_column_set_title(column, title);
234 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_FIXED);
235 gtk_tree_view_column_set_fixed_width(column, ADVANCED_EXIF_DATA_COLUMN_WIDTH);
239 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
242 gtk_tree_view_column_set_resizable(column, TRUE);
243 gtk_tree_view_column_set_sort_column_id(column, n);
245 renderer = gtk_cell_renderer_text_new();
246 gtk_tree_view_column_pack_start(column, renderer, TRUE);
247 gtk_tree_view_column_add_attribute(column, renderer, "text", n);
248 gtk_tree_view_append_column(GTK_TREE_VIEW(listview), column);
251 static void advanced_exif_window_get_geometry(ExifWin *ew)
254 LayoutWindow *lw = nullptr;
258 if (!ew || !lw) return;
260 window = gtk_widget_get_window(ew->window);
261 gdk_window_get_position(window, &lw->options.advanced_exif_window.x, &lw->options.advanced_exif_window.y);
262 lw->options.advanced_exif_window.w = gdk_window_get_width(window);
263 lw->options.advanced_exif_window.h = gdk_window_get_height(window);
266 void advanced_exif_close(ExifWin *ew)
270 advanced_exif_window_get_geometry(ew);
271 file_data_unref(ew->fd);
273 gq_gtk_widget_destroy(ew->window);
278 static gboolean advanced_exif_delete_cb(GtkWidget *, GdkEvent *, gpointer data)
280 auto ew = static_cast<ExifWin *>(data);
282 if (!ew) return FALSE;
284 advanced_exif_window_get_geometry(ew);
285 file_data_unref(ew->fd);
292 static gint advanced_exif_sort_cb(GtkTreeModel *model, GtkTreeIter *a, GtkTreeIter *b, gpointer data)
294 gint n = GPOINTER_TO_INT(data);
299 case EXIF_ADVCOL_DESCRIPTION:
300 case EXIF_ADVCOL_VALUE:
301 case EXIF_ADVCOL_NAME:
302 case EXIF_ADVCOL_TAG:
303 case EXIF_ADVCOL_FORMAT:
304 case EXIF_ADVCOL_ELEMENTS:
308 gtk_tree_model_get(model, a, n, &s1, -1);
309 gtk_tree_model_get(model, b, n, &s2, -1);
313 if (!s1 && !s2) break;
318 ret = g_utf8_collate(s1, s2);
327 g_return_val_if_reached(0);
334 static gboolean advanced_exif_mouseclick(GtkWidget *, GdkEventButton *, gpointer data)
336 /* @FIXME GTK4 stub */
340 static gboolean advanced_exif_mouseclick(GtkWidget *, GdkEventButton *, gpointer data)
342 auto ew = static_cast<ExifWin *>(data);
344 GtkTreeViewColumn *column;
350 GtkClipboard *clipboard;
352 gtk_tree_view_get_cursor(GTK_TREE_VIEW(ew->listview), &path, &column);
355 store = gtk_tree_view_get_model(GTK_TREE_VIEW(ew->listview));
356 gtk_tree_model_get_iter(store, &iter, path);
358 cols = gtk_tree_view_get_columns(GTK_TREE_VIEW(ew->listview));
359 col_num = g_list_index(cols, column);
360 gtk_tree_model_get(store, &iter, display_order[col_num], &value, -1);
362 clipboard = gtk_clipboard_get(GDK_SELECTION_PRIMARY);
363 gtk_clipboard_set_text(clipboard, value, -1);
368 gtk_tree_view_set_search_column(GTK_TREE_VIEW(ew->listview), gtk_tree_view_column_get_sort_column_id(column));
375 static gboolean advanced_exif_keypress(GtkWidget *, GdkEventKey *event, gpointer data)
377 auto ew = static_cast<ExifWin *>(data);
378 gboolean stop_signal = FALSE;
380 if (event->state & GDK_CONTROL_MASK)
382 switch (event->keyval)
385 advanced_exif_close(ew);
389 } // if (event->state & GDK_CONTROL...
390 if (!stop_signal && is_help_key(event))
392 help_window_show("GuideOtherWindowsExif.html");
399 static gboolean search_function_cb(GtkTreeModel *model, gint column, const gchar *key, GtkTreeIter *iter, gpointer)
402 gchar *field_contents;
403 gchar *field_contents_nocase;
406 gtk_tree_model_get(model, iter, column, &field_contents, -1);
408 field_contents_nocase = g_utf8_casefold(field_contents, -1);
409 key_nocase = g_utf8_casefold(key, -1);
411 if (g_strstr_len(field_contents_nocase, -1, key_nocase))
416 g_free(field_contents);
417 g_free(field_contents_nocase);
423 GtkWidget *advanced_exif_new(LayoutWindow *lw)
427 GdkGeometry geometry;
428 GtkTreeSortable *sortable;
432 ew = g_new0(ExifWin, 1);
434 ew->window = window_new("view", nullptr, nullptr, _("Metadata"));
435 DEBUG_NAME(ew->window);
437 geometry.min_width = 900;
438 geometry.min_height = 600;
439 gtk_window_set_geometry_hints(GTK_WINDOW(ew->window), nullptr, &geometry, GDK_HINT_MIN_SIZE);
441 gtk_window_set_resizable(GTK_WINDOW(ew->window), TRUE);
443 gtk_window_resize(GTK_WINDOW(ew->window), lw->options.advanced_exif_window.w, lw->options.advanced_exif_window.h);
444 if (lw->options.advanced_exif_window.x != 0 && lw->options.advanced_exif_window.y != 0)
446 gq_gtk_window_move(GTK_WINDOW(ew->window), lw->options.advanced_exif_window.x, lw->options.advanced_exif_window.y);
449 g_object_set_data(G_OBJECT(ew->window), "advanced_exif_data", ew);
450 g_signal_connect(G_OBJECT(ew->window), "delete_event", G_CALLBACK(advanced_exif_delete_cb), ew);
452 ew->vbox = gtk_box_new(GTK_ORIENTATION_VERTICAL, PREF_PAD_GAP);
453 gq_gtk_container_add(GTK_WIDGET(ew->window), ew->vbox);
454 gtk_widget_show(ew->vbox);
456 box = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 0);
458 ew->label_file_name = gtk_label_new("");
459 gtk_label_set_ellipsize(GTK_LABEL(ew->label_file_name), PANGO_ELLIPSIZE_START);
460 gtk_label_set_selectable(GTK_LABEL(ew->label_file_name), TRUE);
461 gtk_label_set_xalign(GTK_LABEL(ew->label_file_name), 0.5);
462 gtk_label_set_yalign(GTK_LABEL(ew->label_file_name), 0.5);
464 gq_gtk_box_pack_start(GTK_BOX(box), ew->label_file_name, TRUE, TRUE, 0);
465 gtk_widget_show(ew->label_file_name);
467 gq_gtk_box_pack_start(GTK_BOX(ew->vbox), box, FALSE, FALSE, 0);
468 gtk_widget_show(box);
471 store = gtk_list_store_new(7, G_TYPE_BOOLEAN,
472 G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING,
473 G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING);
476 sortable = GTK_TREE_SORTABLE(store);
477 for (n = EXIF_ADVCOL_DESCRIPTION; n <= EXIF_ADVCOL_ELEMENTS; n++)
478 gtk_tree_sortable_set_sort_func(sortable, n, advanced_exif_sort_cb,
479 GINT_TO_POINTER(n), nullptr);
481 /* set initial sort order */
482 gtk_tree_sortable_set_sort_column_id(sortable, EXIF_ADVCOL_NAME, GTK_SORT_ASCENDING);
484 ew->listview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
485 g_object_unref(store);
487 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(ew->listview), TRUE);
489 advanced_exif_add_column(ew->listview, _("Description"), EXIF_ADVCOL_DESCRIPTION, FALSE);
490 advanced_exif_add_column(ew->listview, _("Value"), EXIF_ADVCOL_VALUE, TRUE);
491 advanced_exif_add_column(ew->listview, _("Name"), EXIF_ADVCOL_NAME, FALSE);
492 advanced_exif_add_column(ew->listview, _("Tag"), EXIF_ADVCOL_TAG, FALSE);
493 advanced_exif_add_column(ew->listview, _("Format"), EXIF_ADVCOL_FORMAT, FALSE);
494 advanced_exif_add_column(ew->listview, _("Elements"), EXIF_ADVCOL_ELEMENTS, FALSE);
496 gtk_tree_view_set_enable_search(GTK_TREE_VIEW(ew->listview), TRUE);
497 gtk_tree_view_set_search_column(GTK_TREE_VIEW(ew->listview), EXIF_ADVCOL_DESCRIPTION);
498 gtk_tree_view_set_search_equal_func(GTK_TREE_VIEW(ew->listview), search_function_cb, ew, nullptr);
500 gtk_drag_source_set(ew->listview,
501 static_cast<GdkModifierType>(GDK_BUTTON1_MASK | GDK_BUTTON2_MASK),
502 advanced_exif_drag_types, n_exif_drag_types,
503 static_cast<GdkDragAction>(GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK));
505 g_signal_connect(G_OBJECT(ew->listview), "drag_data_get",
506 G_CALLBACK(advanced_exif_dnd_get), ew);
508 g_signal_connect(G_OBJECT(ew->listview), "drag_begin",
509 G_CALLBACK(advanced_exif_dnd_begin), ew);
511 g_signal_connect(G_OBJECT(ew->window), "key_press_event",
512 G_CALLBACK(advanced_exif_keypress), ew);
514 g_signal_connect(G_OBJECT(ew->listview), "button_release_event",
515 G_CALLBACK(advanced_exif_mouseclick), ew);
517 ew->scrolled = gq_gtk_scrolled_window_new(nullptr, nullptr);
518 gq_gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(ew->scrolled), GTK_SHADOW_IN);
519 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(ew->scrolled),
520 GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
521 gq_gtk_box_pack_start(GTK_BOX(ew->vbox), ew->scrolled, TRUE, TRUE, 0);
522 gq_gtk_container_add(GTK_WIDGET(ew->scrolled), ew->listview);
523 gtk_widget_show(ew->listview);
524 gtk_widget_show(ew->scrolled);
526 gtk_widget_show(ew->window);
529 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */