summaryrefslogtreecommitdiffstats
path: root/app/gui/gui.c
diff options
context:
space:
mode:
Diffstat (limited to 'app/gui/gui.c')
-rw-r--r--app/gui/gui.c1077
1 files changed, 1077 insertions, 0 deletions
diff --git a/app/gui/gui.c b/app/gui/gui.c
new file mode 100644
index 0000000..e5928eb
--- /dev/null
+++ b/app/gui/gui.c
@@ -0,0 +1,1077 @@
+/* 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 <stdlib.h>
+
+#include <gegl.h>
+#include <gtk/gtk.h>
+
+#include "libgimpbase/gimpbase.h"
+#include "libgimpwidgets/gimpwidgets.h"
+#include "libgimpwidgets/gimpwidgets-private.h"
+
+#include "gui-types.h"
+
+#include "config/gimpguiconfig.h"
+
+#include "core/gimp.h"
+#include "core/gimpcontainer.h"
+#include "core/gimpcontext.h"
+#include "core/gimpimage.h"
+#include "core/gimptoolinfo.h"
+
+#include "plug-in/gimpenvirontable.h"
+#include "plug-in/gimppluginmanager.h"
+
+#include "display/gimpdisplay.h"
+#include "display/gimpdisplay-foreach.h"
+#include "display/gimpdisplayshell.h"
+#include "display/gimpstatusbar.h"
+
+#include "tools/gimp-tools.h"
+#include "tools/gimptool.h"
+#include "tools/tool_manager.h"
+
+#include "widgets/gimpaction.h"
+#include "widgets/gimpactiongroup.h"
+#include "widgets/gimpaction-history.h"
+#include "widgets/gimpclipboard.h"
+#include "widgets/gimpcolorselectorpalette.h"
+#include "widgets/gimpcontrollers.h"
+#include "widgets/gimpdevices.h"
+#include "widgets/gimpdialogfactory.h"
+#include "widgets/gimpdnd.h"
+#include "widgets/gimprender.h"
+#include "widgets/gimphelp.h"
+#include "widgets/gimphelp-ids.h"
+#include "widgets/gimpmenufactory.h"
+#include "widgets/gimpmessagebox.h"
+#include "widgets/gimpsessioninfo.h"
+#include "widgets/gimpuimanager.h"
+#include "widgets/gimpwidgets-utils.h"
+#include "widgets/gimplanguagestore-parser.h"
+
+#include "actions/actions.h"
+#include "actions/windows-commands.h"
+
+#include "menus/menus.h"
+
+#include "dialogs/dialogs.h"
+
+#include "gimpuiconfigurer.h"
+#include "gui.h"
+#include "gui-unique.h"
+#include "gui-vtable.h"
+#include "icon-themes.h"
+#include "session.h"
+#include "splash.h"
+#include "themes.h"
+
+#ifdef GDK_WINDOWING_QUARTZ
+#import <AppKit/AppKit.h>
+#include <gtkosxapplication.h>
+#include <gdk/gdkquartz.h>
+
+/* Forward declare since we are building against old SDKs. */
+#if !defined(MAC_OS_X_VERSION_10_12) || \
+ MAC_OS_X_VERSION_MIN_REQUIRED < MAC_OS_X_VERSION_10_12
+
+@interface NSWindow(ForwardDeclarations)
++ (void)setAllowsAutomaticWindowTabbing:(BOOL)allow;
+@end
+
+#endif
+
+#endif /* GDK_WINDOWING_QUARTZ */
+
+#include "gimp-intl.h"
+
+
+/* local function prototypes */
+
+static gchar * gui_sanity_check (void);
+static void gui_help_func (const gchar *help_id,
+ gpointer help_data);
+static gboolean gui_get_background_func (GimpRGB *color);
+static gboolean gui_get_foreground_func (GimpRGB *color);
+
+static void gui_initialize_after_callback (Gimp *gimp,
+ GimpInitStatusFunc callback);
+
+static void gui_restore_callback (Gimp *gimp,
+ GimpInitStatusFunc callback);
+static void gui_restore_after_callback (Gimp *gimp,
+ GimpInitStatusFunc callback);
+
+static gboolean gui_exit_callback (Gimp *gimp,
+ gboolean force);
+static gboolean gui_exit_after_callback (Gimp *gimp,
+ gboolean force);
+
+static void gui_show_tooltips_notify (GimpGuiConfig *gui_config,
+ GParamSpec *pspec,
+ Gimp *gimp);
+static void gui_show_help_button_notify (GimpGuiConfig *gui_config,
+ GParamSpec *pspec,
+ Gimp *gimp);
+static void gui_user_manual_notify (GimpGuiConfig *gui_config,
+ GParamSpec *pspec,
+ Gimp *gimp);
+static void gui_single_window_mode_notify (GimpGuiConfig *gui_config,
+ GParamSpec *pspec,
+ GimpUIConfigurer *ui_configurer);
+static void gui_tearoff_menus_notify (GimpGuiConfig *gui_config,
+ GParamSpec *pspec,
+ GtkUIManager *manager);
+
+static void gui_clipboard_changed (Gimp *gimp);
+
+static void gui_menu_show_tooltip (GimpUIManager *manager,
+ const gchar *tooltip,
+ Gimp *gimp);
+static void gui_menu_hide_tooltip (GimpUIManager *manager,
+ Gimp *gimp);
+
+static void gui_display_changed (GimpContext *context,
+ GimpDisplay *display,
+ Gimp *gimp);
+
+static void gui_compare_accelerator (gpointer data,
+ const gchar *accel_path,
+ guint accel_key,
+ GdkModifierType accel_mods,
+ gboolean changed);
+static void gui_check_unique_accelerator (gpointer data,
+ const gchar *accel_path,
+ guint accel_key,
+ GdkModifierType accel_mods,
+ gboolean changed);
+static gboolean gui_check_action_exists (const gchar *accel_path);
+
+
+/* private variables */
+
+static Gimp *the_gui_gimp = NULL;
+static GimpUIManager *image_ui_manager = NULL;
+static GimpUIConfigurer *ui_configurer = NULL;
+static GdkScreen *initial_screen = NULL;
+static gint initial_monitor = -1;
+
+
+/* public functions */
+
+void
+gui_libs_init (GOptionContext *context)
+{
+ g_return_if_fail (context != NULL);
+
+ g_option_context_add_group (context, gtk_get_option_group (TRUE));
+}
+
+void
+gui_abort (const gchar *abort_message)
+{
+ GtkWidget *dialog;
+ GtkWidget *box;
+
+ g_return_if_fail (abort_message != NULL);
+
+ dialog = gimp_dialog_new (_("GIMP Message"), "gimp-abort",
+ NULL, GTK_DIALOG_MODAL, NULL, NULL,
+
+ _("_OK"), GTK_RESPONSE_OK,
+
+ NULL);
+
+ gtk_window_set_resizable (GTK_WINDOW (dialog), FALSE);
+
+ box = g_object_new (GIMP_TYPE_MESSAGE_BOX,
+ "icon-name", GIMP_ICON_WILBER_EEK,
+ "border-width", 12,
+ NULL);
+
+ gimp_message_box_set_text (GIMP_MESSAGE_BOX (box), "%s", abort_message);
+
+ gtk_box_pack_start (GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (dialog))),
+ box, TRUE, TRUE, 0);
+ gtk_widget_show (box);
+
+ gimp_dialog_run (GIMP_DIALOG (dialog));
+
+ exit (EXIT_FAILURE);
+}
+
+GimpInitStatusFunc
+gui_init (Gimp *gimp,
+ gboolean no_splash)
+{
+ GimpInitStatusFunc status_callback = NULL;
+ gchar *abort_message;
+
+ g_return_val_if_fail (GIMP_IS_GIMP (gimp), NULL);
+ g_return_val_if_fail (the_gui_gimp == NULL, NULL);
+
+ abort_message = gui_sanity_check ();
+ if (abort_message)
+ gui_abort (abort_message);
+
+ the_gui_gimp = gimp;
+
+ /* TRANSLATORS: there is no need to translate this in GIMP. This uses
+ * "gtk20" domain as a special trick to determine language direction,
+ * but xgettext extracts it anyway mistakenly into GIMP po files.
+ * Leave an empty string as translation. It does not matter.
+ */
+ if (g_strcmp0 (dgettext ("gtk20", "default:LTR"), "default:RTL") == 0)
+ /* Normally this should have been taken care of during command line
+ * parsing as a post-parse hook of gtk_get_option_group(), using the
+ * system locales.
+ * But user config may have overridden the language, therefore we must
+ * check the widget directions again.
+ */
+ gtk_widget_set_default_direction (GTK_TEXT_DIR_RTL);
+ else
+ gtk_widget_set_default_direction (GTK_TEXT_DIR_LTR);
+
+ gui_unique_init (gimp);
+ gimp_language_store_parser_init ();
+
+ /* initialize icon themes before gimp_widgets_init() so we avoid
+ * setting the configured theme twice
+ */
+ icon_themes_init (gimp);
+
+ gimp_widgets_init (gui_help_func,
+ gui_get_foreground_func,
+ gui_get_background_func,
+ NULL);
+
+ g_type_class_ref (GIMP_TYPE_COLOR_SELECT);
+
+ /* disable automatic startup notification */
+ gtk_window_set_auto_startup_notification (FALSE);
+
+#ifdef GDK_WINDOWING_QUARTZ
+ /* Before the first window is created (typically the splash window),
+ * we need to disable automatic tabbing behavior introduced on Sierra.
+ * This is known to cause all kinds of weird issues (see for instance
+ * Bugzilla #776294) and needs proper GTK+ support if we would want to
+ * enable it.
+ */
+ if ([NSWindow respondsToSelector:@selector(setAllowsAutomaticWindowTabbing:)])
+ [NSWindow setAllowsAutomaticWindowTabbing:NO];
+
+ /* MacOS 11 (Big Sur) has added a new, dynamic "accent" as default.
+ * This uses a 10-bit colorspace so every GIMP drawing operation
+ * has the additional cost of an 8-bit (ARGB) to 10-bit conversion.
+ * Let's disable this mode to regain the lost performance.
+ */
+ if (gdk_quartz_osx_version () >= GDK_OSX_BIG_SUR)
+ {
+ NSUserDefaults *userDefaults = [NSUserDefaults standardUserDefaults];
+ [userDefaults setBool: NO forKey:@"NSViewUsesAutomaticLayerBackingStores"];
+ }
+
+#endif /* GDK_WINDOWING_QUARTZ */
+
+ gimp_dnd_init (gimp);
+
+ themes_init (gimp);
+
+ initial_monitor = gimp_get_monitor_at_pointer (&initial_screen);
+ gtk_widget_set_default_colormap (gdk_screen_get_rgb_colormap (initial_screen));
+
+ if (! no_splash)
+ {
+ splash_create (gimp->be_verbose, initial_screen, initial_monitor);
+ status_callback = splash_update;
+ }
+
+ g_signal_connect_after (gimp, "initialize",
+ G_CALLBACK (gui_initialize_after_callback),
+ NULL);
+
+ g_signal_connect (gimp, "restore",
+ G_CALLBACK (gui_restore_callback),
+ NULL);
+ g_signal_connect_after (gimp, "restore",
+ G_CALLBACK (gui_restore_after_callback),
+ NULL);
+
+ g_signal_connect (gimp, "exit",
+ G_CALLBACK (gui_exit_callback),
+ NULL);
+ g_signal_connect_after (gimp, "exit",
+ G_CALLBACK (gui_exit_after_callback),
+ NULL);
+
+ return status_callback;
+}
+
+/*
+ * gui_recover:
+ * @n_recoveries: number of recovered files.
+ *
+ * Query the user interactively if files were saved from a previous
+ * crash, asking whether to try and recover or discard them.
+ *
+ * Returns: TRUE if answer is to try and recover, FALSE otherwise.
+ */
+gboolean
+gui_recover (gint n_recoveries)
+{
+ GtkWidget *dialog;
+ GtkWidget *box;
+ gboolean recover;
+
+ dialog = gimp_dialog_new (_("Image Recovery"), "gimp-recovery",
+ NULL, GTK_DIALOG_MODAL, NULL, NULL,
+ _("_Discard"), GTK_RESPONSE_CANCEL,
+ _("_Recover"), GTK_RESPONSE_OK,
+ NULL);
+ gtk_dialog_set_default_response (GTK_DIALOG (dialog),
+ GTK_RESPONSE_OK);
+
+ box = gimp_message_box_new (GIMP_ICON_WILBER_EEK);
+ gtk_box_pack_start (GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (dialog))),
+ box, TRUE, TRUE, 0);
+ gtk_widget_show (box);
+
+ gimp_message_box_set_primary_text (GIMP_MESSAGE_BOX (box),
+ _("Eeek! It looks like GIMP recovered from a crash!"));
+
+ gimp_message_box_set_text (GIMP_MESSAGE_BOX (box),
+ /* TRANSLATORS: even if English singular form does
+ * not use %d, you can use %d for translation in
+ * any singular/plural form of your language if
+ * suited. It will just work and be replaced by the
+ * number of images as expected.
+ */
+ ngettext ("An image was salvaged from the crash. "
+ "Do you want to try and recover it?",
+ "%d images were salvaged from the crash. "
+ "Do you want to try and recover them?",
+ n_recoveries), n_recoveries);
+
+ recover = (gimp_dialog_run (GIMP_DIALOG (dialog)) == GTK_RESPONSE_OK);
+ gtk_widget_destroy (dialog);
+
+ return recover;
+}
+
+gint
+gui_get_initial_monitor (Gimp *gimp,
+ GdkScreen **screen)
+{
+ g_return_val_if_fail (GIMP_IS_GIMP (gimp), 0);
+ g_return_val_if_fail (screen != NULL, 0);
+
+ *screen = initial_screen;
+
+ return initial_monitor;
+}
+
+
+/* private functions */
+
+static gchar *
+gui_sanity_check (void)
+{
+#define GTK_REQUIRED_MAJOR 2
+#define GTK_REQUIRED_MINOR 24
+#define GTK_REQUIRED_MICRO 10
+
+ const gchar *mismatch = gtk_check_version (GTK_REQUIRED_MAJOR,
+ GTK_REQUIRED_MINOR,
+ GTK_REQUIRED_MICRO);
+
+ if (mismatch)
+ {
+ return g_strdup_printf
+ ("%s\n\n"
+ "GIMP requires GTK+ version %d.%d.%d or later.\n"
+ "Installed GTK+ version is %d.%d.%d.\n\n"
+ "Somehow you or your software packager managed\n"
+ "to install GIMP with an older GTK+ version.\n\n"
+ "Please upgrade to GTK+ version %d.%d.%d or later.",
+ mismatch,
+ GTK_REQUIRED_MAJOR, GTK_REQUIRED_MINOR, GTK_REQUIRED_MICRO,
+ gtk_major_version, gtk_minor_version, gtk_micro_version,
+ GTK_REQUIRED_MAJOR, GTK_REQUIRED_MINOR, GTK_REQUIRED_MICRO);
+ }
+
+#undef GTK_REQUIRED_MAJOR
+#undef GTK_REQUIRED_MINOR
+#undef GTK_REQUIRED_MICRO
+
+ return NULL;
+}
+
+static void
+gui_help_func (const gchar *help_id,
+ gpointer help_data)
+{
+ g_return_if_fail (GIMP_IS_GIMP (the_gui_gimp));
+
+ gimp_help (the_gui_gimp, NULL, NULL, help_id);
+}
+
+static gboolean
+gui_get_foreground_func (GimpRGB *color)
+{
+ g_return_val_if_fail (color != NULL, FALSE);
+ g_return_val_if_fail (GIMP_IS_GIMP (the_gui_gimp), FALSE);
+
+ gimp_context_get_foreground (gimp_get_user_context (the_gui_gimp), color);
+
+ return TRUE;
+}
+
+static gboolean
+gui_get_background_func (GimpRGB *color)
+{
+ g_return_val_if_fail (color != NULL, FALSE);
+ g_return_val_if_fail (GIMP_IS_GIMP (the_gui_gimp), FALSE);
+
+ gimp_context_get_background (gimp_get_user_context (the_gui_gimp), color);
+
+ return TRUE;
+}
+
+static void
+gui_initialize_after_callback (Gimp *gimp,
+ GimpInitStatusFunc status_callback)
+{
+ const gchar *name = NULL;
+
+ g_return_if_fail (GIMP_IS_GIMP (gimp));
+
+ if (gimp->be_verbose)
+ g_print ("INIT: %s\n", G_STRFUNC);
+
+#if defined (GDK_WINDOWING_X11)
+ name = "DISPLAY";
+#elif defined (GDK_WINDOWING_DIRECTFB) || defined (GDK_WINDOWING_FB)
+ name = "GDK_DISPLAY";
+#endif
+
+ /* TODO: Need to care about display migration with GTK+ 2.2 at some point */
+
+ if (name)
+ {
+ gchar *display = gdk_get_display ();
+
+ gimp_environ_table_add (gimp->plug_in_manager->environ_table,
+ name, display, NULL);
+ g_free (display);
+ }
+
+ gimp_tools_init (gimp);
+
+ gimp_context_set_tool (gimp_get_user_context (gimp),
+ gimp_tool_info_get_standard (gimp));
+}
+
+static void
+gui_restore_callback (Gimp *gimp,
+ GimpInitStatusFunc status_callback)
+{
+ GimpDisplayConfig *display_config = GIMP_DISPLAY_CONFIG (gimp->config);
+ GimpGuiConfig *gui_config = GIMP_GUI_CONFIG (gimp->config);
+
+ if (gimp->be_verbose)
+ g_print ("INIT: %s\n", G_STRFUNC);
+
+ gui_vtable_init (gimp);
+
+ if (! gui_config->show_tooltips)
+ gimp_help_disable_tooltips ();
+
+ g_signal_connect (gui_config, "notify::show-tooltips",
+ G_CALLBACK (gui_show_tooltips_notify),
+ gimp);
+
+ gimp_dialogs_show_help_button (gui_config->use_help &&
+ gui_config->show_help_button);
+
+ g_signal_connect (gui_config, "notify::use-help",
+ G_CALLBACK (gui_show_help_button_notify),
+ gimp);
+ g_signal_connect (gui_config, "notify::user-manual-online",
+ G_CALLBACK (gui_user_manual_notify),
+ gimp);
+ g_signal_connect (gui_config, "notify::show-help-button",
+ G_CALLBACK (gui_show_help_button_notify),
+ gimp);
+
+ g_signal_connect (gimp_get_user_context (gimp), "display-changed",
+ G_CALLBACK (gui_display_changed),
+ gimp);
+
+ /* make sure the monitor resolution is valid */
+ if (display_config->monitor_res_from_gdk ||
+ display_config->monitor_xres < GIMP_MIN_RESOLUTION ||
+ display_config->monitor_yres < GIMP_MIN_RESOLUTION)
+ {
+ gdouble xres, yres;
+
+ gimp_get_monitor_resolution (initial_screen,
+ initial_monitor,
+ &xres, &yres);
+
+ g_object_set (gimp->config,
+ "monitor-xresolution", xres,
+ "monitor-yresolution", yres,
+ "monitor-resolution-from-windowing-system", TRUE,
+ NULL);
+ }
+
+ actions_init (gimp);
+ menus_init (gimp, global_action_factory);
+ gimp_render_init (gimp);
+
+ dialogs_init (gimp, global_menu_factory);
+
+ gimp_clipboard_init (gimp);
+ if (gimp_get_clipboard_image (gimp))
+ gimp_clipboard_set_image (gimp, gimp_get_clipboard_image (gimp));
+ else
+ gimp_clipboard_set_buffer (gimp, gimp_get_clipboard_buffer (gimp));
+
+ g_signal_connect (gimp, "clipboard-changed",
+ G_CALLBACK (gui_clipboard_changed),
+ NULL);
+
+ gimp_devices_init (gimp);
+ gimp_controllers_init (gimp);
+ session_init (gimp);
+
+ g_type_class_unref (g_type_class_ref (GIMP_TYPE_COLOR_SELECTOR_PALETTE));
+
+ status_callback (NULL, _("Tool Options"), 1.0);
+ gimp_tools_restore (gimp);
+}
+
+#ifdef GDK_WINDOWING_QUARTZ
+static void
+gui_add_to_app_menu (GimpUIManager *ui_manager,
+ GtkosxApplication *osx_app,
+ const gchar *action_path,
+ gint index)
+{
+ GtkWidget *item;
+
+ item = gimp_ui_manager_get_widget (ui_manager, action_path);
+
+ if (GTK_IS_MENU_ITEM (item))
+ gtkosx_application_insert_app_menu_item (osx_app, GTK_WIDGET (item), index);
+}
+
+static gboolean
+gui_quartz_quit_callback (GtkosxApplication *osx_app,
+ GimpUIManager *ui_manager)
+{
+ gimp_ui_manager_activate_action (ui_manager, "file", "file-quit");
+
+ return TRUE;
+}
+#endif
+
+static void
+gui_restore_after_callback (Gimp *gimp,
+ GimpInitStatusFunc status_callback)
+{
+ GimpGuiConfig *gui_config = GIMP_GUI_CONFIG (gimp->config);
+ GimpDisplay *display;
+
+ if (gimp->be_verbose)
+ g_print ("INIT: %s\n", G_STRFUNC);
+
+ gimp->message_handler = GIMP_MESSAGE_BOX;
+
+ /* load the recent documents after gimp_real_restore() because we
+ * need the mime-types implemented by plug-ins
+ */
+ status_callback (NULL, _("Documents"), 0.9);
+ gimp_recent_list_load (gimp);
+
+ /* enable this to always have icons everywhere */
+ if (g_getenv ("GIMP_ICONS_LIKE_A_BOSS"))
+ {
+ GdkScreen *screen = gdk_screen_get_default ();
+
+ g_object_set (G_OBJECT (gtk_settings_get_for_screen (screen)),
+ "gtk-button-images", TRUE,
+ "gtk-menu-images", TRUE,
+ NULL);
+ }
+
+ if (gui_config->restore_accels)
+ menus_restore (gimp);
+
+ ui_configurer = g_object_new (GIMP_TYPE_UI_CONFIGURER,
+ "gimp", gimp,
+ NULL);
+
+ image_ui_manager = gimp_menu_factory_manager_new (global_menu_factory,
+ "<Image>",
+ gimp,
+ gui_config->tearoff_menus);
+ gimp_ui_manager_update (image_ui_manager, gimp);
+
+ /* Check that every accelerator is unique. */
+ gtk_accel_map_foreach_unfiltered (NULL,
+ gui_check_unique_accelerator);
+
+ gimp_action_history_init (gimp);
+
+ g_signal_connect_object (gui_config, "notify::single-window-mode",
+ G_CALLBACK (gui_single_window_mode_notify),
+ ui_configurer, 0);
+ g_signal_connect_object (gui_config, "notify::tearoff-menus",
+ G_CALLBACK (gui_tearoff_menus_notify),
+ image_ui_manager, 0);
+ g_signal_connect (image_ui_manager, "show-tooltip",
+ G_CALLBACK (gui_menu_show_tooltip),
+ gimp);
+ g_signal_connect (image_ui_manager, "hide-tooltip",
+ G_CALLBACK (gui_menu_hide_tooltip),
+ gimp);
+
+ gimp_devices_restore (gimp);
+ gimp_controllers_restore (gimp, image_ui_manager);
+
+ if (status_callback == splash_update)
+ splash_destroy ();
+
+ if (gimp_get_show_gui (gimp))
+ {
+ GimpDisplayShell *shell;
+ GtkWidget *toplevel;
+
+ /* create the empty display */
+ display = GIMP_DISPLAY (gimp_create_display (gimp, NULL,
+ GIMP_UNIT_PIXEL, 1.0,
+ G_OBJECT (initial_screen),
+ initial_monitor));
+
+ shell = gimp_display_get_shell (display);
+
+ if (gui_config->restore_session)
+ session_restore (gimp,
+ initial_screen,
+ initial_monitor);
+
+ toplevel = gtk_widget_get_toplevel (GTK_WIDGET (shell));
+
+#ifdef GDK_WINDOWING_QUARTZ
+ {
+ GtkosxApplication *osx_app;
+ GtkWidget *menu;
+ GtkWidget *item;
+
+ [[NSUserDefaults standardUserDefaults] setObject:@"NO"
+ forKey:@"NSTreatUnknownArgumentsAsOpen"];
+
+ osx_app = gtkosx_application_get ();
+
+ menu = gimp_ui_manager_get_widget (image_ui_manager,
+ "/image-menubar");
+ /* menu should have window parent for accelerator support */
+ gtk_widget_set_parent(menu, toplevel);
+
+ if (GTK_IS_MENU_ITEM (menu))
+ menu = gtk_menu_item_get_submenu (GTK_MENU_ITEM (menu));
+
+ /* do not activate OSX menu if tests are running */
+ if (! g_getenv ("GIMP_TESTING_ABS_TOP_SRCDIR"))
+ gtkosx_application_set_menu_bar (osx_app, GTK_MENU_SHELL (menu));
+
+ gtkosx_application_set_use_quartz_accelerators (osx_app, FALSE);
+
+ gui_add_to_app_menu (image_ui_manager, osx_app,
+ "/image-menubar/Help/dialogs-about", 0);
+ gui_add_to_app_menu (image_ui_manager, osx_app,
+ "/image-menubar/Help/dialogs-search-action", 1);
+
+#define PREFERENCES "/image-menubar/Edit/Preferences/"
+
+ gui_add_to_app_menu (image_ui_manager, osx_app,
+ PREFERENCES "dialogs-preferences", 3);
+ gui_add_to_app_menu (image_ui_manager, osx_app,
+ PREFERENCES "dialogs-input-devices", 4);
+ gui_add_to_app_menu (image_ui_manager, osx_app,
+ PREFERENCES "dialogs-keyboard-shortcuts", 5);
+ gui_add_to_app_menu (image_ui_manager, osx_app,
+ PREFERENCES "dialogs-module-dialog", 6);
+ gui_add_to_app_menu (image_ui_manager, osx_app,
+ PREFERENCES "plug-in-unit-editor", 7);
+
+#undef PREFERENCES
+
+ item = gtk_separator_menu_item_new ();
+ gtkosx_application_insert_app_menu_item (osx_app, item, 8);
+
+ item = gimp_ui_manager_get_widget (image_ui_manager,
+ "/image-menubar/File/file-quit");
+ gtk_widget_hide (item);
+
+ g_signal_connect (osx_app, "NSApplicationBlockTermination",
+ G_CALLBACK (gui_quartz_quit_callback),
+ image_ui_manager);
+
+ gtkosx_application_ready (osx_app);
+ }
+#endif /* GDK_WINDOWING_QUARTZ */
+
+ /* move keyboard focus to the display */
+ gtk_window_present (GTK_WINDOW (toplevel));
+ }
+
+ /* indicate that the application has finished loading */
+ gdk_notify_startup_complete ();
+
+ /* clear startup monitor variables */
+ initial_screen = NULL;
+ initial_monitor = -1;
+}
+
+static gboolean
+gui_exit_callback (Gimp *gimp,
+ gboolean force)
+{
+ GimpGuiConfig *gui_config = GIMP_GUI_CONFIG (gimp->config);
+ GimpTool *active_tool;
+
+ if (gimp->be_verbose)
+ g_print ("EXIT: %s\n", G_STRFUNC);
+
+ if (! force && gimp_displays_dirty (gimp))
+ {
+ GdkScreen *screen;
+ gint monitor;
+
+ monitor = gimp_get_monitor_at_pointer (&screen);
+
+ gimp_dialog_factory_dialog_raise (gimp_dialog_factory_get_singleton (),
+ screen, monitor,
+ "gimp-quit-dialog", -1);
+
+ return TRUE; /* stop exit for now */
+ }
+
+ gimp->message_handler = GIMP_CONSOLE;
+
+ gui_unique_exit ();
+
+ /* If any modifier is set when quitting (typically when exiting with
+ * Ctrl-q for instance!), when serializing the tool options, it will
+ * save any alternate value instead of the main one. Make sure that
+ * any modifier is reset before saving options.
+ */
+ active_tool = tool_manager_get_active (gimp);
+ if (active_tool && active_tool->focus_display)
+ gimp_tool_set_modifier_state (active_tool, 0, active_tool->focus_display);
+
+ if (gui_config->save_session_info)
+ session_save (gimp, FALSE);
+
+ if (gui_config->save_device_status)
+ gimp_devices_save (gimp, FALSE);
+
+ if (TRUE /* gui_config->save_controllers */)
+ gimp_controllers_save (gimp);
+
+ g_signal_handlers_disconnect_by_func (gimp_get_user_context (gimp),
+ gui_display_changed,
+ gimp);
+
+ gimp_displays_delete (gimp);
+
+ if (gui_config->save_accels)
+ menus_save (gimp, FALSE);
+
+ gimp_tools_save (gimp, gui_config->save_tool_options, FALSE);
+ gimp_tools_exit (gimp);
+
+ gimp_language_store_parser_clean ();
+
+ return FALSE; /* continue exiting */
+}
+
+static gboolean
+gui_exit_after_callback (Gimp *gimp,
+ gboolean force)
+{
+ if (gimp->be_verbose)
+ g_print ("EXIT: %s\n", G_STRFUNC);
+
+ g_signal_handlers_disconnect_by_func (gimp->config,
+ gui_show_help_button_notify,
+ gimp);
+ g_signal_handlers_disconnect_by_func (gimp->config,
+ gui_user_manual_notify,
+ gimp);
+ g_signal_handlers_disconnect_by_func (gimp->config,
+ gui_show_tooltips_notify,
+ gimp);
+
+ gimp_action_history_exit (gimp);
+
+ g_object_unref (image_ui_manager);
+ image_ui_manager = NULL;
+
+ g_object_unref (ui_configurer);
+ ui_configurer = NULL;
+
+ /* exit the clipboard before shutting down the GUI because it runs
+ * a whole lot of code paths. See bug #731389.
+ */
+ g_signal_handlers_disconnect_by_func (gimp,
+ G_CALLBACK (gui_clipboard_changed),
+ NULL);
+ gimp_clipboard_exit (gimp);
+
+ session_exit (gimp);
+ menus_exit (gimp);
+ actions_exit (gimp);
+ gimp_render_exit (gimp);
+
+ gimp_controllers_exit (gimp);
+ gimp_devices_exit (gimp);
+ dialogs_exit (gimp);
+ themes_exit (gimp);
+
+ g_type_class_unref (g_type_class_peek (GIMP_TYPE_COLOR_SELECT));
+
+ return FALSE; /* continue exiting */
+}
+
+static void
+gui_show_tooltips_notify (GimpGuiConfig *gui_config,
+ GParamSpec *param_spec,
+ Gimp *gimp)
+{
+ if (gui_config->show_tooltips)
+ gimp_help_enable_tooltips ();
+ else
+ gimp_help_disable_tooltips ();
+}
+
+static void
+gui_show_help_button_notify (GimpGuiConfig *gui_config,
+ GParamSpec *param_spec,
+ Gimp *gimp)
+{
+ gimp_dialogs_show_help_button (gui_config->use_help &&
+ gui_config->show_help_button);
+}
+
+static void
+gui_user_manual_notify (GimpGuiConfig *gui_config,
+ GParamSpec *param_spec,
+ Gimp *gimp)
+{
+ gimp_help_user_manual_changed (gimp);
+}
+
+static void
+gui_single_window_mode_notify (GimpGuiConfig *gui_config,
+ GParamSpec *pspec,
+ GimpUIConfigurer *ui_configurer)
+{
+ gimp_ui_configurer_configure (ui_configurer,
+ gui_config->single_window_mode);
+}
+static void
+gui_tearoff_menus_notify (GimpGuiConfig *gui_config,
+ GParamSpec *pspec,
+ GtkUIManager *manager)
+{
+ gtk_ui_manager_set_add_tearoffs (manager, gui_config->tearoff_menus);
+}
+
+static void
+gui_clipboard_changed (Gimp *gimp)
+{
+ if (gimp_get_clipboard_image (gimp))
+ gimp_clipboard_set_image (gimp, gimp_get_clipboard_image (gimp));
+ else
+ gimp_clipboard_set_buffer (gimp, gimp_get_clipboard_buffer (gimp));
+}
+
+static void
+gui_menu_show_tooltip (GimpUIManager *manager,
+ const gchar *tooltip,
+ Gimp *gimp)
+{
+ GimpContext *context = gimp_get_user_context (gimp);
+ GimpDisplay *display = gimp_context_get_display (context);
+
+ if (display)
+ {
+ GimpDisplayShell *shell = gimp_display_get_shell (display);
+ GimpStatusbar *statusbar = gimp_display_shell_get_statusbar (shell);
+
+ gimp_statusbar_push (statusbar, "menu-tooltip",
+ NULL, "%s", tooltip);
+ }
+}
+
+static void
+gui_menu_hide_tooltip (GimpUIManager *manager,
+ Gimp *gimp)
+{
+ GimpContext *context = gimp_get_user_context (gimp);
+ GimpDisplay *display = gimp_context_get_display (context);
+
+ if (display)
+ {
+ GimpDisplayShell *shell = gimp_display_get_shell (display);
+ GimpStatusbar *statusbar = gimp_display_shell_get_statusbar (shell);
+
+ gimp_statusbar_pop (statusbar, "menu-tooltip");
+ }
+}
+
+static void
+gui_display_changed (GimpContext *context,
+ GimpDisplay *display,
+ Gimp *gimp)
+{
+ if (! display)
+ {
+ GimpImage *image = gimp_context_get_image (context);
+
+ if (image)
+ {
+ GList *list;
+
+ for (list = gimp_get_display_iter (gimp);
+ list;
+ list = g_list_next (list))
+ {
+ GimpDisplay *display2 = list->data;
+
+ if (gimp_display_get_image (display2) == image)
+ {
+ gimp_context_set_display (context, display2);
+
+ /* stop the emission of the original signal
+ * (the emission of the recursive signal is finished)
+ */
+ g_signal_stop_emission_by_name (context, "display-changed");
+ return;
+ }
+ }
+
+ gimp_context_set_image (context, NULL);
+ }
+ }
+
+ gimp_ui_manager_update (image_ui_manager, display);
+}
+
+typedef struct
+{
+ const gchar *path;
+ guint key;
+ GdkModifierType mods;
+}
+accelData;
+
+static void
+gui_compare_accelerator (gpointer data,
+ const gchar *accel_path,
+ guint accel_key,
+ GdkModifierType accel_mods,
+ gboolean changed)
+{
+ accelData *accel = data;
+
+ if (accel->key == accel_key && accel->mods == accel_mods &&
+ g_strcmp0 (accel->path, accel_path))
+ {
+ g_printerr ("Actions \"%s\" and \"%s\" use the same accelerator.\n"
+ " Disabling the accelerator on \"%s\".\n",
+ accel->path, accel_path, accel_path);
+ gtk_accel_map_change_entry (accel_path, 0, 0, FALSE);
+ }
+}
+
+static void
+gui_check_unique_accelerator (gpointer data,
+ const gchar *accel_path,
+ guint accel_key,
+ GdkModifierType accel_mods,
+ gboolean changed)
+{
+ if (gtk_accelerator_valid (accel_key, accel_mods) &&
+ gui_check_action_exists (accel_path))
+ {
+ accelData accel;
+
+ accel.path = accel_path;
+ accel.key = accel_key;
+ accel.mods = accel_mods;
+
+ gtk_accel_map_foreach_unfiltered (&accel,
+ gui_compare_accelerator);
+ }
+}
+
+static gboolean
+gui_check_action_exists (const gchar *accel_path)
+{
+ GimpUIManager *manager;
+ gboolean action_exists = FALSE;
+ GList *list;
+
+ manager = gimp_ui_managers_from_name ("<Image>")->data;
+
+ for (list = gimp_ui_manager_get_action_groups (manager);
+ list;
+ list = g_list_next (list))
+ {
+ GimpActionGroup *group = list->data;
+ GList *actions = NULL;
+ GList *list2;
+
+ actions = gimp_action_group_list_actions (group);
+
+ for (list2 = actions; list2; list2 = g_list_next (list2))
+ {
+ GimpAction *action = list2->data;
+ const gchar *path = gimp_action_get_accel_path (action);
+
+ if (g_strcmp0 (path, accel_path) == 0)
+ {
+ action_exists = TRUE;
+ break;
+ }
+ }
+
+ g_list_free (actions);
+
+ if (action_exists)
+ break;
+ }
+
+ return action_exists;
+}