diff options
Diffstat (limited to '')
-rw-r--r-- | app/gui/gui-vtable.c | 934 |
1 files changed, 934 insertions, 0 deletions
diff --git a/app/gui/gui-vtable.c b/app/gui/gui-vtable.c new file mode 100644 index 0000000..28c999a --- /dev/null +++ b/app/gui/gui-vtable.c @@ -0,0 +1,934 @@ +/* GIMP - The GNU Image Manipulation Program + * Copyright (C) 1995 Spencer Kimball and Peter Mattis + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 3 of the License, or + * (at your option) any later version. + * + * This program 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 General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see <https://www.gnu.org/licenses/>. + */ + +#include "config.h" + +#include <string.h> +#include <errno.h> + +#include <gegl.h> +#include <gtk/gtk.h> + +#ifdef GDK_WINDOWING_X11 +#include <gdk/gdkx.h> +#endif + +#ifdef G_OS_WIN32 +#include <windows.h> +#include <fcntl.h> +#include <io.h> + +#ifndef pipe +#define pipe(fds) _pipe(fds, 4096, _O_BINARY) +#endif +#else +#include <sys/types.h> +#include <sys/wait.h> +#include <signal.h> +#include <unistd.h> +#endif + +#include "libgimpbase/gimpbase.h" +#include "libgimpwidgets/gimpwidgets.h" + +#include "gui-types.h" + +#include "config/gimpguiconfig.h" + +#include "core/gimp.h" +#include "core/gimp-parallel.h" +#include "core/gimp-spawn.h" +#include "core/gimp-utils.h" +#include "core/gimpasync.h" +#include "core/gimpbrush.h" +#include "core/gimpcancelable.h" +#include "core/gimpcontainer.h" +#include "core/gimpcontext.h" +#include "core/gimpgradient.h" +#include "core/gimpimage.h" +#include "core/gimpimagefile.h" +#include "core/gimplist.h" +#include "core/gimppalette.h" +#include "core/gimppattern.h" +#include "core/gimpprogress.h" +#include "core/gimpwaitable.h" + +#include "text/gimpfont.h" + +#include "pdb/gimppdb.h" +#include "pdb/gimpprocedure.h" + +#include "plug-in/gimppluginmanager-file.h" + +#include "widgets/gimpactiongroup.h" +#include "widgets/gimpbrushselect.h" +#include "widgets/gimpdialogfactory.h" +#include "widgets/gimpdocked.h" +#include "widgets/gimpfontselect.h" +#include "widgets/gimpgradientselect.h" +#include "widgets/gimphelp.h" +#include "widgets/gimphelp-ids.h" +#include "widgets/gimpmenufactory.h" +#include "widgets/gimppaletteselect.h" +#include "widgets/gimppatternselect.h" +#include "widgets/gimpprogressdialog.h" +#include "widgets/gimpuimanager.h" +#include "widgets/gimpwidgets-utils.h" + +#include "display/gimpdisplay.h" +#include "display/gimpdisplay-foreach.h" +#include "display/gimpdisplayshell.h" +#include "display/gimpsinglewindowstrategy.h" +#include "display/gimpmultiwindowstrategy.h" + +#include "actions/plug-in-actions.h" + +#include "menus/menus.h" + +#include "dialogs/color-profile-import-dialog.h" + +#include "gui.h" +#include "gui-message.h" +#include "gui-vtable.h" +#include "icon-themes.h" +#include "themes.h" + + +/* local function prototypes */ + +static void gui_ungrab (Gimp *gimp); + +static void gui_threads_enter (Gimp *gimp); +static void gui_threads_leave (Gimp *gimp); + +static void gui_set_busy (Gimp *gimp); +static void gui_unset_busy (Gimp *gimp); + +static void gui_help (Gimp *gimp, + GimpProgress *progress, + const gchar *help_domain, + const gchar *help_id); +static const gchar * gui_get_program_class (Gimp *gimp); +static gchar * gui_get_display_name (Gimp *gimp, + gint display_ID, + GObject **screen, + gint *monitor); +static guint32 gui_get_user_time (Gimp *gimp); +static GFile * gui_get_theme_dir (Gimp *gimp); +static GFile * gui_get_icon_theme_dir (Gimp *gimp); +static GimpObject * gui_get_window_strategy (Gimp *gimp); +static GimpObject * gui_get_empty_display (Gimp *gimp); +static GimpObject * gui_display_get_by_ID (Gimp *gimp, + gint ID); +static gint gui_display_get_ID (GimpObject *display); +static guint32 gui_display_get_window_id (GimpObject *display); +static GimpObject * gui_display_create (Gimp *gimp, + GimpImage *image, + GimpUnit unit, + gdouble scale, + GObject *screen, + gint monitor); +static void gui_display_delete (GimpObject *display); +static void gui_displays_reconnect (Gimp *gimp, + GimpImage *old_image, + GimpImage *new_image); +static gboolean gui_wait (Gimp *gimp, + GimpWaitable *waitable, + const gchar *message); +static GimpProgress * gui_new_progress (Gimp *gimp, + GimpObject *display); +static void gui_free_progress (Gimp *gimp, + GimpProgress *progress); +static gboolean gui_pdb_dialog_new (Gimp *gimp, + GimpContext *context, + GimpProgress *progress, + GimpContainer *container, + const gchar *title, + const gchar *callback_name, + const gchar *object_name, + va_list args); +static gboolean gui_pdb_dialog_set (Gimp *gimp, + GimpContainer *container, + const gchar *callback_name, + const gchar *object_name, + va_list args); +static gboolean gui_pdb_dialog_close (Gimp *gimp, + GimpContainer *container, + const gchar *callback_name); +static gboolean gui_recent_list_add_file (Gimp *gimp, + GFile *file, + const gchar *mime_type); +static void gui_recent_list_load (Gimp *gimp); + +static GMountOperation + * gui_get_mount_operation (Gimp *gimp, + GimpProgress *progress); + +static GimpColorProfilePolicy + gui_query_profile_policy (Gimp *gimp, + GimpImage *image, + GimpContext *context, + GimpColorProfile **dest_profile, + GimpColorRenderingIntent *intent, + gboolean *bpc, + gboolean *dont_ask); + + +/* public functions */ + +void +gui_vtable_init (Gimp *gimp) +{ + g_return_if_fail (GIMP_IS_GIMP (gimp)); + + gimp->gui.ungrab = gui_ungrab; + gimp->gui.threads_enter = gui_threads_enter; + gimp->gui.threads_leave = gui_threads_leave; + gimp->gui.set_busy = gui_set_busy; + gimp->gui.unset_busy = gui_unset_busy; + gimp->gui.show_message = gui_message; + gimp->gui.help = gui_help; + gimp->gui.get_program_class = gui_get_program_class; + gimp->gui.get_display_name = gui_get_display_name; + gimp->gui.get_user_time = gui_get_user_time; + gimp->gui.get_theme_dir = gui_get_theme_dir; + gimp->gui.get_icon_theme_dir = gui_get_icon_theme_dir; + gimp->gui.get_window_strategy = gui_get_window_strategy; + gimp->gui.get_empty_display = gui_get_empty_display; + gimp->gui.display_get_by_id = gui_display_get_by_ID; + gimp->gui.display_get_id = gui_display_get_ID; + gimp->gui.display_get_window_id = gui_display_get_window_id; + gimp->gui.display_create = gui_display_create; + gimp->gui.display_delete = gui_display_delete; + gimp->gui.displays_reconnect = gui_displays_reconnect; + gimp->gui.wait = gui_wait; + gimp->gui.progress_new = gui_new_progress; + gimp->gui.progress_free = gui_free_progress; + gimp->gui.pdb_dialog_new = gui_pdb_dialog_new; + gimp->gui.pdb_dialog_set = gui_pdb_dialog_set; + gimp->gui.pdb_dialog_close = gui_pdb_dialog_close; + gimp->gui.recent_list_add_file = gui_recent_list_add_file; + gimp->gui.recent_list_load = gui_recent_list_load; + gimp->gui.get_mount_operation = gui_get_mount_operation; + gimp->gui.query_profile_policy = gui_query_profile_policy; +} + + +/* private functions */ + +static void +gui_ungrab (Gimp *gimp) +{ + GdkDisplay *display = gdk_display_get_default (); + + if (display) + { + gdk_display_pointer_ungrab (display, GDK_CURRENT_TIME); + gdk_display_keyboard_ungrab (display, GDK_CURRENT_TIME); + } +} + +static void +gui_threads_enter (Gimp *gimp) +{ + GDK_THREADS_ENTER (); +} + +static void +gui_threads_leave (Gimp *gimp) +{ + GDK_THREADS_LEAVE (); +} + +static void +gui_set_busy (Gimp *gimp) +{ + gimp_displays_set_busy (gimp); + gimp_dialog_factory_set_busy (gimp_dialog_factory_get_singleton ()); + + gdk_flush (); +} + +static void +gui_unset_busy (Gimp *gimp) +{ + gimp_displays_unset_busy (gimp); + gimp_dialog_factory_unset_busy (gimp_dialog_factory_get_singleton ()); + + gdk_flush (); +} + +static void +gui_help (Gimp *gimp, + GimpProgress *progress, + const gchar *help_domain, + const gchar *help_id) +{ + gimp_help_show (gimp, progress, help_domain, help_id); +} + +static const gchar * +gui_get_program_class (Gimp *gimp) +{ + return gdk_get_program_class (); +} + +static gchar * +gui_get_display_name (Gimp *gimp, + gint display_ID, + GObject **screen, + gint *monitor) +{ + GimpDisplay *display = NULL; + GdkScreen *my_screen = NULL; + + if (display_ID > 0) + display = gimp_display_get_by_ID (gimp, display_ID); + + if (display) + { + GimpDisplayShell *shell = gimp_display_get_shell (display); + GdkWindow *window = gtk_widget_get_window (GTK_WIDGET (shell)); + + my_screen = gtk_widget_get_screen (GTK_WIDGET (shell)); + *monitor = gdk_screen_get_monitor_at_window (my_screen, window); + } + else + { + *monitor = gui_get_initial_monitor (gimp, &my_screen); + + if (*monitor == -1) + *monitor = gimp_get_monitor_at_pointer (&my_screen); + } + + *screen = G_OBJECT (my_screen); + + if (my_screen) + return gdk_screen_make_display_name (my_screen); + + return NULL; +} + +static guint32 +gui_get_user_time (Gimp *gimp) +{ +#ifdef GDK_WINDOWING_X11 + return gdk_x11_display_get_user_time (gdk_display_get_default ()); +#endif + return 0; +} + +static GFile * +gui_get_theme_dir (Gimp *gimp) +{ + return themes_get_theme_dir (gimp, GIMP_GUI_CONFIG (gimp->config)->theme); +} + +static GFile * +gui_get_icon_theme_dir (Gimp *gimp) +{ + return icon_themes_get_theme_dir (gimp, GIMP_GUI_CONFIG (gimp->config)->icon_theme); +} + +static GimpObject * +gui_get_window_strategy (Gimp *gimp) +{ + if (GIMP_GUI_CONFIG (gimp->config)->single_window_mode) + return gimp_single_window_strategy_get_singleton (); + else + return gimp_multi_window_strategy_get_singleton (); +} + +static GimpObject * +gui_get_empty_display (Gimp *gimp) +{ + GimpObject *display = NULL; + + if (gimp_container_get_n_children (gimp->displays) == 1) + { + display = gimp_container_get_first_child (gimp->displays); + + if (gimp_display_get_image (GIMP_DISPLAY (display))) + { + /* The display was not empty */ + display = NULL; + } + } + + return display; +} + +static GimpObject * +gui_display_get_by_ID (Gimp *gimp, + gint ID) +{ + return (GimpObject *) gimp_display_get_by_ID (gimp, ID); +} + +static gint +gui_display_get_ID (GimpObject *display) +{ + return gimp_display_get_ID (GIMP_DISPLAY (display)); +} + +static guint32 +gui_display_get_window_id (GimpObject *display) +{ + GimpDisplay *disp = GIMP_DISPLAY (display); + GimpDisplayShell *shell = gimp_display_get_shell (disp); + + if (shell) + { + GtkWidget *toplevel = gtk_widget_get_toplevel (GTK_WIDGET (shell)); + + if (GTK_IS_WINDOW (toplevel)) + return gimp_window_get_native_id (GTK_WINDOW (toplevel)); + } + + return 0; +} + +static GimpObject * +gui_display_create (Gimp *gimp, + GimpImage *image, + GimpUnit unit, + gdouble scale, + GObject *screen, + gint monitor) +{ + GimpContext *context = gimp_get_user_context (gimp); + GimpDisplay *display = GIMP_DISPLAY (gui_get_empty_display (gimp)); + + if (! screen) + monitor = gimp_get_monitor_at_pointer ((GdkScreen **) &screen); + + if (display) + { + gimp_display_fill (display, image, unit, scale); + } + else + { + GList *image_managers = gimp_ui_managers_from_name ("<Image>"); + + g_return_val_if_fail (image_managers != NULL, NULL); + + display = gimp_display_new (gimp, image, unit, scale, + image_managers->data, + gimp_dialog_factory_get_singleton (), + GDK_SCREEN (screen), + monitor); + } + + if (gimp_context_get_display (context) == display) + { + gimp_context_set_image (context, image); + gimp_context_display_changed (context); + } + else + { + gimp_context_set_display (context, display); + } + + return GIMP_OBJECT (display); +} + +static void +gui_display_delete (GimpObject *display) +{ + gimp_display_close (GIMP_DISPLAY (display)); +} + +static void +gui_displays_reconnect (Gimp *gimp, + GimpImage *old_image, + GimpImage *new_image) +{ + gimp_displays_reconnect (gimp, old_image, new_image); +} + +static void +gui_wait_input_async (GimpAsync *async, + const gint input_pipe[2]) +{ + guint8 buffer[1]; + + while (read (input_pipe[0], buffer, sizeof (buffer)) == -1 && + errno == EINTR); + + gimp_async_finish (async, NULL); +} + +static gboolean +gui_wait (Gimp *gimp, + GimpWaitable *waitable, + const gchar *message) +{ + GimpProcedure *procedure; + GimpValueArray *args; + GimpAsync *input_async = NULL; + GError *error = NULL; + gint input_pipe[2]; + gint output_pipe[2]; + + procedure = gimp_pdb_lookup_procedure (gimp->pdb, "plug-in-busy-dialog"); + + if (! procedure) + return FALSE; + + if (pipe (input_pipe)) + return FALSE; + + if (pipe (output_pipe)) + { + close (input_pipe[0]); + close (input_pipe[1]); + + return FALSE; + } + + gimp_spawn_set_cloexec (input_pipe[0]); + gimp_spawn_set_cloexec (output_pipe[1]); + + args = gimp_procedure_get_arguments (procedure); + gimp_value_array_truncate (args, 5); + + g_value_set_int (gimp_value_array_index (args, 0), + GIMP_RUN_INTERACTIVE); + g_value_set_int (gimp_value_array_index (args, 1), + output_pipe[0]); + g_value_set_int (gimp_value_array_index (args, 2), + input_pipe[1]); + g_value_set_string (gimp_value_array_index (args, 3), + message); + g_value_set_int (gimp_value_array_index (args, 4), + GIMP_IS_CANCELABLE (waitable)); + + gimp_procedure_execute_async (procedure, gimp, + gimp_get_user_context (gimp), + NULL, args, NULL, &error); + + gimp_value_array_unref (args); + + close (input_pipe[1]); + close (output_pipe[0]); + + if (error) + { + g_clear_error (&error); + + close (input_pipe[0]); + close (output_pipe[1]); + + return FALSE; + } + + if (GIMP_IS_CANCELABLE (waitable)) + { + /* listens for a cancellation request */ + input_async = gimp_parallel_run_async_independent ( + (GimpRunAsyncFunc) gui_wait_input_async, + input_pipe); + + while (! gimp_waitable_wait_for (waitable, 0.1 * G_TIME_SPAN_SECOND)) + { + /* check for a cancellation request */ + if (gimp_waitable_try_wait (GIMP_WAITABLE (input_async))) + { + gimp_cancelable_cancel (GIMP_CANCELABLE (waitable)); + + break; + } + } + } + + gimp_waitable_wait (waitable); + + /* signal completion to the plug-in */ + close (output_pipe[1]); + + if (input_async) + { + gimp_waitable_wait (GIMP_WAITABLE (input_async)); + + g_object_unref (input_async); + } + + close (input_pipe[0]); + + return TRUE; +} + +static GimpProgress * +gui_new_progress (Gimp *gimp, + GimpObject *display) +{ + g_return_val_if_fail (display == NULL || GIMP_IS_DISPLAY (display), NULL); + + if (display) + return GIMP_PROGRESS (display); + + return GIMP_PROGRESS (gimp_progress_dialog_new ()); +} + +static void +gui_free_progress (Gimp *gimp, + GimpProgress *progress) +{ + g_return_if_fail (GIMP_IS_PROGRESS_DIALOG (progress)); + + if (GIMP_IS_PROGRESS_DIALOG (progress)) + gtk_widget_destroy (GTK_WIDGET (progress)); +} + +static gboolean +gui_pdb_dialog_present (GtkWindow *window) +{ + gtk_window_present (window); + + return FALSE; +} + +static gboolean +gui_pdb_dialog_new (Gimp *gimp, + GimpContext *context, + GimpProgress *progress, + GimpContainer *container, + const gchar *title, + const gchar *callback_name, + const gchar *object_name, + va_list args) +{ + GType dialog_type = G_TYPE_NONE; + const gchar *dialog_role = NULL; + const gchar *help_id = NULL; + + if (gimp_container_get_children_type (container) == GIMP_TYPE_BRUSH) + { + dialog_type = GIMP_TYPE_BRUSH_SELECT; + dialog_role = "gimp-brush-selection"; + help_id = GIMP_HELP_BRUSH_DIALOG; + } + else if (gimp_container_get_children_type (container) == GIMP_TYPE_FONT) + { + dialog_type = GIMP_TYPE_FONT_SELECT; + dialog_role = "gimp-font-selection"; + help_id = GIMP_HELP_FONT_DIALOG; + } + else if (gimp_container_get_children_type (container) == GIMP_TYPE_GRADIENT) + { + dialog_type = GIMP_TYPE_GRADIENT_SELECT; + dialog_role = "gimp-gradient-selection"; + help_id = GIMP_HELP_GRADIENT_DIALOG; + } + else if (gimp_container_get_children_type (container) == GIMP_TYPE_PALETTE) + { + dialog_type = GIMP_TYPE_PALETTE_SELECT; + dialog_role = "gimp-palette-selection"; + help_id = GIMP_HELP_PALETTE_DIALOG; + } + else if (gimp_container_get_children_type (container) == GIMP_TYPE_PATTERN) + { + dialog_type = GIMP_TYPE_PATTERN_SELECT; + dialog_role = "gimp-pattern-selection"; + help_id = GIMP_HELP_PATTERN_DIALOG; + } + + if (dialog_type != G_TYPE_NONE) + { + GimpObject *object = NULL; + + if (object_name && strlen (object_name)) + object = gimp_container_get_child_by_name (container, object_name); + + if (! object) + object = gimp_context_get_by_type (context, + gimp_container_get_children_type (container)); + + if (object) + { + gint n_properties = 0; + gchar **names = NULL; + GValue *values = NULL; + GtkWidget *dialog; + GtkWidget *view; + + names = gimp_properties_append (dialog_type, + &n_properties, names, &values, + "title", title, + "role", dialog_role, + "help-func", gimp_standard_help_func, + "help-id", help_id, + "pdb", gimp->pdb, + "context", context, + "select-type", gimp_container_get_children_type (container), + "initial-object", object, + "callback-name", callback_name, + "menu-factory", global_menu_factory, + NULL); + + names = gimp_properties_append_valist (dialog_type, + &n_properties, names, &values, + args); + + dialog = (GtkWidget *) + g_object_new_with_properties (dialog_type, + n_properties, + (const gchar **) names, + (const GValue *) values); + + gimp_properties_free (n_properties, names, values); + + view = GIMP_PDB_DIALOG (dialog)->view; + if (view) + gimp_docked_set_show_button_bar (GIMP_DOCKED (view), FALSE); + + if (progress) + { + guint32 window_id = gimp_progress_get_window_id (progress); + + if (window_id) + gimp_window_set_transient_for (GTK_WINDOW (dialog), window_id); + } + + gtk_widget_show (dialog); + + /* workaround for bug #360106 */ + { + GSource *source = g_timeout_source_new (100); + GClosure *closure; + + closure = g_cclosure_new_object (G_CALLBACK (gui_pdb_dialog_present), + G_OBJECT (dialog)); + + g_source_set_closure (source, closure); + g_source_attach (source, NULL); + g_source_unref (source); + } + + return TRUE; + } + } + + return FALSE; +} + +static gboolean +gui_pdb_dialog_set (Gimp *gimp, + GimpContainer *container, + const gchar *callback_name, + const gchar *object_name, + va_list args) +{ + GimpPdbDialogClass *klass = NULL; + + if (gimp_container_get_children_type (container) == GIMP_TYPE_BRUSH) + klass = g_type_class_peek (GIMP_TYPE_BRUSH_SELECT); + else if (gimp_container_get_children_type (container) == GIMP_TYPE_FONT) + klass = g_type_class_peek (GIMP_TYPE_FONT_SELECT); + else if (gimp_container_get_children_type (container) == GIMP_TYPE_GRADIENT) + klass = g_type_class_peek (GIMP_TYPE_GRADIENT_SELECT); + else if (gimp_container_get_children_type (container) == GIMP_TYPE_PALETTE) + klass = g_type_class_peek (GIMP_TYPE_PALETTE_SELECT); + else if (gimp_container_get_children_type (container) == GIMP_TYPE_PATTERN) + klass = g_type_class_peek (GIMP_TYPE_PATTERN_SELECT); + + if (klass) + { + GimpPdbDialog *dialog; + + dialog = gimp_pdb_dialog_get_by_callback (klass, callback_name); + + if (dialog && dialog->select_type == gimp_container_get_children_type (container)) + { + GimpObject *object; + + object = gimp_container_get_child_by_name (container, object_name); + + if (object) + { + const gchar *prop_name = va_arg (args, const gchar *); + + gimp_context_set_by_type (dialog->context, dialog->select_type, + object); + + if (prop_name) + g_object_set_valist (G_OBJECT (dialog), prop_name, args); + + gtk_window_present (GTK_WINDOW (dialog)); + + return TRUE; + } + } + } + + return FALSE; +} + +static gboolean +gui_pdb_dialog_close (Gimp *gimp, + GimpContainer *container, + const gchar *callback_name) +{ + GimpPdbDialogClass *klass = NULL; + + if (gimp_container_get_children_type (container) == GIMP_TYPE_BRUSH) + klass = g_type_class_peek (GIMP_TYPE_BRUSH_SELECT); + else if (gimp_container_get_children_type (container) == GIMP_TYPE_FONT) + klass = g_type_class_peek (GIMP_TYPE_FONT_SELECT); + else if (gimp_container_get_children_type (container) == GIMP_TYPE_GRADIENT) + klass = g_type_class_peek (GIMP_TYPE_GRADIENT_SELECT); + else if (gimp_container_get_children_type (container) == GIMP_TYPE_PALETTE) + klass = g_type_class_peek (GIMP_TYPE_PALETTE_SELECT); + else if (gimp_container_get_children_type (container) == GIMP_TYPE_PATTERN) + klass = g_type_class_peek (GIMP_TYPE_PATTERN_SELECT); + + if (klass) + { + GimpPdbDialog *dialog; + + dialog = gimp_pdb_dialog_get_by_callback (klass, callback_name); + + if (dialog && dialog->select_type == gimp_container_get_children_type (container)) + { + gtk_widget_destroy (GTK_WIDGET (dialog)); + return TRUE; + } + } + + return FALSE; +} + +static gboolean +gui_recent_list_add_file (Gimp *gimp, + GFile *file, + const gchar *mime_type) +{ + GtkRecentData recent; + const gchar *groups[2] = { "Graphics", NULL }; + gchar *uri; + gboolean success; + + g_return_val_if_fail (GIMP_IS_GIMP (gimp), FALSE); + g_return_val_if_fail (G_IS_FILE (file), FALSE); + + /* use last part of the URI */ + recent.display_name = NULL; + + /* no special description */ + recent.description = NULL; + recent.mime_type = (mime_type ? + (gchar *) mime_type : "application/octet-stream"); + recent.app_name = "GNU Image Manipulation Program"; + recent.app_exec = GIMP_COMMAND " %u"; + recent.groups = (gchar **) groups; + recent.is_private = FALSE; + + uri = g_file_get_uri (file); + + success = gtk_recent_manager_add_full (gtk_recent_manager_get_default (), + uri, &recent); + + g_free (uri); + + return success; +} + +static gint +gui_recent_list_compare (gconstpointer a, + gconstpointer b) +{ + return (gtk_recent_info_get_modified ((GtkRecentInfo *) a) - + gtk_recent_info_get_modified ((GtkRecentInfo *) b)); +} + +static void +gui_recent_list_load (Gimp *gimp) +{ + GList *items; + GList *list; + + g_return_if_fail (GIMP_IS_GIMP (gimp)); + + gimp_container_freeze (gimp->documents); + gimp_container_clear (gimp->documents); + + items = gtk_recent_manager_get_items (gtk_recent_manager_get_default ()); + + items = g_list_sort (items, gui_recent_list_compare); + + for (list = items; list; list = list->next) + { + GtkRecentInfo *info = list->data; + + if (gtk_recent_info_has_application (info, + "GNU Image Manipulation Program")) + { + const gchar *mime_type = gtk_recent_info_get_mime_type (info); + + if (mime_type && + gimp_plug_in_manager_file_procedure_find_by_mime_type (gimp->plug_in_manager, + GIMP_FILE_PROCEDURE_GROUP_OPEN, + mime_type)) + { + GimpImagefile *imagefile; + GFile *file; + + file = g_file_new_for_uri (gtk_recent_info_get_uri (info)); + imagefile = gimp_imagefile_new (gimp, file); + g_object_unref (file); + + gimp_imagefile_set_mime_type (imagefile, mime_type); + + gimp_container_add (gimp->documents, GIMP_OBJECT (imagefile)); + g_object_unref (imagefile); + } + } + + gtk_recent_info_unref (info); + } + + g_list_free (items); + + gimp_container_thaw (gimp->documents); +} + +static GMountOperation * +gui_get_mount_operation (Gimp *gimp, + GimpProgress *progress) +{ + GtkWidget *toplevel = NULL; + + if (GTK_IS_WIDGET (progress)) + toplevel = gtk_widget_get_toplevel (GTK_WIDGET (progress)); + + return gtk_mount_operation_new (GTK_WINDOW (toplevel)); +} + +static GimpColorProfilePolicy +gui_query_profile_policy (Gimp *gimp, + GimpImage *image, + GimpContext *context, + GimpColorProfile **dest_profile, + GimpColorRenderingIntent *intent, + gboolean *bpc, + gboolean *dont_ask) +{ + return color_profile_import_dialog_run (image, context, NULL, + dest_profile, + intent, bpc, + dont_ask); +} |