summaryrefslogtreecommitdiffstats
path: root/app/widgets/gimpdockbook.c
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-27 16:23:22 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-27 16:23:22 +0000
commite42129241681dde7adae7d20697e7b421682fbb4 (patch)
treeaf1fe815a5e639e68e59fabd8395ec69458b3e5e /app/widgets/gimpdockbook.c
parentInitial commit. (diff)
downloadgimp-e42129241681dde7adae7d20697e7b421682fbb4.tar.xz
gimp-e42129241681dde7adae7d20697e7b421682fbb4.zip
Adding upstream version 2.10.22.upstream/2.10.22upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'app/widgets/gimpdockbook.c')
-rw-r--r--app/widgets/gimpdockbook.c1846
1 files changed, 1846 insertions, 0 deletions
diff --git a/app/widgets/gimpdockbook.c b/app/widgets/gimpdockbook.c
new file mode 100644
index 0000000..d4d5208
--- /dev/null
+++ b/app/widgets/gimpdockbook.c
@@ -0,0 +1,1846 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995 Spencer Kimball and Peter Mattis
+ *
+ * gimpdockbook.c
+ * Copyright (C) 2001-2007 Michael Natterer <mitch@gimp.org>
+ *
+ * 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 <gegl.h>
+#include <gtk/gtk.h>
+
+#include "libgimpwidgets/gimpwidgets.h"
+
+#include "widgets-types.h"
+
+#include "config/gimpguiconfig.h"
+
+#include "core/gimp.h"
+#include "core/gimpcontext.h"
+#include "core/gimpmarshal.h"
+
+#include "gimpactiongroup.h"
+#include "gimpdialogfactory.h"
+#include "gimpdnd.h"
+#include "gimpdock.h"
+#include "gimpdockable.h"
+#include "gimpdockbook.h"
+#include "gimpdocked.h"
+#include "gimpdockcontainer.h"
+#include "gimpdockwindow.h"
+#include "gimphelp-ids.h"
+#include "gimpmenufactory.h"
+#include "gimppanedbox.h"
+#include "gimpstringaction.h"
+#include "gimpuimanager.h"
+#include "gimpview.h"
+#include "gimpwidgets-utils.h"
+
+#include "gimp-log.h"
+#include "gimp-intl.h"
+
+#define DEFAULT_TAB_BORDER 0
+#define DEFAULT_TAB_ICON_SIZE GTK_ICON_SIZE_BUTTON
+#define DND_WIDGET_ICON_SIZE GTK_ICON_SIZE_BUTTON
+#define MENU_WIDGET_ICON_SIZE GTK_ICON_SIZE_MENU
+#define MENU_WIDGET_SPACING 4
+#define TAB_HOVER_TIMEOUT 500
+#define GIMP_DOCKABLE_DETACH_REF_KEY "gimp-dockable-detach-ref"
+
+
+enum
+{
+ DOCKABLE_ADDED,
+ DOCKABLE_REMOVED,
+ DOCKABLE_REORDERED,
+ LAST_SIGNAL
+};
+
+/* List of candidates for the automatic style, starting with the
+ * biggest first
+ */
+static const GimpTabStyle gimp_tab_style_candidates[] =
+{
+ GIMP_TAB_STYLE_PREVIEW_BLURB,
+ GIMP_TAB_STYLE_PREVIEW_NAME,
+ GIMP_TAB_STYLE_PREVIEW
+};
+
+
+typedef struct
+{
+ GimpDockbookDragCallback callback;
+ gpointer data;
+} GimpDockbookDragCallbackData;
+
+struct _GimpDockbookPrivate
+{
+ GimpDock *dock;
+ GimpUIManager *ui_manager;
+
+ guint tab_hover_timeout;
+ GimpDockable *tab_hover_dockable;
+
+ GimpPanedBox *drag_handler;
+
+ /* Cache for "what actual tab style for automatic styles can we use
+ * for a given dockbook width
+ */
+ gint min_width_for_style[G_N_ELEMENTS (gimp_tab_style_candidates)];
+
+ /* We need a list separate from the GtkContainer children list,
+ * because we need to do calculations for all dockables before we
+ * can add a dockable as a child, namely automatic tab style
+ * calculations
+ */
+ GList *dockables;
+
+ GtkWidget *menu_button;
+};
+
+
+static void gimp_dockbook_dispose (GObject *object);
+static void gimp_dockbook_finalize (GObject *object);
+static void gimp_dockbook_size_allocate (GtkWidget *widget,
+ GtkAllocation *allocation);
+static void gimp_dockbook_style_set (GtkWidget *widget,
+ GtkStyle *prev_style);
+static void gimp_dockbook_drag_leave (GtkWidget *widget,
+ GdkDragContext *context,
+ guint time);
+static gboolean gimp_dockbook_drag_motion (GtkWidget *widget,
+ GdkDragContext *context,
+ gint x,
+ gint y,
+ guint time);
+static gboolean gimp_dockbook_drag_drop (GtkWidget *widget,
+ GdkDragContext *context,
+ gint x,
+ gint y,
+ guint time);
+static gboolean gimp_dockbook_popup_menu (GtkWidget *widget);
+static gboolean gimp_dockbook_menu_button_press (GimpDockbook *dockbook,
+ GdkEventButton *bevent,
+ GtkWidget *button);
+static gboolean gimp_dockbook_show_menu (GimpDockbook *dockbook);
+static void gimp_dockbook_menu_end (GimpDockable *dockable);
+static void gimp_dockbook_dockable_added (GimpDockbook *dockbook,
+ GimpDockable *dockable);
+static void gimp_dockbook_dockable_removed (GimpDockbook *dockbook,
+ GimpDockable *dockable);
+static void gimp_dockbook_recreate_tab_widgets (GimpDockbook *dockbook,
+ gboolean only_auto);
+static void gimp_dockbook_tab_drag_source_setup (GtkWidget *widget,
+ GimpDockable *dockable);
+static void gimp_dockbook_tab_drag_begin (GtkWidget *widget,
+ GdkDragContext *context,
+ GimpDockable *dockable);
+static void gimp_dockbook_tab_drag_end (GtkWidget *widget,
+ GdkDragContext *context,
+ GimpDockable *dockable);
+static gboolean gimp_dockbook_tab_drag_failed (GtkWidget *widget,
+ GdkDragContext *context,
+ GtkDragResult result,
+ GimpDockable *dockable);
+static void gimp_dockbook_tab_drag_leave (GtkWidget *widget,
+ GdkDragContext *context,
+ guint time,
+ GimpDockable *dockable);
+static gboolean gimp_dockbook_tab_drag_motion (GtkWidget *widget,
+ GdkDragContext *context,
+ gint x,
+ gint y,
+ guint time,
+ GimpDockable *dockable);
+static gboolean gimp_dockbook_tab_drag_drop (GtkWidget *widget,
+ GdkDragContext *context,
+ gint x,
+ gint y,
+ guint time);
+static GimpTabStyle gimp_dockbook_tab_style_to_preferred (GimpTabStyle tab_style,
+ GimpDockable *dockable);
+static void gimp_dockbook_refresh_tab_layout_lut (GimpDockbook *dockbook);
+static void gimp_dockbook_update_automatic_tab_style (GimpDockbook *dockbook);
+static GtkWidget * gimp_dockable_create_event_box_tab_widget (GimpDockable *dockable,
+ GimpContext *context,
+ GimpTabStyle tab_style,
+ GtkIconSize size);
+static GtkIconSize gimp_dockbook_get_tab_icon_size (GimpDockbook *dockbook);
+static gint gimp_dockbook_get_tab_border (GimpDockbook *dockbook);
+static void gimp_dockbook_add_tab_timeout (GimpDockbook *dockbook,
+ GimpDockable *dockable);
+static void gimp_dockbook_remove_tab_timeout (GimpDockbook *dockbook);
+static gboolean gimp_dockbook_tab_timeout (GimpDockbook *dockbook);
+static void gimp_dockbook_tab_locked_notify (GimpDockable *dockable,
+ GParamSpec *pspec,
+ GimpDockbook *dockbook);
+static void gimp_dockbook_help_func (const gchar *help_id,
+ gpointer help_data);
+static const gchar *gimp_dockbook_get_tab_style_name (GimpTabStyle tab_style);
+
+static void gimp_dockbook_config_size_changed (GimpGuiConfig *config,
+ GimpDockbook *dockbook);
+
+
+G_DEFINE_TYPE_WITH_PRIVATE (GimpDockbook, gimp_dockbook, GTK_TYPE_NOTEBOOK)
+
+#define parent_class gimp_dockbook_parent_class
+
+static guint dockbook_signals[LAST_SIGNAL] = { 0 };
+
+static const GtkTargetEntry dialog_target_table[] = { GIMP_TARGET_DIALOG };
+
+static GList *drag_callbacks = NULL;
+
+
+static void
+gimp_dockbook_class_init (GimpDockbookClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+ GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
+
+ dockbook_signals[DOCKABLE_ADDED] =
+ g_signal_new ("dockable-added",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_FIRST,
+ G_STRUCT_OFFSET (GimpDockbookClass, dockable_added),
+ NULL, NULL,
+ gimp_marshal_VOID__OBJECT,
+ G_TYPE_NONE, 1,
+ GIMP_TYPE_DOCKABLE);
+
+ dockbook_signals[DOCKABLE_REMOVED] =
+ g_signal_new ("dockable-removed",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_FIRST,
+ G_STRUCT_OFFSET (GimpDockbookClass, dockable_removed),
+ NULL, NULL,
+ gimp_marshal_VOID__OBJECT,
+ G_TYPE_NONE, 1,
+ GIMP_TYPE_DOCKABLE);
+
+ dockbook_signals[DOCKABLE_REORDERED] =
+ g_signal_new ("dockable-reordered",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_FIRST,
+ G_STRUCT_OFFSET (GimpDockbookClass, dockable_reordered),
+ NULL, NULL,
+ gimp_marshal_VOID__OBJECT,
+ G_TYPE_NONE, 1,
+ GIMP_TYPE_DOCKABLE);
+
+ object_class->dispose = gimp_dockbook_dispose;
+ object_class->finalize = gimp_dockbook_finalize;
+
+ widget_class->size_allocate = gimp_dockbook_size_allocate;
+ widget_class->style_set = gimp_dockbook_style_set;
+ widget_class->drag_leave = gimp_dockbook_drag_leave;
+ widget_class->drag_motion = gimp_dockbook_drag_motion;
+ widget_class->drag_drop = gimp_dockbook_drag_drop;
+ widget_class->popup_menu = gimp_dockbook_popup_menu;
+
+ klass->dockable_added = gimp_dockbook_dockable_added;
+ klass->dockable_removed = gimp_dockbook_dockable_removed;
+ klass->dockable_reordered = NULL;
+
+ gtk_widget_class_install_style_property (widget_class,
+ g_param_spec_int ("tab-border",
+ NULL, NULL,
+ 0, G_MAXINT,
+ DEFAULT_TAB_BORDER,
+ GIMP_PARAM_READABLE));
+ gtk_widget_class_install_style_property (widget_class,
+ g_param_spec_enum ("tab-icon-size",
+ NULL, NULL,
+ GTK_TYPE_ICON_SIZE,
+ DEFAULT_TAB_ICON_SIZE,
+ GIMP_PARAM_READABLE));
+}
+
+static void
+gimp_dockbook_init (GimpDockbook *dockbook)
+{
+ GtkNotebook *notebook = GTK_NOTEBOOK (dockbook);
+ GtkWidget *image = NULL;
+
+ dockbook->p = gimp_dockbook_get_instance_private (dockbook);
+
+ /* Various init */
+ gtk_notebook_popup_enable (notebook);
+ gtk_notebook_set_scrollable (notebook, TRUE);
+ gtk_notebook_set_show_border (notebook, FALSE);
+ gtk_notebook_set_show_tabs (notebook, TRUE);
+
+ gtk_drag_dest_set (GTK_WIDGET (dockbook),
+ 0,
+ dialog_target_table, G_N_ELEMENTS (dialog_target_table),
+ GDK_ACTION_MOVE);
+
+ /* Menu button */
+ dockbook->p->menu_button = gtk_button_new ();
+ gtk_widget_set_can_focus (dockbook->p->menu_button, FALSE);
+ gtk_button_set_relief (GTK_BUTTON (dockbook->p->menu_button),
+ GTK_RELIEF_NONE);
+ gtk_notebook_set_action_widget (notebook,
+ dockbook->p->menu_button,
+ GTK_PACK_END);
+ gtk_widget_show (dockbook->p->menu_button);
+
+ image = gtk_image_new_from_icon_name (GIMP_ICON_MENU_LEFT,
+ GTK_ICON_SIZE_MENU);
+ gtk_image_set_pixel_size (GTK_IMAGE (image), 12);
+ gtk_image_set_from_icon_name (GTK_IMAGE (image), GIMP_ICON_MENU_LEFT,
+ GTK_ICON_SIZE_MENU);
+ gtk_container_add (GTK_CONTAINER (dockbook->p->menu_button), image);
+ gtk_widget_show (image);
+
+ gimp_help_set_help_data (dockbook->p->menu_button, _("Configure this tab"),
+ GIMP_HELP_DOCK_TAB_MENU);
+
+ g_signal_connect_swapped (dockbook->p->menu_button, "button-press-event",
+ G_CALLBACK (gimp_dockbook_menu_button_press),
+ dockbook);
+}
+
+static void
+gimp_dockbook_dispose (GObject *object)
+{
+ GimpDockbook *dockbook = GIMP_DOCKBOOK (object);
+
+ g_signal_handlers_disconnect_by_func (dockbook->p->ui_manager->gimp->config,
+ gimp_dockbook_config_size_changed,
+ dockbook);
+
+ gimp_dockbook_remove_tab_timeout (dockbook);
+
+ while (dockbook->p->dockables)
+ {
+ GimpDockable *dockable = dockbook->p->dockables->data;
+
+ g_object_ref (dockable);
+ gimp_dockbook_remove (dockbook, dockable);
+ gtk_widget_destroy (GTK_WIDGET (dockable));
+ g_object_unref (dockable);
+ }
+
+ G_OBJECT_CLASS (parent_class)->dispose (object);
+}
+
+static void
+gimp_dockbook_finalize (GObject *object)
+{
+ GimpDockbook *dockbook = GIMP_DOCKBOOK (object);
+
+ g_clear_object (&dockbook->p->ui_manager);
+
+ G_OBJECT_CLASS (parent_class)->finalize (object);
+}
+
+static void
+gimp_dockbook_size_allocate (GtkWidget *widget,
+ GtkAllocation *allocation)
+{
+ GimpDockbook *dockbook = GIMP_DOCKBOOK (widget);
+
+ GTK_WIDGET_CLASS (parent_class)->size_allocate (widget, allocation);
+
+ /* Update tab styles, also recreates if changed */
+ gimp_dockbook_update_automatic_tab_style (dockbook);
+}
+
+static void
+gimp_dockbook_style_set (GtkWidget *widget,
+ GtkStyle *prev_style)
+{
+ GTK_WIDGET_CLASS (parent_class)->style_set (widget, prev_style);
+
+ /* Don't attempt to construct widgets that require a GimpContext if
+ * we are detached from a top-level, we're either on our way to
+ * destruction, in which case we don't care, or we will be given a
+ * new parent, in which case the widget style will be reset again
+ * anyway, i.e. this function will be called again
+ */
+ if (! gtk_widget_is_toplevel (gtk_widget_get_toplevel (widget)))
+ return;
+
+ gimp_dockbook_recreate_tab_widgets (GIMP_DOCKBOOK (widget),
+ FALSE /*only_auto*/);
+}
+
+static void
+gimp_dockbook_drag_leave (GtkWidget *widget,
+ GdkDragContext *context,
+ guint time)
+{
+ gimp_highlight_widget (widget, FALSE);
+}
+
+static gboolean
+gimp_dockbook_drag_motion (GtkWidget *widget,
+ GdkDragContext *context,
+ gint x,
+ gint y,
+ guint time)
+{
+ GimpDockbook *dockbook = GIMP_DOCKBOOK (widget);
+
+ if (gimp_paned_box_will_handle_drag (dockbook->p->drag_handler,
+ widget,
+ context,
+ x, y,
+ time))
+ {
+ gdk_drag_status (context, 0, time);
+ gimp_highlight_widget (widget, FALSE);
+
+ return FALSE;
+ }
+
+ gdk_drag_status (context, GDK_ACTION_MOVE, time);
+ gimp_highlight_widget (widget, TRUE);
+
+ /* Return TRUE so drag_leave() is called */
+ return TRUE;
+}
+
+static gboolean
+gimp_dockbook_drag_drop (GtkWidget *widget,
+ GdkDragContext *context,
+ gint x,
+ gint y,
+ guint time)
+{
+ GimpDockbook *dockbook = GIMP_DOCKBOOK (widget);
+ gboolean dropped;
+
+ if (gimp_paned_box_will_handle_drag (dockbook->p->drag_handler,
+ widget,
+ context,
+ x, y,
+ time))
+ {
+ return FALSE;
+ }
+
+ dropped = gimp_dockbook_drop_dockable (dockbook,
+ gtk_drag_get_source_widget (context));
+
+ gtk_drag_finish (context, dropped, TRUE, time);
+
+ return TRUE;
+}
+
+static gboolean
+gimp_dockbook_popup_menu (GtkWidget *widget)
+{
+ return gimp_dockbook_show_menu (GIMP_DOCKBOOK (widget));
+}
+
+static gboolean
+gimp_dockbook_menu_button_press (GimpDockbook *dockbook,
+ GdkEventButton *bevent,
+ GtkWidget *button)
+{
+ gboolean handled = FALSE;
+
+ if (bevent->button == 1 && bevent->type == GDK_BUTTON_PRESS)
+ handled = gimp_dockbook_show_menu (dockbook);
+
+ return handled;
+}
+
+static void
+gimp_dockbook_menu_position (GtkMenu *menu,
+ gint *x,
+ gint *y,
+ gpointer data)
+{
+ GimpDockbook *dockbook = GIMP_DOCKBOOK (data);
+
+ gimp_button_menu_position (dockbook->p->menu_button, menu, GTK_POS_LEFT, x, y);
+}
+
+static gboolean
+gimp_dockbook_show_menu (GimpDockbook *dockbook)
+{
+ GimpUIManager *dockbook_ui_manager;
+ GimpUIManager *dialog_ui_manager;
+ const gchar *dialog_ui_path;
+ gpointer dialog_popup_data;
+ GtkWidget *parent_menu_widget;
+ GimpAction *parent_menu_action;
+ GimpDockable *dockable;
+ gint page_num;
+
+ dockbook_ui_manager = gimp_dockbook_get_ui_manager (dockbook);
+
+ if (! dockbook_ui_manager)
+ return FALSE;
+
+ parent_menu_widget =
+ gimp_ui_manager_get_widget (dockbook_ui_manager,
+ "/dockable-popup/dockable-menu");
+ parent_menu_action =
+ gimp_ui_manager_get_action (dockbook_ui_manager,
+ "/dockable-popup/dockable-menu");
+
+ if (! parent_menu_widget || ! parent_menu_action)
+ return FALSE;
+
+ page_num = gtk_notebook_get_current_page (GTK_NOTEBOOK (dockbook));
+ dockable = GIMP_DOCKABLE (gtk_notebook_get_nth_page (GTK_NOTEBOOK (dockbook),
+ page_num));
+
+ if (! dockable)
+ return FALSE;
+
+ dialog_ui_manager = gimp_dockable_get_menu (dockable,
+ &dialog_ui_path,
+ &dialog_popup_data);
+
+ if (dialog_ui_manager && dialog_ui_path)
+ {
+ GtkWidget *child_menu_widget;
+ GimpAction *child_menu_action;
+ gchar *label;
+
+ child_menu_widget =
+ gimp_ui_manager_get_widget (dialog_ui_manager, dialog_ui_path);
+
+ if (! child_menu_widget)
+ {
+ g_warning ("%s: UI manager '%s' has no widget at path '%s'",
+ G_STRFUNC, dialog_ui_manager->name, dialog_ui_path);
+ return FALSE;
+ }
+
+ child_menu_action =
+ gimp_ui_manager_get_action (dialog_ui_manager,
+ dialog_ui_path);
+
+ if (! child_menu_action)
+ {
+ g_warning ("%s: UI manager '%s' has no action at path '%s'",
+ G_STRFUNC, dialog_ui_manager->name, dialog_ui_path);
+ return FALSE;
+ }
+
+ g_object_get (child_menu_action,
+ "label", &label,
+ NULL);
+
+ g_object_set (parent_menu_action,
+ "label", label,
+ "icon-name", gimp_dockable_get_icon_name (dockable),
+ "visible", TRUE,
+ NULL);
+
+ g_free (label);
+
+ if (! GTK_IS_MENU (child_menu_widget))
+ {
+ g_warning ("%s: child_menu_widget (%p) is not a GtkMenu",
+ G_STRFUNC, child_menu_widget);
+ return FALSE;
+ }
+
+ {
+ GtkWidget *image = gimp_dockable_get_icon (dockable,
+ GTK_ICON_SIZE_MENU);
+
+ gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (parent_menu_widget),
+ image);
+ gtk_image_menu_item_set_always_show_image (GTK_IMAGE_MENU_ITEM (parent_menu_widget),
+ TRUE);
+ gtk_widget_show (image);
+ }
+
+ gtk_menu_item_set_submenu (GTK_MENU_ITEM (parent_menu_widget),
+ child_menu_widget);
+
+ gimp_ui_manager_update (dialog_ui_manager, dialog_popup_data);
+ }
+ else
+ {
+ g_object_set (parent_menu_action, "visible", FALSE, NULL);
+ }
+
+ /* an action callback may destroy both dockable and dockbook, so
+ * reference them for gimp_dockbook_menu_end()
+ */
+ g_object_ref (dockable);
+ g_object_set_data_full (G_OBJECT (dockable), GIMP_DOCKABLE_DETACH_REF_KEY,
+ g_object_ref (dockbook),
+ g_object_unref);
+
+ gimp_ui_manager_update (dockbook_ui_manager, dockable);
+ gimp_ui_manager_ui_popup (dockbook_ui_manager, "/dockable-popup",
+ GTK_WIDGET (dockable),
+ gimp_dockbook_menu_position, dockbook,
+ (GDestroyNotify) gimp_dockbook_menu_end, dockable);
+
+ return TRUE;
+}
+
+static void
+gimp_dockbook_menu_end (GimpDockable *dockable)
+{
+ GimpUIManager *dialog_ui_manager;
+ const gchar *dialog_ui_path;
+ gpointer dialog_popup_data;
+
+ dialog_ui_manager = gimp_dockable_get_menu (dockable,
+ &dialog_ui_path,
+ &dialog_popup_data);
+
+ if (dialog_ui_manager && dialog_ui_path)
+ {
+ GtkWidget *child_menu_widget =
+ gimp_ui_manager_get_widget (dialog_ui_manager, dialog_ui_path);
+
+ if (child_menu_widget)
+ gtk_menu_detach (GTK_MENU (child_menu_widget));
+ }
+
+ /* release gimp_dockbook_show_menu()'s references */
+ g_object_set_data (G_OBJECT (dockable), GIMP_DOCKABLE_DETACH_REF_KEY, NULL);
+ g_object_unref (dockable);
+}
+
+static void
+gimp_dockbook_dockable_added (GimpDockbook *dockbook,
+ GimpDockable *dockable)
+{
+ gtk_notebook_set_current_page (GTK_NOTEBOOK (dockbook),
+ gtk_notebook_page_num (GTK_NOTEBOOK (dockbook),
+ GTK_WIDGET (dockable)));
+}
+
+static void
+gimp_dockbook_dockable_removed (GimpDockbook *dockbook,
+ GimpDockable *dockable)
+{
+}
+
+/**
+ * gimp_dockbook_get_dockable_tab_width:
+ * @dockable:
+ * @tab_style:
+ *
+ * Returns: Width of tab when the dockable is using the specified tab
+ * style.
+ **/
+static gint
+gimp_dockbook_get_dockable_tab_width (GimpDockbook *dockbook,
+ GimpDockable *dockable,
+ GimpTabStyle tab_style)
+{
+ GtkRequisition dockable_request;
+ GtkWidget *tab_widget;
+
+ tab_widget =
+ gimp_dockable_create_event_box_tab_widget (dockable,
+ gimp_dock_get_context (dockbook->p->dock),
+ tab_style,
+ gimp_dockbook_get_tab_icon_size (dockbook));
+
+ /* So font-scale is applied. We can't apply styles without having a
+ * GdkScreen :(
+ */
+ gimp_dock_temp_add (dockbook->p->dock, tab_widget);
+
+ gtk_widget_size_request (tab_widget, &dockable_request);
+
+ /* Also destroys the widget */
+ gimp_dock_temp_remove (dockbook->p->dock, tab_widget);
+
+ return dockable_request.width;
+}
+
+/**
+ * gimp_dockbook_tab_style_to_preferred:
+ * @tab_style:
+ * @dockable:
+ *
+ * The list of tab styles to try in automatic mode only consists of
+ * preview styles. For some dockables, like the tool options dockable,
+ * we rather want to use the icon tab styles for the automatic
+ * mode. This function is used to convert tab styles for such
+ * dockables.
+ *
+ * Returns: An icon tab style if the dockable prefers icon tab styles
+ * in automatic mode.
+ **/
+static GimpTabStyle
+gimp_dockbook_tab_style_to_preferred (GimpTabStyle tab_style,
+ GimpDockable *dockable)
+{
+ GimpDocked *docked = GIMP_DOCKED (gtk_bin_get_child (GTK_BIN (dockable)));
+
+ if (gimp_docked_get_prefer_icon (docked))
+ tab_style = gimp_preview_tab_style_to_icon (tab_style);
+
+ return tab_style;
+}
+
+/**
+ * gimp_dockbook_refresh_tab_layout_lut:
+ * @dockbook:
+ *
+ * For each given set of tab widgets, there is a fixed mapping between
+ * the width of the dockbook and the actual tab style to use for auto
+ * tab widgets. This function refreshes that look-up table.
+ **/
+static void
+gimp_dockbook_refresh_tab_layout_lut (GimpDockbook *dockbook)
+{
+ GList *auto_dockables = NULL;
+ GList *iter = NULL;
+ gint fixed_tab_style_space = 0;
+ int i = 0;
+
+ /* Calculate space taken by dockables with fixed tab styles */
+ fixed_tab_style_space = 0;
+ for (iter = dockbook->p->dockables; iter; iter = g_list_next (iter))
+ {
+ GimpDockable *dockable = GIMP_DOCKABLE (iter->data);
+ GimpTabStyle tab_style = gimp_dockable_get_tab_style (dockable);
+
+ if (tab_style == GIMP_TAB_STYLE_AUTOMATIC)
+ auto_dockables = g_list_prepend (auto_dockables, dockable);
+ else
+ fixed_tab_style_space +=
+ gimp_dockbook_get_dockable_tab_width (dockbook,
+ dockable,
+ tab_style);
+ }
+
+ /* Calculate space taken with auto tab style for all candidates */
+ for (i = 0; i < G_N_ELEMENTS (gimp_tab_style_candidates); i++)
+ {
+ gint size_with_candidate = 0;
+ GimpTabStyle candidate = gimp_tab_style_candidates[i];
+
+ for (iter = auto_dockables; iter; iter = g_list_next (iter))
+ {
+ GimpDockable *dockable = GIMP_DOCKABLE (iter->data);
+ GimpTabStyle style_to_use;
+
+ style_to_use = gimp_dockbook_tab_style_to_preferred (candidate,
+ dockable);
+ size_with_candidate +=
+ gimp_dockbook_get_dockable_tab_width (dockbook,
+ dockable,
+ style_to_use);
+ }
+
+ dockbook->p->min_width_for_style[i] =
+ fixed_tab_style_space + size_with_candidate;
+
+ GIMP_LOG (AUTO_TAB_STYLE, "Total tab space taken for auto tab style %s = %d",
+ gimp_dockbook_get_tab_style_name (candidate),
+ dockbook->p->min_width_for_style[i]);
+ }
+
+ g_list_free (auto_dockables);
+}
+
+/**
+ * gimp_dockbook_update_automatic_tab_style:
+ * @dockbook:
+ *
+ * Based on widget allocation, sets actual tab style for dockables
+ * with automatic tab styles. Takes care of recreating tab widgets if
+ * necessary.
+ **/
+static void
+gimp_dockbook_update_automatic_tab_style (GimpDockbook *dockbook)
+{
+ GtkWidget *widget = GTK_WIDGET (dockbook);
+ gboolean changed = FALSE;
+ GList *iter = NULL;
+ GtkAllocation dockbook_allocation = { 0, };
+ GtkAllocation button_allocation = { 0, };
+ GimpTabStyle tab_style = 0;
+ int i = 0;
+ gint available_space = 0;
+ guint tab_hborder = 0;
+ gint xthickness = 0;
+ gint tab_curvature = 0;
+ gint focus_width = 0;
+ gint tab_overlap = 0;
+ gint tab_padding = 0;
+ gint border_loss = 0;
+ gint action_widget_size = 0;
+
+ xthickness = gtk_widget_get_style (widget)->xthickness;
+ g_object_get (widget,
+ "tab-hborder", &tab_hborder,
+ NULL);
+ gtk_widget_style_get (widget,
+ "tab-curvature", &tab_curvature,
+ "focus-line-width", &focus_width,
+ "tab-overlap", &tab_overlap,
+ NULL);
+ gtk_widget_get_allocation (dockbook->p->menu_button,
+ &button_allocation);
+
+ /* Calculate available space. Based on code in GTK+ internal
+ * functions gtk_notebook_size_request() and
+ * gtk_notebook_pages_allocate()
+ */
+ gtk_widget_get_allocation (widget, &dockbook_allocation);
+
+ /* Border on both sides */
+ border_loss = gtk_container_get_border_width (GTK_CONTAINER (dockbook)) * 2;
+
+ /* Space taken by action widget */
+ action_widget_size = button_allocation.width + xthickness;
+
+ /* Space taken by the tabs but not the tab widgets themselves */
+ tab_padding = gtk_notebook_get_n_pages (GTK_NOTEBOOK (dockbook)) *
+ (2 * (xthickness + tab_curvature + focus_width + tab_hborder) -
+ tab_overlap);
+
+ available_space = dockbook_allocation.width
+ - border_loss
+ - action_widget_size
+ - tab_padding
+ - tab_overlap;
+
+ GIMP_LOG (AUTO_TAB_STYLE, "\n"
+ " available_space = %d where\n"
+ " dockbook_allocation.width = %d\n"
+ " border_loss = %d\n"
+ " action_widget_size = %d\n"
+ " tab_padding = %d\n"
+ " tab_overlap = %d\n",
+ available_space,
+ dockbook_allocation.width,
+ border_loss,
+ action_widget_size,
+ tab_padding,
+ tab_overlap);
+
+ /* Try all candidates, if we don't get any hit we still end up on
+ * the smallest style (which we always fall back to if we don't get
+ * a better match)
+ */
+ for (i = 0; i < G_N_ELEMENTS (gimp_tab_style_candidates); i++)
+ {
+ tab_style = gimp_tab_style_candidates[i];
+ if (available_space > dockbook->p->min_width_for_style[i])
+ {
+ GIMP_LOG (AUTO_TAB_STYLE, "Choosing tab style %s",
+ gimp_dockbook_get_tab_style_name (tab_style));
+ break;
+ }
+ }
+
+ for (iter = dockbook->p->dockables; iter; iter = g_list_next (iter))
+ {
+ GimpDockable *dockable = GIMP_DOCKABLE (iter->data);
+ GimpTabStyle actual_tab_style = tab_style;
+
+ if (gimp_dockable_get_tab_style (dockable) != GIMP_TAB_STYLE_AUTOMATIC)
+ continue;
+
+ actual_tab_style = gimp_dockbook_tab_style_to_preferred (tab_style,
+ dockable);
+
+ if (gimp_dockable_set_actual_tab_style (dockable, actual_tab_style))
+ changed = TRUE;
+ }
+
+ if (changed)
+ gimp_dockbook_recreate_tab_widgets (dockbook,
+ TRUE /*only_auto*/);
+}
+
+GtkWidget *
+gimp_dockbook_new (GimpMenuFactory *menu_factory)
+{
+ GimpDockbook *dockbook;
+
+ g_return_val_if_fail (GIMP_IS_MENU_FACTORY (menu_factory), NULL);
+
+ dockbook = g_object_new (GIMP_TYPE_DOCKBOOK, NULL);
+
+ dockbook->p->ui_manager = gimp_menu_factory_manager_new (menu_factory,
+ "<Dockable>",
+ dockbook,
+ FALSE);
+
+ g_signal_connect (dockbook->p->ui_manager->gimp->config,
+ "size-changed",
+ G_CALLBACK (gimp_dockbook_config_size_changed),
+ dockbook);
+
+ gimp_help_connect (GTK_WIDGET (dockbook), gimp_dockbook_help_func,
+ GIMP_HELP_DOCK, dockbook);
+
+ return GTK_WIDGET (dockbook);
+}
+
+GimpDock *
+gimp_dockbook_get_dock (GimpDockbook *dockbook)
+{
+ g_return_val_if_fail (GIMP_IS_DOCKBOOK (dockbook), NULL);
+
+ return dockbook->p->dock;
+}
+
+void
+gimp_dockbook_set_dock (GimpDockbook *dockbook,
+ GimpDock *dock)
+{
+ g_return_if_fail (GIMP_IS_DOCKBOOK (dockbook));
+ g_return_if_fail (dock == NULL || GIMP_IS_DOCK (dock));
+
+ dockbook->p->dock = dock;
+}
+
+GimpUIManager *
+gimp_dockbook_get_ui_manager (GimpDockbook *dockbook)
+{
+ g_return_val_if_fail (GIMP_IS_DOCKBOOK (dockbook), NULL);
+
+ return dockbook->p->ui_manager;
+}
+
+void
+gimp_dockbook_add (GimpDockbook *dockbook,
+ GimpDockable *dockable,
+ gint position)
+{
+ GtkWidget *tab_widget;
+ GtkWidget *menu_widget;
+
+ g_return_if_fail (GIMP_IS_DOCKBOOK (dockbook));
+ g_return_if_fail (dockbook->p->dock != NULL);
+ g_return_if_fail (GIMP_IS_DOCKABLE (dockable));
+ g_return_if_fail (gimp_dockable_get_dockbook (dockable) == NULL);
+
+ GIMP_LOG (DND, "Adding GimpDockable %p to GimpDockbook %p", dockable, dockbook);
+
+ /* Add to internal list before doing automatic tab style
+ * calculations
+ */
+ dockbook->p->dockables = g_list_insert (dockbook->p->dockables,
+ dockable,
+ position);
+
+ gimp_dockbook_update_auto_tab_style (dockbook);
+
+ /* Create the new tab widget, it will get the correct tab style now */
+ tab_widget = gimp_dockbook_create_tab_widget (dockbook, dockable);
+
+ g_return_if_fail (GTK_IS_WIDGET (tab_widget));
+
+ gimp_dockable_set_drag_handler (dockable, dockbook->p->drag_handler);
+
+ /* For the notebook right-click menu, always use the icon style */
+ menu_widget =
+ gimp_dockable_create_tab_widget (dockable,
+ gimp_dock_get_context (dockbook->p->dock),
+ GIMP_TAB_STYLE_ICON_BLURB,
+ MENU_WIDGET_ICON_SIZE);
+
+ g_return_if_fail (GTK_IS_WIDGET (menu_widget));
+
+ if (position == -1)
+ {
+ gtk_notebook_append_page_menu (GTK_NOTEBOOK (dockbook),
+ GTK_WIDGET (dockable),
+ tab_widget,
+ menu_widget);
+ }
+ else
+ {
+ gtk_notebook_insert_page_menu (GTK_NOTEBOOK (dockbook),
+ GTK_WIDGET (dockable),
+ tab_widget,
+ menu_widget,
+ position);
+ }
+
+ gtk_widget_show (GTK_WIDGET (dockable));
+
+ gimp_dockable_set_dockbook (dockable, dockbook);
+
+ gimp_dockable_set_context (dockable, gimp_dock_get_context (dockbook->p->dock));
+
+ g_signal_connect (dockable, "notify::locked",
+ G_CALLBACK (gimp_dockbook_tab_locked_notify),
+ dockbook);
+
+ g_signal_emit (dockbook, dockbook_signals[DOCKABLE_ADDED], 0, dockable);
+}
+
+/**
+ * gimp_dockbook_add_from_dialog_factory:
+ * @dockbook: The #DockBook
+ * @identifiers: The dockable identifier(s)
+ * @position: The insert position
+ *
+ * Add a dockable from the dialog factory associated with the dockbook.
+ **/
+GtkWidget *
+gimp_dockbook_add_from_dialog_factory (GimpDockbook *dockbook,
+ const gchar *identifiers,
+ gint position)
+{
+ GtkWidget *dockable;
+ GimpDock *dock;
+ gchar *identifier;
+ gchar *p;
+
+ g_return_val_if_fail (GIMP_IS_DOCKBOOK (dockbook), NULL);
+ g_return_val_if_fail (identifiers != NULL, NULL);
+
+ identifier = g_strdup (identifiers);
+
+ p = strchr (identifier, '|');
+
+ if (p)
+ *p = '\0';
+
+ dock = gimp_dockbook_get_dock (dockbook);
+ dockable = gimp_dialog_factory_dockable_new (gimp_dock_get_dialog_factory (dock),
+ dock,
+ identifier, -1);
+
+ g_free (identifier);
+
+ /* Maybe gimp_dialog_factory_dockable_new() returned an already
+ * existing singleton dockable, so check if it already is
+ * attached to a dockbook.
+ */
+ if (dockable && ! gimp_dockable_get_dockbook (GIMP_DOCKABLE (dockable)))
+ gimp_dockbook_add (dockbook, GIMP_DOCKABLE (dockable), position);
+
+ if (dockable)
+ gimp_dockable_set_drag_pos (GIMP_DOCKABLE (dockable),
+ GIMP_DOCKABLE_DRAG_OFFSET,
+ GIMP_DOCKABLE_DRAG_OFFSET);
+ return dockable;
+}
+
+void
+gimp_dockbook_remove (GimpDockbook *dockbook,
+ GimpDockable *dockable)
+{
+ g_return_if_fail (GIMP_IS_DOCKBOOK (dockbook));
+ g_return_if_fail (GIMP_IS_DOCKABLE (dockable));
+ g_return_if_fail (gimp_dockable_get_dockbook (dockable) == dockbook);
+
+ GIMP_LOG (DND, "Removing GimpDockable %p from GimpDockbook %p", dockable, dockbook);
+
+ gimp_dockable_set_drag_handler (dockable, NULL);
+
+ g_object_ref (dockable);
+
+ g_signal_handlers_disconnect_by_func (dockable,
+ G_CALLBACK (gimp_dockbook_tab_locked_notify),
+ dockbook);
+
+ if (dockbook->p->tab_hover_dockable == dockable)
+ gimp_dockbook_remove_tab_timeout (dockbook);
+
+ gimp_dockable_set_dockbook (dockable, NULL);
+
+ gimp_dockable_set_context (dockable, NULL);
+
+ gtk_container_remove (GTK_CONTAINER (dockbook), GTK_WIDGET (dockable));
+ dockbook->p->dockables = g_list_remove (dockbook->p->dockables,
+ dockable);
+
+ g_signal_emit (dockbook, dockbook_signals[DOCKABLE_REMOVED], 0, dockable);
+
+ g_object_unref (dockable);
+
+ if (dockbook->p->dock)
+ {
+ GList *children = gtk_container_get_children (GTK_CONTAINER (dockbook));
+
+ if (children)
+ gimp_dockbook_update_auto_tab_style (dockbook);
+ else
+ gimp_dock_remove_book (dockbook->p->dock, dockbook);
+
+ g_list_free (children);
+ }
+}
+
+/**
+ * gimp_dockbook_update_with_context:
+ * @dockbook:
+ * @context:
+ *
+ * Set @context on all dockables in @dockbook.
+ **/
+void
+gimp_dockbook_update_with_context (GimpDockbook *dockbook,
+ GimpContext *context)
+{
+ GList *children = gtk_container_get_children (GTK_CONTAINER (dockbook));
+ GList *iter = NULL;
+
+ for (iter = children;
+ iter;
+ iter = g_list_next (iter))
+ {
+ GimpDockable *dockable = GIMP_DOCKABLE (iter->data);
+
+ gimp_dockable_set_context (dockable, context);
+ }
+
+ g_list_free (children);
+}
+
+GtkWidget *
+gimp_dockbook_create_tab_widget (GimpDockbook *dockbook,
+ GimpDockable *dockable)
+{
+ GtkWidget *tab_widget;
+ GimpDockWindow *dock_window;
+ GimpAction *action = NULL;
+
+ tab_widget =
+ gimp_dockable_create_event_box_tab_widget (dockable,
+ gimp_dock_get_context (dockbook->p->dock),
+ gimp_dockable_get_actual_tab_style (dockable),
+ gimp_dockbook_get_tab_icon_size (dockbook));
+
+ /* EEK */
+ dock_window = gimp_dock_window_from_dock (dockbook->p->dock);
+ if (dock_window &&
+ gimp_dock_container_get_ui_manager (GIMP_DOCK_CONTAINER (dock_window)))
+ {
+ const gchar *dialog_id;
+
+ dialog_id = g_object_get_data (G_OBJECT (dockable),
+ "gimp-dialog-identifier");
+
+ if (dialog_id)
+ {
+ GimpDockContainer *dock_container;
+ GimpActionGroup *group;
+
+ dock_container = GIMP_DOCK_CONTAINER (dock_window);
+
+ group = gimp_ui_manager_get_action_group
+ (gimp_dock_container_get_ui_manager (dock_container), "dialogs");
+
+ if (group)
+ {
+ GList *actions;
+ GList *list;
+
+ actions = gimp_action_group_list_actions (group);
+
+ for (list = actions; list; list = g_list_next (list))
+ {
+ if (GIMP_IS_STRING_ACTION (list->data) &&
+ strstr (GIMP_STRING_ACTION (list->data)->value,
+ dialog_id))
+ {
+ action = list->data;
+ break;
+ }
+ }
+
+ g_list_free (actions);
+ }
+ }
+ }
+
+ if (action)
+ gimp_widget_set_accel_help (tab_widget, action);
+ else
+ gimp_help_set_help_data (tab_widget,
+ gimp_dockable_get_blurb (dockable),
+ gimp_dockable_get_help_id (dockable));
+
+ g_object_set_data (G_OBJECT (tab_widget), "gimp-dockable", dockable);
+
+ gimp_dockbook_tab_drag_source_setup (tab_widget, dockable);
+
+ g_signal_connect_object (tab_widget, "drag-begin",
+ G_CALLBACK (gimp_dockbook_tab_drag_begin),
+ dockable, 0);
+ g_signal_connect_object (tab_widget, "drag-end",
+ G_CALLBACK (gimp_dockbook_tab_drag_end),
+ dockable, 0);
+ g_signal_connect_object (tab_widget, "drag-failed",
+ G_CALLBACK (gimp_dockbook_tab_drag_failed),
+ dockable, 0);
+
+ g_signal_connect_object (dockable, "drag-begin",
+ G_CALLBACK (gimp_dockbook_tab_drag_begin),
+ dockable, 0);
+ g_signal_connect_object (dockable, "drag-end",
+ G_CALLBACK (gimp_dockbook_tab_drag_end),
+ dockable, 0);
+ g_signal_connect_object (dockable, "drag-failed",
+ G_CALLBACK (gimp_dockbook_tab_drag_failed),
+ dockable, 0);
+
+ gtk_drag_dest_set (tab_widget,
+ 0,
+ dialog_target_table, G_N_ELEMENTS (dialog_target_table),
+ GDK_ACTION_MOVE);
+ g_signal_connect_object (tab_widget, "drag-leave",
+ G_CALLBACK (gimp_dockbook_tab_drag_leave),
+ dockable, 0);
+ g_signal_connect_object (tab_widget, "drag-motion",
+ G_CALLBACK (gimp_dockbook_tab_drag_motion),
+ dockable, 0);
+ g_signal_connect_object (tab_widget, "drag-drop",
+ G_CALLBACK (gimp_dockbook_tab_drag_drop),
+ dockbook, 0);
+
+ return tab_widget;
+}
+
+/**
+ * gimp_dockbook_update_auto_tab_style:
+ * @dockbook:
+ *
+ * Refresh the table that we use to map dockbook width to actual auto
+ * tab style, then update auto tabs (also recreate tab widgets if
+ * necessary).
+ **/
+void
+gimp_dockbook_update_auto_tab_style (GimpDockbook *dockbook)
+{
+ g_return_if_fail (GIMP_IS_DOCKBOOK (dockbook));
+
+ gimp_dockbook_refresh_tab_layout_lut (dockbook);
+ gimp_dockbook_update_automatic_tab_style (dockbook);
+}
+
+gboolean
+gimp_dockbook_drop_dockable (GimpDockbook *dockbook,
+ GtkWidget *drag_source)
+{
+ g_return_val_if_fail (GIMP_IS_DOCKBOOK (dockbook), FALSE);
+
+ if (drag_source)
+ {
+ GimpDockable *dockable =
+ gimp_dockbook_drag_source_to_dockable (drag_source);
+
+ if (dockable)
+ {
+ if (gimp_dockable_get_dockbook (dockable) == dockbook)
+ {
+ gtk_notebook_reorder_child (GTK_NOTEBOOK (dockbook),
+ GTK_WIDGET (dockable), -1);
+ }
+ else
+ {
+ g_object_ref (dockable);
+
+ gimp_dockbook_remove (gimp_dockable_get_dockbook (dockable), dockable);
+ gimp_dockbook_add (dockbook, dockable, -1);
+
+ g_object_unref (dockable);
+ }
+
+ return TRUE;
+ }
+ }
+
+ return FALSE;
+}
+
+/**
+ * gimp_dockable_set_drag_handler:
+ * @dockable:
+ * @handler:
+ *
+ * Set a drag handler that will be asked if it will handle drag events
+ * before the dockbook handles the event itself.
+ **/
+void
+gimp_dockbook_set_drag_handler (GimpDockbook *dockbook,
+ GimpPanedBox *drag_handler)
+{
+ g_return_if_fail (GIMP_IS_DOCKBOOK (dockbook));
+
+ dockbook->p->drag_handler = drag_handler;
+}
+
+/**
+ * gimp_dockbook_drag_source_to_dockable:
+ * @drag_source: A drag-and-drop source widget
+ *
+ * Gets the dockable associated with a drag-and-drop source. If
+ * successful, the function will also cleanup the dockable.
+ *
+ * Returns: The dockable
+ **/
+GimpDockable *
+gimp_dockbook_drag_source_to_dockable (GtkWidget *drag_source)
+{
+ GimpDockable *dockable = NULL;
+
+ if (GIMP_IS_DOCKABLE (drag_source))
+ dockable = GIMP_DOCKABLE (drag_source);
+ else
+ dockable = g_object_get_data (G_OBJECT (drag_source),
+ "gimp-dockable");
+ if (dockable)
+ g_object_set_data (G_OBJECT (dockable),
+ "gimp-dock-drag-widget", NULL);
+
+ return dockable;
+}
+
+void
+gimp_dockbook_add_drag_callback (GimpDockbookDragCallback callback,
+ gpointer data)
+{
+ GimpDockbookDragCallbackData *callback_data;
+
+ callback_data = g_slice_new (GimpDockbookDragCallbackData);
+
+ callback_data->callback = callback;
+ callback_data->data = data;
+
+ drag_callbacks = g_list_prepend (drag_callbacks, callback_data);
+}
+
+void
+gimp_dockbook_remove_drag_callback (GimpDockbookDragCallback callback,
+ gpointer data)
+{
+ GList *iter;
+
+ iter = drag_callbacks;
+
+ while (iter)
+ {
+ GimpDockbookDragCallbackData *callback_data = iter->data;
+ GList *next = g_list_next (iter);
+
+ if (callback_data->callback == callback &&
+ callback_data->data == data)
+ {
+ g_slice_free (GimpDockbookDragCallbackData, callback_data);
+
+ drag_callbacks = g_list_delete_link (drag_callbacks, iter);
+ }
+
+ iter = next;
+ }
+}
+
+/* tab DND source side */
+
+static void
+gimp_dockbook_recreate_tab_widgets (GimpDockbook *dockbook,
+ gboolean only_auto)
+{
+ GList *dockables = gtk_container_get_children (GTK_CONTAINER (dockbook));
+ GList *iter = NULL;
+
+ g_object_set (dockbook,
+ "tab-border", gimp_dockbook_get_tab_border (dockbook),
+ NULL);
+
+ for (iter = dockables; iter; iter = g_list_next (iter))
+ {
+ GimpDockable *dockable = GIMP_DOCKABLE (iter->data);
+ GtkWidget *tab_widget;
+
+ if (only_auto &&
+ ! (gimp_dockable_get_tab_style (dockable) == GIMP_TAB_STYLE_AUTOMATIC))
+ continue;
+
+ tab_widget = gimp_dockbook_create_tab_widget (dockbook, dockable);
+
+ gtk_notebook_set_tab_label (GTK_NOTEBOOK (dockbook),
+ GTK_WIDGET (dockable),
+ tab_widget);
+ }
+
+ g_list_free (dockables);
+}
+
+static void
+gimp_dockbook_tab_drag_source_setup (GtkWidget *widget,
+ GimpDockable *dockable)
+{
+ if (gimp_dockable_is_locked (dockable))
+ {
+ if (widget)
+ gtk_drag_source_unset (widget);
+
+ gtk_drag_source_unset (GTK_WIDGET (dockable));
+ }
+ else
+ {
+ if (widget)
+ gtk_drag_source_set (widget,
+ GDK_BUTTON1_MASK | GDK_BUTTON2_MASK,
+ dialog_target_table,
+ G_N_ELEMENTS (dialog_target_table),
+ GDK_ACTION_MOVE);
+
+ gtk_drag_source_set (GTK_WIDGET (dockable),
+ GDK_BUTTON1_MASK | GDK_BUTTON2_MASK,
+ dialog_target_table,
+ G_N_ELEMENTS (dialog_target_table),
+ GDK_ACTION_MOVE);
+ }
+}
+
+static void
+gimp_dockbook_tab_drag_begin (GtkWidget *widget,
+ GdkDragContext *context,
+ GimpDockable *dockable)
+{
+ GtkAllocation allocation;
+ GtkWidget *window;
+ GtkWidget *view;
+ GList *iter;
+ GtkRequisition requisition;
+ gint drag_x;
+ gint drag_y;
+
+ gtk_widget_get_allocation (widget, &allocation);
+
+ window = gtk_window_new (GTK_WINDOW_POPUP);
+ gtk_window_set_type_hint (GTK_WINDOW (window), GDK_WINDOW_TYPE_HINT_DND);
+ gtk_window_set_screen (GTK_WINDOW (window), gtk_widget_get_screen (widget));
+
+ view = gimp_dockable_create_drag_widget (dockable);
+ gtk_container_add (GTK_CONTAINER (window), view);
+ gtk_widget_show (view);
+
+ gtk_widget_size_request (view, &requisition);
+
+ if (requisition.width < allocation.width)
+ gtk_widget_set_size_request (view, allocation.width, -1);
+
+ gtk_widget_show (window);
+
+ g_object_set_data_full (G_OBJECT (dockable), "gimp-dock-drag-widget",
+ window,
+ (GDestroyNotify) gtk_widget_destroy);
+
+ gimp_dockable_get_drag_pos (dockable, &drag_x, &drag_y);
+ gtk_drag_set_icon_widget (context, window, drag_x, drag_y);
+
+ iter = drag_callbacks;
+
+ while (iter)
+ {
+ GimpDockbookDragCallbackData *callback_data = iter->data;
+
+ iter = g_list_next (iter);
+
+ callback_data->callback (context, TRUE, callback_data->data);
+ }
+}
+
+static void
+gimp_dockbook_tab_drag_end (GtkWidget *widget,
+ GdkDragContext *context,
+ GimpDockable *dockable)
+{
+ GList *iter;
+
+ iter = drag_callbacks;
+
+ while (iter)
+ {
+ GimpDockbookDragCallbackData *callback_data = iter->data;
+
+ iter = g_list_next (iter);
+
+ callback_data->callback (context, FALSE, callback_data->data);
+ }
+}
+
+static gboolean
+gimp_dockbook_tab_drag_failed (GtkWidget *widget,
+ GdkDragContext *context,
+ GtkDragResult result,
+ GimpDockable *dockable)
+{
+ /* XXX The proper way is to handle "drag-end" signal instead.
+ * Unfortunately this signal seems to be broken in various cases on
+ * macOS/GTK+2 (see #1924). As a consequence, we made sure we don't
+ * have anything to clean unconditionally (for instance we used to set
+ * the dockable unsensitive, which anyway was only a visual clue and
+ * is not really useful). Only thing left to handle is the dockable
+ * detachment when dropping in a non-droppable area.
+ */
+ GtkWidget *drag_widget;
+ GtkWidget *window;
+
+ if (result == GTK_DRAG_RESULT_SUCCESS)
+ {
+ /* I don't think this should happen, considering we are in the
+ * "drag-failed" handler, but let's be complete as it is a
+ * possible GtkDragResult value. Just in case!
+ */
+ return FALSE;
+ }
+
+ drag_widget = g_object_get_data (G_OBJECT (dockable),
+ "gimp-dock-drag-widget");
+
+ /* The drag_widget should be present if the drop was not successful,
+ * in which case, we pop up a new dock and move the dockable there.
+ */
+ g_return_val_if_fail (drag_widget, FALSE);
+
+ g_object_set_data (G_OBJECT (dockable), "gimp-dock-drag-widget", NULL);
+ gimp_dockable_detach (dockable);
+
+ window = gtk_widget_get_toplevel (GTK_WIDGET (dockable));
+ gtk_window_present (GTK_WINDOW (window));
+
+ return TRUE;
+}
+
+
+/* tab DND target side */
+
+static void
+gimp_dockbook_tab_drag_leave (GtkWidget *widget,
+ GdkDragContext *context,
+ guint time,
+ GimpDockable *dockable)
+{
+ GimpDockbook *dockbook = gimp_dockable_get_dockbook (dockable);
+
+ gimp_dockbook_remove_tab_timeout (dockbook);
+
+ gimp_highlight_widget (widget, FALSE);
+}
+
+static gboolean
+gimp_dockbook_tab_drag_motion (GtkWidget *widget,
+ GdkDragContext *context,
+ gint x,
+ gint y,
+ guint time,
+ GimpDockable *dockable)
+{
+ GimpDockbook *dockbook = gimp_dockable_get_dockbook (dockable);
+ GtkTargetList *target_list;
+ GdkAtom target_atom;
+ gboolean handle;
+
+ if (gimp_paned_box_will_handle_drag (dockbook->p->drag_handler,
+ widget,
+ context,
+ x, y,
+ time))
+ {
+ gdk_drag_status (context, 0, time);
+ gimp_highlight_widget (widget, FALSE);
+
+ return FALSE;
+ }
+
+ if (! dockbook->p->tab_hover_timeout ||
+ dockbook->p->tab_hover_dockable != dockable)
+ {
+ gint page_num;
+
+ gimp_dockbook_remove_tab_timeout (dockbook);
+
+ page_num = gtk_notebook_page_num (GTK_NOTEBOOK (dockbook),
+ GTK_WIDGET (dockable));
+
+ if (page_num != gtk_notebook_get_current_page (GTK_NOTEBOOK (dockbook)))
+ gimp_dockbook_add_tab_timeout (dockbook, dockable);
+ }
+
+ target_list = gtk_drag_dest_get_target_list (widget);
+ target_atom = gtk_drag_dest_find_target (widget, context, target_list);
+
+ handle = gtk_target_list_find (target_list, target_atom, NULL);
+
+ gdk_drag_status (context, handle ? GDK_ACTION_MOVE : 0, time);
+ gimp_highlight_widget (widget, handle);
+
+ /* Return TRUE so drag_leave() is called */
+ return TRUE;
+}
+
+static gboolean
+gimp_dockbook_tab_drag_drop (GtkWidget *widget,
+ GdkDragContext *context,
+ gint x,
+ gint y,
+ guint time)
+{
+ GimpDockable *dest_dockable;
+ GtkWidget *source;
+ gboolean dropped = FALSE;
+
+ dest_dockable = g_object_get_data (G_OBJECT (widget), "gimp-dockable");
+
+ source = gtk_drag_get_source_widget (context);
+
+ if (gimp_paned_box_will_handle_drag (gimp_dockable_get_drag_handler (dest_dockable),
+ widget,
+ context,
+ x, y,
+ time))
+ {
+ return FALSE;
+ }
+
+ if (dest_dockable && source)
+ {
+ GimpDockable *src_dockable =
+ gimp_dockbook_drag_source_to_dockable (source);
+
+ if (src_dockable)
+ {
+ gint dest_index;
+
+ dest_index =
+ gtk_notebook_page_num (GTK_NOTEBOOK (gimp_dockable_get_dockbook (dest_dockable)),
+ GTK_WIDGET (dest_dockable));
+
+ if (gimp_dockable_get_dockbook (src_dockable) !=
+ gimp_dockable_get_dockbook (dest_dockable))
+ {
+ g_object_ref (src_dockable);
+
+ gimp_dockbook_remove (gimp_dockable_get_dockbook (src_dockable), src_dockable);
+ gimp_dockbook_add (gimp_dockable_get_dockbook (dest_dockable), src_dockable,
+ dest_index);
+
+ g_object_unref (src_dockable);
+
+ dropped = TRUE;
+ }
+ else if (src_dockable != dest_dockable)
+ {
+ gtk_notebook_reorder_child (GTK_NOTEBOOK (gimp_dockable_get_dockbook (src_dockable)),
+ GTK_WIDGET (src_dockable),
+ dest_index);
+
+ g_signal_emit (gimp_dockable_get_dockbook (src_dockable),
+ dockbook_signals[DOCKABLE_REORDERED], 0,
+ src_dockable);
+
+ dropped = TRUE;
+ }
+ }
+ }
+
+ gtk_drag_finish (context, dropped, TRUE, time);
+
+ return TRUE;
+}
+
+static GtkWidget *
+gimp_dockable_create_event_box_tab_widget (GimpDockable *dockable,
+ GimpContext *context,
+ GimpTabStyle tab_style,
+ GtkIconSize size)
+{
+ GtkWidget *tab_widget;
+
+ tab_widget =
+ gimp_dockable_create_tab_widget (dockable,
+ context,
+ tab_style,
+ size);
+
+ if (! GIMP_IS_VIEW (tab_widget))
+ {
+ GtkWidget *event_box;
+
+ event_box = gtk_event_box_new ();
+ gtk_event_box_set_visible_window (GTK_EVENT_BOX (event_box), FALSE);
+ gtk_event_box_set_above_child (GTK_EVENT_BOX (event_box), TRUE);
+ gtk_container_add (GTK_CONTAINER (event_box), tab_widget);
+ gtk_widget_show (tab_widget);
+
+ tab_widget = event_box;
+ }
+
+ return tab_widget;
+}
+
+static GtkIconSize
+gimp_dockbook_get_tab_icon_size (GimpDockbook *dockbook)
+{
+ Gimp *gimp = dockbook->p->ui_manager->gimp;
+ GtkIconSize tab_size = DEFAULT_TAB_ICON_SIZE;
+ GimpIconSize size;
+
+ size = gimp_gui_config_detect_icon_size (GIMP_GUI_CONFIG (gimp->config));
+ /* Match GimpIconSize with GtkIconSize. */
+ switch (size)
+ {
+ case GIMP_ICON_SIZE_SMALL:
+ case GIMP_ICON_SIZE_MEDIUM:
+ tab_size = GTK_ICON_SIZE_MENU;
+ break;
+ case GIMP_ICON_SIZE_LARGE:
+ tab_size = GTK_ICON_SIZE_LARGE_TOOLBAR;
+ break;
+ case GIMP_ICON_SIZE_HUGE:
+ tab_size = GTK_ICON_SIZE_DND;
+ break;
+ default:
+ /* GIMP_ICON_SIZE_DEFAULT:
+ * let's use the size set by the theme. */
+ gtk_widget_style_get (GTK_WIDGET (dockbook),
+ "tab-icon-size", &tab_size,
+ NULL);
+ break;
+ }
+
+ return tab_size;
+}
+
+static gint
+gimp_dockbook_get_tab_border (GimpDockbook *dockbook)
+{
+ Gimp *gimp = dockbook->p->ui_manager->gimp;
+ gint tab_border = DEFAULT_TAB_BORDER;
+ GimpIconSize size;
+
+ gtk_widget_style_get (GTK_WIDGET (dockbook),
+ "tab-border", &tab_border,
+ NULL);
+
+ size = gimp_gui_config_detect_icon_size (GIMP_GUI_CONFIG (gimp->config));
+ /* Match GimpIconSize with GtkIconSize. */
+ switch (size)
+ {
+ case GIMP_ICON_SIZE_SMALL:
+ tab_border /= 2;
+ break;
+ case GIMP_ICON_SIZE_LARGE:
+ tab_border *= 2;
+ break;
+ case GIMP_ICON_SIZE_HUGE:
+ tab_border *= 3;
+ break;
+ default:
+ /* GIMP_ICON_SIZE_MEDIUM and GIMP_ICON_SIZE_DEFAULT:
+ * let's use the size set by the theme. */
+ break;
+ }
+
+ return tab_border;
+}
+
+static void
+gimp_dockbook_add_tab_timeout (GimpDockbook *dockbook,
+ GimpDockable *dockable)
+{
+ dockbook->p->tab_hover_timeout =
+ g_timeout_add (TAB_HOVER_TIMEOUT,
+ (GSourceFunc) gimp_dockbook_tab_timeout,
+ dockbook);
+
+ dockbook->p->tab_hover_dockable = dockable;
+}
+
+static void
+gimp_dockbook_remove_tab_timeout (GimpDockbook *dockbook)
+{
+ if (dockbook->p->tab_hover_timeout)
+ {
+ g_source_remove (dockbook->p->tab_hover_timeout);
+ dockbook->p->tab_hover_timeout = 0;
+ dockbook->p->tab_hover_dockable = NULL;
+ }
+}
+
+static gboolean
+gimp_dockbook_tab_timeout (GimpDockbook *dockbook)
+{
+ gint page_num;
+
+ GDK_THREADS_ENTER ();
+
+ page_num = gtk_notebook_page_num (GTK_NOTEBOOK (dockbook),
+ GTK_WIDGET (dockbook->p->tab_hover_dockable));
+ gtk_notebook_set_current_page (GTK_NOTEBOOK (dockbook), page_num);
+
+ dockbook->p->tab_hover_timeout = 0;
+ dockbook->p->tab_hover_dockable = NULL;
+
+ GDK_THREADS_LEAVE ();
+
+ return FALSE;
+}
+
+static void
+gimp_dockbook_tab_locked_notify (GimpDockable *dockable,
+ GParamSpec *pspec,
+ GimpDockbook *dockbook)
+{
+ GtkWidget *tab_widget;
+
+ tab_widget = gtk_notebook_get_tab_label (GTK_NOTEBOOK (dockbook),
+ GTK_WIDGET (dockable));
+
+ gimp_dockbook_tab_drag_source_setup (tab_widget, dockable);
+}
+
+static void
+gimp_dockbook_help_func (const gchar *help_id,
+ gpointer help_data)
+{
+ GimpDockbook *dockbook = GIMP_DOCKBOOK (help_data);
+ GtkWidget *dockable;
+ gint page_num;
+
+ page_num = gtk_notebook_get_current_page (GTK_NOTEBOOK (dockbook));
+
+ dockable = gtk_notebook_get_nth_page (GTK_NOTEBOOK (dockbook), page_num);
+
+ if (GIMP_IS_DOCKABLE (dockable))
+ gimp_standard_help_func (gimp_dockable_get_help_id (GIMP_DOCKABLE (dockable)),
+ NULL);
+ else
+ gimp_standard_help_func (GIMP_HELP_DOCK, NULL);
+}
+
+static const gchar *
+gimp_dockbook_get_tab_style_name (GimpTabStyle tab_style)
+{
+ return g_enum_get_value (g_type_class_peek (GIMP_TYPE_TAB_STYLE),
+ tab_style)->value_name;
+}
+
+static void
+gimp_dockbook_config_size_changed (GimpGuiConfig *config,
+ GimpDockbook *dockbook)
+{
+ gimp_dockbook_recreate_tab_widgets (dockbook, TRUE);
+}