mirror of https://github.com/GNOME/gimp.git
613 lines
16 KiB
C
613 lines
16 KiB
C
/* The GIMP -- an image manipulation program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* gimpdock.c
|
|
* Copyright (C) 2001 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 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 <gtk/gtk.h>
|
|
|
|
#include "libgimpwidgets/gimpwidgets.h"
|
|
|
|
#include "widgets-types.h"
|
|
|
|
#include "core/gimpcontext.h"
|
|
|
|
#include "gimpdialogfactory.h"
|
|
#include "gimpdnd.h"
|
|
#include "gimpdock.h"
|
|
#include "gimpdockable.h"
|
|
#include "gimpdockbook.h"
|
|
|
|
|
|
#define DEFAULT_SEPARATOR_HEIGHT 6
|
|
|
|
|
|
static void gimp_dock_class_init (GimpDockClass *klass);
|
|
static void gimp_dock_init (GimpDock *dock);
|
|
|
|
static GtkWidget * gimp_dock_separator_new (GimpDock *dock);
|
|
|
|
static void gimp_dock_destroy (GtkObject *object);
|
|
static void gimp_dock_style_set (GtkWidget *widget,
|
|
GtkStyle *prev_style);
|
|
|
|
static gboolean gimp_dock_separator_button_press (GtkWidget *widget,
|
|
GdkEventButton *bevent,
|
|
gpointer data);
|
|
static gboolean gimp_dock_separator_button_release (GtkWidget *widget,
|
|
GdkEventButton *bevent,
|
|
gpointer data);
|
|
|
|
/*
|
|
static void gimp_dock_separator_drag_begin (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gpointer data);
|
|
static void gimp_dock_separator_drag_end (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gpointer data);
|
|
*/
|
|
|
|
static gboolean gimp_dock_separator_drag_drop (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gint x,
|
|
gint y,
|
|
guint time,
|
|
gpointer data);
|
|
|
|
|
|
static GtkWindowClass *parent_class = NULL;
|
|
|
|
static GtkTargetEntry dialog_target_table[] =
|
|
{
|
|
GIMP_TARGET_DIALOG
|
|
};
|
|
|
|
|
|
GType
|
|
gimp_dock_get_type (void)
|
|
{
|
|
static GType dock_type = 0;
|
|
|
|
if (! dock_type)
|
|
{
|
|
static const GTypeInfo dock_info =
|
|
{
|
|
sizeof (GimpDockClass),
|
|
NULL, /* base_init */
|
|
NULL, /* base_finalize */
|
|
(GClassInitFunc) gimp_dock_class_init,
|
|
NULL, /* class_finalize */
|
|
NULL, /* class_data */
|
|
sizeof (GimpDock),
|
|
0, /* n_preallocs */
|
|
(GInstanceInitFunc) gimp_dock_init,
|
|
};
|
|
|
|
dock_type = g_type_register_static (GTK_TYPE_WINDOW,
|
|
"GimpDock",
|
|
&dock_info, 0);
|
|
}
|
|
|
|
return dock_type;
|
|
}
|
|
|
|
static void
|
|
gimp_dock_class_init (GimpDockClass *klass)
|
|
{
|
|
GtkObjectClass *object_class;
|
|
GtkWidgetClass *widget_class;
|
|
|
|
object_class = GTK_OBJECT_CLASS (klass);
|
|
widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
parent_class = g_type_class_peek_parent (klass);
|
|
|
|
object_class->destroy = gimp_dock_destroy;
|
|
|
|
widget_class->style_set = gimp_dock_style_set;
|
|
|
|
gtk_widget_class_install_style_property (widget_class,
|
|
g_param_spec_int ("separator_height",
|
|
NULL, NULL,
|
|
0,
|
|
G_MAXINT,
|
|
DEFAULT_SEPARATOR_HEIGHT,
|
|
G_PARAM_READABLE));
|
|
}
|
|
|
|
static void
|
|
gimp_dock_init (GimpDock *dock)
|
|
{
|
|
GtkWidget *separator;
|
|
|
|
dock->context = NULL;
|
|
dock->destroy_if_empty = FALSE;
|
|
|
|
gtk_window_set_wmclass (GTK_WINDOW (dock), "dock", "Gimp");
|
|
gtk_window_set_resizable (GTK_WINDOW (dock), TRUE);
|
|
|
|
dock->main_vbox = gtk_vbox_new (FALSE, 0);
|
|
gtk_container_add (GTK_CONTAINER (dock), dock->main_vbox);
|
|
gtk_widget_show (dock->main_vbox);
|
|
|
|
dock->vbox = gtk_vbox_new (FALSE, 0);
|
|
gtk_container_add (GTK_CONTAINER (dock->main_vbox), dock->vbox);
|
|
gtk_widget_show (dock->vbox);
|
|
|
|
separator = gimp_dock_separator_new (dock);
|
|
gtk_box_pack_start (GTK_BOX (dock->vbox), separator, FALSE, FALSE, 0);
|
|
gtk_widget_show (separator);
|
|
}
|
|
|
|
static void
|
|
gimp_dock_destroy (GtkObject *object)
|
|
{
|
|
GimpDock *dock;
|
|
|
|
dock = GIMP_DOCK (object);
|
|
|
|
while (dock->dockbooks)
|
|
gimp_dock_remove_book (dock, GIMP_DOCKBOOK (dock->dockbooks->data));
|
|
|
|
if (dock->context)
|
|
{
|
|
g_object_unref (G_OBJECT (dock->context));
|
|
dock->context = NULL;
|
|
}
|
|
|
|
if (GTK_OBJECT_CLASS (parent_class))
|
|
GTK_OBJECT_CLASS (parent_class)->destroy (object);
|
|
}
|
|
|
|
static void
|
|
gimp_dock_style_set (GtkWidget *widget,
|
|
GtkStyle *prev_style)
|
|
{
|
|
GimpDock *dock;
|
|
GList *children;
|
|
GList *list;
|
|
gint separator_height;
|
|
|
|
dock = GIMP_DOCK (widget);
|
|
|
|
if (GTK_WIDGET_CLASS (parent_class)->style_set)
|
|
GTK_WIDGET_CLASS (parent_class)->style_set (widget, prev_style);
|
|
|
|
gtk_widget_style_get (widget,
|
|
"separator_height", &separator_height,
|
|
NULL);
|
|
|
|
children = gtk_container_get_children (GTK_CONTAINER (dock->vbox));
|
|
|
|
for (list = children; list; list = g_list_next (list))
|
|
{
|
|
if (GTK_IS_EVENT_BOX (list->data))
|
|
{
|
|
gtk_widget_set_size_request (GTK_WIDGET (list->data),
|
|
-1, separator_height);
|
|
}
|
|
}
|
|
|
|
g_list_free (children);
|
|
}
|
|
|
|
static GtkWidget *
|
|
gimp_dock_separator_new (GimpDock *dock)
|
|
{
|
|
GtkWidget *event_box;
|
|
GtkWidget *frame;
|
|
gint separator_height;
|
|
|
|
event_box = gtk_event_box_new ();
|
|
|
|
gtk_widget_set_name (event_box, "dock-separator");
|
|
|
|
gtk_widget_style_get (GTK_WIDGET (dock),
|
|
"separator_height", &separator_height,
|
|
NULL);
|
|
|
|
gtk_widget_set_size_request (event_box, -1, separator_height);
|
|
|
|
frame = gtk_frame_new (NULL);
|
|
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_OUT);
|
|
gtk_container_add (GTK_CONTAINER (event_box), frame);
|
|
gtk_widget_show (frame);
|
|
|
|
gtk_drag_dest_set (GTK_WIDGET (event_box),
|
|
GTK_DEST_DEFAULT_ALL,
|
|
dialog_target_table, G_N_ELEMENTS (dialog_target_table),
|
|
GDK_ACTION_MOVE);
|
|
g_signal_connect (G_OBJECT (event_box), "drag_drop",
|
|
G_CALLBACK (gimp_dock_separator_drag_drop),
|
|
dock);
|
|
|
|
g_signal_connect (G_OBJECT (event_box), "button_press_event",
|
|
G_CALLBACK (gimp_dock_separator_button_press),
|
|
dock);
|
|
g_signal_connect (G_OBJECT (event_box), "button_release_event",
|
|
G_CALLBACK (gimp_dock_separator_button_release),
|
|
dock);
|
|
|
|
return event_box;
|
|
}
|
|
|
|
gboolean
|
|
gimp_dock_construct (GimpDock *dock,
|
|
GimpDialogFactory *dialog_factory,
|
|
GimpContext *context,
|
|
gboolean destroy_if_empty)
|
|
{
|
|
g_return_val_if_fail (GIMP_IS_DOCK (dock), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_DIALOG_FACTORY (dialog_factory), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_CONTEXT (context), FALSE);
|
|
|
|
dock->dialog_factory = dialog_factory;
|
|
dock->context = context;
|
|
dock->destroy_if_empty = destroy_if_empty ? TRUE : FALSE;
|
|
|
|
g_object_ref (G_OBJECT (dock->context));
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
gimp_dock_add (GimpDock *dock,
|
|
GimpDockable *dockable,
|
|
gint section,
|
|
gint position)
|
|
{
|
|
GimpDockbook *dockbook;
|
|
|
|
g_return_if_fail (GIMP_IS_DOCK (dock));
|
|
g_return_if_fail (GIMP_IS_DOCKABLE (dockable));
|
|
|
|
g_return_if_fail (dockable->dockbook == NULL);
|
|
|
|
dockbook = GIMP_DOCKBOOK (dock->dockbooks->data);
|
|
|
|
gimp_dockbook_add (dockbook, dockable, position);
|
|
}
|
|
|
|
void
|
|
gimp_dock_remove (GimpDock *dock,
|
|
GimpDockable *dockable)
|
|
{
|
|
g_return_if_fail (GIMP_IS_DOCK (dock));
|
|
g_return_if_fail (GIMP_IS_DOCKABLE (dockable));
|
|
|
|
g_return_if_fail (dockable->dockbook != NULL);
|
|
g_return_if_fail (dockable->dockbook->dock != NULL);
|
|
g_return_if_fail (dockable->dockbook->dock == dock);
|
|
|
|
gimp_dockbook_remove (dockable->dockbook, dockable);
|
|
}
|
|
|
|
void
|
|
gimp_dock_add_book (GimpDock *dock,
|
|
GimpDockbook *dockbook,
|
|
gint index)
|
|
{
|
|
gint old_length;
|
|
|
|
g_return_if_fail (GIMP_IS_DOCK (dock));
|
|
g_return_if_fail (GIMP_IS_DOCKBOOK (dockbook));
|
|
|
|
g_return_if_fail (dockbook->dock == NULL);
|
|
|
|
old_length = g_list_length (dock->dockbooks);
|
|
|
|
if (index >= old_length || index < 0)
|
|
index = old_length;
|
|
|
|
dockbook->dock = dock;
|
|
dock->dockbooks = g_list_insert (dock->dockbooks, dockbook, index);
|
|
|
|
if (old_length == 0)
|
|
{
|
|
GtkWidget *separator;
|
|
|
|
gtk_box_pack_start (GTK_BOX (dock->vbox), GTK_WIDGET (dockbook),
|
|
TRUE, TRUE, 0);
|
|
|
|
separator = gimp_dock_separator_new (dock);
|
|
gtk_box_pack_end (GTK_BOX (dock->vbox), separator, FALSE, FALSE, 0);
|
|
gtk_widget_show (separator);
|
|
}
|
|
else
|
|
{
|
|
GtkWidget *old_book;
|
|
GtkWidget *parent;
|
|
GtkWidget *paned;
|
|
|
|
if (index == 0)
|
|
old_book = g_list_nth_data (dock->dockbooks, index + 1);
|
|
else
|
|
old_book = g_list_nth_data (dock->dockbooks, index - 1);
|
|
|
|
parent = old_book->parent;
|
|
|
|
if ((old_length > 1) && (index > 0))
|
|
{
|
|
GtkWidget *grandparent;
|
|
|
|
grandparent = parent->parent;
|
|
|
|
old_book = parent;
|
|
parent = grandparent;
|
|
}
|
|
|
|
g_object_ref (G_OBJECT (old_book));
|
|
|
|
gtk_container_remove (GTK_CONTAINER (parent), old_book);
|
|
|
|
paned = gtk_vpaned_new ();
|
|
|
|
if (GTK_IS_VPANED (parent))
|
|
gtk_paned_pack1 (GTK_PANED (parent), paned, TRUE, FALSE);
|
|
else
|
|
gtk_box_pack_start (GTK_BOX (parent), paned, TRUE, TRUE, 0);
|
|
|
|
gtk_widget_show (paned);
|
|
|
|
if (index == 0)
|
|
{
|
|
gtk_paned_pack1 (GTK_PANED (paned), GTK_WIDGET (dockbook),
|
|
TRUE, FALSE);
|
|
gtk_paned_pack2 (GTK_PANED (paned), old_book,
|
|
TRUE, FALSE);
|
|
}
|
|
else
|
|
{
|
|
gtk_paned_pack1 (GTK_PANED (paned), old_book,
|
|
TRUE, FALSE);
|
|
gtk_paned_pack2 (GTK_PANED (paned), GTK_WIDGET (dockbook),
|
|
TRUE, FALSE);
|
|
}
|
|
|
|
g_object_unref (G_OBJECT (old_book));
|
|
}
|
|
|
|
gtk_widget_show (GTK_WIDGET (dockbook));
|
|
}
|
|
|
|
void
|
|
gimp_dock_remove_book (GimpDock *dock,
|
|
GimpDockbook *dockbook)
|
|
{
|
|
gint old_length;
|
|
gint index;
|
|
|
|
g_return_if_fail (GIMP_IS_DOCK (dock));
|
|
g_return_if_fail (GIMP_IS_DOCKBOOK (dockbook));
|
|
|
|
g_return_if_fail (dockbook->dock == dock);
|
|
|
|
old_length = g_list_length (dock->dockbooks);
|
|
index = g_list_index (dock->dockbooks, dockbook);
|
|
|
|
dockbook->dock = NULL;
|
|
dock->dockbooks = g_list_remove (dock->dockbooks, dockbook);
|
|
|
|
if (old_length == 1)
|
|
{
|
|
GtkWidget *separator;
|
|
GList *children;
|
|
|
|
children = gtk_container_get_children (GTK_CONTAINER (dock->vbox));
|
|
|
|
separator = g_list_nth_data (children, 2);
|
|
|
|
gtk_container_remove (GTK_CONTAINER (dock->vbox), separator);
|
|
gtk_container_remove (GTK_CONTAINER (dock->vbox), GTK_WIDGET (dockbook));
|
|
|
|
g_list_free (children);
|
|
}
|
|
else
|
|
{
|
|
GtkWidget *other_book;
|
|
GtkWidget *parent;
|
|
GtkWidget *grandparent;
|
|
|
|
parent = GTK_WIDGET (dockbook)->parent;
|
|
grandparent = parent->parent;
|
|
|
|
if (index == 0)
|
|
other_book = GTK_PANED (parent)->child2;
|
|
else
|
|
other_book = GTK_PANED (parent)->child1;
|
|
|
|
g_object_ref (G_OBJECT (other_book));
|
|
|
|
gtk_container_remove (GTK_CONTAINER (parent), other_book);
|
|
gtk_container_remove (GTK_CONTAINER (parent), GTK_WIDGET (dockbook));
|
|
|
|
gtk_container_remove (GTK_CONTAINER (grandparent), parent);
|
|
|
|
if (GTK_IS_VPANED (grandparent))
|
|
gtk_paned_pack1 (GTK_PANED (grandparent), other_book, TRUE, FALSE);
|
|
else
|
|
gtk_box_pack_start (GTK_BOX (dock->vbox), other_book, TRUE, TRUE, 0);
|
|
|
|
g_object_unref (G_OBJECT (other_book));
|
|
}
|
|
|
|
if ((old_length == 1) && (dock->destroy_if_empty))
|
|
{
|
|
gtk_widget_destroy (GTK_WIDGET (dock));
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gimp_dock_separator_button_press (GtkWidget *widget,
|
|
GdkEventButton *bevent,
|
|
gpointer data)
|
|
{
|
|
if (bevent->type == GDK_BUTTON_PRESS)
|
|
{
|
|
if (bevent->button == 1)
|
|
{
|
|
gtk_grab_add (widget);
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
gimp_dock_separator_button_release (GtkWidget *widget,
|
|
GdkEventButton *bevent,
|
|
gpointer data)
|
|
{
|
|
if (bevent->button == 1)
|
|
{
|
|
gtk_grab_remove (widget);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
static void
|
|
gimp_dock_tab_drag_begin (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gpointer data)
|
|
{
|
|
GimpDockable *dockable;
|
|
GtkWidget *window;
|
|
GtkWidget *frame;
|
|
GtkWidget *label;
|
|
|
|
dockable = GIMP_DOCKABLE (data);
|
|
|
|
window = gtk_window_new (GTK_WINDOW_POPUP);
|
|
|
|
frame = gtk_frame_new (NULL);
|
|
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_OUT);
|
|
gtk_container_add (GTK_CONTAINER (window), frame);
|
|
gtk_widget_show (frame);
|
|
|
|
label = gtk_label_new (dockable->name);
|
|
gtk_misc_set_padding (GTK_MISC (label), 10, 5);
|
|
gtk_container_add (GTK_CONTAINER (frame), label);
|
|
gtk_widget_show (label);
|
|
|
|
gtk_widget_show (window);
|
|
|
|
g_object_set_data_full (G_OBJECT (dockable), "gimp-dock-drag-widget",
|
|
window,
|
|
(GDestroyNotify) gtk_widget_destroy);
|
|
|
|
gtk_drag_set_icon_widget (context, window,
|
|
-8, -8);
|
|
}
|
|
|
|
static void
|
|
gimp_dock_tab_drag_end (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gpointer data)
|
|
{
|
|
GimpDockable *dockable;
|
|
GtkWidget *drag_widget;
|
|
|
|
dockable = GIMP_DOCKABLE (data);
|
|
|
|
drag_widget = g_object_get_data (G_OBJECT (dockable),
|
|
"gimp-dock-drag-widget");
|
|
|
|
if (drag_widget)
|
|
{
|
|
GtkWidget *dock;
|
|
|
|
g_object_set_data (G_OBJECT (dockable), "gimp-dock-drag-widget", NULL);
|
|
|
|
dock = gimp_dock_new ();
|
|
|
|
gtk_window_set_position (GTK_WINDOW (dock), GTK_WIN_POS_MOUSE);
|
|
|
|
g_object_ref (G_OBJECT (dockable));
|
|
|
|
gimp_dock_remove (dockable->dock, dockable);
|
|
gimp_dock_add (GIMP_DOCK (dock), dockable, -1, -1);
|
|
|
|
g_object_unref (G_OBJECT (dockable));
|
|
|
|
gtk_widget_show (dock);
|
|
}
|
|
}
|
|
*/
|
|
|
|
static gboolean
|
|
gimp_dock_separator_drag_drop (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gint x,
|
|
gint y,
|
|
guint time,
|
|
gpointer data)
|
|
{
|
|
GimpDock *dock;
|
|
GtkWidget *source;
|
|
|
|
dock = GIMP_DOCK (data);
|
|
|
|
source = gtk_drag_get_source_widget (context);
|
|
|
|
if (source)
|
|
{
|
|
GimpDockable *src_dockable;
|
|
|
|
src_dockable = (GimpDockable *) g_object_get_data (G_OBJECT (source),
|
|
"gimp-dockable");
|
|
|
|
if (src_dockable)
|
|
{
|
|
GtkWidget *dockbook;
|
|
GList *children;
|
|
gint index;
|
|
|
|
g_object_set_data (G_OBJECT (src_dockable),
|
|
"gimp-dock-drag-widget", NULL);
|
|
|
|
children = gtk_container_get_children (GTK_CONTAINER (widget->parent));
|
|
index = g_list_index (children, widget);
|
|
g_list_free (children);
|
|
|
|
if (index == 0)
|
|
index = 0;
|
|
else if (index == 2)
|
|
index = -1;
|
|
|
|
g_object_ref (G_OBJECT (src_dockable));
|
|
|
|
gimp_dockbook_remove (src_dockable->dockbook, src_dockable);
|
|
|
|
dockbook = gimp_dockbook_new ();
|
|
gimp_dock_add_book (dock, GIMP_DOCKBOOK (dockbook), index);
|
|
|
|
gimp_dockbook_add (GIMP_DOCKBOOK (dockbook), src_dockable, -1);
|
|
|
|
g_object_unref (G_OBJECT (src_dockable));
|
|
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|