/* -*- Mode: C; c-file-style: "gnu"; tab-width: 8 -*- */ /* GTK+: gtkfilechooserbutton.c * * Copyright (c) 2004 James M. Cape * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library. If not, see . */ #include "config.h" #include #include #ifdef HAVE_UNISTD_H #include #endif #include #include #include "gtkintl.h" #include "gtkbutton.h" #include "gtkcelllayout.h" #include "gtkcellrenderertext.h" #include "gtkcellrendererpixbuf.h" #include "gtkcombobox.h" #include "gtkcssiconthemevalueprivate.h" #include "gtkdnd.h" #include "gtkdragdest.h" #include "gtkicontheme.h" #include "gtkimage.h" #include "gtklabel.h" #include "gtkliststore.h" #include "gtktreemodelfilter.h" #include "gtkseparator.h" #include "gtkfilechooserdialog.h" #include "gtkfilechoosernative.h" #include "gtkfilechooserprivate.h" #include "gtkfilechooserutils.h" #include "gtkmarshalers.h" #include "gtkfilechooserbutton.h" #include "gtkorientable.h" #include "gtktypebuiltins.h" #include "gtkprivate.h" #include "gtksettings.h" #include "gtkstylecontextprivate.h" #include "gtkbitmaskprivate.h" /** * SECTION:gtkfilechooserbutton * @Short_description: A button to launch a file selection dialog * @Title: GtkFileChooserButton * @See_also:#GtkFileChooserDialog * * The #GtkFileChooserButton is a widget that lets the user select a * file. It implements the #GtkFileChooser interface. Visually, it is a * file name with a button to bring up a #GtkFileChooserDialog. * The user can then use that dialog to change the file associated with * that button. This widget does not support setting the * #GtkFileChooser:select-multiple property to %TRUE. * * ## Create a button to let the user select a file in /etc * * |[ * { * GtkWidget *button; * * button = gtk_file_chooser_button_new (_("Select a file"), * GTK_FILE_CHOOSER_ACTION_OPEN); * gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (button), * "/etc"); * } * ]| * * The #GtkFileChooserButton supports the #GtkFileChooserActions * %GTK_FILE_CHOOSER_ACTION_OPEN and %GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER. * * > The #GtkFileChooserButton will ellipsize the label, and will thus * > request little horizontal space. To give the button more space, * > you should call gtk_widget_get_preferred_size(), * > gtk_file_chooser_button_set_width_chars(), or pack the button in * > such a way that other interface elements give space to the * > widget. */ /* **************** * * Private Macros * * **************** */ #define FALLBACK_ICON_SIZE 16 #define DEFAULT_TITLE N_("Select a File") #define DESKTOP_DISPLAY_NAME N_("Desktop") #define FALLBACK_DISPLAY_NAME N_("(None)") /* this string is used in gtk+/gtk/tests/filechooser.c - change it there if you change it here */ /* ********************** * * Private Enumerations * * ********************** */ /* Property IDs */ enum { PROP_0, PROP_DIALOG, PROP_TITLE, PROP_WIDTH_CHARS }; /* Signals */ enum { FILE_SET, LAST_SIGNAL }; /* TreeModel Columns */ enum { ICON_COLUMN, DISPLAY_NAME_COLUMN, TYPE_COLUMN, DATA_COLUMN, IS_FOLDER_COLUMN, CANCELLABLE_COLUMN, NUM_COLUMNS }; /* TreeModel Row Types */ typedef enum { ROW_TYPE_SPECIAL, ROW_TYPE_VOLUME, ROW_TYPE_SHORTCUT, ROW_TYPE_BOOKMARK_SEPARATOR, ROW_TYPE_BOOKMARK, ROW_TYPE_CURRENT_FOLDER_SEPARATOR, ROW_TYPE_CURRENT_FOLDER, ROW_TYPE_OTHER_SEPARATOR, ROW_TYPE_OTHER, ROW_TYPE_EMPTY_SELECTION, ROW_TYPE_INVALID = -1 } RowType; /* ******************** * * Private Structures * * ******************** */ struct _GtkFileChooserButtonPrivate { GtkFileChooser *chooser; /* Points to either dialog or native, depending on which is set */ GtkWidget *dialog; /* Set if you explicitly enable */ GtkFileChooserNative *native; /* Otherwise this is set */ GtkWidget *box; GtkWidget *button; GtkWidget *image; GtkWidget *label; GtkWidget *combo_box; GtkCellRenderer *icon_cell; GtkCellRenderer *name_cell; /* Currently visible child (either priv->combo_box or priv->button) */ GtkWidget *child; GtkTreeModel *model; GtkTreeModel *filter_model; GtkFileSystem *fs; GFile *selection_while_inactive; GFile *current_folder_while_inactive; gulong fs_volumes_changed_id; GCancellable *dnd_select_folder_cancellable; GCancellable *update_button_cancellable; GSList *change_icon_theme_cancellables; GtkBookmarksManager *bookmarks_manager; gint icon_size; guint8 n_special; guint8 n_volumes; guint8 n_shortcuts; guint8 n_bookmarks; guint has_bookmark_separator : 1; guint has_current_folder_separator : 1; guint has_current_folder : 1; guint has_other_separator : 1; /* Used for hiding/showing the dialog when the button is hidden */ guint active : 1; /* Whether the next async callback from GIO should emit the "selection-changed" signal */ guint is_changing_selection : 1; }; /* ************* * * DnD Support * * ************* */ enum { TEXT_PLAIN, TEXT_URI_LIST }; /* ********************* * * Function Prototypes * * ********************* */ /* GtkFileChooserIface Functions */ static void gtk_file_chooser_button_file_chooser_iface_init (GtkFileChooserIface *iface); static gboolean gtk_file_chooser_button_set_current_folder (GtkFileChooser *chooser, GFile *file, GError **error); static GFile *gtk_file_chooser_button_get_current_folder (GtkFileChooser *chooser); static gboolean gtk_file_chooser_button_select_file (GtkFileChooser *chooser, GFile *file, GError **error); static void gtk_file_chooser_button_unselect_file (GtkFileChooser *chooser, GFile *file); static void gtk_file_chooser_button_unselect_all (GtkFileChooser *chooser); static GSList *gtk_file_chooser_button_get_files (GtkFileChooser *chooser); static gboolean gtk_file_chooser_button_add_shortcut_folder (GtkFileChooser *chooser, GFile *file, GError **error); static gboolean gtk_file_chooser_button_remove_shortcut_folder (GtkFileChooser *chooser, GFile *file, GError **error); /* GObject Functions */ static void gtk_file_chooser_button_constructed (GObject *object); static void gtk_file_chooser_button_set_property (GObject *object, guint param_id, const GValue *value, GParamSpec *pspec); static void gtk_file_chooser_button_get_property (GObject *object, guint param_id, GValue *value, GParamSpec *pspec); static void gtk_file_chooser_button_finalize (GObject *object); /* GtkWidget Functions */ static void gtk_file_chooser_button_destroy (GtkWidget *widget); static void gtk_file_chooser_button_drag_data_received (GtkWidget *widget, GdkDragContext *context, gint x, gint y, GtkSelectionData *data, guint type, guint drag_time); static void gtk_file_chooser_button_show (GtkWidget *widget); static void gtk_file_chooser_button_hide (GtkWidget *widget); static void gtk_file_chooser_button_map (GtkWidget *widget); static gboolean gtk_file_chooser_button_mnemonic_activate (GtkWidget *widget, gboolean group_cycling); static void gtk_file_chooser_button_style_updated (GtkWidget *widget); static void gtk_file_chooser_button_display_changed (GtkWidget *widget, GdkDisplay *old_display); static void gtk_file_chooser_button_state_flags_changed (GtkWidget *widget, GtkStateFlags previous_state); /* Utility Functions */ static GtkIconTheme *get_icon_theme (GtkWidget *widget); static void set_info_for_file_at_iter (GtkFileChooserButton *fs, GFile *file, GtkTreeIter *iter); static gint model_get_type_position (GtkFileChooserButton *button, RowType row_type); static void model_free_row_data (GtkFileChooserButton *button, GtkTreeIter *iter); static void model_add_special (GtkFileChooserButton *button); static void model_add_other (GtkFileChooserButton *button); static void model_add_empty_selection (GtkFileChooserButton *button); static void model_add_volumes (GtkFileChooserButton *button, GSList *volumes); static void model_add_bookmarks (GtkFileChooserButton *button, GSList *bookmarks); static void model_update_current_folder (GtkFileChooserButton *button, GFile *file); static void model_remove_rows (GtkFileChooserButton *button, gint pos, gint n_rows); static gboolean filter_model_visible_func (GtkTreeModel *model, GtkTreeIter *iter, gpointer user_data); static gboolean combo_box_row_separator_func (GtkTreeModel *model, GtkTreeIter *iter, gpointer user_data); static void name_cell_data_func (GtkCellLayout *layout, GtkCellRenderer *cell, GtkTreeModel *model, GtkTreeIter *iter, gpointer user_data); static void open_dialog (GtkFileChooserButton *button); static void update_combo_box (GtkFileChooserButton *button); static void update_label_and_image (GtkFileChooserButton *button); /* Child Object Callbacks */ static void fs_volumes_changed_cb (GtkFileSystem *fs, gpointer user_data); static void bookmarks_changed_cb (gpointer user_data); static void combo_box_changed_cb (GtkComboBox *combo_box, gpointer user_data); static void combo_box_notify_popup_shown_cb (GObject *object, GParamSpec *pspec, gpointer user_data); static void button_clicked_cb (GtkButton *real_button, gpointer user_data); static void chooser_update_preview_cb (GtkFileChooser *dialog, gpointer user_data); static void chooser_notify_cb (GObject *dialog, GParamSpec *pspec, gpointer user_data); static gboolean dialog_delete_event_cb (GtkWidget *dialog, GdkEvent *event, gpointer user_data); static void dialog_response_cb (GtkDialog *dialog, gint response, gpointer user_data); static void native_response_cb (GtkFileChooserNative *native, gint response, gpointer user_data); static guint file_chooser_button_signals[LAST_SIGNAL] = { 0 }; /* ******************* * * GType Declaration * * ******************* */ G_DEFINE_TYPE_WITH_CODE (GtkFileChooserButton, gtk_file_chooser_button, GTK_TYPE_WIDGET, G_ADD_PRIVATE (GtkFileChooserButton) G_IMPLEMENT_INTERFACE (GTK_TYPE_FILE_CHOOSER, gtk_file_chooser_button_file_chooser_iface_init)) static void gtk_file_chooser_button_measure (GtkWidget *widget, GtkOrientation orientation, int for_size, int *minimum, int *natural, int *minimum_baseline, int *natural_baseline) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (widget); GtkFileChooserButtonPrivate *priv = gtk_file_chooser_button_get_instance_private (button); gtk_widget_measure (priv->child, orientation, for_size, minimum, natural, minimum_baseline, natural_baseline); } static void gtk_file_chooser_button_size_allocate (GtkWidget *widget, const GtkAllocation *allocation, int baseline, GtkAllocation *out_clip) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (widget); GtkFileChooserButtonPrivate *priv = gtk_file_chooser_button_get_instance_private (button); gtk_widget_size_allocate (priv->child, allocation, baseline, out_clip); } static void gtk_file_chooser_button_class_init (GtkFileChooserButtonClass * class) { GObjectClass *gobject_class; GtkWidgetClass *widget_class; gobject_class = G_OBJECT_CLASS (class); widget_class = GTK_WIDGET_CLASS (class); gobject_class->constructed = gtk_file_chooser_button_constructed; gobject_class->set_property = gtk_file_chooser_button_set_property; gobject_class->get_property = gtk_file_chooser_button_get_property; gobject_class->finalize = gtk_file_chooser_button_finalize; widget_class->destroy = gtk_file_chooser_button_destroy; widget_class->drag_data_received = gtk_file_chooser_button_drag_data_received; widget_class->show = gtk_file_chooser_button_show; widget_class->hide = gtk_file_chooser_button_hide; widget_class->map = gtk_file_chooser_button_map; widget_class->style_updated = gtk_file_chooser_button_style_updated; widget_class->display_changed = gtk_file_chooser_button_display_changed; widget_class->mnemonic_activate = gtk_file_chooser_button_mnemonic_activate; widget_class->state_flags_changed = gtk_file_chooser_button_state_flags_changed; widget_class->measure = gtk_file_chooser_button_measure; widget_class->size_allocate = gtk_file_chooser_button_size_allocate; /** * GtkFileChooserButton::file-set: * @widget: the object which received the signal. * * The ::file-set signal is emitted when the user selects a file. * * Note that this signal is only emitted when the user * changes the file. * * Since: 2.12 */ file_chooser_button_signals[FILE_SET] = g_signal_new (I_("file-set"), G_TYPE_FROM_CLASS (gobject_class), G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET (GtkFileChooserButtonClass, file_set), NULL, NULL, NULL, G_TYPE_NONE, 0); /** * GtkFileChooserButton:dialog: * * Instance of the #GtkFileChooserDialog associated with the button. * * Since: 2.6 */ g_object_class_install_property (gobject_class, PROP_DIALOG, g_param_spec_object ("dialog", P_("Dialog"), P_("The file chooser dialog to use."), GTK_TYPE_FILE_CHOOSER, (GTK_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY))); /** * GtkFileChooserButton:title: * * Title to put on the #GtkFileChooserDialog associated with the button. * * Since: 2.6 */ g_object_class_install_property (gobject_class, PROP_TITLE, g_param_spec_string ("title", P_("Title"), P_("The title of the file chooser dialog."), _(DEFAULT_TITLE), GTK_PARAM_READWRITE)); /** * GtkFileChooserButton:width-chars: * * The width of the entry and label inside the button, in characters. * * Since: 2.6 */ g_object_class_install_property (gobject_class, PROP_WIDTH_CHARS, g_param_spec_int ("width-chars", P_("Width In Characters"), P_("The desired width of the button widget, in characters."), -1, G_MAXINT, -1, GTK_PARAM_READWRITE)); _gtk_file_chooser_install_properties (gobject_class); gtk_widget_class_set_css_name (widget_class, "filechooserbutton"); } static void gtk_file_chooser_button_init (GtkFileChooserButton *button) { GtkFileChooserButtonPrivate *priv; GtkWidget *box; GtkWidget *icon; GtkTargetList *target_list; gtk_widget_set_has_window (GTK_WIDGET (button), FALSE); priv = button->priv = gtk_file_chooser_button_get_instance_private (button); priv->icon_size = FALLBACK_ICON_SIZE; priv->button = gtk_button_new (); g_signal_connect (priv->button, "clicked", G_CALLBACK (button_clicked_cb), button); priv->image = gtk_image_new (); priv->label = gtk_label_new (_(FALLBACK_DISPLAY_NAME)); gtk_label_set_xalign (GTK_LABEL (priv->label), 0.0f); gtk_widget_set_hexpand (priv->label, TRUE); icon = gtk_image_new_from_icon_name ("document-open-symbolic", GTK_ICON_SIZE_BUTTON); box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0); gtk_widget_set_valign (priv->image, GTK_ALIGN_BASELINE); gtk_container_add (GTK_CONTAINER (box), priv->image); gtk_widget_set_valign (priv->label, GTK_ALIGN_BASELINE); gtk_container_add (GTK_CONTAINER (box), priv->label); gtk_widget_set_valign (icon, GTK_ALIGN_BASELINE); gtk_container_add (GTK_CONTAINER (box), icon); gtk_container_add (GTK_CONTAINER (priv->button), box); gtk_widget_set_parent (priv->button, GTK_WIDGET (button)); priv->model = GTK_TREE_MODEL (gtk_list_store_new (NUM_COLUMNS, CAIRO_GOBJECT_TYPE_SURFACE, G_TYPE_STRING, G_TYPE_CHAR, G_TYPE_POINTER, G_TYPE_BOOLEAN, G_TYPE_POINTER)); priv->combo_box = gtk_combo_box_new (); g_signal_connect (priv->combo_box, "changed", G_CALLBACK (combo_box_changed_cb), button); g_signal_connect (priv->combo_box, "notify::popup-shown", G_CALLBACK (combo_box_notify_popup_shown_cb), button); priv->icon_cell = gtk_cell_renderer_pixbuf_new (); priv->name_cell = gtk_cell_renderer_text_new (); g_object_set (priv->name_cell, "xpad", 6, NULL); gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (priv->combo_box), priv->icon_cell, FALSE); gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (priv->combo_box), priv->icon_cell, "surface", 0, NULL); gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (priv->combo_box), priv->name_cell, FALSE); gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (priv->combo_box), priv->name_cell, "text", 1, NULL); gtk_widget_hide (priv->combo_box); gtk_widget_set_parent (priv->combo_box, GTK_WIDGET (button)); priv->child = priv->button; /* Bookmarks manager */ priv->bookmarks_manager = _gtk_bookmarks_manager_new (bookmarks_changed_cb, button); gtk_cell_layout_set_cell_data_func (GTK_CELL_LAYOUT (priv->combo_box), priv->name_cell, name_cell_data_func, NULL, NULL); /* DnD */ gtk_drag_dest_set (GTK_WIDGET (button), (GTK_DEST_DEFAULT_ALL), NULL, 0, GDK_ACTION_COPY); target_list = gtk_target_list_new (NULL, 0); gtk_target_list_add_uri_targets (target_list, TEXT_URI_LIST); gtk_target_list_add_text_targets (target_list, TEXT_PLAIN); gtk_drag_dest_set_target_list (GTK_WIDGET (button), target_list); gtk_target_list_unref (target_list); } /* ******************************* * * GtkFileChooserIface Functions * * ******************************* */ static void gtk_file_chooser_button_file_chooser_iface_init (GtkFileChooserIface *iface) { _gtk_file_chooser_delegate_iface_init (iface); iface->set_current_folder = gtk_file_chooser_button_set_current_folder; iface->get_current_folder = gtk_file_chooser_button_get_current_folder; iface->select_file = gtk_file_chooser_button_select_file; iface->unselect_file = gtk_file_chooser_button_unselect_file; iface->unselect_all = gtk_file_chooser_button_unselect_all; iface->get_files = gtk_file_chooser_button_get_files; iface->add_shortcut_folder = gtk_file_chooser_button_add_shortcut_folder; iface->remove_shortcut_folder = gtk_file_chooser_button_remove_shortcut_folder; } static void emit_selection_changed_if_changing_selection (GtkFileChooserButton *button) { GtkFileChooserButtonPrivate *priv = button->priv; if (priv->is_changing_selection) { priv->is_changing_selection = FALSE; g_signal_emit_by_name (button, "selection-changed"); } } static gboolean gtk_file_chooser_button_set_current_folder (GtkFileChooser *chooser, GFile *file, GError **error) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (chooser); GtkFileChooserButtonPrivate *priv = button->priv; if (priv->current_folder_while_inactive) g_object_unref (priv->current_folder_while_inactive); priv->current_folder_while_inactive = g_object_ref (file); update_combo_box (button); g_signal_emit_by_name (button, "current-folder-changed"); if (priv->active) gtk_file_chooser_set_current_folder_file (GTK_FILE_CHOOSER (priv->chooser), file, NULL); return TRUE; } static GFile * gtk_file_chooser_button_get_current_folder (GtkFileChooser *chooser) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (chooser); GtkFileChooserButtonPrivate *priv = button->priv; if (priv->current_folder_while_inactive) return g_object_ref (priv->current_folder_while_inactive); else return NULL; } static gboolean gtk_file_chooser_button_select_file (GtkFileChooser *chooser, GFile *file, GError **error) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (chooser); GtkFileChooserButtonPrivate *priv = button->priv; if (priv->selection_while_inactive) g_object_unref (priv->selection_while_inactive); priv->selection_while_inactive = g_object_ref (file); priv->is_changing_selection = TRUE; update_label_and_image (button); update_combo_box (button); if (priv->active) gtk_file_chooser_select_file (GTK_FILE_CHOOSER (priv->chooser), file, NULL); return TRUE; } static void unselect_current_file (GtkFileChooserButton *button) { GtkFileChooserButtonPrivate *priv = button->priv; if (priv->selection_while_inactive) { g_object_unref (priv->selection_while_inactive); priv->selection_while_inactive = NULL; } priv->is_changing_selection = TRUE; update_label_and_image (button); update_combo_box (button); } static void gtk_file_chooser_button_unselect_file (GtkFileChooser *chooser, GFile *file) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (chooser); GtkFileChooserButtonPrivate *priv = button->priv; if (g_file_equal (priv->selection_while_inactive, file)) unselect_current_file (button); if (priv->active) gtk_file_chooser_unselect_file (GTK_FILE_CHOOSER (priv->chooser), file); } static void gtk_file_chooser_button_unselect_all (GtkFileChooser *chooser) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (chooser); GtkFileChooserButtonPrivate *priv = button->priv; unselect_current_file (button); if (priv->active) gtk_file_chooser_unselect_all (GTK_FILE_CHOOSER (priv->chooser)); } static GFile * get_selected_file (GtkFileChooserButton *button) { GtkFileChooserButtonPrivate *priv = button->priv; GFile *retval; retval = NULL; if (priv->selection_while_inactive) retval = priv->selection_while_inactive; else if (priv->chooser && gtk_file_chooser_get_action (priv->chooser) == GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER) { /* If there is no "real" selection in SELECT_FOLDER mode, then we'll just return * the current folder, since that is what GtkFileChooserWidget would do. */ if (priv->current_folder_while_inactive) retval = priv->current_folder_while_inactive; } if (retval) return g_object_ref (retval); else return NULL; } static GSList * gtk_file_chooser_button_get_files (GtkFileChooser *chooser) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (chooser); GFile *file; file = get_selected_file (button); if (file) return g_slist_prepend (NULL, file); else return NULL; } static gboolean gtk_file_chooser_button_add_shortcut_folder (GtkFileChooser *chooser, GFile *file, GError **error) { GtkFileChooser *delegate; gboolean retval; delegate = g_object_get_qdata (G_OBJECT (chooser), GTK_FILE_CHOOSER_DELEGATE_QUARK); retval = _gtk_file_chooser_add_shortcut_folder (delegate, file, error); if (retval) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (chooser); GtkFileChooserButtonPrivate *priv = button->priv; GtkTreeIter iter; gint pos; pos = model_get_type_position (button, ROW_TYPE_SHORTCUT); pos += priv->n_shortcuts; gtk_list_store_insert (GTK_LIST_STORE (priv->model), &iter, pos); gtk_list_store_set (GTK_LIST_STORE (priv->model), &iter, ICON_COLUMN, NULL, DISPLAY_NAME_COLUMN, _(FALLBACK_DISPLAY_NAME), TYPE_COLUMN, ROW_TYPE_SHORTCUT, DATA_COLUMN, g_object_ref (file), IS_FOLDER_COLUMN, FALSE, -1); set_info_for_file_at_iter (button, file, &iter); priv->n_shortcuts++; gtk_tree_model_filter_refilter (GTK_TREE_MODEL_FILTER (priv->filter_model)); } return retval; } static gboolean gtk_file_chooser_button_remove_shortcut_folder (GtkFileChooser *chooser, GFile *file, GError **error) { GtkFileChooser *delegate; gboolean retval; delegate = g_object_get_qdata (G_OBJECT (chooser), GTK_FILE_CHOOSER_DELEGATE_QUARK); retval = _gtk_file_chooser_remove_shortcut_folder (delegate, file, error); if (retval) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (chooser); GtkFileChooserButtonPrivate *priv = button->priv; GtkTreeIter iter; gint pos; gchar type; pos = model_get_type_position (button, ROW_TYPE_SHORTCUT); gtk_tree_model_iter_nth_child (priv->model, &iter, NULL, pos); do { gpointer data; gtk_tree_model_get (priv->model, &iter, TYPE_COLUMN, &type, DATA_COLUMN, &data, -1); if (type == ROW_TYPE_SHORTCUT && data && g_file_equal (data, file)) { model_free_row_data (GTK_FILE_CHOOSER_BUTTON (chooser), &iter); gtk_list_store_remove (GTK_LIST_STORE (priv->model), &iter); priv->n_shortcuts--; gtk_tree_model_filter_refilter (GTK_TREE_MODEL_FILTER (priv->filter_model)); update_combo_box (GTK_FILE_CHOOSER_BUTTON (chooser)); break; } } while (type == ROW_TYPE_SHORTCUT && gtk_tree_model_iter_next (priv->model, &iter)); } return retval; } /* ******************* * * GObject Functions * * ******************* */ static void gtk_file_chooser_button_constructed (GObject *object) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (object); GtkFileChooserButtonPrivate *priv = button->priv; GSList *list; G_OBJECT_CLASS (gtk_file_chooser_button_parent_class)->constructed (object); if (!priv->dialog) { priv->native = gtk_file_chooser_native_new (NULL, NULL, GTK_FILE_CHOOSER_ACTION_OPEN, NULL, NULL); priv->chooser = GTK_FILE_CHOOSER (priv->native); gtk_file_chooser_button_set_title (button, _(DEFAULT_TITLE)); g_signal_connect (priv->native, "response", G_CALLBACK (native_response_cb), object); } else /* dialog set */ { priv->chooser = GTK_FILE_CHOOSER (priv->dialog); if (!gtk_window_get_title (GTK_WINDOW (priv->dialog))) gtk_file_chooser_button_set_title (button, _(DEFAULT_TITLE)); g_signal_connect (priv->dialog, "delete-event", G_CALLBACK (dialog_delete_event_cb), object); g_signal_connect (priv->dialog, "response", G_CALLBACK (dialog_response_cb), object); g_object_add_weak_pointer (G_OBJECT (priv->dialog), (gpointer) (&priv->dialog)); } g_signal_connect (priv->chooser, "notify", G_CALLBACK (chooser_notify_cb), object); /* This is used, instead of the standard delegate, to ensure that signals are only * delegated when the OK button is pressed. */ g_object_set_qdata (object, GTK_FILE_CHOOSER_DELEGATE_QUARK, priv->chooser); priv->fs = g_object_ref (_gtk_file_chooser_get_file_system (priv->chooser)); model_add_special (button); list = _gtk_file_system_list_volumes (priv->fs); model_add_volumes (button, list); g_slist_free (list); list = _gtk_bookmarks_manager_list_bookmarks (priv->bookmarks_manager); model_add_bookmarks (button, list); g_slist_free_full (list, g_object_unref); model_add_other (button); model_add_empty_selection (button); priv->filter_model = gtk_tree_model_filter_new (priv->model, NULL); gtk_tree_model_filter_set_visible_func (GTK_TREE_MODEL_FILTER (priv->filter_model), filter_model_visible_func, object, NULL); gtk_combo_box_set_model (GTK_COMBO_BOX (priv->combo_box), priv->filter_model); gtk_combo_box_set_row_separator_func (GTK_COMBO_BOX (priv->combo_box), combo_box_row_separator_func, NULL, NULL); /* set up the action for a user-provided dialog, this also updates * the label, image and combobox */ g_object_set (object, "action", gtk_file_chooser_get_action (GTK_FILE_CHOOSER (priv->chooser)), NULL); priv->fs_volumes_changed_id = g_signal_connect (priv->fs, "volumes-changed", G_CALLBACK (fs_volumes_changed_cb), object); update_label_and_image (button); update_combo_box (button); } static void gtk_file_chooser_button_set_property (GObject *object, guint param_id, const GValue *value, GParamSpec *pspec) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (object); GtkFileChooserButtonPrivate *priv = button->priv; switch (param_id) { case PROP_DIALOG: /* Construct-only */ priv->dialog = g_value_get_object (value); break; case PROP_WIDTH_CHARS: gtk_file_chooser_button_set_width_chars (GTK_FILE_CHOOSER_BUTTON (object), g_value_get_int (value)); break; case GTK_FILE_CHOOSER_PROP_ACTION: switch (g_value_get_enum (value)) { case GTK_FILE_CHOOSER_ACTION_CREATE_FOLDER: case GTK_FILE_CHOOSER_ACTION_SAVE: { GEnumClass *eclass; GEnumValue *eval; eclass = g_type_class_peek (GTK_TYPE_FILE_CHOOSER_ACTION); eval = g_enum_get_value (eclass, g_value_get_enum (value)); g_warning ("%s: Choosers of type '%s' do not support '%s'.", G_STRFUNC, G_OBJECT_TYPE_NAME (object), eval->value_name); g_value_set_enum ((GValue *) value, GTK_FILE_CHOOSER_ACTION_OPEN); } break; default: break; } g_object_set_property (G_OBJECT (priv->chooser), pspec->name, value); update_label_and_image (GTK_FILE_CHOOSER_BUTTON (object)); update_combo_box (GTK_FILE_CHOOSER_BUTTON (object)); switch (g_value_get_enum (value)) { case GTK_FILE_CHOOSER_ACTION_OPEN: gtk_widget_hide (priv->combo_box); gtk_widget_show (priv->button); priv->child = priv->button; gtk_widget_queue_resize (GTK_WIDGET (button)); break; case GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER: gtk_widget_show (priv->combo_box); gtk_widget_hide (priv->button); priv->child = priv->combo_box; gtk_widget_queue_resize (GTK_WIDGET (button)); break; default: g_assert_not_reached (); break; } break; case PROP_TITLE: case GTK_FILE_CHOOSER_PROP_FILTER: case GTK_FILE_CHOOSER_PROP_PREVIEW_WIDGET: case GTK_FILE_CHOOSER_PROP_PREVIEW_WIDGET_ACTIVE: case GTK_FILE_CHOOSER_PROP_USE_PREVIEW_LABEL: case GTK_FILE_CHOOSER_PROP_EXTRA_WIDGET: case GTK_FILE_CHOOSER_PROP_SHOW_HIDDEN: case GTK_FILE_CHOOSER_PROP_DO_OVERWRITE_CONFIRMATION: case GTK_FILE_CHOOSER_PROP_CREATE_FOLDERS: g_object_set_property (G_OBJECT (priv->chooser), pspec->name, value); break; case GTK_FILE_CHOOSER_PROP_LOCAL_ONLY: g_object_set_property (G_OBJECT (priv->chooser), pspec->name, value); fs_volumes_changed_cb (priv->fs, button); bookmarks_changed_cb (button); break; case GTK_FILE_CHOOSER_PROP_SELECT_MULTIPLE: g_warning ("%s: Choosers of type '%s' do not support selecting multiple files.", G_STRFUNC, G_OBJECT_TYPE_NAME (object)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec); break; } } static void gtk_file_chooser_button_get_property (GObject *object, guint param_id, GValue *value, GParamSpec *pspec) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (object); GtkFileChooserButtonPrivate *priv = button->priv; switch (param_id) { case PROP_WIDTH_CHARS: g_value_set_int (value, gtk_label_get_width_chars (GTK_LABEL (priv->label))); break; case PROP_TITLE: case GTK_FILE_CHOOSER_PROP_ACTION: case GTK_FILE_CHOOSER_PROP_FILTER: case GTK_FILE_CHOOSER_PROP_LOCAL_ONLY: case GTK_FILE_CHOOSER_PROP_PREVIEW_WIDGET: case GTK_FILE_CHOOSER_PROP_PREVIEW_WIDGET_ACTIVE: case GTK_FILE_CHOOSER_PROP_USE_PREVIEW_LABEL: case GTK_FILE_CHOOSER_PROP_EXTRA_WIDGET: case GTK_FILE_CHOOSER_PROP_SELECT_MULTIPLE: case GTK_FILE_CHOOSER_PROP_SHOW_HIDDEN: case GTK_FILE_CHOOSER_PROP_DO_OVERWRITE_CONFIRMATION: case GTK_FILE_CHOOSER_PROP_CREATE_FOLDERS: g_object_get_property (G_OBJECT (priv->chooser), pspec->name, value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec); break; } } static void gtk_file_chooser_button_finalize (GObject *object) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (object); GtkFileChooserButtonPrivate *priv = button->priv; if (priv->selection_while_inactive) g_object_unref (priv->selection_while_inactive); if (priv->current_folder_while_inactive) g_object_unref (priv->current_folder_while_inactive); if (priv->model) g_object_unref (priv->model); gtk_widget_unparent (priv->button); gtk_widget_unparent (priv->combo_box); G_OBJECT_CLASS (gtk_file_chooser_button_parent_class)->finalize (object); } /* ********************* * * GtkWidget Functions * * ********************* */ static void gtk_file_chooser_button_state_flags_changed (GtkWidget *widget, GtkStateFlags previous_state) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (widget); GtkFileChooserButtonPrivate *priv = button->priv; if (gtk_widget_get_state_flags (widget) & GTK_STATE_FLAG_DROP_ACTIVE) gtk_widget_set_state_flags (priv->child, GTK_STATE_FLAG_DROP_ACTIVE, FALSE); else gtk_widget_unset_state_flags (priv->child, GTK_STATE_FLAG_DROP_ACTIVE); GTK_WIDGET_CLASS (gtk_file_chooser_button_parent_class)->state_flags_changed (widget, previous_state); } static void gtk_file_chooser_button_destroy (GtkWidget *widget) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (widget); GtkFileChooserButtonPrivate *priv = button->priv; GSList *l; if (priv->dialog != NULL) { gtk_widget_destroy (priv->dialog); priv->dialog = NULL; } if (priv->native) { gtk_native_dialog_destroy (GTK_NATIVE_DIALOG (priv->native)); g_clear_object (&priv->native); } priv->chooser = NULL; if (priv->dnd_select_folder_cancellable) { g_cancellable_cancel (priv->dnd_select_folder_cancellable); priv->dnd_select_folder_cancellable = NULL; } if (priv->update_button_cancellable) { g_cancellable_cancel (priv->update_button_cancellable); priv->update_button_cancellable = NULL; } if (priv->change_icon_theme_cancellables) { for (l = priv->change_icon_theme_cancellables; l; l = l->next) { GCancellable *cancellable = G_CANCELLABLE (l->data); g_cancellable_cancel (cancellable); } g_slist_free (priv->change_icon_theme_cancellables); priv->change_icon_theme_cancellables = NULL; } if (priv->filter_model) { g_object_unref (priv->filter_model); priv->filter_model = NULL; } if (priv->fs) { g_signal_handler_disconnect (priv->fs, priv->fs_volumes_changed_id); g_object_unref (priv->fs); priv->fs = NULL; } if (priv->bookmarks_manager) { _gtk_bookmarks_manager_free (priv->bookmarks_manager); priv->bookmarks_manager = NULL; } GTK_WIDGET_CLASS (gtk_file_chooser_button_parent_class)->destroy (widget); } struct DndSelectFolderData { GtkFileSystem *file_system; GtkFileChooserButton *button; GtkFileChooserAction action; GFile *file; gchar **uris; guint i; gboolean selected; }; static void dnd_select_folder_get_info_cb (GCancellable *cancellable, GFileInfo *info, const GError *error, gpointer user_data) { gboolean cancelled = g_cancellable_is_cancelled (cancellable); struct DndSelectFolderData *data = user_data; if (cancellable != data->button->priv->dnd_select_folder_cancellable) { g_object_unref (data->button); g_object_unref (data->file); g_strfreev (data->uris); g_free (data); g_object_unref (cancellable); return; } data->button->priv->dnd_select_folder_cancellable = NULL; if (!cancelled && !error && info != NULL) { gboolean is_folder; is_folder = _gtk_file_info_consider_as_directory (info); data->selected = (((data->action == GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER && is_folder) || (data->action == GTK_FILE_CHOOSER_ACTION_OPEN && !is_folder)) && gtk_file_chooser_select_file (GTK_FILE_CHOOSER (data->button), data->file, NULL)); } else data->selected = FALSE; if (data->selected || data->uris[++data->i] == NULL) { g_signal_emit (data->button, file_chooser_button_signals[FILE_SET], 0); g_object_unref (data->button); g_object_unref (data->file); g_strfreev (data->uris); g_free (data); g_object_unref (cancellable); return; } if (data->file) g_object_unref (data->file); data->file = g_file_new_for_uri (data->uris[data->i]); data->button->priv->dnd_select_folder_cancellable = _gtk_file_system_get_info (data->file_system, data->file, "standard::type", dnd_select_folder_get_info_cb, user_data); g_object_unref (cancellable); } static void gtk_file_chooser_button_drag_data_received (GtkWidget *widget, GdkDragContext *context, gint x, gint y, GtkSelectionData *data, guint type, guint drag_time) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (widget); GtkFileChooserButtonPrivate *priv = button->priv; GFile *file; gchar *text; if (GTK_WIDGET_CLASS (gtk_file_chooser_button_parent_class)->drag_data_received != NULL) GTK_WIDGET_CLASS (gtk_file_chooser_button_parent_class)->drag_data_received (widget, context, x, y, data, type, drag_time); if (widget == NULL || context == NULL || data == NULL || gtk_selection_data_get_length (data) < 0) return; switch (type) { case TEXT_URI_LIST: { gchar **uris; struct DndSelectFolderData *info; uris = gtk_selection_data_get_uris (data); if (uris == NULL) break; info = g_new0 (struct DndSelectFolderData, 1); info->button = g_object_ref (button); info->i = 0; info->uris = uris; info->selected = FALSE; info->file_system = priv->fs; g_object_get (priv->chooser, "action", &info->action, NULL); info->file = g_file_new_for_uri (info->uris[info->i]); if (priv->dnd_select_folder_cancellable) g_cancellable_cancel (priv->dnd_select_folder_cancellable); priv->dnd_select_folder_cancellable = _gtk_file_system_get_info (priv->fs, info->file, "standard::type", dnd_select_folder_get_info_cb, info); } break; case TEXT_PLAIN: text = (char*) gtk_selection_data_get_text (data); file = g_file_new_for_uri (text); gtk_file_chooser_select_file (GTK_FILE_CHOOSER (priv->chooser), file, NULL); g_object_unref (file); g_free (text); g_signal_emit (button, file_chooser_button_signals[FILE_SET], 0); break; default: break; } gtk_drag_finish (context, TRUE, FALSE, drag_time); } static void gtk_file_chooser_button_show (GtkWidget *widget) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (widget); GtkFileChooserButtonPrivate *priv = button->priv; if (GTK_WIDGET_CLASS (gtk_file_chooser_button_parent_class)->show) GTK_WIDGET_CLASS (gtk_file_chooser_button_parent_class)->show (widget); if (priv->active) open_dialog (GTK_FILE_CHOOSER_BUTTON (widget)); } static void gtk_file_chooser_button_hide (GtkWidget *widget) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (widget); GtkFileChooserButtonPrivate *priv = button->priv; if (priv->dialog) gtk_widget_hide (priv->dialog); else gtk_native_dialog_hide (GTK_NATIVE_DIALOG (priv->native)); if (GTK_WIDGET_CLASS (gtk_file_chooser_button_parent_class)->hide) GTK_WIDGET_CLASS (gtk_file_chooser_button_parent_class)->hide (widget); } static void gtk_file_chooser_button_map (GtkWidget *widget) { GTK_WIDGET_CLASS (gtk_file_chooser_button_parent_class)->map (widget); } static gboolean gtk_file_chooser_button_mnemonic_activate (GtkWidget *widget, gboolean group_cycling) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (widget); GtkFileChooserButtonPrivate *priv = button->priv; switch (gtk_file_chooser_get_action (GTK_FILE_CHOOSER (priv->chooser))) { case GTK_FILE_CHOOSER_ACTION_OPEN: gtk_widget_grab_focus (priv->button); break; case GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER: return gtk_widget_mnemonic_activate (priv->combo_box, group_cycling); break; case GTK_FILE_CHOOSER_ACTION_SAVE: case GTK_FILE_CHOOSER_ACTION_CREATE_FOLDER: default: g_assert_not_reached (); break; } return TRUE; } /* Changes the icons wherever it is needed */ struct ChangeIconThemeData { GtkFileChooserButton *button; GtkTreeRowReference *row_ref; }; static void change_icon_theme_get_info_cb (GCancellable *cancellable, GFileInfo *info, const GError *error, gpointer user_data) { gboolean cancelled = g_cancellable_is_cancelled (cancellable); cairo_surface_t *surface; struct ChangeIconThemeData *data = user_data; if (!g_slist_find (data->button->priv->change_icon_theme_cancellables, cancellable)) goto out; data->button->priv->change_icon_theme_cancellables = g_slist_remove (data->button->priv->change_icon_theme_cancellables, cancellable); if (cancelled || error) goto out; surface = _gtk_file_info_render_icon (info, GTK_WIDGET (data->button), data->button->priv->icon_size); if (surface) { gint width = 0; GtkTreeIter iter; GtkTreePath *path; width = MAX (width, data->button->priv->icon_size); path = gtk_tree_row_reference_get_path (data->row_ref); if (path) { gtk_tree_model_get_iter (data->button->priv->model, &iter, path); gtk_tree_path_free (path); gtk_list_store_set (GTK_LIST_STORE (data->button->priv->model), &iter, ICON_COLUMN, surface, -1); g_object_set (data->button->priv->icon_cell, "width", width, NULL); } cairo_surface_destroy (surface); } out: g_object_unref (data->button); gtk_tree_row_reference_free (data->row_ref); g_free (data); g_object_unref (cancellable); } static void change_icon_theme (GtkFileChooserButton *button) { GtkFileChooserButtonPrivate *priv = button->priv; GtkIconTheme *theme; GtkTreeIter iter; GSList *l; gint width = 0, height = 0; for (l = button->priv->change_icon_theme_cancellables; l; l = l->next) { GCancellable *cancellable = G_CANCELLABLE (l->data); g_cancellable_cancel (cancellable); } g_slist_free (button->priv->change_icon_theme_cancellables); button->priv->change_icon_theme_cancellables = NULL; if (gtk_icon_size_lookup (GTK_ICON_SIZE_MENU, &width, &height)) priv->icon_size = MAX (width, height); else priv->icon_size = FALLBACK_ICON_SIZE; update_label_and_image (button); gtk_tree_model_get_iter_first (priv->model, &iter); theme = get_icon_theme (GTK_WIDGET (button)); do { cairo_surface_t *surface = NULL; gchar type; gpointer data; type = ROW_TYPE_INVALID; gtk_tree_model_get (priv->model, &iter, TYPE_COLUMN, &type, DATA_COLUMN, &data, -1); switch (type) { case ROW_TYPE_SPECIAL: case ROW_TYPE_SHORTCUT: case ROW_TYPE_BOOKMARK: case ROW_TYPE_CURRENT_FOLDER: if (data) { if (g_file_is_native (G_FILE (data))) { GtkTreePath *path; GCancellable *cancellable; struct ChangeIconThemeData *info; info = g_new0 (struct ChangeIconThemeData, 1); info->button = g_object_ref (button); path = gtk_tree_model_get_path (priv->model, &iter); info->row_ref = gtk_tree_row_reference_new (priv->model, path); gtk_tree_path_free (path); cancellable = _gtk_file_system_get_info (priv->fs, data, "standard::icon", change_icon_theme_get_info_cb, info); button->priv->change_icon_theme_cancellables = g_slist_append (button->priv->change_icon_theme_cancellables, cancellable); surface = NULL; } else { /* Don't call get_info for remote paths to avoid latency and * auth dialogs. * If we switch to a better bookmarks file format (XBEL), we * should use mime info to get a better icon. */ surface = gtk_icon_theme_load_surface (theme, "folder-remote", priv->icon_size, gtk_widget_get_scale_factor (GTK_WIDGET (button)), gtk_widget_get_window (GTK_WIDGET (button)), 0, NULL); } } break; case ROW_TYPE_VOLUME: if (data) { surface = _gtk_file_system_volume_render_icon (data, GTK_WIDGET (button), priv->icon_size, NULL); } break; default: continue; break; } if (surface) width = MAX (width, priv->icon_size); gtk_list_store_set (GTK_LIST_STORE (priv->model), &iter, ICON_COLUMN, surface, -1); if (surface) cairo_surface_destroy (surface); } while (gtk_tree_model_iter_next (priv->model, &iter)); g_object_set (button->priv->icon_cell, "width", width, NULL); } static void gtk_file_chooser_button_style_updated (GtkWidget *widget) { GtkStyleContext *context = gtk_widget_get_style_context (widget); GtkCssStyleChange *change = gtk_style_context_get_change (context); GTK_WIDGET_CLASS (gtk_file_chooser_button_parent_class)->style_updated (widget); /* We need to update the icon surface, but only in case * the icon theme really changed. */ if (!change || gtk_css_style_change_changes_property (change, GTK_CSS_PROPERTY_ICON_THEME)) change_icon_theme (GTK_FILE_CHOOSER_BUTTON (widget)); } static void gtk_file_chooser_button_display_changed (GtkWidget *widget, GdkDisplay *old_display) { if (GTK_WIDGET_CLASS (gtk_file_chooser_button_parent_class)->display_changed) GTK_WIDGET_CLASS (gtk_file_chooser_button_parent_class)->display_changed (widget, old_display); change_icon_theme (GTK_FILE_CHOOSER_BUTTON (widget)); } /* ******************* * * Utility Functions * * ******************* */ /* General */ static GtkIconTheme * get_icon_theme (GtkWidget *widget) { return gtk_css_icon_theme_value_get_icon_theme (_gtk_style_context_peek_property (gtk_widget_get_style_context (widget), GTK_CSS_PROPERTY_ICON_THEME)); } struct SetDisplayNameData { GtkFileChooserButton *button; char *label; GtkTreeRowReference *row_ref; }; static void set_info_get_info_cb (GCancellable *cancellable, GFileInfo *info, const GError *error, gpointer callback_data) { gboolean cancelled = g_cancellable_is_cancelled (cancellable); cairo_surface_t *surface; GtkTreePath *path; GtkTreeIter iter; GCancellable *model_cancellable = NULL; struct SetDisplayNameData *data = callback_data; gboolean is_folder; if (!data->button->priv->model) /* button got destroyed */ goto out; path = gtk_tree_row_reference_get_path (data->row_ref); if (!path) /* Cancellable doesn't exist anymore in the model */ goto out; gtk_tree_model_get_iter (data->button->priv->model, &iter, path); gtk_tree_path_free (path); /* Validate the cancellable */ gtk_tree_model_get (data->button->priv->model, &iter, CANCELLABLE_COLUMN, &model_cancellable, -1); if (cancellable != model_cancellable) goto out; gtk_list_store_set (GTK_LIST_STORE (data->button->priv->model), &iter, CANCELLABLE_COLUMN, NULL, -1); if (cancelled || error) /* There was an error, leave the fallback name in there */ goto out; surface = _gtk_file_info_render_icon (info, GTK_WIDGET (data->button), data->button->priv->icon_size); if (!data->label) data->label = g_strdup (g_file_info_get_display_name (info)); is_folder = _gtk_file_info_consider_as_directory (info); gtk_list_store_set (GTK_LIST_STORE (data->button->priv->model), &iter, ICON_COLUMN, surface, DISPLAY_NAME_COLUMN, data->label, IS_FOLDER_COLUMN, is_folder, -1); if (surface) cairo_surface_destroy (surface); out: g_object_unref (data->button); g_free (data->label); gtk_tree_row_reference_free (data->row_ref); g_free (data); g_object_unref (cancellable); } static void set_info_for_file_at_iter (GtkFileChooserButton *button, GFile *file, GtkTreeIter *iter) { struct SetDisplayNameData *data; GtkTreePath *tree_path; GCancellable *cancellable; data = g_new0 (struct SetDisplayNameData, 1); data->button = g_object_ref (button); data->label = _gtk_bookmarks_manager_get_bookmark_label (button->priv->bookmarks_manager, file); tree_path = gtk_tree_model_get_path (button->priv->model, iter); data->row_ref = gtk_tree_row_reference_new (button->priv->model, tree_path); gtk_tree_path_free (tree_path); cancellable = _gtk_file_system_get_info (button->priv->fs, file, "standard::type,standard::icon,standard::display-name", set_info_get_info_cb, data); gtk_list_store_set (GTK_LIST_STORE (button->priv->model), iter, CANCELLABLE_COLUMN, cancellable, -1); } /* Shortcuts Model */ static gint model_get_type_position (GtkFileChooserButton *button, RowType row_type) { gint retval = 0; if (row_type == ROW_TYPE_SPECIAL) return retval; retval += button->priv->n_special; if (row_type == ROW_TYPE_VOLUME) return retval; retval += button->priv->n_volumes; if (row_type == ROW_TYPE_SHORTCUT) return retval; retval += button->priv->n_shortcuts; if (row_type == ROW_TYPE_BOOKMARK_SEPARATOR) return retval; retval += button->priv->has_bookmark_separator; if (row_type == ROW_TYPE_BOOKMARK) return retval; retval += button->priv->n_bookmarks; if (row_type == ROW_TYPE_CURRENT_FOLDER_SEPARATOR) return retval; retval += button->priv->has_current_folder_separator; if (row_type == ROW_TYPE_CURRENT_FOLDER) return retval; retval += button->priv->has_current_folder; if (row_type == ROW_TYPE_OTHER_SEPARATOR) return retval; retval += button->priv->has_other_separator; if (row_type == ROW_TYPE_OTHER) return retval; retval++; if (row_type == ROW_TYPE_EMPTY_SELECTION) return retval; g_assert_not_reached (); return -1; } static void model_free_row_data (GtkFileChooserButton *button, GtkTreeIter *iter) { gchar type; gpointer data; GCancellable *cancellable; gtk_tree_model_get (button->priv->model, iter, TYPE_COLUMN, &type, DATA_COLUMN, &data, CANCELLABLE_COLUMN, &cancellable, -1); if (cancellable) g_cancellable_cancel (cancellable); switch (type) { case ROW_TYPE_SPECIAL: case ROW_TYPE_SHORTCUT: case ROW_TYPE_BOOKMARK: case ROW_TYPE_CURRENT_FOLDER: g_object_unref (data); break; case ROW_TYPE_VOLUME: _gtk_file_system_volume_unref (data); break; default: break; } } static void model_add_special_get_info_cb (GCancellable *cancellable, GFileInfo *info, const GError *error, gpointer user_data) { gboolean cancelled = g_cancellable_is_cancelled (cancellable); GtkTreeIter iter; GtkTreePath *path; cairo_surface_t *surface; GCancellable *model_cancellable = NULL; struct ChangeIconThemeData *data = user_data; gchar *name; if (!data->button->priv->model) /* button got destroyed */ goto out; path = gtk_tree_row_reference_get_path (data->row_ref); if (!path) /* Cancellable doesn't exist anymore in the model */ goto out; gtk_tree_model_get_iter (data->button->priv->model, &iter, path); gtk_tree_path_free (path); gtk_tree_model_get (data->button->priv->model, &iter, CANCELLABLE_COLUMN, &model_cancellable, -1); if (cancellable != model_cancellable) goto out; gtk_list_store_set (GTK_LIST_STORE (data->button->priv->model), &iter, CANCELLABLE_COLUMN, NULL, -1); if (cancelled || error) goto out; surface = _gtk_file_info_render_icon (info, GTK_WIDGET (data->button), data->button->priv->icon_size); if (surface) { gtk_list_store_set (GTK_LIST_STORE (data->button->priv->model), &iter, ICON_COLUMN, surface, -1); cairo_surface_destroy (surface); } gtk_tree_model_get (data->button->priv->model, &iter, DISPLAY_NAME_COLUMN, &name, -1); if (!name) gtk_list_store_set (GTK_LIST_STORE (data->button->priv->model), &iter, DISPLAY_NAME_COLUMN, g_file_info_get_display_name (info), -1); g_free (name); out: g_object_unref (data->button); gtk_tree_row_reference_free (data->row_ref); g_free (data); g_object_unref (cancellable); } static void model_add_special (GtkFileChooserButton *button) { const gchar *homedir; const gchar *desktopdir; GtkListStore *store; GtkTreeIter iter; GFile *file; gint pos; store = GTK_LIST_STORE (button->priv->model); pos = model_get_type_position (button, ROW_TYPE_SPECIAL); homedir = g_get_home_dir (); if (homedir) { GtkTreePath *tree_path; GCancellable *cancellable; struct ChangeIconThemeData *info; file = g_file_new_for_path (homedir); gtk_list_store_insert (store, &iter, pos); pos++; info = g_new0 (struct ChangeIconThemeData, 1); info->button = g_object_ref (button); tree_path = gtk_tree_model_get_path (GTK_TREE_MODEL (store), &iter); info->row_ref = gtk_tree_row_reference_new (GTK_TREE_MODEL (store), tree_path); gtk_tree_path_free (tree_path); cancellable = _gtk_file_system_get_info (button->priv->fs, file, "standard::icon,standard::display-name", model_add_special_get_info_cb, info); gtk_list_store_set (store, &iter, ICON_COLUMN, NULL, DISPLAY_NAME_COLUMN, NULL, TYPE_COLUMN, ROW_TYPE_SPECIAL, DATA_COLUMN, file, IS_FOLDER_COLUMN, TRUE, CANCELLABLE_COLUMN, cancellable, -1); button->priv->n_special++; } desktopdir = g_get_user_special_dir (G_USER_DIRECTORY_DESKTOP); /* "To disable a directory, point it to the homedir." * See http://freedesktop.org/wiki/Software/xdg-user-dirs */ if (g_strcmp0 (desktopdir, g_get_home_dir ()) != 0) { GtkTreePath *tree_path; GCancellable *cancellable; struct ChangeIconThemeData *info; file = g_file_new_for_path (desktopdir); gtk_list_store_insert (store, &iter, pos); pos++; info = g_new0 (struct ChangeIconThemeData, 1); info->button = g_object_ref (button); tree_path = gtk_tree_model_get_path (GTK_TREE_MODEL (store), &iter); info->row_ref = gtk_tree_row_reference_new (GTK_TREE_MODEL (store), tree_path); gtk_tree_path_free (tree_path); cancellable = _gtk_file_system_get_info (button->priv->fs, file, "standard::icon,standard::display-name", model_add_special_get_info_cb, info); gtk_list_store_set (store, &iter, TYPE_COLUMN, ROW_TYPE_SPECIAL, ICON_COLUMN, NULL, DISPLAY_NAME_COLUMN, _(DESKTOP_DISPLAY_NAME), DATA_COLUMN, file, IS_FOLDER_COLUMN, TRUE, CANCELLABLE_COLUMN, cancellable, -1); button->priv->n_special++; } } static void model_add_volumes (GtkFileChooserButton *button, GSList *volumes) { GtkListStore *store; gint pos; gboolean local_only; GSList *l; if (!volumes) return; store = GTK_LIST_STORE (button->priv->model); pos = model_get_type_position (button, ROW_TYPE_VOLUME); local_only = gtk_file_chooser_get_local_only (GTK_FILE_CHOOSER (button->priv->chooser)); for (l = volumes; l; l = l->next) { GtkFileSystemVolume *volume; GtkTreeIter iter; cairo_surface_t *surface; gchar *display_name; volume = l->data; if (local_only) { if (_gtk_file_system_volume_is_mounted (volume)) { GFile *base_file; base_file = _gtk_file_system_volume_get_root (volume); if (base_file != NULL) { if (!_gtk_file_has_native_path (base_file)) { g_object_unref (base_file); continue; } else g_object_unref (base_file); } } } surface = _gtk_file_system_volume_render_icon (volume, GTK_WIDGET (button), button->priv->icon_size, NULL); display_name = _gtk_file_system_volume_get_display_name (volume); gtk_list_store_insert (store, &iter, pos); gtk_list_store_set (store, &iter, ICON_COLUMN, surface, DISPLAY_NAME_COLUMN, display_name, TYPE_COLUMN, ROW_TYPE_VOLUME, DATA_COLUMN, _gtk_file_system_volume_ref (volume), IS_FOLDER_COLUMN, TRUE, -1); if (surface) cairo_surface_destroy (surface); g_free (display_name); button->priv->n_volumes++; pos++; } } static void model_add_bookmarks (GtkFileChooserButton *button, GSList *bookmarks) { GtkListStore *store; GtkTreeIter iter; gint pos; gboolean local_only; GSList *l; if (!bookmarks) return; store = GTK_LIST_STORE (button->priv->model); pos = model_get_type_position (button, ROW_TYPE_BOOKMARK); local_only = gtk_file_chooser_get_local_only (GTK_FILE_CHOOSER (button->priv->chooser)); for (l = bookmarks; l; l = l->next) { GFile *file; file = l->data; if (_gtk_file_has_native_path (file)) { gtk_list_store_insert (store, &iter, pos); gtk_list_store_set (store, &iter, ICON_COLUMN, NULL, DISPLAY_NAME_COLUMN, _(FALLBACK_DISPLAY_NAME), TYPE_COLUMN, ROW_TYPE_BOOKMARK, DATA_COLUMN, g_object_ref (file), IS_FOLDER_COLUMN, FALSE, -1); set_info_for_file_at_iter (button, file, &iter); } else { gchar *label; GtkIconTheme *icon_theme; cairo_surface_t *surface = NULL; if (local_only) continue; /* Don't call get_info for remote paths to avoid latency and * auth dialogs. * If we switch to a better bookmarks file format (XBEL), we * should use mime info to get a better icon. */ label = _gtk_bookmarks_manager_get_bookmark_label (button->priv->bookmarks_manager, file); if (!label) label = _gtk_file_chooser_label_for_file (file); icon_theme = get_icon_theme (GTK_WIDGET (button)); surface = gtk_icon_theme_load_surface (icon_theme, "folder-remote", button->priv->icon_size, gtk_widget_get_scale_factor (GTK_WIDGET (button)), gtk_widget_get_window (GTK_WIDGET (button)), 0, NULL); gtk_list_store_insert (store, &iter, pos); gtk_list_store_set (store, &iter, ICON_COLUMN, surface, DISPLAY_NAME_COLUMN, label, TYPE_COLUMN, ROW_TYPE_BOOKMARK, DATA_COLUMN, g_object_ref (file), IS_FOLDER_COLUMN, TRUE, -1); g_free (label); if (surface) cairo_surface_destroy (surface); } button->priv->n_bookmarks++; pos++; } if (button->priv->n_bookmarks > 0 && !button->priv->has_bookmark_separator) { pos = model_get_type_position (button, ROW_TYPE_BOOKMARK_SEPARATOR); gtk_list_store_insert (store, &iter, pos); gtk_list_store_set (store, &iter, ICON_COLUMN, NULL, DISPLAY_NAME_COLUMN, NULL, TYPE_COLUMN, ROW_TYPE_BOOKMARK_SEPARATOR, DATA_COLUMN, NULL, IS_FOLDER_COLUMN, FALSE, -1); button->priv->has_bookmark_separator = TRUE; } } static void model_update_current_folder (GtkFileChooserButton *button, GFile *file) { GtkListStore *store; GtkTreeIter iter; gint pos; if (!file) return; store = GTK_LIST_STORE (button->priv->model); if (!button->priv->has_current_folder_separator) { pos = model_get_type_position (button, ROW_TYPE_CURRENT_FOLDER_SEPARATOR); gtk_list_store_insert (store, &iter, pos); gtk_list_store_set (store, &iter, ICON_COLUMN, NULL, DISPLAY_NAME_COLUMN, NULL, TYPE_COLUMN, ROW_TYPE_CURRENT_FOLDER_SEPARATOR, DATA_COLUMN, NULL, IS_FOLDER_COLUMN, FALSE, -1); button->priv->has_current_folder_separator = TRUE; } pos = model_get_type_position (button, ROW_TYPE_CURRENT_FOLDER); if (!button->priv->has_current_folder) { gtk_list_store_insert (store, &iter, pos); button->priv->has_current_folder = TRUE; } else { gtk_tree_model_iter_nth_child (button->priv->model, &iter, NULL, pos); model_free_row_data (button, &iter); } if (g_file_is_native (file)) { gtk_list_store_set (store, &iter, ICON_COLUMN, NULL, DISPLAY_NAME_COLUMN, _(FALLBACK_DISPLAY_NAME), TYPE_COLUMN, ROW_TYPE_CURRENT_FOLDER, DATA_COLUMN, g_object_ref (file), IS_FOLDER_COLUMN, FALSE, -1); set_info_for_file_at_iter (button, file, &iter); } else { gchar *label; GtkIconTheme *icon_theme; cairo_surface_t *surface; /* Don't call get_info for remote paths to avoid latency and * auth dialogs. * If we switch to a better bookmarks file format (XBEL), we * should use mime info to get a better icon. */ label = _gtk_bookmarks_manager_get_bookmark_label (button->priv->bookmarks_manager, file); if (!label) label = _gtk_file_chooser_label_for_file (file); icon_theme = get_icon_theme (GTK_WIDGET (button)); if (g_file_is_native (file)) surface = gtk_icon_theme_load_surface (icon_theme, "folder", button->priv->icon_size, gtk_widget_get_scale_factor (GTK_WIDGET (button)), gtk_widget_get_window (GTK_WIDGET (button)), 0, NULL); else surface = gtk_icon_theme_load_surface (icon_theme, "folder-remote", button->priv->icon_size, gtk_widget_get_scale_factor (GTK_WIDGET (button)), gtk_widget_get_window (GTK_WIDGET (button)), 0, NULL); gtk_list_store_set (store, &iter, ICON_COLUMN, surface, DISPLAY_NAME_COLUMN, label, TYPE_COLUMN, ROW_TYPE_CURRENT_FOLDER, DATA_COLUMN, g_object_ref (file), IS_FOLDER_COLUMN, TRUE, -1); g_free (label); if (surface) cairo_surface_destroy (surface); } } static void model_add_other (GtkFileChooserButton *button) { GtkListStore *store; GtkTreeIter iter; gint pos; store = GTK_LIST_STORE (button->priv->model); pos = model_get_type_position (button, ROW_TYPE_OTHER_SEPARATOR); gtk_list_store_insert (store, &iter, pos); gtk_list_store_set (store, &iter, ICON_COLUMN, NULL, DISPLAY_NAME_COLUMN, NULL, TYPE_COLUMN, ROW_TYPE_OTHER_SEPARATOR, DATA_COLUMN, NULL, IS_FOLDER_COLUMN, FALSE, -1); button->priv->has_other_separator = TRUE; pos++; gtk_list_store_insert (store, &iter, pos); gtk_list_store_set (store, &iter, ICON_COLUMN, NULL, DISPLAY_NAME_COLUMN, _("Other…"), TYPE_COLUMN, ROW_TYPE_OTHER, DATA_COLUMN, NULL, IS_FOLDER_COLUMN, FALSE, -1); } static void model_add_empty_selection (GtkFileChooserButton *button) { GtkListStore *store; GtkTreeIter iter; gint pos; store = GTK_LIST_STORE (button->priv->model); pos = model_get_type_position (button, ROW_TYPE_EMPTY_SELECTION); gtk_list_store_insert (store, &iter, pos); gtk_list_store_set (store, &iter, ICON_COLUMN, NULL, DISPLAY_NAME_COLUMN, _(FALLBACK_DISPLAY_NAME), TYPE_COLUMN, ROW_TYPE_EMPTY_SELECTION, DATA_COLUMN, NULL, IS_FOLDER_COLUMN, FALSE, -1); } static void model_remove_rows (GtkFileChooserButton *button, gint pos, gint n_rows) { GtkListStore *store; if (!n_rows) return; store = GTK_LIST_STORE (button->priv->model); do { GtkTreeIter iter; if (!gtk_tree_model_iter_nth_child (button->priv->model, &iter, NULL, pos)) g_assert_not_reached (); model_free_row_data (button, &iter); gtk_list_store_remove (store, &iter); n_rows--; } while (n_rows); } /* Filter Model */ static gboolean test_if_file_is_visible (GtkFileSystem *fs, GFile *file, gboolean local_only, gboolean is_folder) { if (!file) return FALSE; if (local_only && !_gtk_file_has_native_path (file)) return FALSE; if (!is_folder) return FALSE; return TRUE; } static gboolean filter_model_visible_func (GtkTreeModel *model, GtkTreeIter *iter, gpointer user_data) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (user_data); GtkFileChooserButtonPrivate *priv = button->priv; gchar type; gpointer data; gboolean local_only, retval, is_folder; type = ROW_TYPE_INVALID; data = NULL; local_only = gtk_file_chooser_get_local_only (GTK_FILE_CHOOSER (priv->chooser)); gtk_tree_model_get (model, iter, TYPE_COLUMN, &type, DATA_COLUMN, &data, IS_FOLDER_COLUMN, &is_folder, -1); switch (type) { case ROW_TYPE_CURRENT_FOLDER: retval = TRUE; break; case ROW_TYPE_SPECIAL: case ROW_TYPE_SHORTCUT: case ROW_TYPE_BOOKMARK: retval = test_if_file_is_visible (priv->fs, data, local_only, is_folder); break; case ROW_TYPE_VOLUME: { retval = TRUE; if (local_only) { if (_gtk_file_system_volume_is_mounted (data)) { GFile *base_file; base_file = _gtk_file_system_volume_get_root (data); if (base_file) { if (!_gtk_file_has_native_path (base_file)) retval = FALSE; g_object_unref (base_file); } else retval = FALSE; } } } break; case ROW_TYPE_EMPTY_SELECTION: { gboolean popup_shown; g_object_get (priv->combo_box, "popup-shown", &popup_shown, NULL); if (popup_shown) retval = FALSE; else { GFile *selected; /* When the combo box is not popped up... */ selected = get_selected_file (button); if (selected) retval = FALSE; /* ... nonempty selection means the ROW_TYPE_EMPTY_SELECTION is *not* visible... */ else retval = TRUE; /* ... and empty selection means the ROW_TYPE_EMPTY_SELECTION *is* visible */ if (selected) g_object_unref (selected); } break; } default: retval = TRUE; break; } return retval; } /* Combo Box */ static void name_cell_data_func (GtkCellLayout *layout, GtkCellRenderer *cell, GtkTreeModel *model, GtkTreeIter *iter, gpointer user_data) { gchar type; type = 0; gtk_tree_model_get (model, iter, TYPE_COLUMN, &type, -1); if (type == ROW_TYPE_CURRENT_FOLDER) g_object_set (cell, "ellipsize", PANGO_ELLIPSIZE_END, NULL); else if (type == ROW_TYPE_BOOKMARK || type == ROW_TYPE_SHORTCUT) g_object_set (cell, "ellipsize", PANGO_ELLIPSIZE_MIDDLE, NULL); else g_object_set (cell, "ellipsize", PANGO_ELLIPSIZE_NONE, NULL); } static gboolean combo_box_row_separator_func (GtkTreeModel *model, GtkTreeIter *iter, gpointer user_data) { gchar type = ROW_TYPE_INVALID; gtk_tree_model_get (model, iter, TYPE_COLUMN, &type, -1); return (type == ROW_TYPE_BOOKMARK_SEPARATOR || type == ROW_TYPE_CURRENT_FOLDER_SEPARATOR || type == ROW_TYPE_OTHER_SEPARATOR); } static void select_combo_box_row_no_notify (GtkFileChooserButton *button, int pos) { GtkFileChooserButtonPrivate *priv = button->priv; GtkTreeIter iter, filter_iter; gtk_tree_model_iter_nth_child (priv->model, &iter, NULL, pos); gtk_tree_model_filter_convert_child_iter_to_iter (GTK_TREE_MODEL_FILTER (priv->filter_model), &filter_iter, &iter); g_signal_handlers_block_by_func (priv->combo_box, combo_box_changed_cb, button); gtk_combo_box_set_active_iter (GTK_COMBO_BOX (priv->combo_box), &filter_iter); g_signal_handlers_unblock_by_func (priv->combo_box, combo_box_changed_cb, button); } static void update_combo_box (GtkFileChooserButton *button) { GtkFileChooserButtonPrivate *priv = button->priv; GFile *file; GtkTreeIter iter; gboolean row_found; file = get_selected_file (button); row_found = FALSE; gtk_tree_model_get_iter_first (priv->filter_model, &iter); do { gchar type; gpointer data; type = ROW_TYPE_INVALID; data = NULL; gtk_tree_model_get (priv->filter_model, &iter, TYPE_COLUMN, &type, DATA_COLUMN, &data, -1); switch (type) { case ROW_TYPE_SPECIAL: case ROW_TYPE_SHORTCUT: case ROW_TYPE_BOOKMARK: case ROW_TYPE_CURRENT_FOLDER: row_found = (file && g_file_equal (data, file)); break; case ROW_TYPE_VOLUME: { GFile *base_file; base_file = _gtk_file_system_volume_get_root (data); if (base_file) { row_found = (file && g_file_equal (base_file, file)); g_object_unref (base_file); } } break; default: row_found = FALSE; break; } if (row_found) { g_signal_handlers_block_by_func (priv->combo_box, combo_box_changed_cb, button); gtk_combo_box_set_active_iter (GTK_COMBO_BOX (priv->combo_box), &iter); g_signal_handlers_unblock_by_func (priv->combo_box, combo_box_changed_cb, button); } } while (!row_found && gtk_tree_model_iter_next (priv->filter_model, &iter)); if (!row_found) { gint pos; /* If it hasn't been found already, update & select the current-folder row. */ if (file) { model_update_current_folder (button, file); pos = model_get_type_position (button, ROW_TYPE_CURRENT_FOLDER); } else { /* No selection; switch to that row */ pos = model_get_type_position (button, ROW_TYPE_EMPTY_SELECTION); } gtk_tree_model_filter_refilter (GTK_TREE_MODEL_FILTER (priv->filter_model)); select_combo_box_row_no_notify (button, pos); } if (file) g_object_unref (file); } /* Button */ static void update_label_get_info_cb (GCancellable *cancellable, GFileInfo *info, const GError *error, gpointer data) { gboolean cancelled = g_cancellable_is_cancelled (cancellable); cairo_surface_t *surface; GtkFileChooserButton *button = data; GtkFileChooserButtonPrivate *priv = button->priv; if (cancellable != priv->update_button_cancellable) goto out; priv->update_button_cancellable = NULL; if (cancelled || error) goto out; gtk_label_set_text (GTK_LABEL (priv->label), g_file_info_get_display_name (info)); surface = _gtk_file_info_render_icon (info, GTK_WIDGET (priv->image), priv->icon_size); gtk_image_set_from_surface (GTK_IMAGE (priv->image), surface); if (surface) cairo_surface_destroy (surface); out: emit_selection_changed_if_changing_selection (button); g_object_unref (button); g_object_unref (cancellable); } static void update_label_and_image (GtkFileChooserButton *button) { GtkFileChooserButtonPrivate *priv = button->priv; gchar *label_text; GFile *file; gboolean done_changing_selection; file = get_selected_file (button); label_text = NULL; done_changing_selection = FALSE; if (priv->update_button_cancellable) { g_cancellable_cancel (priv->update_button_cancellable); priv->update_button_cancellable = NULL; } if (file) { GtkFileSystemVolume *volume = NULL; volume = _gtk_file_system_get_volume_for_file (priv->fs, file); if (volume) { GFile *base_file; base_file = _gtk_file_system_volume_get_root (volume); if (base_file && g_file_equal (base_file, file)) { cairo_surface_t *surface; label_text = _gtk_file_system_volume_get_display_name (volume); surface = _gtk_file_system_volume_render_icon (volume, GTK_WIDGET (button), priv->icon_size, NULL); gtk_image_set_from_surface (GTK_IMAGE (priv->image), surface); if (surface) cairo_surface_destroy (surface); } if (base_file) g_object_unref (base_file); _gtk_file_system_volume_unref (volume); if (label_text) { done_changing_selection = TRUE; goto out; } } if (g_file_is_native (file)) { priv->update_button_cancellable = _gtk_file_system_get_info (priv->fs, file, "standard::icon,standard::display-name", update_label_get_info_cb, g_object_ref (button)); } else { cairo_surface_t *surface; label_text = _gtk_bookmarks_manager_get_bookmark_label (button->priv->bookmarks_manager, file); surface = gtk_icon_theme_load_surface (get_icon_theme (GTK_WIDGET (priv->image)), "text-x-generic", priv->icon_size, gtk_widget_get_scale_factor (GTK_WIDGET (button)), gtk_widget_get_window (GTK_WIDGET (button)), 0, NULL); gtk_image_set_from_surface (GTK_IMAGE (priv->image), surface); if (surface) cairo_surface_destroy (surface); done_changing_selection = TRUE; } } else { /* We know the selection is empty */ done_changing_selection = TRUE; } out: if (file) g_object_unref (file); if (label_text) { gtk_label_set_text (GTK_LABEL (priv->label), label_text); g_free (label_text); } else { gtk_label_set_text (GTK_LABEL (priv->label), _(FALLBACK_DISPLAY_NAME)); gtk_image_set_from_surface (GTK_IMAGE (priv->image), NULL); } if (done_changing_selection) emit_selection_changed_if_changing_selection (button); } /* ************************ * * Child Object Callbacks * * ************************ */ /* File System */ static void fs_volumes_changed_cb (GtkFileSystem *fs, gpointer user_data) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (user_data); GtkFileChooserButtonPrivate *priv = button->priv; GSList *volumes; model_remove_rows (user_data, model_get_type_position (user_data, ROW_TYPE_VOLUME), priv->n_volumes); priv->n_volumes = 0; volumes = _gtk_file_system_list_volumes (fs); model_add_volumes (user_data, volumes); g_slist_free (volumes); gtk_tree_model_filter_refilter (GTK_TREE_MODEL_FILTER (priv->filter_model)); update_label_and_image (user_data); update_combo_box (user_data); } static void bookmarks_changed_cb (gpointer user_data) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (user_data); GtkFileChooserButtonPrivate *priv = button->priv; GSList *bookmarks; bookmarks = _gtk_bookmarks_manager_list_bookmarks (priv->bookmarks_manager); model_remove_rows (user_data, model_get_type_position (user_data, ROW_TYPE_BOOKMARK_SEPARATOR), priv->n_bookmarks + priv->has_bookmark_separator); priv->has_bookmark_separator = FALSE; priv->n_bookmarks = 0; model_add_bookmarks (user_data, bookmarks); g_slist_free_full (bookmarks, g_object_unref); gtk_tree_model_filter_refilter (GTK_TREE_MODEL_FILTER (priv->filter_model)); update_label_and_image (user_data); update_combo_box (user_data); } static void save_inactive_state (GtkFileChooserButton *button) { GtkFileChooserButtonPrivate *priv = button->priv; if (priv->current_folder_while_inactive) g_object_unref (priv->current_folder_while_inactive); if (priv->selection_while_inactive) g_object_unref (priv->selection_while_inactive); priv->current_folder_while_inactive = gtk_file_chooser_get_current_folder_file (GTK_FILE_CHOOSER (priv->chooser)); priv->selection_while_inactive = gtk_file_chooser_get_file (GTK_FILE_CHOOSER (priv->chooser)); } static void restore_inactive_state (GtkFileChooserButton *button) { GtkFileChooserButtonPrivate *priv = button->priv; if (priv->current_folder_while_inactive) gtk_file_chooser_set_current_folder_file (GTK_FILE_CHOOSER (priv->chooser), priv->current_folder_while_inactive, NULL); if (priv->selection_while_inactive) gtk_file_chooser_select_file (GTK_FILE_CHOOSER (priv->chooser), priv->selection_while_inactive, NULL); else gtk_file_chooser_unselect_all (GTK_FILE_CHOOSER (priv->chooser)); } /* Dialog */ static void open_dialog (GtkFileChooserButton *button) { GtkFileChooserButtonPrivate *priv = button->priv; GtkWidget *toplevel; toplevel = gtk_widget_get_toplevel (GTK_WIDGET (button)); /* Setup the dialog parent to be chooser button's toplevel, and be modal as needed. */ if (priv->dialog != NULL) { if (!gtk_widget_get_visible (priv->dialog)) { if (gtk_widget_is_toplevel (toplevel) && GTK_IS_WINDOW (toplevel)) { if (GTK_WINDOW (toplevel) != gtk_window_get_transient_for (GTK_WINDOW (priv->dialog))) gtk_window_set_transient_for (GTK_WINDOW (priv->dialog), GTK_WINDOW (toplevel)); gtk_window_set_modal (GTK_WINDOW (priv->dialog), gtk_window_get_modal (GTK_WINDOW (toplevel))); } } } else { if (!gtk_native_dialog_get_visible (GTK_NATIVE_DIALOG (priv->native))) { if (gtk_widget_is_toplevel (toplevel) && GTK_IS_WINDOW (toplevel)) { if (GTK_WINDOW (toplevel) != gtk_native_dialog_get_transient_for (GTK_NATIVE_DIALOG (priv->native))) gtk_native_dialog_set_transient_for (GTK_NATIVE_DIALOG (priv->native), GTK_WINDOW (toplevel)); gtk_native_dialog_set_modal (GTK_NATIVE_DIALOG (priv->native), gtk_window_get_modal (GTK_WINDOW (toplevel))); } } } if (!priv->active) { restore_inactive_state (button); priv->active = TRUE; /* Only handle update-preview handler if it is handled on the button */ if (g_signal_has_handler_pending (button, g_signal_lookup ("update-preview", GTK_TYPE_FILE_CHOOSER), 0, TRUE)) { g_signal_connect (priv->chooser, "update-preview", G_CALLBACK (chooser_update_preview_cb), button); } } gtk_widget_set_sensitive (priv->combo_box, FALSE); if (priv->dialog) gtk_window_present (GTK_WINDOW (priv->dialog)); else gtk_native_dialog_show (GTK_NATIVE_DIALOG (priv->native)); } /* Combo Box */ static void combo_box_changed_cb (GtkComboBox *combo_box, gpointer user_data) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (user_data); GtkFileChooserButtonPrivate *priv = button->priv; GtkTreeIter iter; gboolean file_was_set; file_was_set = FALSE; if (gtk_combo_box_get_active_iter (combo_box, &iter)) { gchar type; gpointer data; type = ROW_TYPE_INVALID; data = NULL; gtk_tree_model_get (priv->filter_model, &iter, TYPE_COLUMN, &type, DATA_COLUMN, &data, -1); switch (type) { case ROW_TYPE_SPECIAL: case ROW_TYPE_SHORTCUT: case ROW_TYPE_BOOKMARK: case ROW_TYPE_CURRENT_FOLDER: if (data) { gtk_file_chooser_button_select_file (GTK_FILE_CHOOSER (button), data, NULL); file_was_set = TRUE; } break; case ROW_TYPE_VOLUME: { GFile *base_file; base_file = _gtk_file_system_volume_get_root (data); if (base_file) { gtk_file_chooser_button_select_file (GTK_FILE_CHOOSER (button), base_file, NULL); file_was_set = TRUE; g_object_unref (base_file); } } break; case ROW_TYPE_OTHER: open_dialog (user_data); break; default: break; } } if (file_was_set) g_signal_emit (button, file_chooser_button_signals[FILE_SET], 0); } /* Calback for the "notify::popup-shown" signal on the combo box. * When the combo is popped up, we don’t want the ROW_TYPE_EMPTY_SELECTION to be visible * at all; otherwise we would be showing a “(None)” item in the combo box’s popup. * * However, when the combo box is *not* popped up, we want the empty-selection row * to be visible depending on the selection. * * Since all that is done through the filter_model_visible_func(), this means * that we need to refilter the model when the combo box pops up - hence the * present signal handler. */ static void combo_box_notify_popup_shown_cb (GObject *object, GParamSpec *pspec, gpointer user_data) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (user_data); GtkFileChooserButtonPrivate *priv = button->priv; gboolean popup_shown; g_object_get (priv->combo_box, "popup-shown", &popup_shown, NULL); /* Indicate that the ROW_TYPE_EMPTY_SELECTION will change visibility... */ gtk_tree_model_filter_refilter (GTK_TREE_MODEL_FILTER (priv->filter_model)); /* If the combo box popup got dismissed, go back to showing the ROW_TYPE_EMPTY_SELECTION if needed */ if (!popup_shown) { GFile *selected = get_selected_file (button); if (!selected) { int pos; pos = model_get_type_position (button, ROW_TYPE_EMPTY_SELECTION); select_combo_box_row_no_notify (button, pos); } else g_object_unref (selected); } } /* Button */ static void button_clicked_cb (GtkButton *real_button, gpointer user_data) { open_dialog (user_data); } /* Dialog */ static void chooser_update_preview_cb (GtkFileChooser *dialog, gpointer user_data) { g_signal_emit_by_name (user_data, "update-preview"); } static void chooser_notify_cb (GObject *dialog, GParamSpec *pspec, gpointer user_data) { gpointer iface; iface = g_type_interface_peek (g_type_class_peek (G_OBJECT_TYPE (dialog)), GTK_TYPE_FILE_CHOOSER); if (g_object_interface_find_property (iface, pspec->name)) g_object_notify (user_data, pspec->name); if (g_ascii_strcasecmp (pspec->name, "local-only") == 0) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (user_data); GtkFileChooserButtonPrivate *priv = button->priv; if (priv->has_current_folder) { GtkTreeIter iter; gint pos; gpointer data; pos = model_get_type_position (user_data, ROW_TYPE_CURRENT_FOLDER); gtk_tree_model_iter_nth_child (priv->model, &iter, NULL, pos); data = NULL; gtk_tree_model_get (priv->model, &iter, DATA_COLUMN, &data, -1); /* If the path isn't local but we're in local-only mode now, remove * the custom-folder row */ if (data && _gtk_file_has_native_path (G_FILE (data)) && gtk_file_chooser_get_local_only (GTK_FILE_CHOOSER (priv->chooser))) { pos--; model_remove_rows (user_data, pos, 2); } } gtk_tree_model_filter_refilter (GTK_TREE_MODEL_FILTER (priv->filter_model)); update_combo_box (user_data); } } static gboolean dialog_delete_event_cb (GtkWidget *dialog, GdkEvent *event, gpointer user_data) { g_signal_emit_by_name (dialog, "response", GTK_RESPONSE_DELETE_EVENT); return TRUE; } static void common_response_cb (GtkFileChooserButton *button, gint response) { GtkFileChooserButtonPrivate *priv = button->priv; if (response == GTK_RESPONSE_ACCEPT || response == GTK_RESPONSE_OK) { save_inactive_state (button); g_signal_emit_by_name (button, "current-folder-changed"); g_signal_emit_by_name (button, "selection-changed"); } else { restore_inactive_state (button); } if (priv->active) { priv->active = FALSE; g_signal_handlers_disconnect_by_func (priv->chooser, chooser_update_preview_cb, button); } update_label_and_image (button); update_combo_box (button); gtk_widget_set_sensitive (priv->combo_box, TRUE); } static void dialog_response_cb (GtkDialog *dialog, gint response, gpointer user_data) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (user_data); GtkFileChooserButtonPrivate *priv = button->priv; common_response_cb (button, response); gtk_widget_hide (priv->dialog); if (response == GTK_RESPONSE_ACCEPT || response == GTK_RESPONSE_OK) g_signal_emit (button, file_chooser_button_signals[FILE_SET], 0); } static void native_response_cb (GtkFileChooserNative *native, gint response, gpointer user_data) { GtkFileChooserButton *button = GTK_FILE_CHOOSER_BUTTON (user_data); common_response_cb (button, response); /* dialog already hidden */ if (response == GTK_RESPONSE_ACCEPT || response == GTK_RESPONSE_OK) g_signal_emit (button, file_chooser_button_signals[FILE_SET], 0); } /* ************************************************************************** * * Public API * * ************************************************************************** */ /** * gtk_file_chooser_button_new: * @title: the title of the browse dialog. * @action: the open mode for the widget. * * Creates a new file-selecting button widget. * * Returns: a new button widget. * * Since: 2.6 */ GtkWidget * gtk_file_chooser_button_new (const gchar *title, GtkFileChooserAction action) { g_return_val_if_fail (action == GTK_FILE_CHOOSER_ACTION_OPEN || action == GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER, NULL); return g_object_new (GTK_TYPE_FILE_CHOOSER_BUTTON, "action", action, "title", (title ? title : _(DEFAULT_TITLE)), NULL); } /** * gtk_file_chooser_button_new_with_dialog: * @dialog: (type Gtk.Dialog): the widget to use as dialog * * Creates a #GtkFileChooserButton widget which uses @dialog as its * file-picking window. * * Note that @dialog must be a #GtkDialog (or subclass) which * implements the #GtkFileChooser interface and must not have * %GTK_DIALOG_DESTROY_WITH_PARENT set. * * Also note that the dialog needs to have its confirmative button * added with response %GTK_RESPONSE_ACCEPT or %GTK_RESPONSE_OK in * order for the button to take over the file selected in the dialog. * * Returns: a new button widget. * * Since: 2.6 */ GtkWidget * gtk_file_chooser_button_new_with_dialog (GtkWidget *dialog) { g_return_val_if_fail (GTK_IS_FILE_CHOOSER (dialog) && GTK_IS_DIALOG (dialog), NULL); return g_object_new (GTK_TYPE_FILE_CHOOSER_BUTTON, "dialog", dialog, NULL); } /** * gtk_file_chooser_button_set_title: * @button: the button widget to modify. * @title: the new browse dialog title. * * Modifies the @title of the browse dialog used by @button. * * Since: 2.6 */ void gtk_file_chooser_button_set_title (GtkFileChooserButton *button, const gchar *title) { g_return_if_fail (GTK_IS_FILE_CHOOSER_BUTTON (button)); if (button->priv->dialog) gtk_window_set_title (GTK_WINDOW (button->priv->dialog), title); else gtk_native_dialog_set_title (GTK_NATIVE_DIALOG (button->priv->native), title); g_object_notify (G_OBJECT (button), "title"); } /** * gtk_file_chooser_button_get_title: * @button: the button widget to examine. * * Retrieves the title of the browse dialog used by @button. The returned value * should not be modified or freed. * * Returns: a pointer to the browse dialog’s title. * * Since: 2.6 */ const gchar * gtk_file_chooser_button_get_title (GtkFileChooserButton *button) { g_return_val_if_fail (GTK_IS_FILE_CHOOSER_BUTTON (button), NULL); if (button->priv->dialog) return gtk_window_get_title (GTK_WINDOW (button->priv->dialog)); else return gtk_native_dialog_get_title (GTK_NATIVE_DIALOG (button->priv->native)); } /** * gtk_file_chooser_button_get_width_chars: * @button: the button widget to examine. * * Retrieves the width in characters of the @button widget’s entry and/or label. * * Returns: an integer width (in characters) that the button will use to size itself. * * Since: 2.6 */ gint gtk_file_chooser_button_get_width_chars (GtkFileChooserButton *button) { g_return_val_if_fail (GTK_IS_FILE_CHOOSER_BUTTON (button), -1); return gtk_label_get_width_chars (GTK_LABEL (button->priv->label)); } /** * gtk_file_chooser_button_set_width_chars: * @button: the button widget to examine. * @n_chars: the new width, in characters. * * Sets the width (in characters) that @button will use to @n_chars. * * Since: 2.6 */ void gtk_file_chooser_button_set_width_chars (GtkFileChooserButton *button, gint n_chars) { g_return_if_fail (GTK_IS_FILE_CHOOSER_BUTTON (button)); gtk_label_set_width_chars (GTK_LABEL (button->priv->label), n_chars); g_object_notify (G_OBJECT (button), "width-chars"); }