
2005-07-25 Michael Natterer <mitch@gimp.org> Some DND fixes / cleanup: * app/widgets/widgets-enums.h: renamed GIMP_DND_TYPE_TOOL to GIMP_DND_TYPE_TOOL_INFO. * app/widgets/gimpselectiondata.[ch]: s/tool/tool_info/g. Moved private functions to the end of the file. Include GIMP's PID in all GtkSelectionData strings which are used to pass around stuff by reference. For things which are referenced by name, also encode the object's address in the GtkSelectionData so having a brush called "Standard" or a named buffer called "Global Buffer" will work together with DND. * app/widgets/gimpdnd.[ch]: s/tool/tool_info/g. Renamed gimp_dnd_get_data_data() to gimp_dnd_get_object_data() since it's not limited to GimpData objects. Follow above selection data API changes. Cleanup. * libgimp/gimpbrushmenu.c * libgimp/gimpdrawablecombobox.c * libgimp/gimpfontselectbutton.c * libgimp/gimpgradientmenu.c * libgimp/gimpimagecombobox.c * libgimp/gimppalettemenu.c * libgimp/gimppatternmenu.c: follow GtkSelectionData format change and check the dropped things' PID against the return value of gimp_getpid().
425 lines
13 KiB
C
425 lines
13 KiB
C
/* LIBGIMP - The GIMP Library
|
|
* Copyright (C) 1995-1997 Peter Mattis and Spencer Kimball
|
|
*
|
|
* gimpdrawablecombobox.c
|
|
* Copyright (C) 2004 Sven Neumann <sven@gimp.org>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
#include "libgimpwidgets/gimpwidgets.h"
|
|
|
|
#include "gimp.h"
|
|
|
|
#include "gimpuitypes.h"
|
|
#include "gimpdrawablecombobox.h"
|
|
#include "gimppixbuf.h"
|
|
|
|
|
|
#define THUMBNAIL_SIZE 24
|
|
#define WIDTH_REQUEST 200
|
|
|
|
|
|
typedef struct _GimpDrawableComboBoxClass GimpDrawableComboBoxClass;
|
|
typedef struct _GimpChannelComboBoxClass GimpChannelComboBoxClass;
|
|
typedef struct _GimpLayerComboBoxClass GimpLayerComboBoxClass;
|
|
|
|
struct _GimpDrawableComboBox
|
|
{
|
|
GimpIntComboBox parent_instance;
|
|
};
|
|
|
|
struct _GimpDrawableComboBoxClass
|
|
{
|
|
GimpIntComboBoxClass parent_class;
|
|
};
|
|
|
|
struct _GimpChannelComboBox
|
|
{
|
|
GimpIntComboBox parent_instance;
|
|
};
|
|
|
|
struct _GimpChannelComboBoxClass
|
|
{
|
|
GimpIntComboBoxClass parent_class;
|
|
};
|
|
|
|
struct _GimpLayerComboBox
|
|
{
|
|
GimpIntComboBox parent_instance;
|
|
};
|
|
|
|
struct _GimpLayerComboBoxClass
|
|
{
|
|
GimpIntComboBoxClass parent_class;
|
|
};
|
|
|
|
|
|
static void gimp_drawable_combo_box_model_add (GtkListStore *store,
|
|
gint32 image,
|
|
gint num_drawables,
|
|
gint32 *drawables,
|
|
GimpDrawableConstraintFunc constraint,
|
|
gpointer data);
|
|
|
|
static void gimp_drawable_combo_box_drag_data_received (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gint x,
|
|
gint y,
|
|
GtkSelectionData *selection,
|
|
guint info,
|
|
guint time);
|
|
|
|
|
|
static const GtkTargetEntry targets[] =
|
|
{
|
|
{ "application/x-gimp-channel-id", 0 },
|
|
{ "application/x-gimp-layer-id", 0 }
|
|
};
|
|
|
|
|
|
G_DEFINE_TYPE(GimpDrawableComboBox,
|
|
gimp_drawable_combo_box,
|
|
GIMP_TYPE_INT_COMBO_BOX);
|
|
|
|
static void
|
|
gimp_drawable_combo_box_class_init (GimpDrawableComboBoxClass *klass)
|
|
{
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
widget_class->drag_data_received = gimp_drawable_combo_box_drag_data_received;
|
|
}
|
|
|
|
static void
|
|
gimp_drawable_combo_box_init (GimpDrawableComboBox *combo_box)
|
|
{
|
|
gtk_drag_dest_set (GTK_WIDGET (combo_box),
|
|
GTK_DEST_DEFAULT_HIGHLIGHT |
|
|
GTK_DEST_DEFAULT_MOTION |
|
|
GTK_DEST_DEFAULT_DROP,
|
|
targets, 2,
|
|
GDK_ACTION_COPY);
|
|
}
|
|
|
|
/**
|
|
* gimp_drawable_combo_box_new:
|
|
* @constraint: a #GimpDrawableConstraintFunc or %NULL
|
|
* @data: a pointer that is passed to @constraint
|
|
*
|
|
* Creates a new #GimpIntComboBox filled with all currently opened
|
|
* drawables. If a @constraint function is specified, it is called for
|
|
* each drawable and only if the function returns %TRUE, the drawable
|
|
* is added to the combobox.
|
|
*
|
|
* You should use gimp_int_combo_box_connect() to initialize and connect
|
|
* the combo. Use gimp_int_combo_box_set_active() to get the active
|
|
* drawable ID and gimp_int_combo_box_get_active() to retrieve the ID
|
|
* of the selected drawable.
|
|
*
|
|
* Return value: a new #GimpIntComboBox.
|
|
*
|
|
* Since: GIMP 2.2
|
|
**/
|
|
GtkWidget *
|
|
gimp_drawable_combo_box_new (GimpDrawableConstraintFunc constraint,
|
|
gpointer data)
|
|
{
|
|
GtkWidget *combo_box;
|
|
GtkTreeModel *model;
|
|
GtkTreeIter iter;
|
|
gint32 *images;
|
|
gint num_images;
|
|
gint i;
|
|
|
|
combo_box = g_object_new (GIMP_TYPE_DRAWABLE_COMBO_BOX,
|
|
"width-request", WIDTH_REQUEST,
|
|
"ellipsize", PANGO_ELLIPSIZE_MIDDLE,
|
|
NULL);
|
|
|
|
model = gtk_combo_box_get_model (GTK_COMBO_BOX (combo_box));
|
|
|
|
images = gimp_image_list (&num_images);
|
|
|
|
for (i = 0; i < num_images; i++)
|
|
{
|
|
gint32 *drawables;
|
|
gint num_drawables;
|
|
|
|
drawables = gimp_image_get_layers (images[i], &num_drawables);
|
|
gimp_drawable_combo_box_model_add (GTK_LIST_STORE (model),
|
|
images[i],
|
|
num_drawables, drawables,
|
|
constraint, data);
|
|
g_free (drawables);
|
|
|
|
drawables = gimp_image_get_channels (images[i], &num_drawables);
|
|
gimp_drawable_combo_box_model_add (GTK_LIST_STORE (model),
|
|
images[i],
|
|
num_drawables, drawables,
|
|
constraint, data);
|
|
g_free (drawables);
|
|
}
|
|
|
|
g_free (images);
|
|
|
|
if (gtk_tree_model_get_iter_first (model, &iter))
|
|
gtk_combo_box_set_active_iter (GTK_COMBO_BOX (combo_box), &iter);
|
|
|
|
return combo_box;
|
|
}
|
|
|
|
|
|
G_DEFINE_TYPE(GimpChannelComboBox,
|
|
gimp_channel_combo_box,
|
|
GIMP_TYPE_INT_COMBO_BOX);
|
|
|
|
static void
|
|
gimp_channel_combo_box_class_init (GimpChannelComboBoxClass *klass)
|
|
{
|
|
GtkWidgetClass *widget_class;
|
|
|
|
widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
widget_class->drag_data_received = gimp_drawable_combo_box_drag_data_received;
|
|
}
|
|
|
|
static void
|
|
gimp_channel_combo_box_init (GimpChannelComboBox *combo_box)
|
|
{
|
|
gtk_drag_dest_set (GTK_WIDGET (combo_box),
|
|
GTK_DEST_DEFAULT_HIGHLIGHT |
|
|
GTK_DEST_DEFAULT_MOTION |
|
|
GTK_DEST_DEFAULT_DROP,
|
|
targets, 1,
|
|
GDK_ACTION_COPY);
|
|
}
|
|
|
|
/**
|
|
* gimp_channel_combo_box_new:
|
|
* @constraint: a #GimpDrawableConstraintFunc or %NULL
|
|
* @data: a pointer that is passed to @constraint
|
|
*
|
|
* Creates a new #GimpIntComboBox filled with all currently opened
|
|
* channels. See gimp_drawable_combo_box() for more info.
|
|
*
|
|
* Return value: a new #GimpIntComboBox.
|
|
*
|
|
* Since: GIMP 2.2
|
|
**/
|
|
GtkWidget *
|
|
gimp_channel_combo_box_new (GimpDrawableConstraintFunc constraint,
|
|
gpointer data)
|
|
{
|
|
GtkWidget *combo_box;
|
|
GtkTreeModel *model;
|
|
GtkTreeIter iter;
|
|
gint32 *images;
|
|
gint num_images;
|
|
gint i;
|
|
|
|
combo_box = g_object_new (GIMP_TYPE_CHANNEL_COMBO_BOX,
|
|
"width-request", WIDTH_REQUEST,
|
|
"ellipsize", PANGO_ELLIPSIZE_MIDDLE,
|
|
NULL);
|
|
|
|
model = gtk_combo_box_get_model (GTK_COMBO_BOX (combo_box));
|
|
|
|
images = gimp_image_list (&num_images);
|
|
|
|
for (i = 0; i < num_images; i++)
|
|
{
|
|
gint32 *drawables;
|
|
gint num_drawables;
|
|
|
|
drawables = gimp_image_get_channels (images[i], &num_drawables);
|
|
gimp_drawable_combo_box_model_add (GTK_LIST_STORE (model),
|
|
images[i],
|
|
num_drawables, drawables,
|
|
constraint, data);
|
|
g_free (drawables);
|
|
}
|
|
|
|
g_free (images);
|
|
|
|
if (gtk_tree_model_get_iter_first (model, &iter))
|
|
gtk_combo_box_set_active_iter (GTK_COMBO_BOX (combo_box), &iter);
|
|
|
|
return combo_box;
|
|
}
|
|
|
|
|
|
G_DEFINE_TYPE(GimpLayerComboBox,
|
|
gimp_layer_combo_box,
|
|
GIMP_TYPE_INT_COMBO_BOX);
|
|
|
|
static void
|
|
gimp_layer_combo_box_class_init (GimpLayerComboBoxClass *klass)
|
|
{
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
widget_class->drag_data_received = gimp_drawable_combo_box_drag_data_received;
|
|
}
|
|
|
|
static void
|
|
gimp_layer_combo_box_init (GimpLayerComboBox *combo_box)
|
|
{
|
|
gtk_drag_dest_set (GTK_WIDGET (combo_box),
|
|
GTK_DEST_DEFAULT_HIGHLIGHT |
|
|
GTK_DEST_DEFAULT_MOTION |
|
|
GTK_DEST_DEFAULT_DROP,
|
|
targets + 1, 1,
|
|
GDK_ACTION_COPY);
|
|
}
|
|
|
|
/**
|
|
* gimp_layer_combo_box_new:
|
|
* @constraint: a #GimpDrawableConstraintFunc or %NULL
|
|
* @data: a pointer that is passed to @constraint
|
|
*
|
|
* Creates a new #GimpIntComboBox filled with all currently opened
|
|
* layers. See gimp_drawable_combo_box() for more info.
|
|
*
|
|
* Return value: a new #GimpIntComboBox.
|
|
*
|
|
* Since: GIMP 2.2
|
|
**/
|
|
GtkWidget *
|
|
gimp_layer_combo_box_new (GimpDrawableConstraintFunc constraint,
|
|
gpointer data)
|
|
{
|
|
GtkWidget *combo_box;
|
|
GtkTreeModel *model;
|
|
GtkTreeIter iter;
|
|
gint32 *images;
|
|
gint num_images;
|
|
gint i;
|
|
|
|
combo_box = g_object_new (GIMP_TYPE_LAYER_COMBO_BOX,
|
|
"width-request", WIDTH_REQUEST,
|
|
"ellipsize", PANGO_ELLIPSIZE_MIDDLE,
|
|
NULL);
|
|
|
|
model = gtk_combo_box_get_model (GTK_COMBO_BOX (combo_box));
|
|
|
|
images = gimp_image_list (&num_images);
|
|
|
|
for (i = 0; i < num_images; i++)
|
|
{
|
|
gint32 *drawables;
|
|
gint num_drawables;
|
|
|
|
drawables = gimp_image_get_layers (images[i], &num_drawables);
|
|
gimp_drawable_combo_box_model_add (GTK_LIST_STORE (model),
|
|
images[i],
|
|
num_drawables, drawables,
|
|
constraint, data);
|
|
g_free (drawables);
|
|
}
|
|
|
|
g_free (images);
|
|
|
|
if (gtk_tree_model_get_iter_first (model, &iter))
|
|
gtk_combo_box_set_active_iter (GTK_COMBO_BOX (combo_box), &iter);
|
|
|
|
return combo_box;
|
|
}
|
|
|
|
static void
|
|
gimp_drawable_combo_box_model_add (GtkListStore *store,
|
|
gint32 image,
|
|
gint num_drawables,
|
|
gint32 *drawables,
|
|
GimpDrawableConstraintFunc constraint,
|
|
gpointer data)
|
|
{
|
|
GtkTreeIter iter;
|
|
gint i;
|
|
|
|
for (i = 0; i < num_drawables; i++)
|
|
{
|
|
if (! constraint || (* constraint) (image, drawables[i], data))
|
|
{
|
|
gchar *image_name = gimp_image_get_name (image);
|
|
gchar *drawable_name = gimp_drawable_get_name (drawables[i]);
|
|
gchar *label;
|
|
GdkPixbuf *thumb;
|
|
|
|
label = g_strdup_printf ("%s-%d/%s-%d",
|
|
image_name, image,
|
|
drawable_name, drawables[i]);
|
|
|
|
g_free (drawable_name);
|
|
g_free (image_name);
|
|
|
|
thumb = gimp_drawable_get_thumbnail (drawables[i],
|
|
THUMBNAIL_SIZE, THUMBNAIL_SIZE,
|
|
GIMP_PIXBUF_SMALL_CHECKS);
|
|
|
|
gtk_list_store_append (store, &iter);
|
|
gtk_list_store_set (store, &iter,
|
|
GIMP_INT_STORE_VALUE, drawables[i],
|
|
GIMP_INT_STORE_LABEL, label,
|
|
GIMP_INT_STORE_PIXBUF, thumb,
|
|
-1);
|
|
|
|
if (thumb)
|
|
g_object_unref (thumb);
|
|
|
|
g_free (label);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
gimp_drawable_combo_box_drag_data_received (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gint x,
|
|
gint y,
|
|
GtkSelectionData *selection,
|
|
guint info,
|
|
guint time)
|
|
{
|
|
gchar *str;
|
|
|
|
if ((selection->format != 8) || (selection->length < 1))
|
|
{
|
|
g_warning ("Received invalid drawable ID data!");
|
|
return;
|
|
}
|
|
|
|
str = g_strndup (selection->data, selection->length);
|
|
|
|
if (g_utf8_validate (str, -1, NULL))
|
|
{
|
|
gint pid;
|
|
gint ID;
|
|
|
|
if (sscanf (str, "%i:%i", &pid, &ID) == 2 &&
|
|
pid == gimp_getpid ())
|
|
{
|
|
gimp_int_combo_box_set_active (GIMP_INT_COMBO_BOX (widget), ID);
|
|
}
|
|
}
|
|
|
|
g_free (str);
|
|
}
|