forked from AuroraMiddleware/gtk
ef982b7d46
Rename the can-pick property to can-target, and redo the pick flags with more descriptive names.
379 lines
10 KiB
C
379 lines
10 KiB
C
/* Drag-and-Drop
|
|
*
|
|
* I can't believe its not glade!
|
|
*/
|
|
|
|
#include <glib/gi18n.h>
|
|
#include <gtk/gtk.h>
|
|
#include <string.h>
|
|
|
|
typedef struct _GtkDemoWidget GtkDemoWidget;
|
|
struct _GtkDemoWidget
|
|
{
|
|
GType type;
|
|
union {
|
|
char *text;
|
|
gboolean active;
|
|
};
|
|
};
|
|
|
|
static gpointer
|
|
copy_demo_widget (gpointer data)
|
|
{
|
|
GtkDemoWidget *demo = g_memdup (data, sizeof (GtkDemoWidget));
|
|
|
|
if (demo->type == GTK_TYPE_LABEL)
|
|
demo->text = g_strdup (demo->text);
|
|
|
|
return demo;
|
|
}
|
|
|
|
static void
|
|
free_demo_widget (gpointer data)
|
|
{
|
|
GtkDemoWidget *demo = data;
|
|
|
|
if (demo->type == GTK_TYPE_LABEL)
|
|
g_free (demo->text);
|
|
|
|
g_free (demo);
|
|
}
|
|
|
|
#define GTK_TYPE_DEMO_WIDGET (gtk_demo_widget_get_type ())
|
|
|
|
G_DEFINE_BOXED_TYPE (GtkDemoWidget, gtk_demo_widget, copy_demo_widget, free_demo_widget)
|
|
|
|
static GtkDemoWidget *
|
|
serialize_widget (GtkWidget *widget)
|
|
{
|
|
GtkDemoWidget *demo;
|
|
|
|
demo = g_new0 (GtkDemoWidget, 1);
|
|
demo->type = G_OBJECT_TYPE (widget);
|
|
|
|
if (GTK_IS_LABEL (widget))
|
|
{
|
|
demo->text = g_strdup (gtk_label_get_text (GTK_LABEL (widget)));
|
|
}
|
|
else if (GTK_IS_SPINNER (widget))
|
|
{
|
|
g_object_get (widget, "active", &demo->active, NULL);
|
|
}
|
|
else
|
|
{
|
|
g_print ("Type %s not supported\n", g_type_name (demo->type));
|
|
}
|
|
|
|
return demo;
|
|
}
|
|
|
|
static GtkWidget *
|
|
deserialize_widget (GtkDemoWidget *demo)
|
|
{
|
|
GtkWidget *widget = NULL;
|
|
|
|
if (demo->type == GTK_TYPE_LABEL)
|
|
{
|
|
widget = gtk_label_new (demo->text);
|
|
}
|
|
else if (demo->type == GTK_TYPE_SPINNER)
|
|
{
|
|
widget = g_object_new (demo->type, "active", demo->active, NULL);
|
|
gtk_style_context_add_class (gtk_widget_get_style_context (widget), "demo");
|
|
}
|
|
else
|
|
{
|
|
g_print ("Type %s not supported\n", g_type_name (demo->type));
|
|
}
|
|
|
|
return widget;
|
|
}
|
|
|
|
static double pos_x, pos_y;
|
|
|
|
static void
|
|
new_label_cb (GtkMenuItem *item,
|
|
gpointer data)
|
|
{
|
|
GtkFixed *fixed = data;
|
|
GtkWidget *widget;
|
|
|
|
widget = gtk_label_new ("Label");
|
|
gtk_fixed_put (fixed, widget, pos_x, pos_y);
|
|
}
|
|
|
|
static void
|
|
new_spinner_cb (GtkMenuItem *item,
|
|
gpointer data)
|
|
{
|
|
GtkFixed *fixed = data;
|
|
GtkWidget *widget;
|
|
|
|
widget = gtk_spinner_new ();
|
|
gtk_style_context_add_class (gtk_widget_get_style_context (widget), "demo");
|
|
gtk_spinner_start (GTK_SPINNER (widget));
|
|
gtk_fixed_put (fixed, widget, pos_x, pos_y);
|
|
}
|
|
|
|
static void
|
|
copy_cb (GtkWidget *child)
|
|
{
|
|
GdkClipboard *clipboard;
|
|
GtkDemoWidget *demo;
|
|
|
|
g_print ("Copy %s\n", G_OBJECT_TYPE_NAME (child));
|
|
|
|
demo = serialize_widget (child);
|
|
|
|
clipboard = gdk_display_get_clipboard (gdk_display_get_default ());
|
|
gdk_clipboard_set (clipboard, GTK_TYPE_DEMO_WIDGET, demo);
|
|
}
|
|
|
|
static void
|
|
delete_cb (GtkWidget *child)
|
|
{
|
|
gtk_widget_destroy (child);
|
|
}
|
|
|
|
static void
|
|
cut_cb (GtkWidget *child)
|
|
{
|
|
copy_cb (child);
|
|
delete_cb (child);
|
|
}
|
|
|
|
static void
|
|
value_read (GObject *source,
|
|
GAsyncResult *res,
|
|
gpointer data)
|
|
{
|
|
GdkClipboard *clipboard = GDK_CLIPBOARD (source);
|
|
GError *error = NULL;
|
|
const GValue *value;
|
|
GtkDemoWidget *demo;
|
|
GtkWidget *widget = NULL;
|
|
|
|
value = gdk_clipboard_read_value_finish (clipboard, res, &error);
|
|
|
|
if (value == NULL)
|
|
{
|
|
g_print ("error: %s\n", error->message);
|
|
g_error_free (error);
|
|
return;
|
|
}
|
|
|
|
if (!G_VALUE_HOLDS (value, GTK_TYPE_DEMO_WIDGET))
|
|
{
|
|
g_print ("can't handle clipboard contents\n");
|
|
return;
|
|
}
|
|
|
|
demo = g_value_get_boxed (value);
|
|
widget = deserialize_widget (demo);
|
|
|
|
gtk_fixed_put (GTK_FIXED (data), widget, pos_x, pos_y);
|
|
}
|
|
|
|
static void
|
|
paste_cb (GtkWidget *fixed)
|
|
{
|
|
GdkClipboard *clipboard;
|
|
|
|
clipboard = gdk_display_get_clipboard (gdk_display_get_default ());
|
|
if (gdk_content_formats_contain_gtype (gdk_clipboard_get_formats (clipboard), GTK_TYPE_DEMO_WIDGET))
|
|
{
|
|
g_print ("Paste %s\n", g_type_name (GTK_TYPE_DEMO_WIDGET));
|
|
gdk_clipboard_read_value_async (clipboard, GTK_TYPE_DEMO_WIDGET, 0, NULL, value_read, fixed);
|
|
}
|
|
else
|
|
g_print ("Don't know how to handle clipboard contents\n");
|
|
}
|
|
|
|
static void
|
|
edit_label_done (GtkWidget *entry, gpointer data)
|
|
{
|
|
GtkWidget *fixed = gtk_widget_get_parent (entry);
|
|
GtkWidget *label;
|
|
int x, y;
|
|
|
|
gtk_fixed_get_child_position (GTK_FIXED (fixed), entry, &x, &y);
|
|
|
|
label = GTK_WIDGET (g_object_get_data (G_OBJECT (entry), "label"));
|
|
gtk_label_set_text (GTK_LABEL (label), gtk_editable_get_text (GTK_EDITABLE (entry)));
|
|
|
|
gtk_widget_destroy (entry);
|
|
}
|
|
|
|
static void
|
|
edit_cb (GtkWidget *child)
|
|
{
|
|
GtkWidget *fixed = gtk_widget_get_parent (child);
|
|
int x, y;
|
|
|
|
gtk_fixed_get_child_position (GTK_FIXED (fixed), child, &x, &y);
|
|
|
|
if (GTK_IS_LABEL (child))
|
|
{
|
|
GtkWidget *entry = gtk_entry_new ();
|
|
|
|
g_object_set_data (G_OBJECT (entry), "label", child);
|
|
|
|
gtk_editable_set_text (GTK_EDITABLE (entry), gtk_label_get_text (GTK_LABEL (child)));
|
|
g_signal_connect (entry, "activate", G_CALLBACK (edit_label_done), NULL);
|
|
gtk_fixed_put (GTK_FIXED (fixed), entry, x, y);
|
|
gtk_widget_grab_focus (entry);
|
|
}
|
|
else if (GTK_IS_SPINNER (child))
|
|
{
|
|
gboolean active;
|
|
|
|
g_object_get (child, "active", &active, NULL);
|
|
g_object_set (child, "active", !active, NULL);
|
|
}
|
|
}
|
|
|
|
static void
|
|
pressed_cb (GtkGesture *gesture,
|
|
int n_press,
|
|
double x,
|
|
double y,
|
|
gpointer data)
|
|
{
|
|
GtkWidget *widget;
|
|
GtkWidget *child;
|
|
|
|
widget = gtk_event_controller_get_widget (GTK_EVENT_CONTROLLER (gesture));
|
|
child = gtk_widget_pick (widget, x, y, GTK_PICK_DEFAULT);
|
|
|
|
if (gtk_gesture_single_get_current_button (GTK_GESTURE_SINGLE (gesture)) == GDK_BUTTON_SECONDARY)
|
|
{
|
|
GdkRectangle rect;
|
|
GtkWidget *menu;
|
|
GtkWidget *item;
|
|
GdkClipboard *clipboard;
|
|
|
|
pos_x = x;
|
|
pos_y = y;
|
|
|
|
menu = gtk_menu_new ();
|
|
item = gtk_menu_item_new_with_label ("New Label");
|
|
g_signal_connect (item, "activate", G_CALLBACK (new_label_cb), widget);
|
|
gtk_container_add (GTK_CONTAINER (menu), item);
|
|
item = gtk_menu_item_new_with_label ("New Spinner");
|
|
g_signal_connect (item, "activate", G_CALLBACK (new_spinner_cb), widget);
|
|
gtk_container_add (GTK_CONTAINER (menu), item);
|
|
|
|
item = gtk_separator_menu_item_new ();
|
|
gtk_container_add (GTK_CONTAINER (menu), item);
|
|
|
|
item = gtk_menu_item_new_with_label ("Edit");
|
|
gtk_widget_set_sensitive (item, child != NULL && child != widget);
|
|
g_signal_connect_swapped (item, "activate", G_CALLBACK (edit_cb), child);
|
|
gtk_container_add (GTK_CONTAINER (menu), item);
|
|
|
|
item = gtk_separator_menu_item_new ();
|
|
gtk_container_add (GTK_CONTAINER (menu), item);
|
|
|
|
item = gtk_menu_item_new_with_label ("Cut");
|
|
gtk_widget_set_sensitive (item, child != NULL && child != widget);
|
|
g_signal_connect_swapped (item, "activate", G_CALLBACK (cut_cb), child);
|
|
gtk_container_add (GTK_CONTAINER (menu), item);
|
|
item = gtk_menu_item_new_with_label ("Copy");
|
|
gtk_widget_set_sensitive (item, child != NULL && child != widget);
|
|
g_signal_connect_swapped (item, "activate", G_CALLBACK (copy_cb), child);
|
|
gtk_container_add (GTK_CONTAINER (menu), item);
|
|
item = gtk_menu_item_new_with_label ("Paste");
|
|
clipboard = gdk_display_get_clipboard (gdk_display_get_default ());
|
|
gtk_widget_set_sensitive (item,
|
|
gdk_content_formats_contain_gtype (gdk_clipboard_get_formats (clipboard), GTK_TYPE_DEMO_WIDGET));
|
|
g_signal_connect_swapped (item, "activate", G_CALLBACK (paste_cb), widget);
|
|
gtk_container_add (GTK_CONTAINER (menu), item);
|
|
item = gtk_menu_item_new_with_label ("Delete");
|
|
gtk_widget_set_sensitive (item, child != NULL && child != widget);
|
|
g_signal_connect_swapped (item, "activate", G_CALLBACK (delete_cb), child);
|
|
gtk_container_add (GTK_CONTAINER (menu), item);
|
|
|
|
rect.x = x;
|
|
rect.y = y;
|
|
rect.width = 0;
|
|
rect.height = 0;
|
|
|
|
gtk_menu_popup_at_rect (GTK_MENU (menu),
|
|
gtk_widget_get_surface (widget),
|
|
&rect,
|
|
GDK_GRAVITY_NORTH_WEST,
|
|
GDK_GRAVITY_NORTH_WEST,
|
|
NULL);
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void
|
|
released_cb (GtkGesture *gesture,
|
|
int n_press,
|
|
double x,
|
|
double y,
|
|
gpointer data)
|
|
{
|
|
GtkWidget *widget;
|
|
GtkWidget *child;
|
|
|
|
widget = gtk_event_controller_get_widget (GTK_EVENT_CONTROLLER (gesture));
|
|
child = gtk_widget_pick (widget, x, y, 0);
|
|
|
|
if (gtk_gesture_single_get_current_button (GTK_GESTURE_SINGLE (gesture)) == GDK_BUTTON_PRIMARY)
|
|
{
|
|
if (child != NULL && child != widget)
|
|
edit_cb (child);
|
|
}
|
|
}
|
|
|
|
static GtkWidget *window = NULL;
|
|
|
|
GtkWidget *
|
|
do_dnd (GtkWidget *do_widget)
|
|
{
|
|
if (!window)
|
|
{
|
|
GtkWidget *vbox, *fixed;
|
|
GtkGesture *multipress;
|
|
GtkCssProvider *provider;
|
|
|
|
window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
|
|
gtk_window_set_display (GTK_WINDOW (window),
|
|
gtk_widget_get_display (do_widget));
|
|
gtk_window_set_title (GTK_WINDOW (window), "Drag-and-drop");
|
|
|
|
g_signal_connect (window, "destroy",
|
|
G_CALLBACK (gtk_widget_destroyed), &window);
|
|
|
|
vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
|
gtk_container_add (GTK_CONTAINER (window), vbox);
|
|
|
|
fixed = gtk_fixed_new ();
|
|
gtk_container_add (GTK_CONTAINER (vbox), fixed);
|
|
gtk_widget_set_hexpand (fixed, TRUE);
|
|
gtk_widget_set_vexpand (fixed, TRUE);
|
|
|
|
multipress = gtk_gesture_multi_press_new ();
|
|
gtk_gesture_single_set_button (GTK_GESTURE_SINGLE (multipress), 0);
|
|
g_signal_connect (multipress, "pressed", G_CALLBACK (pressed_cb), NULL);
|
|
g_signal_connect (multipress, "released", G_CALLBACK (released_cb), NULL);
|
|
gtk_widget_add_controller (fixed, GTK_EVENT_CONTROLLER (multipress));
|
|
|
|
provider = gtk_css_provider_new ();
|
|
gtk_css_provider_load_from_resource (provider, "/dnd/dnd.css");
|
|
gtk_style_context_add_provider_for_display (gdk_display_get_default (),
|
|
GTK_STYLE_PROVIDER (provider),
|
|
800);
|
|
}
|
|
|
|
if (!gtk_widget_get_visible (window))
|
|
gtk_widget_show (window);
|
|
else
|
|
gtk_widget_destroy (window);
|
|
|
|
return window;
|
|
}
|