/* EggFileFormatChooser * Copyright (C) 2007 Mathias Hasselmann * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, * Boston, MA 02110-1301, USA. */ #include "eggfileformatchooser.h" #include "egg-macros.h" #include <glib/gi18n.h> #include <gtk/gtk.h> #include <string.h> #include <ctype.h> typedef struct _EggFileFormatFilterInfo EggFileFormatFilterInfo; typedef struct _EggFileFormatSearch EggFileFormatSearch; enum { MODEL_COLUMN_ID, MODEL_COLUMN_NAME, MODEL_COLUMN_ICON, MODEL_COLUMN_EXTENSIONS, MODEL_COLUMN_FILTER, MODEL_COLUMN_DATA, MODEL_COLUMN_DESTROY }; enum { SIGNAL_SELECTION_CHANGED, SIGNAL_LAST }; struct _EggFileFormatChooserPrivate { GtkTreeStore *model; GtkTreeSelection *selection; guint idle_hack; guint last_id; gulong size_changed_event; GtkFileChooser *chooser; GtkFileFilter *all_files; GtkFileFilter *supported_files; }; struct _EggFileFormatFilterInfo { GHashTable *extension_set; GSList *extension_list; gboolean show_extensions; gchar *name; }; struct _EggFileFormatSearch { gboolean success; GtkTreeIter iter; guint format; const gchar *extension; }; static guint signals[SIGNAL_LAST] = { 0 }; G_DEFINE_TYPE (EggFileFormatChooser, egg_file_format_chooser, GTK_TYPE_EXPANDER); static EGG_DEFINE_QUARK (EggFileFormatFilterInfo, egg_file_format_filter_info); static EggFileFormatFilterInfo* egg_file_format_filter_info_new (const gchar *name, gboolean show_extensions) { EggFileFormatFilterInfo *self; self = g_new0 (EggFileFormatFilterInfo, 1); self->extension_set = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL); self->show_extensions = show_extensions; self->name = g_strdup (name); return self; } static void egg_file_format_filter_info_free (gpointer boxed) { EggFileFormatFilterInfo *self; if (boxed) { self = boxed; g_hash_table_unref (self->extension_set); g_slist_foreach (self->extension_list, (GFunc) g_free, NULL); g_slist_free (self->extension_list); g_free (self->name); g_free (self); } } static gboolean egg_file_format_filter_find (gpointer key, gpointer value G_GNUC_UNUSED, gpointer data) { const GtkFileFilterInfo *info = data; const gchar *pattern = key; return g_str_has_suffix (info->filename, pattern + 1); } static gboolean egg_file_format_filter_filter (const GtkFileFilterInfo *info, gpointer data) { EggFileFormatFilterInfo *self = data; return NULL != g_hash_table_find (self->extension_set, egg_file_format_filter_find, (gpointer) info); } static GtkFileFilter* egg_file_format_filter_new (const gchar *name, gboolean show_extensions) { GtkFileFilter *filter; EggFileFormatFilterInfo *info; filter = gtk_file_filter_new (); gtk_file_filter_set_name (filter, name); info = egg_file_format_filter_info_new (name, show_extensions); gtk_file_filter_add_custom (filter, GTK_FILE_FILTER_FILENAME, egg_file_format_filter_filter, info, NULL); g_object_set_qdata_full (G_OBJECT (filter), egg_file_format_filter_info_quark (), info, egg_file_format_filter_info_free); return filter; } static void egg_file_format_filter_add_extensions (GtkFileFilter *filter, const gchar *extensions) { EggFileFormatFilterInfo *info; GString *filter_name; const gchar *extptr; gchar *pattern; gsize length; g_assert (NULL != extensions); info = g_object_get_qdata (G_OBJECT (filter), egg_file_format_filter_info_quark ()); info->extension_list = g_slist_prepend (info->extension_list, g_strdup (extensions)); if (info->show_extensions) { filter_name = g_string_new (info->name); g_string_append (filter_name, " ("); } else filter_name = NULL; extptr = extensions; while (*extptr) { length = strcspn (extptr, ","); pattern = g_new (gchar, length + 3); memcpy (pattern, "*.", 2); memcpy (pattern + 2, extptr, length); pattern[length + 2] = '\0'; if (filter_name) { if (extptr != extensions) g_string_append (filter_name, ", "); g_string_append (filter_name, pattern); } extptr += length; if (*extptr) extptr += 2; g_hash_table_replace (info->extension_set, pattern, pattern); } if (filter_name) { g_string_append (filter_name, ")"); gtk_file_filter_set_name (filter, filter_name->str); g_string_free (filter_name, TRUE); } } static void selection_changed_cb (GtkTreeSelection *selection, EggFileFormatChooser *self) { gchar *label; gchar *name; GtkFileFilter *filter; GtkTreeModel *model; GtkTreeIter parent; GtkTreeIter iter; if (gtk_tree_selection_get_selected (selection, &model, &iter)) { gtk_tree_model_get (model, &iter, MODEL_COLUMN_NAME, &name, -1); label = g_strdup_printf (_("File _Format: %s"), name); gtk_expander_set_use_underline (GTK_EXPANDER (self), TRUE); gtk_expander_set_label (GTK_EXPANDER (self), label); g_free (name); g_free (label); if (self->priv->chooser) { while (gtk_tree_model_iter_parent (model, &parent, &iter)) iter = parent; gtk_tree_model_get (model, &iter, MODEL_COLUMN_FILTER, &filter, -1); gtk_file_chooser_set_filter (self->priv->chooser, filter); g_object_unref (filter); } g_signal_emit (self, signals[SIGNAL_SELECTION_CHANGED], 0); } } /* XXX This hack is needed, as gtk_expander_set_label seems * not to work from egg_file_format_chooser_init */ static gboolean select_default_file_format (gpointer data) { EggFileFormatChooser *self = EGG_FILE_FORMAT_CHOOSER (data); egg_file_format_chooser_set_format (self, 0); self->priv->idle_hack = 0; return FALSE; } static gboolean find_by_format (GtkTreeModel *model, GtkTreePath *path G_GNUC_UNUSED, GtkTreeIter *iter, gpointer data) { EggFileFormatSearch *search = data; guint id; gtk_tree_model_get (model, iter, MODEL_COLUMN_ID, &id, -1); if (id == search->format) { search->success = TRUE; search->iter = *iter; } return search->success; } static gboolean find_in_list (gchar *list, const gchar *needle) { gchar *saveptr; gchar *token; for (token = strtok_r (list, ",", &saveptr); NULL != token; token = strtok_r (NULL, ",", &saveptr)) { token = g_strstrip (token); if (strcasecmp (needle, token) == 0) return TRUE; } return FALSE; } static gboolean accept_filename (gchar *extensions, const gchar *filename) { const gchar *extptr; gchar *saveptr; gchar *token; gsize length; length = strlen (filename); for (token = strtok_r (extensions, ",", &saveptr); NULL != token; token = strtok_r (NULL, ",", &saveptr)) { token = g_strstrip (token); extptr = filename + length - strlen (token) - 1; if (extptr > filename && '.' == *extptr && !strcmp (extptr + 1, token)) return TRUE; } return FALSE; } static gboolean find_by_extension (GtkTreeModel *model, GtkTreePath *path G_GNUC_UNUSED, GtkTreeIter *iter, gpointer data) { EggFileFormatSearch *search = data; gchar *extensions = NULL; guint format = 0; gtk_tree_model_get (model, iter, MODEL_COLUMN_EXTENSIONS, &extensions, MODEL_COLUMN_ID, &format, -1); if (extensions && find_in_list (extensions, search->extension)) { search->format = format; search->success = TRUE; search->iter = *iter; } g_free (extensions); return search->success; } static int emit_default_size_changed (gpointer user_data) { EggFileFormatChooser *self = user_data; self->priv->size_changed_event = 0; g_signal_emit_by_name (self->priv->chooser, "default-size-changed"); return FALSE; } static void expander_unmap_cb (GtkWidget *widget, gpointer user_data) { EggFileFormatChooser *self = user_data; if (self->priv->size_changed_event == 0) self->priv->size_changed_event = gdk_threads_add_idle (emit_default_size_changed, self); } static void egg_file_format_chooser_init (EggFileFormatChooser *self) { GtkWidget *scroller; GtkWidget *view; GtkTreeViewColumn *column; GtkCellRenderer *cell; GtkTreeIter iter; self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self, EGG_TYPE_FILE_FORMAT_CHOOSER, EggFileFormatChooserPrivate); self->priv->size_changed_event = 0; /* file filters */ self->priv->all_files = g_object_ref_sink (gtk_file_filter_new ()); gtk_file_filter_set_name (self->priv->all_files, _("All Files")); self->priv->supported_files = egg_file_format_filter_new (_("All Supported Files"), FALSE); /* tree model */ self->priv->model = gtk_tree_store_new (7, G_TYPE_UINT, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING, GTK_TYPE_FILE_FILTER, G_TYPE_POINTER, G_TYPE_POINTER); gtk_tree_store_append (self->priv->model, &iter, NULL); gtk_tree_store_set (self->priv->model, &iter, MODEL_COLUMN_NAME, _("By Extension"), MODEL_COLUMN_FILTER, self->priv->supported_files, MODEL_COLUMN_ID, 0, -1); /* tree view */ view = gtk_tree_view_new_with_model (GTK_TREE_MODEL (self->priv->model)); self->priv->selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view)); gtk_tree_view_set_rules_hint (GTK_TREE_VIEW (view), TRUE); /* file format column */ column = gtk_tree_view_column_new (); gtk_tree_view_column_set_expand (column, TRUE); gtk_tree_view_column_set_title (column, _("File Format")); gtk_tree_view_append_column (GTK_TREE_VIEW (view), column); cell = gtk_cell_renderer_pixbuf_new (); gtk_tree_view_column_pack_start (column, cell, FALSE); gtk_tree_view_column_set_attributes (column, cell, "icon-name", MODEL_COLUMN_ICON, NULL); cell = gtk_cell_renderer_text_new (); gtk_tree_view_column_pack_start (column, cell, TRUE); gtk_tree_view_column_set_attributes (column, cell, "text", MODEL_COLUMN_NAME, NULL); /* extensions column */ column = gtk_tree_view_column_new_with_attributes ( _("Extension(s)"), gtk_cell_renderer_text_new (), "text", MODEL_COLUMN_EXTENSIONS, NULL); gtk_tree_view_column_set_expand (column, FALSE); gtk_tree_view_append_column (GTK_TREE_VIEW (view), column); /* selection */ gtk_tree_selection_set_mode (self->priv->selection, GTK_SELECTION_BROWSE); g_signal_connect (self->priv->selection, "changed", G_CALLBACK (selection_changed_cb), self); self->priv->idle_hack = g_idle_add (select_default_file_format, self); /* scroller */ scroller = gtk_scrolled_window_new (NULL, NULL); gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scroller), GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC); gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (scroller), GTK_SHADOW_IN); gtk_widget_set_size_request (scroller, -1, 150); gtk_container_add (GTK_CONTAINER (scroller), view); gtk_widget_show_all (scroller); gtk_container_add (GTK_CONTAINER (self), scroller); g_signal_connect_after (scroller, "unmap", G_CALLBACK (expander_unmap_cb), self); } static void reset_model (EggFileFormatChooser *self) { GtkTreeModel *model = GTK_TREE_MODEL (self->priv->model); GtkTreeIter iter; if (gtk_tree_model_get_iter_first (model, &iter)) { do { GDestroyNotify destroy = NULL; gpointer data = NULL; gtk_tree_model_get (model, &iter, MODEL_COLUMN_DESTROY, &destroy, MODEL_COLUMN_DATA, &data, -1); if (destroy) destroy (data); } while (gtk_tree_model_iter_next (model, &iter)); } gtk_tree_store_clear (self->priv->model); } static void egg_file_format_chooser_dispose (GObject *obj) { EggFileFormatChooser *self = EGG_FILE_FORMAT_CHOOSER (obj); if (NULL != self) { if (self->priv->idle_hack) { g_source_remove (self->priv->idle_hack); self->priv->idle_hack = 0; } if (self->priv->size_changed_event != 0) { g_source_remove (self->priv->size_changed_event); self->priv->size_changed_event = 0; } } G_OBJECT_CLASS (egg_file_format_chooser_parent_class)->dispose (obj); } static void egg_file_format_chooser_finalize (GObject *obj) { EggFileFormatChooser *self = EGG_FILE_FORMAT_CHOOSER (obj); if (NULL != self) { if (self->priv->model) { reset_model (self); g_object_unref (self->priv->model); self->priv->model = NULL; g_object_unref (self->priv->all_files); self->priv->all_files = NULL; } } G_OBJECT_CLASS (egg_file_format_chooser_parent_class)->finalize (obj); } static void filter_changed_cb (GObject *object, GParamSpec *spec, gpointer data) { EggFileFormatChooser *self; GtkFileFilter *current_filter; GtkFileFilter *format_filter; GtkTreeModel *model; GtkTreeIter iter; GtkTreeIter parent; self = EGG_FILE_FORMAT_CHOOSER (data); format_filter = NULL; current_filter = gtk_file_chooser_get_filter (GTK_FILE_CHOOSER (object)); model = GTK_TREE_MODEL (self->priv->model); if (gtk_tree_selection_get_selected (self->priv->selection, &model, &iter)) { while (gtk_tree_model_iter_parent (model, &parent, &iter)) iter = parent; gtk_tree_model_get (model, &iter, MODEL_COLUMN_FILTER, &format_filter, -1); g_object_unref (format_filter); } if (current_filter && current_filter != format_filter && gtk_tree_model_get_iter_first (model, &iter)) { if (current_filter == self->priv->all_files) format_filter = current_filter; else { format_filter = NULL; do { gtk_tree_model_get (model, &iter, MODEL_COLUMN_FILTER, &format_filter, -1); g_object_unref (format_filter); if (format_filter == current_filter) break; } while (gtk_tree_model_iter_next (model, &iter)); } if (format_filter) gtk_tree_selection_select_iter (self->priv->selection, &iter); } } /* Shows an error dialog set as transient for the specified window */ static void error_message_with_parent (GtkWindow *parent, const char *msg, const char *detail) { GtkWidget *dialog; g_warning ("%s: Merge with the code in Gtk{File,Recent}ChooserDefault.", G_STRLOC); dialog = gtk_message_dialog_new (parent, GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_ERROR, GTK_BUTTONS_OK, "%s", msg); gtk_message_dialog_format_secondary_text (GTK_MESSAGE_DIALOG (dialog), "%s", detail); if (gtk_window_get_group (parent)) gtk_window_group_add_window (gtk_window_get_group (parent), GTK_WINDOW (dialog)); gtk_dialog_run (GTK_DIALOG (dialog)); gtk_widget_destroy (dialog); } /* Returns a toplevel GtkWindow, or NULL if none */ static GtkWindow * get_toplevel (GtkWidget *widget) { GtkWidget *toplevel; toplevel = gtk_widget_get_toplevel (widget); if (!gtk_widget_is_toplevel (toplevel)) return NULL; else return GTK_WINDOW (toplevel); } /* Shows an error dialog for the file chooser */ static void error_message (EggFileFormatChooser *impl, const char *msg, const char *detail) { error_message_with_parent (get_toplevel (GTK_WIDGET (impl)), msg, detail); } static void chooser_response_cb (GtkDialog *dialog, gint response_id, gpointer data) { EggFileFormatChooser *self; gchar *filename, *basename; gchar *message; guint format; self = EGG_FILE_FORMAT_CHOOSER (data); if (EGG_IS_POSITIVE_RESPONSE (response_id)) { filename = gtk_file_chooser_get_filename (self->priv->chooser); basename = g_filename_display_basename (filename); g_free (filename); format = egg_file_format_chooser_get_format (self, basename); g_print ("%s: %s - %d\n", G_STRFUNC, basename, format); if (0 == format) { message = g_strdup_printf ( _("The program was not able to find out the file format " "you want to use for `%s'. Please make sure to use a " "known extension for that file or manually choose a " "file format from the list below."), basename); error_message (self, _("File format not recognized"), message); g_free (message); g_signal_stop_emission_by_name (dialog, "response"); } else { filename = egg_file_format_chooser_append_extension (self, basename, format); if (strcmp (filename, basename)) { gtk_file_chooser_set_current_name (self->priv->chooser, filename); g_signal_stop_emission_by_name (dialog, "response"); } g_free (filename); } g_free (basename); } } static void egg_file_format_chooser_realize (GtkWidget *widget) { EggFileFormatChooser *self; GtkWidget *parent; GtkFileFilter *filter; GtkTreeModel *model; GtkTreeIter iter; GTK_WIDGET_CLASS (egg_file_format_chooser_parent_class)->realize (widget); self = EGG_FILE_FORMAT_CHOOSER (widget); g_return_if_fail (NULL == self->priv->chooser); parent = gtk_widget_get_parent (widget); while ((parent != NULL) && !GTK_IS_FILE_CHOOSER (parent)) parent = gtk_widget_get_parent (parent); self->priv->chooser = GTK_FILE_CHOOSER (parent); g_return_if_fail (GTK_IS_FILE_CHOOSER (self->priv->chooser)); g_return_if_fail (gtk_file_chooser_get_action (self->priv->chooser) == GTK_FILE_CHOOSER_ACTION_SAVE); g_object_ref (self->priv->chooser); g_signal_connect (self->priv->chooser, "notify::filter", G_CALLBACK (filter_changed_cb), self); gtk_file_chooser_add_filter (self->priv->chooser, self->priv->all_files); model = GTK_TREE_MODEL (self->priv->model); if (gtk_tree_model_get_iter_first (model, &iter)) { do { gtk_tree_model_get (model, &iter, MODEL_COLUMN_FILTER, &filter, -1); gtk_file_chooser_add_filter (self->priv->chooser, filter); g_object_unref (filter); } while (gtk_tree_model_iter_next (model, &iter)); } gtk_file_chooser_set_filter (self->priv->chooser, self->priv->supported_files); if (GTK_IS_DIALOG (self->priv->chooser)) g_signal_connect (self->priv->chooser, "response", G_CALLBACK (chooser_response_cb), self); } static void egg_file_format_chooser_unrealize (GtkWidget *widget) { EggFileFormatChooser *self; GtkFileFilter *filter; GtkTreeModel *model; GtkTreeIter iter; GTK_WIDGET_CLASS (egg_file_format_chooser_parent_class)->unrealize (widget); self = EGG_FILE_FORMAT_CHOOSER (widget); model = GTK_TREE_MODEL (self->priv->model); g_signal_handlers_disconnect_by_func (self->priv->chooser, filter_changed_cb, self); g_signal_handlers_disconnect_by_func (self->priv->chooser, chooser_response_cb, self); if (gtk_tree_model_get_iter_first (model, &iter)) { do { gtk_tree_model_get (model, &iter, MODEL_COLUMN_FILTER, &filter, -1); gtk_file_chooser_remove_filter (self->priv->chooser, filter); g_object_unref (filter); } while (gtk_tree_model_iter_next (model, &iter)); } gtk_file_chooser_remove_filter (self->priv->chooser, self->priv->all_files); g_object_unref (self->priv->chooser); } static void egg_file_format_chooser_class_init (EggFileFormatChooserClass *cls) { GObjectClass *object_class = G_OBJECT_CLASS (cls); GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (cls); g_type_class_add_private (cls, sizeof (EggFileFormatChooserPrivate)); object_class->dispose = egg_file_format_chooser_dispose; object_class->finalize = egg_file_format_chooser_finalize; widget_class->realize = egg_file_format_chooser_realize; widget_class->unrealize = egg_file_format_chooser_unrealize; signals[SIGNAL_SELECTION_CHANGED] = g_signal_new ( "selection-changed", EGG_TYPE_FILE_FORMAT_CHOOSER, G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET (EggFileFormatChooserClass, selection_changed), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); } GtkWidget* egg_file_format_chooser_new (void) { return g_object_new (EGG_TYPE_FILE_FORMAT_CHOOSER, NULL); } static guint egg_file_format_chooser_add_format_impl (EggFileFormatChooser *self, guint parent, const gchar *name, const gchar *icon, const gchar *extensions) { EggFileFormatSearch search; GtkFileFilter *filter; GtkTreeIter iter; search.success = FALSE; search.format = parent; filter = NULL; if (parent > 0) { gtk_tree_model_foreach (GTK_TREE_MODEL (self->priv->model), find_by_format, &search); g_return_val_if_fail (search.success, -1); } else filter = egg_file_format_filter_new (name, TRUE); gtk_tree_store_append (self->priv->model, &iter, parent > 0 ? &search.iter : NULL); gtk_tree_store_set (self->priv->model, &iter, MODEL_COLUMN_ID, ++self->priv->last_id, MODEL_COLUMN_EXTENSIONS, extensions, MODEL_COLUMN_FILTER, filter, MODEL_COLUMN_NAME, name, MODEL_COLUMN_ICON, icon, -1); if (extensions) { if (parent > 0) gtk_tree_model_get (GTK_TREE_MODEL (self->priv->model), &search.iter, MODEL_COLUMN_FILTER, &filter, -1); egg_file_format_filter_add_extensions (self->priv->supported_files, extensions); egg_file_format_filter_add_extensions (filter, extensions); if (parent > 0) g_object_unref (filter); } return self->priv->last_id; } guint egg_file_format_chooser_add_format (EggFileFormatChooser *self, guint parent, const gchar *name, const gchar *icon, ...) { GString *buffer = NULL; const gchar* extptr; va_list extensions; guint id; g_return_val_if_fail (EGG_IS_FILE_FORMAT_CHOOSER (self), 0); g_return_val_if_fail (NULL != name, 0); va_start (extensions, icon); while (NULL != (extptr = va_arg (extensions, const gchar*))) { if (NULL == buffer) buffer = g_string_new (NULL); else g_string_append (buffer, ", "); g_string_append (buffer, extptr); } va_end (extensions); id = egg_file_format_chooser_add_format_impl (self, parent, name, icon, buffer ? buffer->str : NULL); if (buffer) g_string_free (buffer, TRUE); return id; } static gchar* get_icon_name (const gchar *mime_type) { static gboolean first_call = TRUE; gchar *name = NULL; gchar *s; if (first_call) { g_warning ("%s: Replace by g_content_type_get_icon " "when GVFS is merged into GLib.", G_STRLOC); first_call = FALSE; } if (mime_type) { name = g_strconcat ("mate-mime-", mime_type, NULL); for(s = name; *s; ++s) { if (!isalpha (*s) || !isascii (*s)) *s = '-'; } } if (!name || !gtk_icon_theme_has_icon (gtk_icon_theme_get_default (), name)) { g_free (name); name = g_strdup ("mate-mime-image"); } return name; } void egg_file_format_chooser_add_pixbuf_formats (EggFileFormatChooser *self, guint parent G_GNUC_UNUSED, guint **formats) { GSList *pixbuf_formats = NULL; GSList *iter; gint i; g_return_if_fail (EGG_IS_FILE_FORMAT_CHOOSER (self)); pixbuf_formats = gdk_pixbuf_get_formats (); if (formats) *formats = g_new0 (guint, g_slist_length (pixbuf_formats) + 1); for(iter = pixbuf_formats, i = 0; iter; iter = iter->next, ++i) { GdkPixbufFormat *format = iter->data; gchar *description, *name, *extensions, *icon; gchar **mime_types, **extension_list; guint id; if (gdk_pixbuf_format_is_disabled (format) || !gdk_pixbuf_format_is_writable (format)) continue; mime_types = gdk_pixbuf_format_get_mime_types (format); icon = get_icon_name (mime_types[0]); g_strfreev (mime_types); extension_list = gdk_pixbuf_format_get_extensions (format); extensions = g_strjoinv (", ", extension_list); g_strfreev (extension_list); description = gdk_pixbuf_format_get_description (format); name = gdk_pixbuf_format_get_name (format); id = egg_file_format_chooser_add_format_impl (self, parent, description, icon, extensions); g_free (description); g_free (extensions); g_free (icon); egg_file_format_chooser_set_format_data (self, id, name, g_free); if (formats) *formats[i] = id; } g_slist_free (pixbuf_formats); } void egg_file_format_chooser_remove_format (EggFileFormatChooser *self, guint format) { GDestroyNotify destroy = NULL; gpointer data = NULL; EggFileFormatSearch search; GtkFileFilter *filter; GtkTreeModel *model; g_return_if_fail (EGG_IS_FILE_FORMAT_CHOOSER (self)); search.success = FALSE; search.format = format; model = GTK_TREE_MODEL (self->priv->model); gtk_tree_model_foreach (model, find_by_format, &search); g_return_if_fail (search.success); gtk_tree_model_get (model, &search.iter, MODEL_COLUMN_FILTER, &filter, MODEL_COLUMN_DESTROY, &destroy, MODEL_COLUMN_DATA, &data, -1); if (destroy) destroy (data); if (filter) { if (self->priv->chooser) gtk_file_chooser_remove_filter (self->priv->chooser, filter); g_object_unref (filter); } else g_warning ("TODO: Remove extensions from parent filter"); gtk_tree_store_remove (self->priv->model, &search.iter); } void egg_file_format_chooser_set_format (EggFileFormatChooser *self, guint format) { EggFileFormatSearch search; GtkTreeModel *model; GtkTreePath *path; GtkTreeView *view; g_return_if_fail (EGG_IS_FILE_FORMAT_CHOOSER (self)); search.success = FALSE; search.format = format; model = GTK_TREE_MODEL (self->priv->model); gtk_tree_model_foreach (model, find_by_format, &search); g_return_if_fail (search.success); path = gtk_tree_model_get_path (model, &search.iter); view = gtk_tree_selection_get_tree_view (self->priv->selection); gtk_tree_view_expand_to_path (view, path); gtk_tree_selection_unselect_all (self->priv->selection); gtk_tree_selection_select_path (self->priv->selection, path); gtk_tree_path_free (path); if (self->priv->idle_hack > 0) { g_source_remove (self->priv->idle_hack); self->priv->idle_hack = 0; } } guint egg_file_format_chooser_get_format (EggFileFormatChooser *self, const gchar *filename) { GtkTreeModel *model; GtkTreeIter iter; guint format = 0; g_return_val_if_fail (EGG_IS_FILE_FORMAT_CHOOSER (self), -1); if (gtk_tree_selection_get_selected (self->priv->selection, &model, &iter)) gtk_tree_model_get (model, &iter, MODEL_COLUMN_ID, &format, -1); if (0 == format && NULL != filename) { EggFileFormatSearch search; search.extension = strrchr(filename, '.'); search.success = FALSE; if (search.extension++) gtk_tree_model_foreach (model, find_by_extension, &search); if (search.success) format = search.format; } return format; } void egg_file_format_chooser_set_format_data (EggFileFormatChooser *self, guint format, gpointer data, GDestroyNotify destroy) { EggFileFormatSearch search; g_return_if_fail (EGG_IS_FILE_FORMAT_CHOOSER (self)); search.success = FALSE; search.format = format; gtk_tree_model_foreach (GTK_TREE_MODEL (self->priv->model), find_by_format, &search); g_return_if_fail (search.success); gtk_tree_store_set (self->priv->model, &search.iter, MODEL_COLUMN_DESTROY, destroy, MODEL_COLUMN_DATA, data, -1); } gpointer egg_file_format_chooser_get_format_data (EggFileFormatChooser *self, guint format) { EggFileFormatSearch search; gpointer data = NULL; GtkTreeModel *model; g_return_val_if_fail (EGG_IS_FILE_FORMAT_CHOOSER (self), NULL); search.success = FALSE; search.format = format; model = GTK_TREE_MODEL (self->priv->model); gtk_tree_model_foreach (model, find_by_format, &search); g_return_val_if_fail (search.success, NULL); gtk_tree_model_get (model, &search.iter, MODEL_COLUMN_DATA, &data, -1); return data; } gchar* egg_file_format_chooser_append_extension (EggFileFormatChooser *self, const gchar *filename, guint format) { EggFileFormatSearch search; GtkTreeModel *model; GtkTreeIter child; gchar *extensions; gchar *result; g_return_val_if_fail (EGG_IS_FILE_FORMAT_CHOOSER (self), NULL); g_return_val_if_fail (NULL != filename, NULL); if (0 == format) format = egg_file_format_chooser_get_format (self, NULL); if (0 == format) { g_warning ("%s: No file format selected. Cannot append extension.", G_STRFUNC); return NULL; } search.success = FALSE; search.format = format; model = GTK_TREE_MODEL (self->priv->model); gtk_tree_model_foreach (model, find_by_format, &search); g_return_val_if_fail (search.success, NULL); gtk_tree_model_get (model, &search.iter, MODEL_COLUMN_EXTENSIONS, &extensions, -1); if (NULL == extensions && gtk_tree_model_iter_nth_child (model, &child, &search.iter, 0)) { gtk_tree_model_get (model, &child, MODEL_COLUMN_EXTENSIONS, &extensions, -1); } if (NULL == extensions) { g_warning ("%s: File format %d doesn't provide file extensions. " "Cannot append extension.", G_STRFUNC, format); return NULL; } if (accept_filename (extensions, filename)) result = g_strdup (filename); else result = g_strconcat (filename, ".", extensions, NULL); g_assert (NULL == strchr(extensions, ',')); g_free (extensions); return result; } void egg_file_format_chooser_emit_size_changed (EggFileFormatChooser *self) { if (self->priv->size_changed_event == 0) self->priv->size_changed_event = gdk_threads_add_idle (emit_default_size_changed, self); } /* vim: set sw=2 sta et: */