Implementing CTRL-W to close advanced exif window
[geeqie.git] / src / advanced_exif.c
1 /*
2  * Geeqie
3  * (C) 2004 John Ellis
4  * Copyright (C) 2008 - 2009 The Geeqie Team
5  *
6  * Author: John Ellis
7  *
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!
11  */
12
13
14 #include "main.h"
15 #include "advanced_exif.h"
16
17 #include "exif.h"
18 #include "metadata.h"
19 #include "filedata.h"
20 #include "history_list.h"
21 #include "misc.h"
22 #include "ui_misc.h"
23 #include "window.h"
24 #include "dnd.h"
25
26 /* FIXME: not needed when bar_exif.c is improved */
27 #include "bar_exif.h"
28
29 #include <math.h>
30
31 #define ADVANCED_EXIF_DATA_COLUMN_WIDTH 200
32
33 /*
34  *-------------------------------------------------------------------
35  * EXIF window
36  *-------------------------------------------------------------------
37  */
38
39 typedef struct _ExifWin ExifWin;
40 struct _ExifWin
41 {
42         GtkWidget *window;
43         GtkWidget *vbox;
44         GtkWidget *scrolled;
45         GtkWidget *listview;
46         GtkWidget *label_file_name;
47
48         FileData *fd;
49 };
50
51 enum {
52         EXIF_ADVCOL_ENABLED = 0,
53         EXIF_ADVCOL_TAG,
54         EXIF_ADVCOL_NAME,
55         EXIF_ADVCOL_VALUE,
56         EXIF_ADVCOL_FORMAT,
57         EXIF_ADVCOL_ELEMENTS,
58         EXIF_ADVCOL_DESCRIPTION,
59         EXIF_ADVCOL_COUNT
60 };
61
62 static gboolean advanced_exif_row_enabled(const gchar *name)
63 {
64         GList *list;
65
66         if (!name) return FALSE;
67
68         list = history_list_get_by_key("exif_extras");
69         while (list)
70                 {
71                 if (strcmp(name, (gchar *)(list->data)) == 0) return TRUE;
72                 list = list->next;
73         }
74
75         return FALSE;
76 }
77
78 static void advanced_exif_update(ExifWin *ew)
79 {
80         ExifData *exif;
81
82         GtkListStore *store;
83         GtkTreeIter iter;
84         ExifData *exif_original;
85         ExifItem *item;
86
87         exif = exif_read_fd(ew->fd);
88         
89         gtk_widget_set_sensitive(ew->scrolled, !!exif);
90
91         if (!exif) return;
92         
93         exif_original = exif_get_original(exif);
94
95         store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(ew->listview)));
96         gtk_list_store_clear(store);
97
98         item = exif_get_first_item(exif_original);
99         while (item)
100                 {
101                 gchar *tag;
102                 gchar *tag_name;
103                 gchar *text;
104                 gchar *utf8_text;
105                 const gchar *format;
106                 gchar *elements;
107                 gchar *description;
108
109                 tag = g_strdup_printf("0x%04x", exif_item_get_tag_id(item));
110                 tag_name = exif_item_get_tag_name(item);
111                 format = exif_item_get_format_name(item, TRUE);
112                 text = exif_item_get_data_as_text(item);
113                 utf8_text = utf8_validate_or_convert(text);
114                 g_free(text);
115                 elements = g_strdup_printf("%d", exif_item_get_elements(item));
116                 description = exif_item_get_description(item);
117                 if (!description || *description == '\0') 
118                         {
119                         g_free(description);
120                         description = g_strdup(tag_name);
121                         }
122
123                 gtk_list_store_append(store, &iter);
124                 gtk_list_store_set(store, &iter,
125                                 EXIF_ADVCOL_ENABLED, advanced_exif_row_enabled(tag_name),
126                                 EXIF_ADVCOL_TAG, tag,
127                                 EXIF_ADVCOL_NAME, tag_name,
128                                 EXIF_ADVCOL_VALUE, utf8_text,
129                                 EXIF_ADVCOL_FORMAT, format,
130                                 EXIF_ADVCOL_ELEMENTS, elements,
131                                 EXIF_ADVCOL_DESCRIPTION, description, -1);
132                 g_free(tag);
133                 g_free(utf8_text);
134                 g_free(elements);
135                 g_free(description);
136                 g_free(tag_name);
137                 item = exif_get_next_item(exif_original);
138                 }
139         exif_free_fd(ew->fd, exif);
140
141 }
142
143 static void advanced_exif_clear(ExifWin *ew)
144 {
145         GtkListStore *store;
146
147         store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(ew->listview)));
148         gtk_list_store_clear(store);
149 }
150
151 void advanced_exif_set_fd(GtkWidget *window, FileData *fd)
152 {
153         ExifWin *ew;
154
155         ew = g_object_get_data(G_OBJECT(window), "advanced_exif_data");
156         if (!ew) return;
157
158         /* store this, advanced view toggle needs to reload data */
159         file_data_unref(ew->fd);
160         ew->fd = file_data_ref(fd);
161
162         gtk_label_set_text(GTK_LABEL(ew->label_file_name), (ew->fd) ? ew->fd->path : "");
163
164         advanced_exif_clear(ew);
165         advanced_exif_update(ew);
166 }
167
168 #if 0
169 static void advanced_exif_row_toggled_cb(GtkCellRendererToggle *toggle, const gchar *path, gpointer data)
170 {
171         GtkWidget *listview = data;
172         GtkTreeModel *store;
173         GtkTreeIter iter;
174         GtkTreePath *tpath;
175         gchar *name = NULL;
176         gboolean active;
177
178         store = gtk_tree_view_get_model(GTK_TREE_VIEW(listview));
179
180         tpath = gtk_tree_path_new_from_string(path);
181         gtk_tree_model_get_iter(store, &iter, tpath);
182         gtk_tree_path_free(tpath);
183
184         gtk_tree_model_get(store, &iter, EXIF_ADVCOL_ENABLED, &active,
185                                          EXIF_ADVCOL_NAME, &name, -1);
186         active = (!active);
187
188         if (active &&
189             g_list_length(history_list_get_by_key("exif_extras")) >= EXIF_BAR_CUSTOM_COUNT)
190                 {
191                 active = FALSE;
192                 }
193
194         gtk_list_store_set(GTK_LIST_STORE(store), &iter, EXIF_ADVCOL_ENABLED, active, -1);
195
196         if (active)
197                 {
198                 history_list_add_to_key("exif_extras", name, EXIF_BAR_CUSTOM_COUNT);
199                 }
200         else
201                 {
202                 history_list_item_change("exif_extras", name, NULL);
203                 }
204
205         g_free(name);
206 }
207 #endif 
208
209 #if 0
210 static void advanced_exif_add_column_check(GtkWidget *listview, const gchar *title, gint n)
211 {
212         GtkTreeViewColumn *column;
213         GtkCellRenderer *renderer;
214
215         column = gtk_tree_view_column_new();
216         gtk_tree_view_column_set_title(column, title);
217         gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
218
219         renderer = gtk_cell_renderer_toggle_new();
220         gtk_tree_view_column_pack_start(column, renderer, TRUE);
221         gtk_tree_view_column_add_attribute(column, renderer, "active", n);
222         gtk_tree_view_append_column(GTK_TREE_VIEW(listview), column);
223
224         g_signal_connect(G_OBJECT(renderer), "toggled",
225                          G_CALLBACK(advanced_exif_row_toggled_cb), listview);
226 }
227 #endif
228
229 static GtkTargetEntry advanced_exif_drag_types[] = {
230         { "text/plain", 0, TARGET_TEXT_PLAIN }
231 };
232 static gint n_exif_drag_types = 1;
233
234
235 static void advanced_exif_dnd_get(GtkWidget *listview, GdkDragContext *context,
236                                   GtkSelectionData *selection_data, guint info,
237                                   guint time, gpointer data)
238 {
239         //ExifWin *ew = data;
240         GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(listview)); 
241         GtkTreeIter iter;
242
243         if (gtk_tree_selection_get_selected(sel, NULL, &iter)) 
244                 {
245                 GtkTreeModel *store = gtk_tree_view_get_model(GTK_TREE_VIEW(listview));
246                 gchar *key;
247
248                 gtk_tree_model_get(store, &iter, EXIF_ADVCOL_NAME, &key, -1);
249                 gtk_selection_data_set_text(selection_data, key, -1);
250                 //printf("%s\n",key);
251                 g_free(key);
252                 }
253
254 }
255
256
257 static void advanced_exif_dnd_begin(GtkWidget *listview, GdkDragContext *context, gpointer data)
258 {
259         //ExifWin *ew = data;
260         GtkTreeSelection *sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(listview)); 
261         GtkTreeIter iter;
262
263         if (gtk_tree_selection_get_selected(sel, NULL, &iter)) 
264                 {
265                 GtkTreeModel *store = gtk_tree_view_get_model(GTK_TREE_VIEW(listview));
266                 gchar *key;
267
268                 gtk_tree_model_get(store, &iter, EXIF_ADVCOL_NAME, &key, -1);
269
270                 dnd_set_drag_label(listview, context, key);
271                 g_free(key);
272                 }
273 }
274
275
276
277 static void advanced_exif_add_column(GtkWidget *listview, const gchar *title, gint n, gboolean sizable)
278 {
279         GtkTreeViewColumn *column;
280         GtkCellRenderer *renderer;
281
282         column = gtk_tree_view_column_new();
283         gtk_tree_view_column_set_title(column, title);
284
285         if (sizable)
286                 {
287                 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_FIXED);
288                 gtk_tree_view_column_set_fixed_width(column, ADVANCED_EXIF_DATA_COLUMN_WIDTH);
289                 }
290         else
291                 {
292                 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
293                 }
294         
295         gtk_tree_view_column_set_resizable(column, TRUE);
296         gtk_tree_view_column_set_sort_column_id(column, n);
297
298         renderer = gtk_cell_renderer_text_new();
299         gtk_tree_view_column_pack_start(column, renderer, TRUE);
300         gtk_tree_view_column_add_attribute(column, renderer, "text", n);
301         gtk_tree_view_append_column(GTK_TREE_VIEW(listview), column);
302 }
303
304 void advanced_exif_close(ExifWin *ew)
305 {
306         if (!ew) return;
307
308         gtk_widget_destroy(ew->window);
309 }
310
311 static void advanced_exif_destroy(GtkWidget *widget, gpointer data)
312 {
313         ExifWin *ew = data;
314         file_data_unref(ew->fd);
315         g_free(ew);
316 }
317
318 static gint advanced_exif_sort_cb(GtkTreeModel *model, GtkTreeIter *a, GtkTreeIter *b, gpointer data)
319 {
320         gint n = GPOINTER_TO_INT(data);
321         gint ret = 0;
322
323         switch (n)
324                 {
325                 case EXIF_ADVCOL_DESCRIPTION:
326                 case EXIF_ADVCOL_VALUE:
327                 case EXIF_ADVCOL_NAME:
328                 case EXIF_ADVCOL_TAG:
329                 case EXIF_ADVCOL_FORMAT:
330                 case EXIF_ADVCOL_ELEMENTS:
331                         {
332                         gchar *s1, *s2;
333
334                         gtk_tree_model_get(model, a, n, &s1, -1);
335                         gtk_tree_model_get(model, b, n, &s2, -1);
336
337                         if (!s1 || !s2)
338                                 {
339                                 if (!s1 && !s2) break;
340                                 ret = s1 ? 1 : -1;
341                                 }
342                         else
343                                 {
344                                 ret = g_utf8_collate(s1, s2);
345                                 }
346
347                         g_free(s1);
348                         g_free(s2);
349                         }
350                         break;
351
352                 default:
353                         g_return_val_if_reached(0);
354                 }
355
356         return ret;
357 }
358
359 static gboolean advanced_exif_keypress(GtkWidget *widget, GdkEventKey *event, gpointer data)
360 {
361         ExifWin *ew = data;
362         gboolean stop_signal = FALSE;
363
364         if (event->state & GDK_CONTROL_MASK)
365                 {
366                 switch (event->keyval)
367                         {
368                         case 'W': case 'w':
369                                 advanced_exif_close(ew);
370                                 stop_signal = TRUE;
371                                 break;
372                         }
373                 } // if (event->state & GDK_CONTROL...
374
375         return stop_signal;
376 } // static gboolean advanced_exif_...
377
378 GtkWidget *advanced_exif_new(void)
379 {
380         ExifWin *ew;
381         GtkListStore *store;
382         GdkGeometry geometry;
383         GtkTreeSortable *sortable;
384         GtkWidget *box;
385         gint n;
386
387         ew = g_new0(ExifWin, 1);
388
389         ew->window = window_new(GTK_WINDOW_TOPLEVEL, "view", NULL, NULL, _("Metadata"));
390
391         geometry.min_width = 900;
392         geometry.min_height = 600;
393         gtk_window_set_geometry_hints(GTK_WINDOW(ew->window), NULL, &geometry, GDK_HINT_MIN_SIZE);
394
395         gtk_window_set_resizable(GTK_WINDOW(ew->window), TRUE);
396
397         g_object_set_data(G_OBJECT(ew->window), "advanced_exif_data", ew);
398         g_signal_connect_after(G_OBJECT(ew->window), "destroy",
399                                G_CALLBACK(advanced_exif_destroy), ew);
400
401         ew->vbox = gtk_vbox_new(FALSE, PREF_PAD_GAP);
402         gtk_container_add(GTK_CONTAINER(ew->window), ew->vbox);
403         gtk_widget_show(ew->vbox);
404
405         box = gtk_hbox_new(FALSE, 0);
406
407         ew->label_file_name = gtk_label_new("");
408         gtk_label_set_ellipsize(GTK_LABEL(ew->label_file_name), PANGO_ELLIPSIZE_START);
409         gtk_label_set_selectable(GTK_LABEL(ew->label_file_name), TRUE);
410         gtk_misc_set_alignment(GTK_MISC(ew->label_file_name), 0.5, 0.5);
411         gtk_box_pack_start(GTK_BOX(box), ew->label_file_name, TRUE, TRUE, 0);
412         gtk_widget_show(ew->label_file_name);
413
414         gtk_box_pack_start(GTK_BOX(ew->vbox), box, FALSE, FALSE, 0);
415         gtk_widget_show(box);
416
417
418         store = gtk_list_store_new(7, G_TYPE_BOOLEAN,
419                                       G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING,
420                                       G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING);
421
422         /* set up sorting */
423         sortable = GTK_TREE_SORTABLE(store);
424         for (n = EXIF_ADVCOL_DESCRIPTION; n <= EXIF_ADVCOL_ELEMENTS; n++)
425                 gtk_tree_sortable_set_sort_func(sortable, n, advanced_exif_sort_cb,
426                                                 GINT_TO_POINTER(n), NULL);
427
428         /* set initial sort order */
429         gtk_tree_sortable_set_sort_column_id(sortable, EXIF_ADVCOL_NAME, GTK_SORT_ASCENDING);
430
431         ew->listview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
432         g_object_unref(store);
433
434         gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(ew->listview), TRUE);
435         gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(ew->listview), TRUE);
436
437         advanced_exif_add_column(ew->listview, _("Description"), EXIF_ADVCOL_DESCRIPTION, FALSE);
438         advanced_exif_add_column(ew->listview, _("Value"), EXIF_ADVCOL_VALUE, TRUE);
439         advanced_exif_add_column(ew->listview, _("Name"), EXIF_ADVCOL_NAME, FALSE);
440         advanced_exif_add_column(ew->listview, _("Tag"), EXIF_ADVCOL_TAG, FALSE);
441         advanced_exif_add_column(ew->listview, _("Format"), EXIF_ADVCOL_FORMAT, FALSE);
442         advanced_exif_add_column(ew->listview, _("Elements"), EXIF_ADVCOL_ELEMENTS, FALSE);
443         
444
445         gtk_drag_source_set(ew->listview,
446                            GDK_BUTTON1_MASK | GDK_BUTTON2_MASK,
447                            advanced_exif_drag_types, n_exif_drag_types,
448                            GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK);
449
450         g_signal_connect(G_OBJECT(ew->listview), "drag_data_get",
451                          G_CALLBACK(advanced_exif_dnd_get), ew);
452
453         g_signal_connect(G_OBJECT(ew->listview), "drag_begin",
454                          G_CALLBACK(advanced_exif_dnd_begin), ew);
455
456         g_signal_connect(G_OBJECT(ew->window), "key_press_event",
457                          G_CALLBACK(advanced_exif_keypress), ew);
458
459         ew->scrolled = gtk_scrolled_window_new(NULL, NULL);
460         gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(ew->scrolled), GTK_SHADOW_IN);
461         gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(ew->scrolled),
462                                        GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
463         gtk_box_pack_start(GTK_BOX(ew->vbox), ew->scrolled, TRUE, TRUE, 0);
464         gtk_container_add(GTK_CONTAINER(ew->scrolled), ew->listview);
465         gtk_widget_show(ew->listview);
466         gtk_widget_show(ew->scrolled);
467
468         gtk_widget_show(ew->window);
469         return ew->window;
470 }
471 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */