/* The GIMP -- an 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 2 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, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include "config.h" #include #include #include #include "libgimpbase/gimpbase.h" #include "libgimpwidgets/gimpwidgets.h" #include "libgimpwidgets/gimpwidgets-private.h" #include "gui-types.h" #include "config/gimpguiconfig.h" #include "config/gimpconfig-path.h" #include "core/gimp.h" #include "core/gimpcontainer.h" #include "core/gimpcontext.h" #include "core/gimpenvirontable.h" #include "core/gimpimage.h" #include "display/gimpdisplay.h" #include "display/gimpdisplay-foreach.h" #include "display/gimpdisplayshell.h" #include "display/gimpdisplayshell-render.h" #include "widgets/gimpdevices.h" #include "widgets/gimpdialogfactory.h" #include "widgets/gimpitemfactory.h" #include "widgets/gimpmenufactory.h" #include "widgets/gimpwidgets-utils.h" #include "device-status-dialog.h" #include "dialogs.h" #include "dialogs-commands.h" #include "error-console-dialog.h" #include "gui.h" #include "menus.h" #include "session.h" #include "gimp-intl.h" /* local function prototypes */ 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_message (Gimp *gimp, const gchar *message); static GimpObject * gui_display_new (GimpImage *gimage, guint scale); static void gui_themes_dir_foreach_func (GimpDatafileData *file_data); static gboolean gui_exit_callback (Gimp *gimp, gboolean kill_it); static gboolean gui_exit_finish_callback (Gimp *gimp, gboolean kill_it); static void gui_really_quit_callback (GtkWidget *button, gboolean quit, gpointer data); static void gui_show_tooltips_notify (GObject *config, GParamSpec *param_spec, Gimp *gimp); static void gui_display_changed (GimpContext *context, GimpDisplay *display, Gimp *gimp); static void gui_image_disconnect (GimpImage *gimage, Gimp *gimp); /* private variables */ static GQuark image_disconnect_handler_id = 0; static GHashTable *themes_hash = NULL; static GimpItemFactory *toolbox_item_factory = NULL; static GimpItemFactory *image_item_factory = NULL; /* forward declarations to avoid inclusion of libgimp-glue.h */ gboolean gimp_palette_get_background (GimpRGB *color); gboolean gimp_palette_get_foreground (GimpRGB *color); /* public functions */ gboolean gui_libs_init (gint *argc, gchar ***argv) { GimpWidgetsVTable vtable; g_return_val_if_fail (argc != NULL, FALSE); g_return_val_if_fail (argv != NULL, FALSE); if (!gtk_init_check (argc, argv)) return FALSE; /* Initialize the eeky vtable needed by libgimpwidgets */ vtable.standard_help_func = gimp_standard_help_func; vtable.palette_get_background = gimp_palette_get_background; vtable.palette_get_foreground = gimp_palette_get_foreground; vtable.unit_get_number_of_units = gimp_unit_get_number_of_units; vtable.unit_get_number_of_built_in_units = gimp_unit_get_number_of_built_in_units; vtable.unit_get_factor = gimp_unit_get_factor; vtable.unit_get_digits = gimp_unit_get_digits; vtable.unit_get_identifier = gimp_unit_get_identifier; vtable.unit_get_symbol = gimp_unit_get_symbol; vtable.unit_get_abbreviation = gimp_unit_get_abbreviation; vtable.unit_get_singular = gimp_unit_get_singular; vtable.unit_get_plural = gimp_unit_get_plural; gimp_widgets_init (&vtable); g_type_class_ref (GIMP_TYPE_COLOR_SELECT); return TRUE; } void gui_environ_init (Gimp *gimp) { gchar *display, *name = NULL; #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) { display = gdk_get_display (); gimp_environ_table_add (gimp->environ_table, name, display, NULL); g_free (display); } } void gui_themes_init (Gimp *gimp) { GimpGuiConfig *config; const gchar *theme_dir; gchar *gtkrc; g_return_if_fail (GIMP_IS_GIMP (gimp)); config = GIMP_GUI_CONFIG (gimp->config); themes_hash = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free); if (config->theme_path) { gchar *path; path = gimp_config_path_expand (config->theme_path, TRUE, NULL); gimp_datafiles_read_directories (path, G_FILE_TEST_IS_DIR, gui_themes_dir_foreach_func, gimp); g_free (path); } theme_dir = gui_themes_get_theme_dir (gimp); if (theme_dir) { gtkrc = g_build_filename (theme_dir, "gtkrc", NULL); } else { /* get the hardcoded default theme gtkrc */ gtkrc = g_strdup (gimp_gtkrc ()); } if (gimp->be_verbose) g_print (_("Parsing '%s'\n"), gtkrc); gtk_rc_parse (gtkrc); g_free (gtkrc); /* parse the user gtkrc */ gtkrc = gimp_personal_rc_file ("gtkrc"); if (gimp->be_verbose) g_print (_("Parsing '%s'\n"), gtkrc); gtk_rc_parse (gtkrc); g_free (gtkrc); if (! config->show_tool_tips) gimp_help_disable_tooltips (); g_signal_connect (gimp->config, "notify::show-tool-tips", G_CALLBACK (gui_show_tooltips_notify), gimp); gdk_rgb_set_min_colors (CLAMP (gimp->config->min_colors, 27, 256)); gdk_rgb_set_install (gimp->config->install_cmap); gtk_widget_set_default_colormap (gdk_rgb_get_colormap ()); } const gchar * gui_themes_get_theme_dir (Gimp *gimp) { GimpGuiConfig *config = GIMP_GUI_CONFIG (gimp->config); if (config->theme) return g_hash_table_lookup (themes_hash, config->theme); return g_hash_table_lookup (themes_hash, "Default"); } void gui_init (Gimp *gimp) { GimpDisplayConfig *display_config; GimpGuiConfig *gui_config; g_return_if_fail (GIMP_IS_GIMP (gimp)); display_config = GIMP_DISPLAY_CONFIG (gimp->config); gui_config = GIMP_GUI_CONFIG (gimp->config); gimp->gui_threads_enter_func = gui_threads_enter; gimp->gui_threads_leave_func = gui_threads_leave; gimp->gui_set_busy_func = gui_set_busy; gimp->gui_unset_busy_func = gui_unset_busy; gimp->gui_message_func = gui_message; gimp->gui_create_display_func = gui_display_new; image_disconnect_handler_id = gimp_container_add_handler (gimp->images, "disconnect", G_CALLBACK (gui_image_disconnect), 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; gui_get_screen_resolution (&xres, &yres); g_object_set (gimp->config, "monitor-xresolution", xres, "monitor-yresolution", yres, "monitor-resolution-from-windowing-system", TRUE, NULL); } menus_init (gimp); render_init (gimp); dialogs_init (gimp); gimp_devices_init (gimp, device_status_dialog_update_current); session_init (gimp); } void gui_restore (Gimp *gimp, gboolean restore_session) { g_return_if_fail (GIMP_IS_GIMP (gimp)); gimp->message_handler = GIMP_MESSAGE_BOX; menus_restore (gimp); toolbox_item_factory = gimp_menu_factory_menu_new (global_menu_factory, "", GTK_TYPE_MENU_BAR, gimp, TRUE); image_item_factory = gimp_menu_factory_menu_new (global_menu_factory, "", GTK_TYPE_MENU, gimp, TRUE); gimp_devices_restore (gimp); if (GIMP_GUI_CONFIG (gimp->config)->restore_session || restore_session) session_restore (gimp); dialogs_show_toolbox (); g_signal_connect (gimp, "exit", G_CALLBACK (gui_exit_callback), NULL); g_signal_connect_after (gimp, "exit", G_CALLBACK (gui_exit_finish_callback), NULL); } void gui_post_init (Gimp *gimp) { g_return_if_fail (GIMP_IS_GIMP (gimp)); if (GIMP_GUI_CONFIG (gimp->config)->show_tips) gimp_dialog_factory_dialog_new (global_dialog_factory, "gimp-tips-dialog", -1); } void gui_get_screen_resolution (gdouble *xres, gdouble *yres) { gint width, height; gint width_mm, height_mm; gdouble x = 0.0; gdouble y = 0.0; g_return_if_fail (xres != NULL); g_return_if_fail (yres != NULL); width = gdk_screen_width (); height = gdk_screen_height (); width_mm = gdk_screen_width_mm (); height_mm = gdk_screen_height_mm (); /* * From xdpyinfo.c: * * there are 2.54 centimeters to an inch; so there are 25.4 millimeters. * * dpi = N pixels / (M millimeters / (25.4 millimeters / 1 inch)) * = N pixels / (M inch / 25.4) * = N * 25.4 pixels / M inch */ if (width_mm > 0 && height_mm > 0) { x = (width * 25.4) / (gdouble) width_mm; y = (height * 25.4) / (gdouble) height_mm; } if (x < GIMP_MIN_RESOLUTION || x > GIMP_MAX_RESOLUTION || y < GIMP_MIN_RESOLUTION || y > GIMP_MAX_RESOLUTION) { g_warning ("GDK returned bogus values for the screen resolution, " "using 75 dpi instead."); x = 75.0; y = 75.0; } /* round the value to full integers to give more pleasant results */ *xres = RINT (x); *yres = RINT (y); } /* private functions */ 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_factories_idle (); gdk_flush (); } static void gui_unset_busy (Gimp *gimp) { gimp_displays_unset_busy (gimp); gimp_dialog_factories_unidle (); gdk_flush (); } static void gui_message (Gimp *gimp, const gchar *message) { switch (gimp->message_handler) { case GIMP_MESSAGE_BOX: gimp_message_box (message, NULL, NULL); break; case GIMP_ERROR_CONSOLE: gimp_dialog_factory_dialog_raise (global_dock_factory, "gimp-error-console", -1); error_console_add (gimp, message); break; default: break; } } static GimpObject * gui_display_new (GimpImage *gimage, guint scale) { GimpDisplayShell *shell; GimpDisplay *gdisp; gdisp = gimp_display_new (gimage, scale, global_menu_factory, image_item_factory); shell = GIMP_DISPLAY_SHELL (gdisp->shell); gimp_context_set_display (gimp_get_user_context (gimage->gimp), gdisp); gimp_item_factory_update (shell->menubar_factory, shell); return GIMP_OBJECT (gdisp); } static void gui_themes_dir_foreach_func (GimpDatafileData *file_data) { Gimp *gimp; gchar *basename; gimp = (Gimp *) file_data->user_data; basename = g_path_get_basename (file_data->filename); if (gimp->be_verbose) g_print (_("Adding theme '%s' (%s)\n"), basename, file_data->filename); g_hash_table_insert (themes_hash, basename, g_strdup (file_data->filename)); } static gboolean gui_exit_callback (Gimp *gimp, gboolean kill_it) { if (! kill_it && gimp_displays_dirty (gimp)) { GtkWidget *dialog; gimp_item_factories_set_sensitive ("", "/File/Quit", FALSE); gimp_item_factories_set_sensitive ("", "/File/Quit", FALSE); dialog = gimp_query_boolean_box (_("Quit The GIMP?"), gimp_standard_help_func, "dialogs/really_quit.html", GIMP_STOCK_WILBER_EEK, _("Some files are unsaved.\n" "\nReally quit The GIMP?"), GTK_STOCK_QUIT, GTK_STOCK_CANCEL, NULL, NULL, gui_really_quit_callback, gimp); gtk_widget_show (dialog); return TRUE; /* stop exit for now */ } gimp->message_handler = GIMP_CONSOLE; session_save (gimp); menus_save (gimp); if (GIMP_GUI_CONFIG (gimp->config)->save_device_status) gimp_devices_save (gimp); gimp_displays_delete (gimp); return FALSE; /* continue exiting */ } static gboolean gui_exit_finish_callback (Gimp *gimp, gboolean kill_it) { g_signal_handlers_disconnect_by_func (gimp->config, gui_show_tooltips_notify, gimp); gimp_container_remove_handler (gimp->images, image_disconnect_handler_id); image_disconnect_handler_id = 0; g_object_unref (toolbox_item_factory); toolbox_item_factory = NULL; g_object_unref (image_item_factory); image_item_factory = NULL; menus_exit (gimp); render_exit (gimp); dialogs_exit (gimp); gimp_devices_exit (gimp); if (themes_hash) { g_hash_table_destroy (themes_hash); themes_hash = NULL; } g_type_class_unref (g_type_class_peek (GIMP_TYPE_COLOR_SELECT)); return FALSE; /* continue exiting */ } static void gui_really_quit_callback (GtkWidget *button, gboolean quit, gpointer data) { Gimp *gimp; gimp = GIMP (data); if (quit) { gimp_exit (gimp, TRUE); } else { gimp_item_factories_set_sensitive ("", "/File/Quit", TRUE); gimp_item_factories_set_sensitive ("", "/File/Quit", TRUE); } } static void gui_show_tooltips_notify (GObject *config, GParamSpec *param_spec, Gimp *gimp) { gboolean show_tool_tips; g_object_get (config, "show-tool-tips", &show_tool_tips, NULL); if (show_tool_tips) gimp_help_enable_tooltips (); else gimp_help_disable_tooltips (); } #ifdef __GNUC__ #warning FIXME: this junk should mostly go to the display subsystem #endif static void gui_display_changed (GimpContext *context, GimpDisplay *display, Gimp *gimp) { GimpItemFactory *item_factory; GimpDisplayShell *shell = NULL; item_factory = gimp_item_factory_from_path (""); if (display) shell = GIMP_DISPLAY_SHELL (display->shell); gimp_item_factory_update (item_factory, shell); } static void gui_image_disconnect (GimpImage *gimage, Gimp *gimp) { /* check if this is the last image and if it had a display */ if (gimp_container_num_children (gimp->images) == 1 && gimage->instance_count > 0) { dialogs_show_toolbox (); } }