Mercurial > geeqie
view src/bar_exif.c @ 1377:dab37628f5d8
layout_status_update_pixel_cb(): cleanup and optimization, drop pango markup in i18n string, use a small function to calculate numbers length, only allocate text when needed.
author | zas_ |
---|---|
date | Wed, 04 Mar 2009 21:33:19 +0000 |
parents | 27ba259d3496 |
children | 32ff4da1cb20 |
line wrap: on
line source
/* * Geeqie * (C) 2004 John Ellis * Copyright (C) 2008 - 2009 The Geeqie Team * * Author: John Ellis * * This software is released under the GNU General Public License (GNU GPL). * Please read the included file COPYING for more information. * This software comes with no warranty of any kind, use at your own risk! */ #include "main.h" #include "bar_exif.h" #include "exif.h" #include "metadata.h" #include "filedata.h" #include "history_list.h" #include "misc.h" #include "ui_misc.h" #include "bar.h" #include "rcfile.h" #include "dnd.h" #include <math.h> #define MIN_HEIGHT 25 /* *------------------------------------------------------------------- * EXIF widget *------------------------------------------------------------------- */ typedef struct _ExifEntry ExifEntry; struct _ExifEntry { GtkWidget *ebox; GtkWidget *hbox; GtkWidget *title_label; GtkWidget *value_label; gchar *key; gchar *title; gboolean if_set; gboolean auto_title; }; typedef struct _PaneExifData PaneExifData; struct _PaneExifData { PaneData pane; GtkWidget *vbox; GtkWidget *widget; GtkSizeGroup *size_group; gint min_height; FileData *fd; }; static void bar_pane_exif_entry_dnd_init(GtkWidget *entry); static void bar_pane_exif_update_entry(PaneExifData *ped, GtkWidget *entry, gboolean update_title); static void bar_pane_exif_entry_destroy(GtkWidget *widget, gpointer data) { ExifEntry *ee = data; g_free(ee->key); g_free(ee->title); g_free(ee); } static GtkWidget *bar_pane_exif_add_entry(PaneExifData *ped, const gchar *key, const gchar *title, gint if_set) { ExifEntry *ee = g_new0(ExifEntry, 1); ee->key = g_strdup(key); if (title && title[0]) { ee->title = g_strdup(title); } else { ee->title = exif_get_description_by_key(key); ee->auto_title = TRUE; } ee->if_set = if_set; ee->ebox = gtk_event_box_new(); g_object_set_data(G_OBJECT(ee->ebox), "entry_data", ee); g_signal_connect_after(G_OBJECT(ee->ebox), "destroy", G_CALLBACK(bar_pane_exif_entry_destroy), ee); ee->hbox = gtk_hbox_new(FALSE, 0); gtk_container_add(GTK_CONTAINER(ee->ebox), ee->hbox); gtk_widget_show(ee->hbox); ee->title_label = gtk_label_new(NULL); gtk_misc_set_alignment(GTK_MISC(ee->title_label), 1.0, 0.5); gtk_size_group_add_widget(ped->size_group, ee->title_label); gtk_box_pack_start(GTK_BOX(ee->hbox), ee->title_label, FALSE, TRUE, 0); gtk_widget_show(ee->title_label); ee->value_label = gtk_label_new(NULL); // gtk_label_set_width_chars(GTK_LABEL(ee->value_label), 20); gtk_label_set_ellipsize(GTK_LABEL(ee->value_label), PANGO_ELLIPSIZE_END); // gtk_widget_set_size_request(ee->value_label, 100, -1); gtk_misc_set_alignment(GTK_MISC(ee->value_label), 0.0, 0.5); gtk_box_pack_start(GTK_BOX(ee->hbox), ee->value_label, TRUE, TRUE, 1); gtk_widget_show(ee->value_label); gtk_box_pack_start(GTK_BOX(ped->vbox), ee->ebox, TRUE, TRUE, 0); bar_pane_exif_entry_dnd_init(ee->ebox); bar_pane_exif_update_entry(ped, ee->ebox, TRUE); return ee->ebox; } static void bar_pane_exif_reparent_entry(GtkWidget *entry, GtkWidget *pane) { GtkWidget *old_pane = entry->parent; PaneExifData *ped = g_object_get_data(G_OBJECT(pane), "pane_data"); PaneExifData *old_ped = g_object_get_data(G_OBJECT(old_pane), "pane_data"); ExifEntry *ee = g_object_get_data(G_OBJECT(entry), "entry_data"); if (!ped || !old_ped || !ee) return; g_object_ref(entry); gtk_size_group_remove_widget(old_ped->size_group, ee->title_label); gtk_container_remove(GTK_CONTAINER(old_ped->vbox), entry); gtk_size_group_add_widget(ped->size_group, ee->title_label); gtk_box_pack_start(GTK_BOX(ped->vbox), entry, TRUE, TRUE, 0); } static void bar_pane_exif_entry_update_title(ExifEntry *ee) { gchar *markup; markup = g_markup_printf_escaped("<span size='small'>%s:</span>", (ee->title) ? ee->title : "fixme"); gtk_label_set_markup(GTK_LABEL(ee->title_label), markup); g_free(markup); } static void bar_pane_exif_update_entry(PaneExifData *ped, GtkWidget *entry, gboolean update_title) { gchar *text; ExifEntry *ee = g_object_get_data(G_OBJECT(entry), "entry_data"); if (!ee) return; text = metadata_read_string(ped->fd, ee->key, METADATA_FORMATTED); if (ee->if_set && (!text || !*text)) { gtk_label_set_text(GTK_LABEL(ee->value_label), NULL); gtk_widget_hide(entry); } else { gtk_label_set_text(GTK_LABEL(ee->value_label), text); #if GTK_CHECK_VERSION(2,12,0) gtk_widget_set_tooltip_text(ee->hbox, text); #endif gtk_widget_show(entry); } g_free(text); if (update_title) bar_pane_exif_entry_update_title(ee); } static void bar_pane_exif_update(PaneExifData *ped) { GList *list, *work; #if 0 ExifData *exif; /* do we have any exif at all ? */ exif = exif_read_fd(ped->fd); if (!exif) { bar_pane_exif_sensitive(ped, FALSE); return; } else { /* we will use high level functions so we can release it for now. it will stay in the cache */ exif_free_fd(ped->fd, exif); exif = NULL; } bar_pane_exif_sensitive(ped, TRUE); #endif list = gtk_container_get_children(GTK_CONTAINER(ped->vbox)); work = list; while (work) { GtkWidget *entry = work->data; work = work->next; bar_pane_exif_update_entry(ped, entry, FALSE); } g_list_free(list); } void bar_pane_exif_set_fd(GtkWidget *widget, FileData *fd) { PaneExifData *ped; ped = g_object_get_data(G_OBJECT(widget), "pane_data"); if (!ped) return; file_data_unref(ped->fd); ped->fd = file_data_ref(fd); bar_pane_exif_update(ped); } /* *------------------------------------------------------------------- * dnd *------------------------------------------------------------------- */ static GtkTargetEntry bar_pane_exif_drag_types[] = { { TARGET_APP_EXIF_ENTRY_STRING, GTK_TARGET_SAME_APP, TARGET_APP_EXIF_ENTRY }, { "text/plain", 0, TARGET_TEXT_PLAIN } }; static gint n_exif_entry_drag_types = 2; static GtkTargetEntry bar_pane_exif_drop_types[] = { { TARGET_APP_EXIF_ENTRY_STRING, GTK_TARGET_SAME_APP, TARGET_APP_EXIF_ENTRY }, { "text/plain", 0, TARGET_TEXT_PLAIN } }; static gint n_exif_entry_drop_types = 2; static void bar_pane_exif_entry_dnd_get(GtkWidget *entry, GdkDragContext *context, GtkSelectionData *selection_data, guint info, guint time, gpointer data) { ExifEntry *ee = g_object_get_data(G_OBJECT(entry), "entry_data"); switch (info) { case TARGET_APP_EXIF_ENTRY: gtk_selection_data_set(selection_data, selection_data->target, 8, (gpointer) &entry, sizeof(entry)); break; case TARGET_TEXT_PLAIN: default: gtk_selection_data_set_text(selection_data, ee->key, -1); break; } } static void bar_pane_exif_dnd_receive(GtkWidget *pane, GdkDragContext *context, gint x, gint y, GtkSelectionData *selection_data, guint info, guint time, gpointer data) { PaneExifData *ped; GList *work, *list; gint pos; GtkWidget *new_entry = NULL; ped = g_object_get_data(G_OBJECT(pane), "pane_data"); if (!ped) return; switch (info) { case TARGET_APP_EXIF_ENTRY: new_entry = *(gpointer *)selection_data->data; if (new_entry->parent && new_entry->parent != ped->vbox) bar_pane_exif_reparent_entry(new_entry, pane); break; default: /* FIXME: this needs a check for valid exif keys */ new_entry = bar_pane_exif_add_entry(ped, (gchar *)selection_data->data, NULL, TRUE); break; } list = gtk_container_get_children(GTK_CONTAINER(ped->vbox)); work = list; pos = 0; while (work) { gint nx, ny; GtkWidget *entry = work->data; work = work->next; if (entry == new_entry) continue; if (GTK_WIDGET_DRAWABLE(entry) && gtk_widget_translate_coordinates(pane, entry, x, y, &nx, &ny) && ny < entry->allocation.height / 2) break; pos++; } g_list_free(list); gtk_box_reorder_child(GTK_BOX(ped->vbox), new_entry, pos); } static void bar_pane_exif_entry_dnd_begin(GtkWidget *widget, GdkDragContext *context, gpointer data) { // gtk_drag_set_icon_default(context); } static void bar_pane_exif_entry_dnd_end(GtkWidget *widget, GdkDragContext *context, gpointer data) { } static void bar_pane_exif_entry_dnd_init(GtkWidget *entry) { ExifEntry *ee = g_object_get_data(G_OBJECT(entry), "entry_data"); gtk_drag_source_set(entry, GDK_BUTTON1_MASK | GDK_BUTTON2_MASK, bar_pane_exif_drag_types, n_exif_entry_drag_types, GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK); g_signal_connect(G_OBJECT(entry), "drag_data_get", G_CALLBACK(bar_pane_exif_entry_dnd_get), ee); g_signal_connect(G_OBJECT(entry), "drag_begin", G_CALLBACK(bar_pane_exif_entry_dnd_begin), ee); g_signal_connect(G_OBJECT(entry), "drag_end", G_CALLBACK(bar_pane_exif_entry_dnd_end), ee); } static void bar_pane_exif_dnd_init(GtkWidget *pane) { gtk_drag_dest_set(pane, GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_HIGHLIGHT | GTK_DEST_DEFAULT_DROP, bar_pane_exif_drop_types, n_exif_entry_drop_types, GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_ASK); g_signal_connect(G_OBJECT(pane), "drag_data_received", G_CALLBACK(bar_pane_exif_dnd_receive), NULL); } static void bar_pane_exif_entry_write_config(GtkWidget *entry, GString *outstr, gint indent) { ExifEntry *ee = g_object_get_data(G_OBJECT(entry), "entry_data"); if (!ee) return; WRITE_STRING("<entry\n"); indent++; WRITE_CHAR(*ee, key); if (!ee->auto_title) WRITE_CHAR(*ee, title); WRITE_BOOL(*ee, if_set); indent--; WRITE_STRING("/>\n"); } static void bar_pane_exif_write_config(GtkWidget *pane, GString *outstr, gint indent) { PaneExifData *ped; GList *work, *list; ped = g_object_get_data(G_OBJECT(pane), "pane_data"); if (!ped) return; WRITE_STRING("<pane_exif\n"); indent++; write_char_option(outstr, indent, "pane.title", gtk_label_get_text(GTK_LABEL(ped->pane.title))); WRITE_BOOL(*ped, pane.expanded); indent--; WRITE_STRING(">\n"); indent++; list = gtk_container_get_children(GTK_CONTAINER(ped->vbox)); work = list; while (work) { GtkWidget *entry = work->data; work = work->next; bar_pane_exif_entry_write_config(entry, outstr, indent); } g_list_free(list); indent--; WRITE_STRING("</pane_exif>\n"); } void bar_pane_exif_close(GtkWidget *widget) { PaneExifData *ped; ped = g_object_get_data(G_OBJECT(widget), "pane_data"); if (!ped) return; gtk_widget_destroy(ped->vbox); } static void bar_pane_exif_destroy(GtkWidget *widget, gpointer data) { PaneExifData *ped = data; g_object_unref(ped->size_group); file_data_unref(ped->fd); g_free(ped); } static void bar_pane_exif_size_request(GtkWidget *pane, GtkRequisition *requisition, gpointer data) { PaneExifData *ped = data; if (requisition->height < ped->min_height) { requisition->height = ped->min_height; } } static void bar_pane_exif_size_allocate(GtkWidget *pane, GtkAllocation *alloc, gpointer data) { PaneExifData *ped = data; ped->min_height = alloc->height; } GtkWidget *bar_pane_exif_new(const gchar *title, gboolean expanded, gboolean populate) { PaneExifData *ped; ped = g_new0(PaneExifData, 1); ped->pane.pane_set_fd = bar_pane_exif_set_fd; ped->pane.pane_write_config = bar_pane_exif_write_config; ped->pane.title = gtk_label_new(title); ped->pane.expanded = expanded; ped->size_group = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL); ped->vbox = gtk_vbox_new(FALSE, PREF_PAD_GAP); ped->widget = ped->vbox; ped->min_height = MIN_HEIGHT; g_object_set_data(G_OBJECT(ped->widget), "pane_data", ped); g_signal_connect_after(G_OBJECT(ped->widget), "destroy", G_CALLBACK(bar_pane_exif_destroy), ped); g_signal_connect(G_OBJECT(ped->widget), "size-request", G_CALLBACK(bar_pane_exif_size_request), ped); g_signal_connect(G_OBJECT(ped->widget), "size-allocate", G_CALLBACK(bar_pane_exif_size_allocate), ped); bar_pane_exif_dnd_init(ped->widget); if (populate) { bar_pane_exif_add_entry(ped, EXIF_FORMATTED("Camera"), NULL, TRUE); bar_pane_exif_add_entry(ped, EXIF_FORMATTED("DateTime"), NULL, TRUE); bar_pane_exif_add_entry(ped, EXIF_FORMATTED("ShutterSpeed"), NULL, TRUE); bar_pane_exif_add_entry(ped, EXIF_FORMATTED("Aperture"), NULL, TRUE); bar_pane_exif_add_entry(ped, EXIF_FORMATTED("ExposureBias"), NULL, TRUE); bar_pane_exif_add_entry(ped, EXIF_FORMATTED("ISOSpeedRating"), NULL, TRUE); bar_pane_exif_add_entry(ped, EXIF_FORMATTED("FocalLength"), NULL, TRUE); bar_pane_exif_add_entry(ped, EXIF_FORMATTED("FocalLength35mmFilm"), NULL, TRUE); bar_pane_exif_add_entry(ped, EXIF_FORMATTED("Flash"), NULL, TRUE); bar_pane_exif_add_entry(ped, "Exif.Photo.ExposureProgram", NULL, TRUE); bar_pane_exif_add_entry(ped, "Exif.Photo.MeteringMode", NULL, TRUE); bar_pane_exif_add_entry(ped, "Exif.Photo.LightSource", NULL, TRUE); bar_pane_exif_add_entry(ped, EXIF_FORMATTED("ColorProfile"), NULL, TRUE); bar_pane_exif_add_entry(ped, EXIF_FORMATTED("SubjectDistance"), NULL, TRUE); bar_pane_exif_add_entry(ped, EXIF_FORMATTED("Resolution"), NULL, TRUE); bar_pane_exif_add_entry(ped, "Exif.Image.Orientation", NULL, TRUE); bar_pane_exif_add_entry(ped, EXIF_FORMATTED("GPSPosition"), NULL, TRUE); bar_pane_exif_add_entry(ped, EXIF_FORMATTED("GPSAltitude"), NULL, TRUE); bar_pane_exif_add_entry(ped, "Exif.Image.ImageDescription", NULL, TRUE); bar_pane_exif_add_entry(ped, "Exif.Image.Copyright", NULL, TRUE); } gtk_widget_show(ped->widget); return ped->widget; } GtkWidget *bar_pane_exif_new_from_config(const gchar **attribute_names, const gchar **attribute_values) { gchar *title = g_strdup(_("NoName")); gboolean expanded = TRUE; while (*attribute_names) { const gchar *option = *attribute_names++; const gchar *value = *attribute_values++; if (READ_CHAR_FULL("pane.title", title)) continue; if (READ_BOOL_FULL("pane.expanded", expanded)) continue; DEBUG_1("unknown attribute %s = %s", option, value); } return bar_pane_exif_new(title, expanded, FALSE); } void bar_pane_exif_entry_add_from_config(GtkWidget *pane, const gchar **attribute_names, const gchar **attribute_values) { PaneExifData *ped; gchar *key = NULL; gchar *title = NULL; gboolean if_set = TRUE; ped = g_object_get_data(G_OBJECT(pane), "pane_data"); if (!ped) return; while (*attribute_names) { const gchar *option = *attribute_names++; const gchar *value = *attribute_values++; if (READ_CHAR_FULL("key", key)) continue; if (READ_CHAR_FULL("title", title)) continue; if (READ_BOOL_FULL("if_set", if_set)) continue; DEBUG_1("unknown attribute %s = %s", option, value); } if (key && key[0]) bar_pane_exif_add_entry(ped, key, title, if_set); } /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */