123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590 |
- /*
- * gui_frame_drag.c - GUI, dragging of frame items
- *
- * Written 2010 by Werner Almesberger
- * Copyright 2010 by Werner Almesberger
- *
- * 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.
- */
- #include <assert.h>
- #include <gtk/gtk.h>
- #include "util.h"
- #include "obj.h"
- #include "gui_util.h"
- #include "gui.h"
- #include "gui_canvas.h"
- #include "gui_frame_drag.h"
- #if 0
- #include "icons/frame.xpm"
- #endif
- enum {
- target_id_var,
- target_id_value,
- target_id_frame,
- target_id_canvas,
- };
- static GtkTargetEntry target_var = {
- .target = "var",
- .flags = GTK_TARGET_SAME_APP,
- .info = target_id_var,
- };
- static GtkTargetEntry target_value = {
- .target = "value",
- .flags = GTK_TARGET_SAME_APP,
- .info = target_id_value,
- };
- static GtkTargetEntry target_frame = {
- .target = "frame",
- .flags = GTK_TARGET_SAME_APP,
- .info = target_id_frame,
- };
- /* ----- dragging status --------------------------------------------------- */
- /*
- * Pointer to whatever it is we're dragging. NULL if not dragging.
- */
- static void *dragging;
- int is_dragging(void *this)
- {
- return this == dragging;
- }
- int is_dragging_anything(void)
- {
- return !!dragging;
- }
- /* ----- helper functions for indexed list --------------------------------- */
- #define NDX(first, item) \
- ({ typeof(first) NDX_walk; \
- int NDX_n = 0; \
- for (NDX_walk = (first); NDX_walk != (item); \
- NDX_walk = NDX_walk->next) \
- NDX_n++; \
- NDX_n; })
- #define NTH(first, n) \
- ({ typeof(first) *NTH_walk; \
- int NTH_n = (n); \
- for (NTH_walk = &(first); NTH_n; NTH_n--) \
- NTH_walk = &(*NTH_walk)->next; \
- NTH_walk; })
- #define FOR_UNORDERED(var, a, b) \
- for (var = (a) < (b) ? (a) : (b); var != ((a) < (b) ? (b) : (a)); \
- var++)
- /* ----- generic helper functions. maybe move to gui_util later ------------ */
- static void get_cell_coords(GtkWidget *widget, guint res[4])
- {
- GtkWidget *tab;
- tab = gtk_widget_get_ancestor(widget, GTK_TYPE_TABLE);
- gtk_container_child_get(GTK_CONTAINER(tab), widget,
- "left-attach", res,
- "right-attach", res+1,
- "top-attach", res+2,
- "bottom-attach", res+3, NULL);
- }
- static void swap_table_cells(GtkWidget *a, GtkWidget *b)
- {
- GtkWidget *tab_a, *tab_b;
- guint pos_a[4], pos_b[4];
- tab_a = gtk_widget_get_ancestor(a, GTK_TYPE_TABLE);
- tab_b = gtk_widget_get_ancestor(b, GTK_TYPE_TABLE);
- get_cell_coords(a, pos_a);
- get_cell_coords(b, pos_b);
- g_object_ref(a);
- g_object_ref(b);
- gtk_container_remove(GTK_CONTAINER(tab_a), a);
- gtk_container_remove(GTK_CONTAINER(tab_b), b);
- gtk_table_attach_defaults(GTK_TABLE(tab_a), b,
- pos_a[0], pos_a[1], pos_a[2], pos_a[3]);
- gtk_table_attach_defaults(GTK_TABLE(tab_b), a,
- pos_b[0], pos_b[1], pos_b[2], pos_b[3]);
- g_object_unref(a);
- g_object_unref(b);
- }
- static GtkWidget *pick_table_cell(GtkWidget *table, int x, int y)
- {
- GList *children, *walk;
- GtkWidget *child;
- guint pos[4];
- children = gtk_container_get_children(GTK_CONTAINER(table));
- for (walk = children; walk; walk = g_list_next(walk)) {
- child = g_list_nth_data(walk, 0);
- assert(child);
- get_cell_coords(child, pos);
- if (pos[0] == x && pos[2] == y)
- break;
- }
- g_list_free(children);
- return walk ? child : NULL;
- }
- static void swap_table_cells_by_coord(GtkWidget *table_a,
- int a_col, int a_row, GtkWidget *table_b, int b_col, int b_row)
- {
- GtkWidget *a, *b;
- a = pick_table_cell(table_a, a_col, a_row);
- b = pick_table_cell(table_b, b_col, b_row);
- if (a) {
- g_object_ref(a);
- gtk_container_remove(GTK_CONTAINER(table_a), a);
- }
- if (b) {
- g_object_ref(b);
- gtk_container_remove(GTK_CONTAINER(table_b), b);
- }
- if (a)
- gtk_table_attach_defaults(GTK_TABLE(table_b), a,
- b_col, b_col+1, b_row, b_row+1);
- if (b)
- gtk_table_attach_defaults(GTK_TABLE(table_a), b,
- a_col, a_col+1, a_row, a_row+1);
- if (a)
- g_object_unref(a);
- if (b)
- g_object_unref(b);
- }
- static void swap_table_rows(GtkWidget *table, int a, int b)
- {
- guint cols;
- int i;
- g_object_get(table, "n-columns", &cols, NULL);
- for (i = 0; i != cols; i++)
- swap_table_cells_by_coord(table, i, a, table, i, b);
- }
- /* ----- swap table items -------------------------------------------------- */
- static void swap_vars(struct table *table, int a, int b)
- {
- struct var **var_a, **var_b;
- var_a = NTH(table->vars, a);
- var_b = NTH(table->vars, b);
- swap_table_cells(box_of_label((*var_a)->widget),
- box_of_label((*var_b)->widget));
- SWAP(*var_a, *var_b);
- SWAP((*var_a)->next, (*var_b)->next);
- }
- static void swap_values(struct row *row, int a, int b)
- {
- struct value **value_a, **value_b;
- value_a = NTH(row->values, a);
- value_b = NTH(row->values, b);
- swap_table_cells(box_of_label((*value_a)->widget),
- box_of_label((*value_b)->widget));
- SWAP(*value_a, *value_b);
- SWAP((*value_a)->next, (*value_b)->next);
- }
- static void swap_cols(struct table *table, int a, int b)
- {
- struct row *row;
- swap_vars(table, a, b);
- for (row = table->rows; row; row = row->next)
- swap_values(row, a, b);
- }
- static void swap_rows(struct row **a, struct row **b)
- {
- struct value *value_a, *value_b;
- value_a = (*a)->values;
- value_b = (*b)->values;
- while (value_a) {
- swap_table_cells(box_of_label(value_a->widget),
- box_of_label(value_b->widget));
- value_a = value_a->next;
- value_b = value_b->next;
- }
- SWAP(*a, *b);
- SWAP((*a)->next, (*b)->next);
- }
- /* ----- swap frames ------------------------------------------------------- */
- static void swap_frames(GtkWidget *table, int a, int b)
- {
- struct frame **frame_a = NTH(frames, a);
- struct frame **frame_b = NTH(frames, b);
- swap_table_rows(table, 2*a+1, 2*b+1);
- swap_table_rows(table, 2*a+2, 2*b+2);
- SWAP(*frame_a, *frame_b);
- SWAP((*frame_a)->next, (*frame_b)->next);
- }
- /* ----- common functions -------------------------------------------------- */
- /*
- * according to
- * http://www.pubbs.net/201004/gtk/22819-re-drag-and-drop-drag-motion-cursor-lockup-fixed-.html
- * http://www.cryingwolf.org/articles/gtk-dnd.html
- */
- static int has_target(GtkWidget *widget, GdkDragContext *drag_context,
- const char *name)
- {
- GdkAtom target;
- target = gtk_drag_dest_find_target(widget, drag_context, NULL);
- /*
- * Force allocation so that we don't have to check for GDK_NONE.
- */
- return target == gdk_atom_intern(name, FALSE);
- }
- static void drag_begin(GtkWidget *widget,
- GtkTextDirection previous_direction, gpointer user_data)
- {
- GdkPixbuf *pixbuf;
- /*
- * Suppress the icon. PixBufs can't be zero-sized, but nobody will
- * notice a lone pixel either :-)
- */
- pixbuf =
- gdk_pixbuf_get_from_drawable(NULL, DA, NULL, 0, 0, 0, 0, 1, 1);
- gtk_drag_source_set_icon_pixbuf(widget, pixbuf);
- g_object_unref(pixbuf);
- dragging = user_data;
- }
- static void drag_end(GtkWidget *widget, GdkDragContext *drag_context,
- gpointer user_data)
- {
- dragging = NULL;
- }
- static void setup_drag_common(GtkWidget *widget, void *user_arg)
- {
- g_signal_connect(G_OBJECT(widget), "drag-begin",
- G_CALLBACK(drag_begin), user_arg);
- g_signal_connect(G_OBJECT(widget), "drag-end",
- G_CALLBACK(drag_end), user_arg);
- }
- /* ----- drag variables ---------------------------------------------------- */
- static gboolean drag_var_motion(GtkWidget *widget,
- GdkDragContext *drag_context, gint x, gint y, guint time_,
- gpointer user_data)
- {
- struct var *from = dragging;
- struct var *to = user_data;
- int from_n, to_n, i;
- if (!has_target(widget, drag_context, "var"))
- return FALSE;
- if (from == to || from->table != to->table)
- return FALSE;
- from_n = NDX(from->table->vars, from);
- to_n = NDX(to->table->vars, to);
- FOR_UNORDERED(i, from_n, to_n)
- swap_cols(from->table, i, i+1);
- return FALSE;
- }
- void setup_var_drag(struct var *var)
- {
- GtkWidget *box;
- box = box_of_label(var->widget);
- gtk_drag_source_set(box, GDK_BUTTON1_MASK,
- &target_var, 1, GDK_ACTION_PRIVATE);
- gtk_drag_dest_set(box, GTK_DEST_DEFAULT_MOTION,
- &target_var, 1, GDK_ACTION_PRIVATE);
- setup_drag_common(box, var);
- g_signal_connect(G_OBJECT(box), "drag-motion",
- G_CALLBACK(drag_var_motion), var);
- }
- /* ----- drag values ------------------------------------------------------- */
- static gboolean drag_value_motion(GtkWidget *widget,
- GdkDragContext *drag_context, gint x, gint y, guint time_,
- gpointer user_data)
- {
- struct value *from = dragging;
- struct value *to = user_data;
- struct table *table;
- struct row **row, *end;
- int from_n, to_n, i;
- if (!has_target(widget, drag_context, "value"))
- return FALSE;
- table = from->row->table;
- if (table != to->row->table)
- return FALSE;
- /* columns */
- from_n = NDX(from->row->values, from);
- to_n = NDX(to->row->values, to);
- FOR_UNORDERED(i, from_n, to_n)
- swap_cols(table, i, i+1);
- /* rows */
- if (from->row == to->row)
- return FALSE;
- row = &table->rows;
- while (1) {
- if (*row == from->row) {
- end = to->row;
- break;
- }
- if (*row == to->row) {
- end = from->row;
- break;
- }
- row = &(*row)->next;
- }
- while (1) {
- swap_rows(row, &(*row)->next);
- if (*row == end)
- break;
- row = &(*row)->next;
- }
- return FALSE;
- }
- void setup_value_drag(struct value *value)
- {
- GtkWidget *box;
- box = box_of_label(value->widget);
- gtk_drag_source_set(box, GDK_BUTTON1_MASK,
- &target_value, 1, GDK_ACTION_PRIVATE);
- gtk_drag_dest_set(box, GTK_DEST_DEFAULT_MOTION,
- &target_value, 1, GDK_ACTION_PRIVATE);
- setup_drag_common(box, value);
- g_signal_connect(G_OBJECT(box), "drag-motion",
- G_CALLBACK(drag_value_motion), value);
- }
- /* ----- frame to canvas helper functions ---------------------------------- */
- static int frame_on_canvas = 0;
- static void leave_canvas(void)
- {
- if (frame_on_canvas)
- canvas_frame_end();
- frame_on_canvas = 0;
- }
- /* ----- drag frame labels ------------------------------------------------- */
- #if 0
- /*
- * Setting our own icon looks nice but it slows things down to the point where
- * cursor movements can lag noticeable and it adds yet another element to an
- * already crowded cursor.
- */
- static void drag_frame_begin(GtkWidget *widget,
- GtkTextDirection previous_direction, gpointer user_data)
- {
- GdkPixmap *pixmap;
- GdkBitmap *mask;
- GdkColormap *cmap;
- pixmap = gdk_pixmap_create_from_xpm_d(DA, &mask, NULL, xpm_frame);
- cmap = gdk_drawable_get_colormap(root->window);
- gtk_drag_source_set_icon(widget, cmap, pixmap, mask);
- g_object_unref(pixmap);
- g_object_unref(mask);
- dragging = user_data;
- }
- #endif
- static gboolean drag_frame_motion(GtkWidget *widget,
- GdkDragContext *drag_context, gint x, gint y, guint time_,
- gpointer user_data)
- {
- struct frame *from = dragging;
- struct frame *to = user_data;
- int from_n, to_n, i;
- if (!has_target(widget, drag_context, "frame"))
- return FALSE;
- assert(from != frames);
- assert(to != frames);
- from_n = NDX(frames, from);
- to_n = NDX(frames, to);
- FOR_UNORDERED(i, from_n, to_n)
- swap_frames(gtk_widget_get_ancestor(widget, GTK_TYPE_TABLE),
- i, i+1);
- return FALSE;
- }
- static void drag_frame_end(GtkWidget *widget, GdkDragContext *drag_context,
- gpointer user_data)
- {
- leave_canvas();
- drag_end(widget, drag_context, user_data);
- }
- void setup_frame_drag(struct frame *frame)
- {
- GtkWidget *box;
- box = box_of_label(frame->label);
- gtk_drag_source_set(box, GDK_BUTTON1_MASK,
- &target_frame, 1, GDK_ACTION_COPY | GDK_ACTION_MOVE);
- gtk_drag_dest_set(box, GTK_DEST_DEFAULT_MOTION,
- &target_frame, 1, GDK_ACTION_MOVE);
- setup_drag_common(box, frame);
- /* override */
- #if 0
- g_signal_connect(G_OBJECT(box), "drag-begin",
- G_CALLBACK(drag_frame_begin), frame);
- #endif
- g_signal_connect(G_OBJECT(box), "drag-end",
- G_CALLBACK(drag_frame_end), frame);
- g_signal_connect(G_OBJECT(box), "drag-motion",
- G_CALLBACK(drag_frame_motion), frame);
- }
- /* ----- drag to the canvas ------------------------------------------------ */
- static gboolean drag_canvas_motion(GtkWidget *widget,
- GdkDragContext *drag_context, gint x, gint y, guint time_,
- gpointer user_data)
- {
- if (!has_target(widget, drag_context, "frame"))
- return FALSE;
- if (!frame_on_canvas) {
- frame_on_canvas = 1;
- canvas_frame_begin(dragging);
- }
- if (canvas_frame_motion(dragging, x, y)) {
- gdk_drag_status(drag_context, GDK_ACTION_COPY, time_);
- return TRUE;
- } else {
- gdk_drag_status(drag_context, 0, time_);
- return FALSE;
- }
- }
- static void drag_canvas_leave(GtkWidget *widget, GdkDragContext *drag_context,
- guint time_, gpointer user_data)
- {
- leave_canvas();
- }
- static gboolean drag_canvas_drop(GtkWidget *widget,
- GdkDragContext *drag_context, gint x, gint y, guint time_,
- gpointer user_data)
- {
- if (!has_target(widget, drag_context, "frame"))
- return FALSE;
- if (!canvas_frame_drop(dragging, x, y))
- return FALSE;
- gtk_drag_finish(drag_context, TRUE, FALSE, time_);
- drag_end(widget, drag_context, user_data);
- return TRUE;
- }
- void setup_canvas_drag(GtkWidget *canvas)
- {
- gtk_drag_dest_set(canvas,
- GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_DROP,
- &target_frame, 1, GDK_ACTION_COPY);
- g_signal_connect(G_OBJECT(canvas), "drag-motion",
- G_CALLBACK(drag_canvas_motion), NULL);
- g_signal_connect(G_OBJECT(canvas), "drag-leave",
- G_CALLBACK(drag_canvas_leave), NULL);
- g_signal_connect(G_OBJECT(canvas), "drag-drop",
- G_CALLBACK(drag_canvas_drop), NULL);
- }
|