mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-01 16:30:15 +00:00
4658d7ea54
This is in preparation of using input streams to show that these coordinates aren't needed most of the time and can otherwise be saved during GtkWidget::drag-drop.
372 lines
11 KiB
C
372 lines
11 KiB
C
#include <gtk/gtk.h>
|
|
|
|
static cairo_surface_t *
|
|
get_image_surface (GtkImage *image,
|
|
int *out_size)
|
|
{
|
|
GtkIconTheme *icon_theme;
|
|
const char *icon_name;
|
|
int width = 48;
|
|
cairo_surface_t *surface;
|
|
|
|
switch (gtk_image_get_storage_type (image))
|
|
{
|
|
case GTK_IMAGE_SURFACE:
|
|
surface = gtk_image_get_surface (image);
|
|
*out_size = cairo_image_surface_get_width (surface);
|
|
return cairo_surface_reference (surface);
|
|
case GTK_IMAGE_ICON_NAME:
|
|
icon_name = gtk_image_get_icon_name (image);
|
|
icon_theme = gtk_icon_theme_get_for_display (gtk_widget_get_display (GTK_WIDGET (image)));
|
|
*out_size = width;
|
|
return gtk_icon_theme_load_surface (icon_theme, icon_name, width, 1, NULL, GTK_ICON_LOOKUP_GENERIC_FALLBACK, NULL);
|
|
default:
|
|
g_warning ("Image storage type %d not handled",
|
|
gtk_image_get_storage_type (image));
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
enum {
|
|
TOP_LEFT,
|
|
CENTER,
|
|
BOTTOM_RIGHT
|
|
};
|
|
|
|
static void
|
|
image_drag_begin (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gpointer data)
|
|
{
|
|
cairo_surface_t *surface;
|
|
gint hotspot;
|
|
gint hot_x, hot_y;
|
|
gint size;
|
|
|
|
surface = get_image_surface (GTK_IMAGE (data), &size);
|
|
hotspot = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (data), "hotspot"));
|
|
switch (hotspot)
|
|
{
|
|
default:
|
|
case TOP_LEFT:
|
|
hot_x = 0;
|
|
hot_y = 0;
|
|
break;
|
|
case CENTER:
|
|
hot_x = size / 2;
|
|
hot_y = size / 2;
|
|
break;
|
|
case BOTTOM_RIGHT:
|
|
hot_x = size;
|
|
hot_y = size;
|
|
break;
|
|
}
|
|
cairo_surface_set_device_offset (surface, hot_x, hot_y);
|
|
gtk_drag_set_icon_surface (context, surface);
|
|
cairo_surface_destroy (surface);
|
|
}
|
|
|
|
static void
|
|
drag_widget_destroyed (GtkWidget *image, gpointer data)
|
|
{
|
|
GtkWidget *widget = data;
|
|
|
|
g_print ("drag widget destroyed\n");
|
|
g_object_unref (image);
|
|
g_object_set_data (G_OBJECT (widget), "drag widget", NULL);
|
|
}
|
|
|
|
static void
|
|
window_drag_end (GtkWidget *widget, GdkDragContext *context, gpointer data)
|
|
{
|
|
GtkWidget *window = data;
|
|
|
|
gtk_widget_destroy (window);
|
|
g_signal_handlers_disconnect_by_func (widget, window_drag_end, data);
|
|
}
|
|
|
|
static void
|
|
window_drag_begin (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gpointer data)
|
|
{
|
|
cairo_surface_t *surface;
|
|
GtkWidget *image;
|
|
int hotspot;
|
|
int size;
|
|
|
|
hotspot = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (data), "hotspot"));
|
|
|
|
image = g_object_get_data (G_OBJECT (widget), "drag widget");
|
|
if (image == NULL)
|
|
{
|
|
g_print ("creating new drag widget\n");
|
|
surface = get_image_surface (GTK_IMAGE (data), &size);
|
|
image = gtk_image_new_from_surface (surface);
|
|
cairo_surface_destroy (surface);
|
|
g_object_ref (image);
|
|
g_object_set_data (G_OBJECT (widget), "drag widget", image);
|
|
g_signal_connect (image, "destroy", G_CALLBACK (drag_widget_destroyed), widget);
|
|
}
|
|
else
|
|
g_print ("reusing drag widget\n");
|
|
|
|
gtk_drag_set_icon_widget (context, image, 0, 0);
|
|
|
|
if (hotspot == CENTER)
|
|
g_signal_connect (widget, "drag-end", G_CALLBACK (window_drag_end), image);
|
|
}
|
|
|
|
static void
|
|
update_source_target_list (GtkWidget *image)
|
|
{
|
|
GdkContentFormats *target_list;
|
|
|
|
target_list = gdk_content_formats_new (NULL, 0);
|
|
|
|
target_list = gtk_content_formats_add_image_targets (target_list, FALSE);
|
|
if (gtk_image_get_storage_type (GTK_IMAGE (image)) == GTK_IMAGE_ICON_NAME)
|
|
target_list = gtk_content_formats_add_text_targets (target_list);
|
|
|
|
gtk_drag_source_set_target_list (image, target_list);
|
|
|
|
gdk_content_formats_unref (target_list);
|
|
}
|
|
|
|
static void
|
|
update_dest_target_list (GtkWidget *image)
|
|
{
|
|
GdkContentFormats *target_list;
|
|
|
|
target_list = gdk_content_formats_new (NULL, 0);
|
|
|
|
target_list = gtk_content_formats_add_image_targets (target_list, FALSE);
|
|
target_list = gtk_content_formats_add_text_targets (target_list);
|
|
|
|
gtk_drag_dest_set_target_list (image, target_list);
|
|
|
|
gdk_content_formats_unref (target_list);
|
|
}
|
|
|
|
void
|
|
image_drag_data_get (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
GtkSelectionData *selection_data,
|
|
guint time,
|
|
gpointer data)
|
|
{
|
|
cairo_surface_t *surface;
|
|
const gchar *name;
|
|
int size;
|
|
|
|
if (gtk_selection_data_targets_include_image (selection_data, TRUE))
|
|
{
|
|
surface = get_image_surface (GTK_IMAGE (data), &size);
|
|
gtk_selection_data_set_surface (selection_data, surface);
|
|
}
|
|
else if (gtk_selection_data_targets_include_text (selection_data))
|
|
{
|
|
if (gtk_image_get_storage_type (GTK_IMAGE (data)) == GTK_IMAGE_ICON_NAME)
|
|
name = gtk_image_get_icon_name (GTK_IMAGE (data));
|
|
else
|
|
name = "Boo!";
|
|
gtk_selection_data_set_text (selection_data, name, -1);
|
|
}
|
|
else
|
|
{
|
|
g_assert_not_reached ();
|
|
}
|
|
}
|
|
|
|
static void
|
|
image_drag_data_received (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
GtkSelectionData *selection_data,
|
|
guint32 time,
|
|
gpointer data)
|
|
{
|
|
cairo_surface_t *surface;
|
|
gchar *text;
|
|
|
|
if (gtk_selection_data_get_length (selection_data) == 0)
|
|
return;
|
|
|
|
if (gtk_selection_data_targets_include_image (selection_data, FALSE))
|
|
{
|
|
surface = gtk_selection_data_get_surface (selection_data);
|
|
gtk_image_set_from_surface (GTK_IMAGE (data), surface);
|
|
cairo_surface_destroy (surface);
|
|
}
|
|
else if (gtk_selection_data_targets_include_text (selection_data))
|
|
{
|
|
text = (gchar *)gtk_selection_data_get_text (selection_data);
|
|
gtk_image_set_from_icon_name (GTK_IMAGE (data), text);
|
|
g_free (text);
|
|
}
|
|
else
|
|
{
|
|
g_assert_not_reached ();
|
|
}
|
|
}
|
|
|
|
|
|
GtkWidget *
|
|
make_image (const gchar *icon_name, int hotspot)
|
|
{
|
|
GtkWidget *image;
|
|
|
|
image = gtk_image_new_from_icon_name (icon_name);
|
|
gtk_image_set_icon_size (GTK_IMAGE (image), GTK_ICON_SIZE_LARGE);
|
|
|
|
gtk_drag_source_set (image, GDK_BUTTON1_MASK, NULL, GDK_ACTION_COPY);
|
|
update_source_target_list (image);
|
|
|
|
g_object_set_data (G_OBJECT (image), "hotspot", GINT_TO_POINTER (hotspot));
|
|
|
|
g_signal_connect (image, "drag-begin", G_CALLBACK (image_drag_begin), image);
|
|
g_signal_connect (image, "drag-data-get", G_CALLBACK (image_drag_data_get), image);
|
|
|
|
gtk_drag_dest_set (image, GTK_DEST_DEFAULT_ALL, NULL, GDK_ACTION_COPY);
|
|
g_signal_connect (image, "drag-data-received", G_CALLBACK (image_drag_data_received), image);
|
|
update_dest_target_list (image);
|
|
|
|
return image;
|
|
}
|
|
|
|
GtkWidget *
|
|
make_image2 (const gchar *icon_name, int hotspot)
|
|
{
|
|
GtkWidget *image;
|
|
|
|
image = gtk_image_new_from_icon_name (icon_name);
|
|
gtk_image_set_icon_size (GTK_IMAGE (image), GTK_ICON_SIZE_LARGE);
|
|
|
|
gtk_drag_source_set (image, GDK_BUTTON1_MASK, NULL, GDK_ACTION_COPY);
|
|
update_source_target_list (image);
|
|
|
|
g_object_set_data (G_OBJECT (image), "hotspot", GINT_TO_POINTER (hotspot));
|
|
|
|
g_signal_connect (image, "drag-begin", G_CALLBACK (window_drag_begin), image);
|
|
g_signal_connect (image, "drag-data-get", G_CALLBACK (image_drag_data_get), image);
|
|
|
|
gtk_drag_dest_set (image, GTK_DEST_DEFAULT_ALL, NULL, GDK_ACTION_COPY);
|
|
g_signal_connect (image, "drag-data-received", G_CALLBACK (image_drag_data_received), image);
|
|
update_dest_target_list (image);
|
|
|
|
return image;
|
|
}
|
|
|
|
static void
|
|
spinner_drag_begin (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gpointer data)
|
|
{
|
|
GtkWidget *spinner;
|
|
|
|
g_print ("GtkWidget::drag-begin\n");
|
|
spinner = g_object_new (GTK_TYPE_SPINNER,
|
|
"visible", TRUE,
|
|
"active", TRUE,
|
|
NULL);
|
|
gtk_drag_set_icon_widget (context, spinner, 0, 0);
|
|
g_object_set_data (G_OBJECT (context), "spinner", spinner);
|
|
}
|
|
|
|
static void
|
|
spinner_drag_end (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gpointer data)
|
|
{
|
|
GtkWidget *spinner;
|
|
|
|
g_print ("GtkWidget::drag-end\n");
|
|
spinner = g_object_get_data (G_OBJECT (context), "spinner");
|
|
gtk_widget_destroy (spinner);
|
|
}
|
|
|
|
static gboolean
|
|
spinner_drag_failed (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
GtkDragResult result,
|
|
gpointer data)
|
|
{
|
|
GTypeClass *class;
|
|
GEnumValue *value;
|
|
|
|
class = g_type_class_ref (GTK_TYPE_DRAG_RESULT);
|
|
value = g_enum_get_value (G_ENUM_CLASS (class), result);
|
|
g_print ("GtkWidget::drag-failed %s\n", value->value_nick);
|
|
g_type_class_unref (class);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
void
|
|
spinner_drag_data_get (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
GtkSelectionData *selection_data,
|
|
guint time,
|
|
gpointer data)
|
|
{
|
|
g_print ("GtkWidget::drag-data-get\n");
|
|
gtk_selection_data_set_text (selection_data, "ACTIVE", -1);
|
|
}
|
|
|
|
static GtkWidget *
|
|
make_spinner (void)
|
|
{
|
|
GtkWidget *spinner;
|
|
|
|
spinner = gtk_spinner_new ();
|
|
gtk_spinner_start (GTK_SPINNER (spinner));
|
|
|
|
gtk_drag_source_set (spinner, GDK_BUTTON1_MASK, NULL, GDK_ACTION_COPY);
|
|
gtk_drag_source_add_text_targets (spinner);
|
|
|
|
g_signal_connect (spinner, "drag-begin", G_CALLBACK (spinner_drag_begin), spinner);
|
|
g_signal_connect (spinner, "drag-end", G_CALLBACK (spinner_drag_end), spinner);
|
|
g_signal_connect (spinner, "drag-failed", G_CALLBACK (spinner_drag_failed), spinner);
|
|
g_signal_connect (spinner, "drag-data-get", G_CALLBACK (spinner_drag_data_get), spinner);
|
|
|
|
return spinner;
|
|
}
|
|
|
|
int
|
|
main (int argc, char *Argv[])
|
|
{
|
|
GtkWidget *window;
|
|
GtkWidget *grid;
|
|
GtkWidget *entry;
|
|
|
|
gtk_init ();
|
|
|
|
window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_window_set_title (GTK_WINDOW (window), "Drag And Drop");
|
|
gtk_window_set_resizable (GTK_WINDOW (window), FALSE);
|
|
|
|
grid = gtk_grid_new ();
|
|
g_object_set (grid,
|
|
"margin", 20,
|
|
"row-spacing", 20,
|
|
"column-spacing", 20,
|
|
NULL);
|
|
gtk_container_add (GTK_CONTAINER (window), grid);
|
|
gtk_grid_attach (GTK_GRID (grid), make_image ("dialog-warning", TOP_LEFT), 0, 0, 1, 1);
|
|
gtk_grid_attach (GTK_GRID (grid), make_image ("process-stop", BOTTOM_RIGHT), 1, 0, 1, 1);
|
|
|
|
entry = gtk_entry_new ();
|
|
gtk_grid_attach (GTK_GRID (grid), entry, 0, 1, 2, 1);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), make_spinner (), 0, 2, 1, 1);
|
|
gtk_grid_attach (GTK_GRID (grid), make_image ("weather-clear", CENTER), 1, 2, 1, 1);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), make_image2 ("dialog-question", TOP_LEFT), 0, 3, 1, 1);
|
|
|
|
gtk_grid_attach (GTK_GRID (grid), make_image2 ("dialog-information", CENTER), 1, 3, 1, 1);
|
|
|
|
gtk_widget_show (window);
|
|
gtk_main ();
|
|
|
|
return 0;
|
|
}
|