split filelist.c to filefilter.c and filedata.c
[geeqie.git] / src / bar_exif.c
1 /*
2  * Geeqie
3  * (C) 2004 John Ellis
4  * Copyright (C) 2008 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 "bar_exif.h"
16
17 #include "exif.h"
18 #include "ui_bookmark.h"
19 #include "ui_misc.h"
20 #include "filedata.h"
21
22 #include <math.h>
23
24 #define EXIF_BAR_SIZE_INCREMENT 48
25 #define EXIF_BAR_ARROW_SIZE 7
26
27 #define EXIF_BAR_CUSTOM_COUNT 20
28
29 #define BAR_EXIF_DATA_COLUMN_WIDTH 250
30
31 ExifUI ExifUIList[]={
32         { 0, 0, EXIF_UI_IFSET,  "formatted.Camera"},
33         { 0, 0, EXIF_UI_IFSET,  "formatted.DateTime"},
34         { 0, 0, EXIF_UI_IFSET,  "formatted.ShutterSpeed"},
35         { 0, 0, EXIF_UI_IFSET,  "formatted.Aperture"},
36         { 0, 0, EXIF_UI_IFSET,  "formatted.ExposureBias"},
37         { 0, 0, EXIF_UI_IFSET,  "formatted.ISOSpeedRating"},
38         { 0, 0, EXIF_UI_IFSET,  "formatted.FocalLength"},
39         { 0, 0, EXIF_UI_IFSET,  "formatted.FocalLength35mmFilm"},
40         { 0, 0, EXIF_UI_IFSET,  "formatted.Flash"},
41         { 0, 0, EXIF_UI_IFSET,  "formatted.SubjectDistance"},
42         { 0, 0, EXIF_UI_IFSET,  "Exif.Photo.ExposureProgram"},
43         { 0, 0, EXIF_UI_IFSET,  "Exif.Photo.MeteringMode"},
44         { 0, 0, EXIF_UI_IFSET,  "Exif.Photo.LightSource"},
45         { 0, 0, EXIF_UI_IFSET,  "formatted.ColorProfile"},
46         { 0, 0, EXIF_UI_IFSET,  "formatted.SubjectDistance"},
47         { 0, 0, EXIF_UI_IFSET,  "formatted.Resolution"},
48         { 0, 0, EXIF_UI_IFSET,  "Exif.Image.Orientation"},
49         { 0, 0, EXIF_UI_IFSET,  "Exif.Image.ImageDescription"},
50         { 0, 0, EXIF_UI_IFSET,  "Exif.Image.Copyright"},
51         { 0, 0, EXIF_UI_OFF,    NULL}
52 };
53
54
55 /*
56  *-------------------------------------------------------------------
57  * table util
58  *-------------------------------------------------------------------
59  */
60
61 static void table_add_line_custom(GtkWidget *table, gint x, gint y,
62                                   const gchar *text1, const gchar *text2,
63                                   GtkWidget **label1, GtkWidget **label2)
64 {
65         GtkWidget *label;
66         gchar *buf;
67
68         buf = g_strconcat((text1) ? text1 : "fixme", ":", NULL);
69         if (!text2) text2 = "";
70
71         label = gtk_label_new(buf);
72         gtk_misc_set_alignment(GTK_MISC(label), 1.0, 0.0);
73         pref_label_bold(label, TRUE, FALSE);
74         gtk_table_attach(GTK_TABLE(table), label,
75                          x, x + 1, y, y + 1,
76                          GTK_FILL, GTK_FILL,
77                          2, 2);
78         *label1 = label;
79
80         label = gtk_label_new(text2);
81         gtk_misc_set_alignment(GTK_MISC(label), 0.0, 0.0);
82         gtk_table_attach(GTK_TABLE(table), label,
83                          x + 1, x + 2, y, y + 1,
84                          GTK_FILL, GTK_FILL,
85                          2, 2);
86         *label2 = label;
87
88         g_free(buf);
89 }
90
91 static GtkWidget *table_add_line(GtkWidget *table, gint x, gint y,
92                                  const gchar *description, const gchar *text,
93                                 GtkWidget **keyret)
94 {
95         GtkWidget *key;
96         GtkWidget *label;
97
98         table_add_line_custom(table, x, y, description, text, &key, &label);
99         gtk_widget_show(key);
100         gtk_widget_show(label);
101         if (keyret) *keyret = key;
102
103         return label;
104 }
105
106
107 /*
108  *-------------------------------------------------------------------
109  * EXIF bar
110  *-------------------------------------------------------------------
111  */
112
113 typedef struct _ExifBar ExifBar;
114 struct _ExifBar
115 {
116         GtkWidget *vbox;
117         GtkWidget *scrolled;
118         GtkWidget *table;
119         GtkWidget *advanced_scrolled;
120         GtkWidget *listview;
121         GtkWidget **keys;
122         GtkWidget **labels;
123
124         GtkWidget *custom_sep;
125         GtkWidget *custom_name[EXIF_BAR_CUSTOM_COUNT];
126         GtkWidget *custom_value[EXIF_BAR_CUSTOM_COUNT];
127
128         FileData *fd;
129
130         gint allow_search;
131 };
132
133 enum {
134         EXIF_ADVCOL_ENABLED = 0,
135         EXIF_ADVCOL_TAG,
136         EXIF_ADVCOL_NAME,
137         EXIF_ADVCOL_VALUE,
138         EXIF_ADVCOL_FORMAT,
139         EXIF_ADVCOL_ELEMENTS,
140         EXIF_ADVCOL_DESCRIPTION,
141         EXIF_ADVCOL_COUNT
142 };
143
144 gchar *bar_exif_validate_text(gchar *text)
145 {
146         if (text && !g_utf8_validate(text, strlen(text), NULL))
147                 {
148                 gchar *tmp = text;
149                 text = g_convert(tmp, strlen(tmp), "UTF-8", "ISO-8859-1", NULL, NULL, NULL);
150                 g_free(tmp);
151                 }
152         return text;
153 }
154
155 static void bar_exif_sensitive(ExifBar *eb, gint enable)
156 {
157         gtk_widget_set_sensitive(eb->table, enable);
158         if (eb->advanced_scrolled) gtk_widget_set_sensitive(eb->advanced_scrolled, enable);
159 }
160
161 static gint bar_exif_row_enabled(const gchar *name)
162 {
163         GList *list;
164
165         list = history_list_get_by_key("exif_extras");
166         while (list)
167                 {
168                 if (name && strcmp(name, (gchar *)(list->data)) == 0) return TRUE;
169                 list = list->next;
170         }
171
172         return FALSE;
173 }
174
175 static void bar_exif_update(ExifBar *eb)
176 {
177         ExifData *exif;
178         gint i;
179
180         exif = exif_read_fd(eb->fd);
181
182         if (!exif)
183                 {
184                 bar_exif_sensitive(eb, FALSE);
185                 return;
186                 }
187
188         bar_exif_sensitive(eb, TRUE);
189
190         if (GTK_WIDGET_VISIBLE(eb->scrolled))
191                 {
192                 GList *list;
193                 for (i = 0; ExifUIList[i].key; i++)
194                         {
195                         gchar *text;
196
197                         if (ExifUIList[i].current == EXIF_UI_OFF)
198                                 {
199                                 gtk_widget_hide(eb->labels[i]);
200                                 gtk_widget_hide(eb->keys[i]);
201                                 continue;
202                                 }
203                         text = exif_get_data_as_text(exif, ExifUIList[i].key);
204                         text = bar_exif_validate_text(text);
205                         if (ExifUIList[i].current == EXIF_UI_IFSET
206                             && (!text || !*text))
207                                 {
208                                 gtk_widget_hide(eb->labels[i]);
209                                 gtk_widget_hide(eb->keys[i]);
210                                 continue;
211                                 }
212                         gtk_widget_show(eb->labels[i]);
213                         gtk_widget_show(eb->keys[i]);
214                         gtk_label_set_text(GTK_LABEL(eb->labels[i]), text);
215                         g_free(text);
216                         }
217
218                 list = g_list_last(history_list_get_by_key("exif_extras"));
219                 if (list)
220                         {
221                         gtk_widget_show(eb->custom_sep);
222                         }
223                 else
224                         {
225                         gtk_widget_hide(eb->custom_sep);
226                         }
227                 i = 0;
228                 while (list && i < EXIF_BAR_CUSTOM_COUNT)
229                         {
230                         gchar *text;
231                         gchar *name;
232                         gchar *buf;
233
234                         name = list->data;
235                         list = list->prev;
236
237                         text = exif_get_data_as_text(exif, name);
238                         text = bar_exif_validate_text(text);
239
240                         buf = g_strconcat(name, ":", NULL);
241                         gtk_label_set_text(GTK_LABEL(eb->custom_name[i]), buf);
242                         g_free(buf);
243                         gtk_label_set_text(GTK_LABEL(eb->custom_value[i]), text);
244                         g_free(text);
245
246                         gtk_widget_show(eb->custom_name[i]);
247                         gtk_widget_show(eb->custom_value[i]);
248
249                         i++;
250                         }
251                 while (i < EXIF_BAR_CUSTOM_COUNT)
252                         {
253                         gtk_widget_hide(eb->custom_name[i]);
254                         gtk_widget_hide(eb->custom_value[i]);
255                         i++;
256                         }
257                 }
258
259         if (eb->advanced_scrolled && GTK_WIDGET_VISIBLE(eb->advanced_scrolled))
260                 {
261                 GtkListStore *store;
262                 GtkTreeIter iter;
263                 ExifItem *item;
264
265                 store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(eb->listview)));
266                 gtk_list_store_clear(store);
267
268                 item = exif_get_first_item(exif);
269                 while (item)
270                         {
271                         gchar *tag;
272                         gchar *tag_name;
273                         gchar *text;
274                         const gchar *format;
275                         gchar *elements;
276                         gchar *description;
277
278                         tag = g_strdup_printf("0x%04x", exif_item_get_tag_id(item));
279                         tag_name = exif_item_get_tag_name(item);
280                         format = exif_item_get_format_name(item, TRUE);
281                         text = exif_item_get_data_as_text(item);
282                         text = bar_exif_validate_text(text);
283                         elements = g_strdup_printf("%d", exif_item_get_elements(item));
284                         description = exif_item_get_description(item);
285                         if (!description) description = g_strdup("");
286                         description = bar_exif_validate_text(description);
287                         gtk_list_store_append(store, &iter);
288                         gtk_list_store_set(store, &iter,
289                                         EXIF_ADVCOL_ENABLED, bar_exif_row_enabled(tag_name),
290                                         EXIF_ADVCOL_TAG, tag,
291                                         EXIF_ADVCOL_NAME, tag_name,
292                                         EXIF_ADVCOL_VALUE, text,
293                                         EXIF_ADVCOL_FORMAT, format,
294                                         EXIF_ADVCOL_ELEMENTS, elements,
295                                         EXIF_ADVCOL_DESCRIPTION, description, -1);
296                         g_free(tag);
297                         g_free(text);
298                         g_free(elements);
299                         g_free(description);
300                         g_free(tag_name);
301                         item = exif_get_next_item(exif);
302                         }
303                 }
304
305         exif_free(exif);
306 }
307
308 static void bar_exif_clear(ExifBar *eb)
309 {
310         gint i;
311
312         if (!GTK_WIDGET_SENSITIVE(eb->labels[0])) return;
313
314         for (i = 0; ExifUIList[i].key; i++)
315                 {
316                 gtk_label_set_text(GTK_LABEL(eb->labels[i]), "");
317                 }
318         for (i = 0; i < EXIF_BAR_CUSTOM_COUNT; i++)
319                 {
320                 gtk_label_set_text(GTK_LABEL(eb->custom_value[i]), "");
321                 }
322
323         if (eb->listview)
324                 {
325                 GtkListStore *store;
326
327                 store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(eb->listview)));
328                 gtk_list_store_clear(store);
329                 }
330 }
331
332 void bar_exif_set(GtkWidget *bar, FileData *fd)
333 {
334         ExifBar *eb;
335
336         eb = g_object_get_data(G_OBJECT(bar), "bar_exif_data");
337         if (!eb) return;
338
339         /* store this, advanced view toggle needs to reload data */
340         file_data_unref(eb->fd);
341         eb->fd = file_data_ref(fd);
342
343         bar_exif_clear(eb);
344         bar_exif_update(eb);
345 }
346
347 static void bar_exif_row_toggled_cb(GtkCellRendererToggle *toggle, const gchar *path, gpointer data)
348 {
349         GtkWidget *listview = data;
350         GtkTreeModel *store;
351         GtkTreeIter iter;
352         GtkTreePath *tpath;
353         gchar *name = NULL;
354         gboolean active;
355
356         store = gtk_tree_view_get_model(GTK_TREE_VIEW(listview));
357
358         tpath = gtk_tree_path_new_from_string(path);
359         gtk_tree_model_get_iter(store, &iter, tpath);
360         gtk_tree_path_free(tpath);
361
362         gtk_tree_model_get(store, &iter, EXIF_ADVCOL_ENABLED, &active,
363                                          EXIF_ADVCOL_NAME, &name, -1);
364         active = (!active);
365
366         if (active &&
367             g_list_length(history_list_get_by_key("exif_extras")) >= EXIF_BAR_CUSTOM_COUNT)
368                 {
369                 active = FALSE;
370                 }
371
372         gtk_list_store_set(GTK_LIST_STORE(store), &iter, EXIF_ADVCOL_ENABLED, active, -1);
373
374         if (active)
375                 {
376                 history_list_add_to_key("exif_extras", name, EXIF_BAR_CUSTOM_COUNT);
377                 }
378         else
379                 {
380                 history_list_item_change("exif_extras", name, NULL);
381                 }
382
383         g_free(name);
384 }
385
386 static void bar_exif_add_column_check(GtkWidget *listview, const gchar *title, gint n)
387 {
388         GtkTreeViewColumn *column;
389         GtkCellRenderer *renderer;
390
391         column = gtk_tree_view_column_new();
392         gtk_tree_view_column_set_title(column, title);
393         gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
394
395         renderer = gtk_cell_renderer_toggle_new();
396         gtk_tree_view_column_pack_start(column, renderer, TRUE);
397         gtk_tree_view_column_add_attribute(column, renderer, "active", n);
398         gtk_tree_view_append_column(GTK_TREE_VIEW(listview), column);
399
400         g_signal_connect(G_OBJECT(renderer), "toggled",
401                          G_CALLBACK(bar_exif_row_toggled_cb), listview);
402 }
403
404 static void bar_exif_add_column(GtkWidget *listview, const gchar *title, gint n, gint sizable)
405 {
406         GtkTreeViewColumn *column;
407         GtkCellRenderer *renderer;
408
409         column = gtk_tree_view_column_new();
410         gtk_tree_view_column_set_title(column, title);
411
412         if (sizable)
413                 {
414                 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_FIXED);
415                 gtk_tree_view_column_set_fixed_width(column, BAR_EXIF_DATA_COLUMN_WIDTH);
416                 gtk_tree_view_column_set_resizable(column, TRUE);
417                 }
418         else
419                 {
420                 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
421                 }
422
423         renderer = gtk_cell_renderer_text_new();
424         gtk_tree_view_column_pack_start(column, renderer, TRUE);
425         gtk_tree_view_column_add_attribute(column, renderer, "text", n);
426         gtk_tree_view_append_column(GTK_TREE_VIEW(listview), column);
427 }
428
429 static void bar_exif_advanced_build_view(ExifBar *eb)
430 {
431         GtkListStore *store;
432
433         if (eb->listview) return;
434
435         store = gtk_list_store_new(7, G_TYPE_BOOLEAN,
436                                       G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING,
437                                       G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING);
438         eb->listview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
439         g_object_unref(store);
440
441         gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(eb->listview), TRUE);
442         gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(eb->listview), TRUE);
443
444         if (eb->allow_search)
445                 {
446                 gtk_tree_view_set_search_column(GTK_TREE_VIEW(eb->listview), EXIF_ADVCOL_NAME);
447                 }
448         else
449                 {
450                 gtk_tree_view_set_enable_search(GTK_TREE_VIEW(eb->listview), FALSE);
451                 }
452
453         bar_exif_add_column_check(eb->listview, "", EXIF_ADVCOL_ENABLED);
454
455         bar_exif_add_column(eb->listview, _("Tag"), EXIF_ADVCOL_TAG, FALSE);
456         bar_exif_add_column(eb->listview, _("Name"), EXIF_ADVCOL_NAME, FALSE);
457         bar_exif_add_column(eb->listview, _("Value"), EXIF_ADVCOL_VALUE, TRUE);
458         bar_exif_add_column(eb->listview, _("Format"), EXIF_ADVCOL_FORMAT, FALSE);
459         bar_exif_add_column(eb->listview, _("Elements"), EXIF_ADVCOL_ELEMENTS, FALSE);
460         bar_exif_add_column(eb->listview, _("Description"), EXIF_ADVCOL_DESCRIPTION, FALSE);
461
462         eb->advanced_scrolled = gtk_scrolled_window_new(NULL, NULL);
463         gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(eb->advanced_scrolled), GTK_SHADOW_IN);
464         gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(eb->advanced_scrolled),
465                                        GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
466         gtk_box_pack_start(GTK_BOX(eb->vbox), eb->advanced_scrolled, TRUE, TRUE, 0);
467         gtk_container_add(GTK_CONTAINER(eb->advanced_scrolled), eb->listview);
468         gtk_widget_show(eb->listview);
469 }
470
471 static void bar_exif_advanced_cb(GtkWidget *widget, gpointer data)
472 {
473         ExifBar *eb = data;
474         gint advanced;
475
476         advanced = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
477
478         if (advanced)
479                 {
480                 gtk_widget_hide(eb->scrolled);
481                 bar_exif_advanced_build_view(eb);
482                 gtk_widget_show(eb->advanced_scrolled);
483                 }
484         else
485                 {
486                 gtk_widget_hide(eb->advanced_scrolled);
487                 gtk_widget_show(eb->scrolled);
488                 }
489
490         bar_exif_update(eb);
491 }
492
493 gint bar_exif_is_advanced(GtkWidget *bar)
494 {
495         ExifBar *eb;
496
497         eb = g_object_get_data(G_OBJECT(bar), "bar_exif_data");
498         if (!eb) return FALSE;
499
500         return (eb->advanced_scrolled && GTK_WIDGET_VISIBLE(eb->advanced_scrolled));
501 }
502
503 void bar_exif_close(GtkWidget *bar)
504 {
505         ExifBar *eb;
506
507         eb = g_object_get_data(G_OBJECT(bar), "bar_exif_data");
508         if (!eb) return;
509
510         gtk_widget_destroy(eb->vbox);
511 }
512
513 static void bar_exif_width(ExifBar *eb, gint val)
514 {
515         gint size;
516
517         size = eb->vbox->allocation.width;
518         size = CLAMP(size + val, EXIF_BAR_SIZE_INCREMENT * 2, EXIF_BAR_SIZE_INCREMENT * 16);
519
520         gtk_widget_set_size_request(eb->vbox, size, -1);
521         options->panels.exif.width = eb->vbox->allocation.width;
522 }
523
524 static void bar_exif_larger(GtkWidget *widget, gpointer data)
525 {
526         ExifBar *eb = data;
527
528         bar_exif_width(eb, EXIF_BAR_SIZE_INCREMENT);
529 }
530
531 static void bar_exif_smaller(GtkWidget *widget, gpointer data)
532 {
533         ExifBar *eb = data;
534
535         bar_exif_width(eb, -EXIF_BAR_SIZE_INCREMENT);
536 }
537
538 static void bar_exif_destroy(GtkWidget *widget, gpointer data)
539 {
540         ExifBar *eb = data;
541
542         g_free(eb->keys);
543         g_free(eb->labels);
544         file_data_unref(eb->fd);
545         g_free(eb);
546 }
547
548 GtkWidget *bar_exif_new(gint show_title, FileData *fd, gint advanced, GtkWidget *bounding_widget)
549 {
550         ExifBar *eb;
551         GtkWidget *table;
552         GtkWidget *viewport;
553         GtkWidget *hbox;
554         GtkWidget *button;
555         gint i;
556         gint exif_len;
557
558         for (exif_len = 0; ExifUIList[exif_len].key; exif_len++)
559               ;
560
561         eb = g_new0(ExifBar, 1);
562
563         eb->keys = g_new0(GtkWidget *, exif_len);
564         eb->labels = g_new0(GtkWidget *, exif_len);
565
566         eb->vbox = gtk_vbox_new(FALSE, PREF_PAD_GAP);
567         g_object_set_data(G_OBJECT(eb->vbox), "bar_exif_data", eb);
568         g_signal_connect_after(G_OBJECT(eb->vbox), "destroy",
569                                G_CALLBACK(bar_exif_destroy), eb);
570
571         eb->allow_search = !show_title;
572
573         if (show_title)
574                 {
575                 GtkWidget *box;
576                 GtkWidget *label;
577                 GtkWidget *button;
578                 GtkWidget *arrow;
579
580                 box = gtk_hbox_new(FALSE, 0);
581
582                 label = sizer_new(eb->vbox, bounding_widget, SIZER_POS_LEFT);
583                 sizer_set_limits(label, EXIF_BAR_SIZE_INCREMENT * 2, -1, -1 , -1);
584                 gtk_box_pack_start(GTK_BOX(box), label, FALSE, FALSE, 0);
585                 gtk_widget_show(label);
586
587                 label = gtk_label_new(_("Exif"));
588                 pref_label_bold(label, TRUE, FALSE);
589                 gtk_box_pack_start(GTK_BOX(box), label, TRUE, TRUE, 0);
590                 gtk_widget_show(label);
591
592                 button = gtk_button_new();
593                 gtk_button_set_relief(GTK_BUTTON(button), GTK_RELIEF_NONE);
594                 g_signal_connect(G_OBJECT(button), "clicked",
595                                  G_CALLBACK(bar_exif_smaller), eb);
596                 gtk_box_pack_end(GTK_BOX(box), button, FALSE, FALSE, 0);
597                 arrow = gtk_arrow_new(GTK_ARROW_RIGHT, GTK_SHADOW_NONE);
598                 gtk_widget_set_size_request(arrow, EXIF_BAR_ARROW_SIZE, EXIF_BAR_ARROW_SIZE);
599                 gtk_container_add(GTK_CONTAINER(button), arrow);
600                 gtk_widget_show(arrow);
601                 gtk_widget_show(button);
602
603                 button = gtk_button_new();
604                 gtk_button_set_relief(GTK_BUTTON(button), GTK_RELIEF_NONE);
605                 g_signal_connect(G_OBJECT(button), "clicked",
606                                  G_CALLBACK(bar_exif_larger), eb);
607                 gtk_box_pack_end(GTK_BOX(box), button, FALSE, FALSE, 0);
608                 arrow = gtk_arrow_new(GTK_ARROW_LEFT, GTK_SHADOW_NONE);
609                 gtk_widget_set_size_request(arrow, EXIF_BAR_ARROW_SIZE, EXIF_BAR_ARROW_SIZE);
610                 gtk_container_add(GTK_CONTAINER(button), arrow);
611                 gtk_widget_show(arrow);
612                 gtk_widget_show(button);
613
614                 gtk_box_pack_start(GTK_BOX(eb->vbox), box, FALSE, FALSE, 0);
615                 gtk_widget_show(box);
616                 }
617
618
619         table = gtk_table_new(2, exif_len + 1 + EXIF_BAR_CUSTOM_COUNT, FALSE);
620
621         eb->table = table;
622
623         for (i = 0; ExifUIList[i].key; i++)
624                 {
625                 const gchar *text;
626
627                 text = exif_get_description_by_key(ExifUIList[i].key);
628                 eb->labels[i] = table_add_line(table, 0, i, text, NULL,
629                       &eb->keys[i]);
630                 }
631
632         eb->custom_sep = gtk_hseparator_new();
633         gtk_table_attach(GTK_TABLE(table), eb->custom_sep, 0, 1,
634                                            exif_len, exif_len + 1,
635                                            GTK_FILL, GTK_FILL, 2, 2);
636
637         for (i = 0; i < EXIF_BAR_CUSTOM_COUNT; i++)
638                 {
639                 table_add_line_custom(table, 0, exif_len + 1 + i,
640                                       "", "",  &eb->custom_name[i], &eb->custom_value[i]);
641                 }
642
643         eb->scrolled = gtk_scrolled_window_new(NULL, NULL);
644         gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(eb->scrolled),
645                                        GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
646
647         viewport = gtk_viewport_new(NULL, NULL);
648         gtk_viewport_set_shadow_type(GTK_VIEWPORT(viewport), GTK_SHADOW_NONE);
649         gtk_container_add(GTK_CONTAINER(eb->scrolled), viewport);
650         gtk_widget_show(viewport);
651
652         gtk_container_add(GTK_CONTAINER(viewport), table);
653         gtk_widget_show(table);
654
655         gtk_box_pack_start(GTK_BOX(eb->vbox), eb->scrolled, TRUE, TRUE, 0);
656
657         hbox = gtk_hbox_new(FALSE, PREF_PAD_SPACE);
658         gtk_box_pack_end(GTK_BOX(eb->vbox), hbox, FALSE, FALSE, 0);
659         gtk_widget_show(hbox);
660
661         button = gtk_check_button_new_with_label(_("Advanced view"));
662         if (advanced) gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button), TRUE);
663         g_signal_connect(G_OBJECT(button), "toggled",
664                          G_CALLBACK(bar_exif_advanced_cb), eb);
665         gtk_box_pack_end(GTK_BOX(hbox), button, FALSE, FALSE, 0);
666         gtk_widget_show(button);
667
668         eb->advanced_scrolled = NULL;
669         eb->listview = NULL;
670
671         if (advanced)
672                 {
673                 bar_exif_advanced_build_view(eb);
674                 gtk_widget_show(eb->advanced_scrolled);
675                 }
676         else
677                 {
678                 gtk_widget_show(eb->scrolled);
679                 }
680
681         eb->fd = file_data_ref(fd);
682         bar_exif_update(eb);
683
684         return eb->vbox;
685 }