/* GIMP - The GNU Image Manipulation Program * Copyright (C) 1995 Spencer Kimball and Peter Mattis * * gimpdockable.c * Copyright (C) 2001-2003 Michael Natterer * * 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 . */ #include "config.h" #include #include #include #include "libgimpbase/gimpbase.h" #include "libgimpwidgets/gimpwidgets.h" #include "widgets-types.h" #include "core/gimpcontext.h" #include "menus/menus.h" #include "gimpdialogfactory.h" #include "gimpdnd.h" #include "gimpdock.h" #include "gimpdockable.h" #include "gimpdockbook.h" #include "gimpdocked.h" #include "gimpdockwindow.h" #include "gimphelp-ids.h" #include "gimppanedbox.h" #include "gimpsessioninfo-aux.h" #include "gimpsessionmanaged.h" #include "gimpuimanager.h" #include "gimpwidgets-utils.h" #include "gimp-intl.h" enum { PROP_0, PROP_LOCKED }; struct _GimpDockablePrivate { gchar *name; gchar *blurb; gchar *icon_name; gchar *help_id; GimpTabStyle tab_style; gboolean locked; GimpDockbook *dockbook; GimpContext *context; }; static void gimp_dockable_session_managed_iface_init (GimpSessionManagedInterface *iface); static void gimp_dockable_dispose (GObject *object); static void gimp_dockable_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec); static void gimp_dockable_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec); static void gimp_dockable_style_updated (GtkWidget *widget); static void gimp_dockable_add (GtkContainer *container, GtkWidget *widget); static GType gimp_dockable_child_type (GtkContainer *container); static GList * gimp_dockable_get_aux_info (GimpSessionManaged *managed); static void gimp_dockable_set_aux_info (GimpSessionManaged *managed, GList *aux_info); G_DEFINE_TYPE_WITH_CODE (GimpDockable, gimp_dockable, GTK_TYPE_BIN, G_ADD_PRIVATE (GimpDockable) G_IMPLEMENT_INTERFACE (GIMP_TYPE_SESSION_MANAGED, gimp_dockable_session_managed_iface_init)) #define parent_class gimp_dockable_parent_class static const GtkTargetEntry dialog_target_table[] = { GIMP_TARGET_NOTEBOOK_TAB }; static void gimp_dockable_class_init (GimpDockableClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); GtkContainerClass *container_class = GTK_CONTAINER_CLASS (klass); object_class->dispose = gimp_dockable_dispose; object_class->set_property = gimp_dockable_set_property; object_class->get_property = gimp_dockable_get_property; widget_class->style_updated = gimp_dockable_style_updated; container_class->add = gimp_dockable_add; container_class->child_type = gimp_dockable_child_type; gtk_container_class_handle_border_width (container_class); g_object_class_install_property (object_class, PROP_LOCKED, g_param_spec_boolean ("locked", NULL, NULL, FALSE, GIMP_PARAM_READWRITE)); gtk_widget_class_install_style_property (widget_class, g_param_spec_int ("content-border", NULL, NULL, 0, G_MAXINT, 0, GIMP_PARAM_READABLE)); } static void gimp_dockable_init (GimpDockable *dockable) { dockable->p = gimp_dockable_get_instance_private (dockable); dockable->p->tab_style = GIMP_TAB_STYLE_PREVIEW; gtk_drag_dest_set (GTK_WIDGET (dockable), 0, dialog_target_table, G_N_ELEMENTS (dialog_target_table), GDK_ACTION_MOVE); } static void gimp_dockable_session_managed_iface_init (GimpSessionManagedInterface *iface) { iface->get_aux_info = gimp_dockable_get_aux_info; iface->set_aux_info = gimp_dockable_set_aux_info; } static void gimp_dockable_dispose (GObject *object) { GimpDockable *dockable = GIMP_DOCKABLE (object); if (dockable->p->context) gimp_dockable_set_context (dockable, NULL); g_clear_pointer (&dockable->p->blurb, g_free); g_clear_pointer (&dockable->p->name, g_free); g_clear_pointer (&dockable->p->icon_name, g_free); g_clear_pointer (&dockable->p->help_id, g_free); G_OBJECT_CLASS (parent_class)->dispose (object); } static void gimp_dockable_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec) { GimpDockable *dockable = GIMP_DOCKABLE (object); switch (property_id) { case PROP_LOCKED: gimp_dockable_set_locked (dockable, g_value_get_boolean (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); break; } } static void gimp_dockable_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec) { GimpDockable *dockable = GIMP_DOCKABLE (object); switch (property_id) { case PROP_LOCKED: g_value_set_boolean (value, dockable->p->locked); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); break; } } static void gimp_dockable_style_updated (GtkWidget *widget) { gint content_border; GTK_WIDGET_CLASS (parent_class)->style_updated (widget); gtk_widget_style_get (widget, "content-border", &content_border, NULL); gtk_container_set_border_width (GTK_CONTAINER (widget), content_border); } static void gimp_dockable_add (GtkContainer *container, GtkWidget *widget) { GimpDockable *dockable; g_return_if_fail (gtk_bin_get_child (GTK_BIN (container)) == NULL); GTK_CONTAINER_CLASS (parent_class)->add (container, widget); /* not all tab styles are supported by all children */ dockable = GIMP_DOCKABLE (container); gimp_dockable_set_tab_style (dockable, dockable->p->tab_style); } static GType gimp_dockable_child_type (GtkContainer *container) { if (gtk_bin_get_child (GTK_BIN (container))) return G_TYPE_NONE; return GIMP_TYPE_DOCKED; } static GtkWidget * gimp_dockable_new_tab_widget_internal (GimpDockable *dockable, GimpContext *context, GimpTabStyle tab_style, GtkIconSize size, gboolean dnd) { GtkWidget *tab_widget = NULL; GtkWidget *label = NULL; GtkWidget *icon = NULL; switch (tab_style) { case GIMP_TAB_STYLE_NAME: case GIMP_TAB_STYLE_ICON_NAME: case GIMP_TAB_STYLE_PREVIEW_NAME: label = gtk_label_new (dockable->p->name); break; case GIMP_TAB_STYLE_BLURB: case GIMP_TAB_STYLE_ICON_BLURB: case GIMP_TAB_STYLE_PREVIEW_BLURB: label = gtk_label_new (dockable->p->blurb); break; default: break; } switch (tab_style) { case GIMP_TAB_STYLE_ICON: case GIMP_TAB_STYLE_ICON_NAME: case GIMP_TAB_STYLE_ICON_BLURB: icon = gimp_dockable_get_icon (dockable, size); break; case GIMP_TAB_STYLE_PREVIEW: case GIMP_TAB_STYLE_PREVIEW_NAME: case GIMP_TAB_STYLE_PREVIEW_BLURB: { GtkWidget *child = gtk_bin_get_child (GTK_BIN (dockable)); if (child) icon = gimp_docked_get_preview (GIMP_DOCKED (child), context, size); if (! icon) icon = gimp_dockable_get_icon (dockable, size); } break; default: break; } if (label && dnd) gimp_label_set_attributes (GTK_LABEL (label), PANGO_ATTR_WEIGHT, PANGO_WEIGHT_SEMIBOLD, -1); switch (tab_style) { case GIMP_TAB_STYLE_ICON: case GIMP_TAB_STYLE_PREVIEW: tab_widget = icon; break; case GIMP_TAB_STYLE_NAME: case GIMP_TAB_STYLE_BLURB: tab_widget = label; break; case GIMP_TAB_STYLE_ICON_NAME: case GIMP_TAB_STYLE_ICON_BLURB: case GIMP_TAB_STYLE_PREVIEW_NAME: case GIMP_TAB_STYLE_PREVIEW_BLURB: tab_widget = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, dnd ? 6 : 2); gtk_box_pack_start (GTK_BOX (tab_widget), icon, FALSE, FALSE, 0); gtk_widget_show (icon); gtk_box_pack_start (GTK_BOX (tab_widget), label, FALSE, FALSE, 0); gtk_widget_show (label); break; } return tab_widget; } /* public functions */ GtkWidget * gimp_dockable_new (const gchar *name, const gchar *blurb, const gchar *icon_name, const gchar *help_id) { GimpDockable *dockable; g_return_val_if_fail (name != NULL, NULL); g_return_val_if_fail (icon_name != NULL, NULL); g_return_val_if_fail (help_id != NULL, NULL); dockable = g_object_new (GIMP_TYPE_DOCKABLE, NULL); dockable->p->name = g_strdup (name); dockable->p->icon_name = g_strdup (icon_name); dockable->p->help_id = g_strdup (help_id); if (blurb) dockable->p->blurb = g_strdup (blurb); else dockable->p->blurb = g_strdup (dockable->p->name); gimp_help_set_help_data (GTK_WIDGET (dockable), NULL, help_id); return GTK_WIDGET (dockable); } void gimp_dockable_set_dockbook (GimpDockable *dockable, GimpDockbook *dockbook) { g_return_if_fail (GIMP_IS_DOCKABLE (dockable)); g_return_if_fail (dockbook == NULL || GIMP_IS_DOCKBOOK (dockbook)); dockable->p->dockbook = dockbook; } GimpDockbook * gimp_dockable_get_dockbook (GimpDockable *dockable) { g_return_val_if_fail (GIMP_IS_DOCKABLE (dockable), NULL); return dockable->p->dockbook; } void gimp_dockable_set_tab_style (GimpDockable *dockable, GimpTabStyle tab_style) { GtkWidget *child; g_return_if_fail (GIMP_IS_DOCKABLE (dockable)); child = gtk_bin_get_child (GTK_BIN (dockable)); if (child && ! GIMP_DOCKED_GET_IFACE (child)->get_preview) tab_style = gimp_preview_tab_style_to_icon (tab_style); dockable->p->tab_style = tab_style; } GimpTabStyle gimp_dockable_get_tab_style (GimpDockable *dockable) { g_return_val_if_fail (GIMP_IS_DOCKABLE (dockable), -1); return dockable->p->tab_style; } void gimp_dockable_set_locked (GimpDockable *dockable, gboolean lock) { g_return_if_fail (GIMP_IS_DOCKABLE (dockable)); if (dockable->p->locked != lock) { dockable->p->locked = lock ? TRUE : FALSE; g_object_notify (G_OBJECT (dockable), "locked"); } } gboolean gimp_dockable_get_locked (GimpDockable *dockable) { g_return_val_if_fail (GIMP_IS_DOCKABLE (dockable), FALSE); return dockable->p->locked; } const gchar * gimp_dockable_get_name (GimpDockable *dockable) { g_return_val_if_fail (GIMP_IS_DOCKABLE (dockable), NULL); return dockable->p->name; } const gchar * gimp_dockable_get_blurb (GimpDockable *dockable) { g_return_val_if_fail (GIMP_IS_DOCKABLE (dockable), NULL); return dockable->p->blurb; } const gchar * gimp_dockable_get_help_id (GimpDockable *dockable) { g_return_val_if_fail (GIMP_IS_DOCKABLE (dockable), NULL); return dockable->p->help_id; } const gchar * gimp_dockable_get_icon_name (GimpDockable *dockable) { g_return_val_if_fail (GIMP_IS_DOCKABLE (dockable), NULL); return dockable->p->icon_name; } GtkWidget * gimp_dockable_get_icon (GimpDockable *dockable, GtkIconSize size) { g_return_val_if_fail (GIMP_IS_DOCKABLE (dockable), NULL); return gtk_image_new_from_icon_name (dockable->p->icon_name, size); } GtkWidget * gimp_dockable_create_tab_widget (GimpDockable *dockable, GimpContext *context, GimpTabStyle tab_style, GtkIconSize size) { g_return_val_if_fail (GIMP_IS_DOCKABLE (dockable), NULL); g_return_val_if_fail (GIMP_IS_CONTEXT (context), NULL); return gimp_dockable_new_tab_widget_internal (dockable, context, tab_style, size, FALSE); } void gimp_dockable_set_context (GimpDockable *dockable, GimpContext *context) { g_return_if_fail (GIMP_IS_DOCKABLE (dockable)); g_return_if_fail (context == NULL || GIMP_IS_CONTEXT (context)); if (context != dockable->p->context) { GtkWidget *child = gtk_bin_get_child (GTK_BIN (dockable)); if (child) gimp_docked_set_context (GIMP_DOCKED (child), context); dockable->p->context = context; } } GimpUIManager * gimp_dockable_get_menu (GimpDockable *dockable, const gchar **ui_path, gpointer *popup_data) { GtkWidget *child; g_return_val_if_fail (GIMP_IS_DOCKABLE (dockable), NULL); g_return_val_if_fail (ui_path != NULL, NULL); g_return_val_if_fail (popup_data != NULL, NULL); child = gtk_bin_get_child (GTK_BIN (dockable)); if (child) return gimp_docked_get_menu (GIMP_DOCKED (child), ui_path, popup_data); return NULL; } void gimp_dockable_detach (GimpDockable *dockable) { GimpDialogFactory *dialog_factory; GimpMenuFactory *menu_factory; GimpDockWindow *src_dock_window; GimpDock *src_dock; GtkWidget *dock; GimpDockWindow *dock_window; GtkWidget *dockbook; g_return_if_fail (GIMP_IS_DOCKABLE (dockable)); g_return_if_fail (GIMP_IS_DOCKBOOK (dockable->p->dockbook)); src_dock = gimp_dockbook_get_dock (dockable->p->dockbook); src_dock_window = gimp_dock_window_from_dock (src_dock); dialog_factory = gimp_dock_get_dialog_factory (src_dock); menu_factory = menus_get_global_menu_factory (gimp_dialog_factory_get_context (dialog_factory)->gimp); dock = gimp_dock_with_window_new (dialog_factory, gimp_widget_get_monitor (GTK_WIDGET (dockable)), FALSE /*toolbox*/); dock_window = gimp_dock_window_from_dock (GIMP_DOCK (dock)); gtk_window_set_position (GTK_WINDOW (dock_window), GTK_WIN_POS_MOUSE); if (src_dock_window) gimp_dock_window_setup (dock_window, src_dock_window); dockbook = gimp_dockbook_new (menu_factory); gimp_dock_add_book (GIMP_DOCK (dock), GIMP_DOCKBOOK (dockbook), 0); g_object_ref (dockable); gtk_container_remove (GTK_CONTAINER (dockable->p->dockbook), GTK_WIDGET (dockable)); gtk_notebook_append_page (GTK_NOTEBOOK (dockbook), GTK_WIDGET (dockable), NULL); g_object_unref (dockable); gtk_widget_show (GTK_WIDGET (dock_window)); gtk_widget_show (dock); } /* private functions */ static GList * gimp_dockable_get_aux_info (GimpSessionManaged *session_managed) { GimpDockable *dockable; GtkWidget *child; g_return_val_if_fail (GIMP_IS_DOCKABLE (session_managed), NULL); dockable = GIMP_DOCKABLE (session_managed); child = gtk_bin_get_child (GTK_BIN (dockable)); if (child) return gimp_docked_get_aux_info (GIMP_DOCKED (child)); return NULL; } static void gimp_dockable_set_aux_info (GimpSessionManaged *session_managed, GList *aux_info) { GimpDockable *dockable; GtkWidget *child; g_return_if_fail (GIMP_IS_DOCKABLE (session_managed)); dockable = GIMP_DOCKABLE (session_managed); child = gtk_bin_get_child (GTK_BIN (dockable)); if (child) gimp_docked_set_aux_info (GIMP_DOCKED (child), aux_info); }