916 lines
30 KiB
C
916 lines
30 KiB
C
/* PIKA - Photo and Image Kooker Application
|
|
* a rebranding of The GNU Image Manipulation Program (created with heckimp)
|
|
* A derived work which may be trivial. However, any changes may be (C)2023 by Aldercone Studio
|
|
*
|
|
* Original copyright, applying to most contents (license remains unchanged):
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* pikacontainertreeview-dnd.c
|
|
* Copyright (C) 2003-2009 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 3 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, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <gegl.h>
|
|
#include <gtk/gtk.h>
|
|
|
|
#include "widgets-types.h"
|
|
|
|
#include "core/pikacontainer.h"
|
|
#include "core/pikaviewable.h"
|
|
|
|
#include "pikacontainertreestore.h"
|
|
#include "pikacontainertreeview.h"
|
|
#include "pikacontainertreeview-dnd.h"
|
|
#include "pikacontainertreeview-private.h"
|
|
#include "pikacontainerview.h"
|
|
#include "pikadnd.h"
|
|
#include "pikaviewrenderer.h"
|
|
#include "pikaselectiondata.h"
|
|
|
|
|
|
static gint
|
|
pika_container_tree_view_viewable_sort (PikaViewable *v1,
|
|
PikaViewable *v2,
|
|
PikaContainerTreeView *tree_view)
|
|
{
|
|
PikaContainerView *view = PIKA_CONTAINER_VIEW (tree_view);
|
|
PikaViewable *parent1;
|
|
PikaViewable *parent2;
|
|
PikaContainer *container1 = NULL;
|
|
PikaContainer *container2 = NULL;
|
|
PikaContainer *container = pika_container_view_get_container (view);
|
|
gint index1 = -1;
|
|
gint index2 = -1;
|
|
gint depth1;
|
|
gint depth2;
|
|
|
|
parent1 = pika_viewable_get_parent (v1);
|
|
parent2 = pika_viewable_get_parent (v2);
|
|
|
|
if (parent1)
|
|
container1 = pika_viewable_get_children (parent1);
|
|
else if (pika_container_have (container, PIKA_OBJECT (v1)))
|
|
container1 = container;
|
|
|
|
if (parent2)
|
|
container2 = pika_viewable_get_children (parent2);
|
|
else if (pika_container_have (container, PIKA_OBJECT (v2)))
|
|
container2 = container;
|
|
|
|
g_return_val_if_fail (container1 && container2, 0);
|
|
|
|
if (container1 == container2)
|
|
{
|
|
index1 = pika_container_get_child_index (container1, PIKA_OBJECT (v1));
|
|
index2 = pika_container_get_child_index (container2, PIKA_OBJECT (v2));
|
|
|
|
return index1 < index2 ? -1 : (index1 > index2 ? 1 : 0);
|
|
}
|
|
|
|
depth1 = pika_viewable_get_depth (v1);
|
|
depth2 = pika_viewable_get_depth (v2);
|
|
|
|
if (depth1 == depth2)
|
|
{
|
|
return pika_container_tree_view_viewable_sort (parent1, parent2, tree_view);
|
|
}
|
|
else if (depth1 > depth2)
|
|
{
|
|
depth1 = pika_viewable_get_depth (parent1);
|
|
while (depth1 > depth2)
|
|
{
|
|
parent1 = pika_viewable_get_parent (parent1);
|
|
depth1 = pika_viewable_get_depth (parent1);
|
|
}
|
|
return pika_container_tree_view_viewable_sort (parent1, v2, tree_view);
|
|
}
|
|
else /* if (depth1 < depth2) */
|
|
{
|
|
depth2 = pika_viewable_get_depth (parent2);
|
|
while (depth1 < depth2)
|
|
{
|
|
parent2 = pika_viewable_get_parent (parent2);
|
|
depth2 = pika_viewable_get_depth (parent2);
|
|
}
|
|
return pika_container_tree_view_viewable_sort (v1, parent2, tree_view);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* pika_container_tree_view_drop_status:
|
|
* @tree_view:
|
|
* @context:
|
|
* @x:
|
|
* @y:
|
|
* @time:
|
|
* @return_path: the #GtkTreePath of the drop position if the drop is
|
|
* possible.
|
|
* @return_atom:
|
|
* @return_src_type: the type of drag'n drop.
|
|
* @return_src: allocated #GList of #PikaViewable being dragged.
|
|
* @return_dest: the #PikaViewable you are dropping on.
|
|
* @return_pos: the drop position (before, after or into @return_dest).
|
|
*
|
|
* Check whether the current drag can be dropped into @tree_view at
|
|
* position (@x, @y). If so, the various return value information will
|
|
* be optionally filled.
|
|
* Note: if @return_src is not %NULL, hence is filled, it must be freed
|
|
* with g_list_free().
|
|
*
|
|
* Returns: %TRUE is the drop is possible, %FALSE otherwise.
|
|
*/
|
|
static gboolean
|
|
pika_container_tree_view_drop_status (PikaContainerTreeView *tree_view,
|
|
GdkDragContext *context,
|
|
gint x,
|
|
gint y,
|
|
guint time,
|
|
GtkTreePath **return_path,
|
|
GdkAtom *return_atom,
|
|
PikaDndType *return_src_type,
|
|
GList **return_src,
|
|
PikaViewable **return_dest,
|
|
GtkTreeViewDropPosition *return_pos)
|
|
{
|
|
GList *src_viewables = NULL;
|
|
PikaViewable *dest_viewable = NULL;
|
|
GtkTreePath *drop_path = NULL;
|
|
GtkTargetList *target_list;
|
|
GdkAtom target_atom;
|
|
PikaDndType src_type;
|
|
GtkTreeViewDropPosition drop_pos = GTK_TREE_VIEW_DROP_BEFORE;
|
|
GdkDragAction drag_action = 0;
|
|
|
|
if (! pika_container_view_get_container (PIKA_CONTAINER_VIEW (tree_view)) ||
|
|
! pika_container_view_get_reorderable (PIKA_CONTAINER_VIEW (tree_view)))
|
|
goto drop_impossible;
|
|
|
|
target_list = gtk_drag_dest_get_target_list (GTK_WIDGET (tree_view->view));
|
|
target_atom = gtk_drag_dest_find_target (GTK_WIDGET (tree_view->view),
|
|
context, target_list);
|
|
if (! gtk_target_list_find (target_list, target_atom, &src_type))
|
|
goto drop_impossible;
|
|
|
|
switch (src_type)
|
|
{
|
|
case PIKA_DND_TYPE_URI_LIST:
|
|
case PIKA_DND_TYPE_TEXT_PLAIN:
|
|
case PIKA_DND_TYPE_NETSCAPE_URL:
|
|
case PIKA_DND_TYPE_COLOR:
|
|
case PIKA_DND_TYPE_SVG:
|
|
case PIKA_DND_TYPE_SVG_XML:
|
|
case PIKA_DND_TYPE_COMPONENT:
|
|
case PIKA_DND_TYPE_PIXBUF:
|
|
break;
|
|
|
|
case PIKA_DND_TYPE_XDS:
|
|
case PIKA_DND_TYPE_IMAGE:
|
|
case PIKA_DND_TYPE_LAYER:
|
|
case PIKA_DND_TYPE_CHANNEL:
|
|
case PIKA_DND_TYPE_LAYER_MASK:
|
|
case PIKA_DND_TYPE_VECTORS:
|
|
case PIKA_DND_TYPE_BRUSH:
|
|
case PIKA_DND_TYPE_PATTERN:
|
|
case PIKA_DND_TYPE_GRADIENT:
|
|
case PIKA_DND_TYPE_PALETTE:
|
|
case PIKA_DND_TYPE_FONT:
|
|
case PIKA_DND_TYPE_BUFFER:
|
|
case PIKA_DND_TYPE_IMAGEFILE:
|
|
case PIKA_DND_TYPE_TEMPLATE:
|
|
case PIKA_DND_TYPE_TOOL_ITEM:
|
|
case PIKA_DND_TYPE_NOTEBOOK_TAB:
|
|
/* Various PikaViewable drag data. */
|
|
{
|
|
GtkWidget *src_widget = gtk_drag_get_source_widget (context);
|
|
PikaViewable *src_viewable = NULL;
|
|
|
|
if (! src_widget)
|
|
goto drop_impossible;
|
|
|
|
src_viewable = pika_dnd_get_drag_viewable (src_widget);
|
|
|
|
if (! PIKA_IS_VIEWABLE (src_viewable))
|
|
goto drop_impossible;
|
|
|
|
src_viewables = g_list_prepend (src_viewables, src_viewable);
|
|
}
|
|
break;
|
|
|
|
case PIKA_DND_TYPE_CHANNEL_LIST:
|
|
case PIKA_DND_TYPE_LAYER_LIST:
|
|
case PIKA_DND_TYPE_VECTORS_LIST:
|
|
/* Various PikaViewable list (GList) drag data. */
|
|
{
|
|
GtkWidget *src_widget = gtk_drag_get_source_widget (context);
|
|
GList *iter;
|
|
|
|
if (! src_widget)
|
|
goto drop_impossible;
|
|
|
|
src_viewables = pika_dnd_get_drag_list (src_widget);
|
|
|
|
if (! src_viewables)
|
|
goto drop_impossible;
|
|
|
|
for (iter = src_viewables; iter; iter = iter->next)
|
|
if (! PIKA_IS_VIEWABLE (iter->data))
|
|
{
|
|
g_warning ("%s: contents of the viewable list has the wrong type '%s'.",
|
|
G_STRFUNC, G_OBJECT_TYPE_NAME (iter->data));
|
|
goto drop_impossible;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
goto drop_impossible;
|
|
break;
|
|
}
|
|
|
|
gtk_tree_view_convert_widget_to_bin_window_coords (tree_view->view, x, y, &x, &y);
|
|
if (gtk_tree_view_get_path_at_pos (tree_view->view, x, y,
|
|
&drop_path, NULL, NULL, NULL))
|
|
{
|
|
PikaViewRenderer *renderer;
|
|
GtkTreeIter iter;
|
|
GdkRectangle cell_area;
|
|
|
|
gtk_tree_model_get_iter (tree_view->model, &iter, drop_path);
|
|
|
|
renderer = pika_container_tree_store_get_renderer (PIKA_CONTAINER_TREE_STORE (tree_view->model),
|
|
&iter);
|
|
|
|
dest_viewable = renderer->viewable;
|
|
|
|
g_object_unref (renderer);
|
|
|
|
gtk_tree_view_get_cell_area (tree_view->view, drop_path, NULL, &cell_area);
|
|
|
|
if (pika_viewable_get_children (dest_viewable))
|
|
{
|
|
if (gtk_tree_view_row_expanded (tree_view->view, drop_path))
|
|
{
|
|
if (y >= (cell_area.y + cell_area.height / 2))
|
|
drop_pos = GTK_TREE_VIEW_DROP_INTO_OR_AFTER;
|
|
else
|
|
drop_pos = GTK_TREE_VIEW_DROP_BEFORE;
|
|
}
|
|
else
|
|
{
|
|
if (y >= (cell_area.y + 2 * (cell_area.height / 3)))
|
|
drop_pos = GTK_TREE_VIEW_DROP_AFTER;
|
|
else if (y <= (cell_area.y + cell_area.height / 3))
|
|
drop_pos = GTK_TREE_VIEW_DROP_BEFORE;
|
|
else
|
|
drop_pos = GTK_TREE_VIEW_DROP_INTO_OR_AFTER;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (y >= (cell_area.y + cell_area.height / 2))
|
|
drop_pos = GTK_TREE_VIEW_DROP_AFTER;
|
|
else
|
|
drop_pos = GTK_TREE_VIEW_DROP_BEFORE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
GtkTreeIter iter;
|
|
gint n_children;
|
|
|
|
n_children = gtk_tree_model_iter_n_children (tree_view->model, NULL);
|
|
|
|
if (n_children > 0 &&
|
|
gtk_tree_model_iter_nth_child (tree_view->model, &iter,
|
|
NULL, n_children - 1))
|
|
{
|
|
PikaViewRenderer *renderer;
|
|
|
|
renderer = pika_container_tree_store_get_renderer (PIKA_CONTAINER_TREE_STORE (tree_view->model),
|
|
&iter);
|
|
|
|
drop_path = gtk_tree_model_get_path (tree_view->model, &iter);
|
|
dest_viewable = renderer->viewable;
|
|
drop_pos = GTK_TREE_VIEW_DROP_AFTER;
|
|
|
|
g_object_unref (renderer);
|
|
}
|
|
}
|
|
|
|
if (dest_viewable || tree_view->priv->dnd_drop_to_empty)
|
|
{
|
|
if (PIKA_CONTAINER_TREE_VIEW_GET_CLASS (tree_view)->drop_possible (tree_view,
|
|
src_type,
|
|
src_viewables,
|
|
dest_viewable,
|
|
drop_path,
|
|
drop_pos,
|
|
&drop_pos,
|
|
&drag_action))
|
|
{
|
|
gdk_drag_status (context, drag_action, time);
|
|
|
|
if (return_path)
|
|
*return_path = drop_path;
|
|
else
|
|
gtk_tree_path_free (drop_path);
|
|
|
|
if (return_atom)
|
|
*return_atom = target_atom;
|
|
|
|
if (return_src)
|
|
{
|
|
src_viewables = g_list_sort_with_data (src_viewables,
|
|
(GCompareDataFunc) pika_container_tree_view_viewable_sort,
|
|
tree_view);
|
|
*return_src = src_viewables;
|
|
}
|
|
else
|
|
{
|
|
g_list_free (src_viewables);
|
|
}
|
|
|
|
if (return_dest)
|
|
*return_dest = dest_viewable;
|
|
|
|
if (return_pos)
|
|
*return_pos = drop_pos;
|
|
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
drop_impossible:
|
|
|
|
g_list_free (src_viewables);
|
|
gtk_tree_path_free (drop_path);
|
|
gdk_drag_status (context, 0, time);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
#define SCROLL_DISTANCE 30
|
|
#define SCROLL_STEP 10
|
|
#define SCROLL_INTERVAL 5
|
|
/* #define SCROLL_DEBUG 1 */
|
|
|
|
static gboolean
|
|
pika_container_tree_view_scroll_timeout (gpointer data)
|
|
{
|
|
PikaContainerTreeView *tree_view = PIKA_CONTAINER_TREE_VIEW (data);
|
|
GtkAdjustment *adj;
|
|
gdouble new_value;
|
|
|
|
adj = gtk_scrollable_get_vadjustment (GTK_SCROLLABLE (tree_view->view));
|
|
|
|
#ifdef SCROLL_DEBUG
|
|
g_print ("scroll_timeout: scrolling by %d\n", SCROLL_STEP);
|
|
#endif
|
|
|
|
if (tree_view->priv->scroll_dir == GDK_SCROLL_UP)
|
|
new_value = gtk_adjustment_get_value (adj) - SCROLL_STEP;
|
|
else
|
|
new_value = gtk_adjustment_get_value (adj) + SCROLL_STEP;
|
|
|
|
new_value = CLAMP (new_value,
|
|
gtk_adjustment_get_lower (adj),
|
|
gtk_adjustment_get_upper (adj) -
|
|
gtk_adjustment_get_page_size (adj));
|
|
|
|
gtk_adjustment_set_value (adj, new_value);
|
|
|
|
if (tree_view->priv->scroll_timeout_id)
|
|
{
|
|
g_source_remove (tree_view->priv->scroll_timeout_id);
|
|
|
|
tree_view->priv->scroll_timeout_id =
|
|
g_timeout_add (tree_view->priv->scroll_timeout_interval,
|
|
pika_container_tree_view_scroll_timeout,
|
|
tree_view);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
void
|
|
pika_container_tree_view_drag_failed (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
GtkDragResult result,
|
|
PikaContainerTreeView *tree_view)
|
|
{
|
|
if (tree_view->priv->scroll_timeout_id)
|
|
{
|
|
g_source_remove (tree_view->priv->scroll_timeout_id);
|
|
tree_view->priv->scroll_timeout_id = 0;
|
|
}
|
|
|
|
gtk_tree_view_set_drag_dest_row (tree_view->view, NULL, 0);
|
|
}
|
|
|
|
void
|
|
pika_container_tree_view_drag_leave (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
guint time,
|
|
PikaContainerTreeView *tree_view)
|
|
{
|
|
if (tree_view->priv->scroll_timeout_id)
|
|
{
|
|
g_source_remove (tree_view->priv->scroll_timeout_id);
|
|
tree_view->priv->scroll_timeout_id = 0;
|
|
}
|
|
|
|
gtk_tree_view_set_drag_dest_row (tree_view->view, NULL, 0);
|
|
}
|
|
|
|
gboolean
|
|
pika_container_tree_view_drag_motion (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gint x,
|
|
gint y,
|
|
guint time,
|
|
PikaContainerTreeView *tree_view)
|
|
{
|
|
GtkAllocation allocation;
|
|
GtkTreePath *drop_path;
|
|
GtkTreeViewDropPosition drop_pos;
|
|
|
|
gtk_widget_get_allocation (widget, &allocation);
|
|
|
|
if (y < SCROLL_DISTANCE || y > (allocation.height - SCROLL_DISTANCE))
|
|
{
|
|
gint distance;
|
|
|
|
if (y < SCROLL_DISTANCE)
|
|
{
|
|
tree_view->priv->scroll_dir = GDK_SCROLL_UP;
|
|
distance = MIN (-y, -1);
|
|
}
|
|
else
|
|
{
|
|
tree_view->priv->scroll_dir = GDK_SCROLL_DOWN;
|
|
distance = MAX (allocation.height - y, 1);
|
|
}
|
|
|
|
tree_view->priv->scroll_timeout_interval = SCROLL_INTERVAL * ABS (distance);
|
|
|
|
#ifdef SCROLL_DEBUG
|
|
g_print ("drag_motion: scroll_distance = %d scroll_interval = %d\n",
|
|
distance, tree_view->priv->scroll_timeout_interval);
|
|
#endif
|
|
|
|
if (! tree_view->priv->scroll_timeout_id)
|
|
tree_view->priv->scroll_timeout_id =
|
|
g_timeout_add (tree_view->priv->scroll_timeout_interval,
|
|
pika_container_tree_view_scroll_timeout,
|
|
tree_view);
|
|
}
|
|
else if (tree_view->priv->scroll_timeout_id)
|
|
{
|
|
g_source_remove (tree_view->priv->scroll_timeout_id);
|
|
tree_view->priv->scroll_timeout_id = 0;
|
|
}
|
|
|
|
if (pika_container_tree_view_drop_status (tree_view,
|
|
context, x, y, time,
|
|
&drop_path, NULL, NULL, NULL, NULL,
|
|
&drop_pos))
|
|
{
|
|
gtk_tree_view_set_drag_dest_row (tree_view->view, drop_path, drop_pos);
|
|
gtk_tree_path_free (drop_path);
|
|
}
|
|
else
|
|
{
|
|
gtk_tree_view_set_drag_dest_row (tree_view->view, NULL, 0);
|
|
}
|
|
|
|
/* always return TRUE so drag_leave() is called */
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
pika_container_tree_view_drag_drop (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gint x,
|
|
gint y,
|
|
guint time,
|
|
PikaContainerTreeView *tree_view)
|
|
{
|
|
PikaDndType src_type;
|
|
GList *src_viewables;
|
|
PikaViewable *dest_viewable;
|
|
GdkAtom target;
|
|
GtkTreeViewDropPosition drop_pos;
|
|
|
|
if (tree_view->priv->scroll_timeout_id)
|
|
{
|
|
g_source_remove (tree_view->priv->scroll_timeout_id);
|
|
tree_view->priv->scroll_timeout_id = 0;
|
|
}
|
|
|
|
if (pika_container_tree_view_drop_status (tree_view,
|
|
context, x, y, time,
|
|
NULL, &target, &src_type,
|
|
&src_viewables,
|
|
&dest_viewable, &drop_pos))
|
|
{
|
|
PikaContainerTreeViewClass *tree_view_class;
|
|
|
|
tree_view_class = PIKA_CONTAINER_TREE_VIEW_GET_CLASS (tree_view);
|
|
|
|
if (src_viewables)
|
|
{
|
|
gboolean success = TRUE;
|
|
|
|
/* XXX: Make PikaContainerTreeViewClass::drop_viewable()
|
|
* return success?
|
|
*/
|
|
tree_view_class->drop_viewables (tree_view, src_viewables,
|
|
dest_viewable, drop_pos);
|
|
|
|
gtk_drag_finish (context, success, FALSE, time);
|
|
g_list_free (src_viewables);
|
|
}
|
|
else
|
|
{
|
|
/* Necessary for instance for dragging color components onto
|
|
* item dialogs.
|
|
*/
|
|
gtk_drag_get_data (widget, context, target, time);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
void
|
|
pika_container_tree_view_drag_data_received (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gint x,
|
|
gint y,
|
|
GtkSelectionData *selection_data,
|
|
guint info,
|
|
guint time,
|
|
PikaContainerTreeView *tree_view)
|
|
{
|
|
PikaViewable *dest_viewable;
|
|
GtkTreeViewDropPosition drop_pos;
|
|
gboolean success = FALSE;
|
|
|
|
if (pika_container_tree_view_drop_status (tree_view,
|
|
context, x, y, time,
|
|
NULL, NULL, NULL, NULL,
|
|
&dest_viewable, &drop_pos))
|
|
{
|
|
PikaContainerTreeViewClass *tree_view_class;
|
|
|
|
tree_view_class = PIKA_CONTAINER_TREE_VIEW_GET_CLASS (tree_view);
|
|
|
|
switch (info)
|
|
{
|
|
case PIKA_DND_TYPE_URI_LIST:
|
|
case PIKA_DND_TYPE_TEXT_PLAIN:
|
|
case PIKA_DND_TYPE_NETSCAPE_URL:
|
|
if (tree_view_class->drop_uri_list)
|
|
{
|
|
GList *uri_list;
|
|
|
|
uri_list = pika_selection_data_get_uri_list (selection_data);
|
|
|
|
if (uri_list)
|
|
{
|
|
tree_view_class->drop_uri_list (tree_view, uri_list,
|
|
dest_viewable, drop_pos);
|
|
|
|
g_list_free_full (uri_list, (GDestroyNotify) g_free);
|
|
|
|
success = TRUE;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case PIKA_DND_TYPE_COLOR:
|
|
if (tree_view_class->drop_color)
|
|
{
|
|
PikaRGB color;
|
|
|
|
if (pika_selection_data_get_color (selection_data, &color))
|
|
{
|
|
tree_view_class->drop_color (tree_view, &color,
|
|
dest_viewable, drop_pos);
|
|
|
|
success = TRUE;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case PIKA_DND_TYPE_SVG:
|
|
case PIKA_DND_TYPE_SVG_XML:
|
|
if (tree_view_class->drop_svg)
|
|
{
|
|
const guchar *stream;
|
|
gsize stream_length;
|
|
|
|
stream = pika_selection_data_get_stream (selection_data,
|
|
&stream_length);
|
|
|
|
if (stream)
|
|
{
|
|
tree_view_class->drop_svg (tree_view,
|
|
(const gchar *) stream,
|
|
stream_length,
|
|
dest_viewable, drop_pos);
|
|
|
|
success = TRUE;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case PIKA_DND_TYPE_COMPONENT:
|
|
if (tree_view_class->drop_component)
|
|
{
|
|
PikaImage *image = NULL;
|
|
PikaChannelType component;
|
|
|
|
if (tree_view->dnd_pika)
|
|
image = pika_selection_data_get_component (selection_data,
|
|
tree_view->dnd_pika,
|
|
&component);
|
|
|
|
if (image)
|
|
{
|
|
tree_view_class->drop_component (tree_view,
|
|
image, component,
|
|
dest_viewable, drop_pos);
|
|
|
|
success = TRUE;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case PIKA_DND_TYPE_PIXBUF:
|
|
if (tree_view_class->drop_pixbuf)
|
|
{
|
|
GdkPixbuf *pixbuf;
|
|
|
|
pixbuf = gtk_selection_data_get_pixbuf (selection_data);
|
|
|
|
if (pixbuf)
|
|
{
|
|
tree_view_class->drop_pixbuf (tree_view,
|
|
pixbuf,
|
|
dest_viewable, drop_pos);
|
|
g_object_unref (pixbuf);
|
|
|
|
success = TRUE;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
gtk_drag_finish (context, success, FALSE, time);
|
|
}
|
|
|
|
gboolean
|
|
pika_container_tree_view_real_drop_possible (PikaContainerTreeView *tree_view,
|
|
PikaDndType src_type,
|
|
GList *src_viewables,
|
|
PikaViewable *dest_viewable,
|
|
GtkTreePath *drop_path,
|
|
GtkTreeViewDropPosition drop_pos,
|
|
GtkTreeViewDropPosition *return_drop_pos,
|
|
GdkDragAction *return_drag_action)
|
|
{
|
|
PikaContainerView *view = PIKA_CONTAINER_VIEW (tree_view);
|
|
PikaContainer *container = pika_container_view_get_container (view);
|
|
PikaContainer *src_container = NULL;
|
|
PikaContainer *dest_container = NULL;
|
|
GList *iter;
|
|
gint src_index = -1;
|
|
gint dest_index = -1;
|
|
|
|
if (dest_viewable)
|
|
{
|
|
PikaViewable *parent;
|
|
|
|
/* dropping on the lower third of a group item drops into that group */
|
|
if (drop_pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER &&
|
|
pika_viewable_get_children (dest_viewable))
|
|
{
|
|
parent = dest_viewable;
|
|
}
|
|
else
|
|
{
|
|
parent = pika_viewable_get_parent (dest_viewable);
|
|
}
|
|
|
|
if (parent)
|
|
dest_container = pika_viewable_get_children (parent);
|
|
else if (pika_container_have (container, PIKA_OBJECT (dest_viewable)))
|
|
dest_container = container;
|
|
|
|
if (parent == dest_viewable)
|
|
dest_index = 0;
|
|
else
|
|
dest_index = pika_container_get_child_index (dest_container,
|
|
PIKA_OBJECT (dest_viewable));
|
|
}
|
|
|
|
if (return_drag_action)
|
|
{
|
|
if (! src_viewables)
|
|
*return_drag_action = GDK_ACTION_COPY;
|
|
else
|
|
*return_drag_action = GDK_ACTION_MOVE;
|
|
}
|
|
|
|
for (iter = src_viewables; iter; iter = iter->next)
|
|
{
|
|
PikaViewable *src_viewable = iter->data;
|
|
PikaViewable *parent;
|
|
|
|
parent = pika_viewable_get_parent (src_viewable);
|
|
|
|
if (parent)
|
|
src_container = pika_viewable_get_children (parent);
|
|
else if (pika_container_have (container, PIKA_OBJECT (src_viewable)))
|
|
src_container = container;
|
|
|
|
if (src_container)
|
|
src_index = pika_container_get_child_index (src_container,
|
|
PIKA_OBJECT (src_viewable));
|
|
|
|
|
|
if (g_type_is_a (G_TYPE_FROM_INSTANCE (src_viewable),
|
|
pika_container_get_children_type (container)))
|
|
{
|
|
/* The drop won't change a thing. This is not a fatal drop
|
|
* failure, unless there is only one source viewable.
|
|
* See also the XXX below.
|
|
*/
|
|
if (src_viewable == dest_viewable && g_list_length (src_viewables) == 1)
|
|
return FALSE;
|
|
|
|
if (src_index == -1 || dest_index == -1)
|
|
return FALSE;
|
|
|
|
/* don't allow dropping a parent node onto one of its descendants
|
|
*/
|
|
if (pika_viewable_is_ancestor (src_viewable, dest_viewable))
|
|
return FALSE;
|
|
}
|
|
|
|
/* XXX only check these for list of 1 viewable for now.
|
|
* Actually this drop failure would also happen for more than 1
|
|
* viewable if all the sources are from the same src_container
|
|
* with successive indexes.
|
|
*/
|
|
if (src_container == dest_container && g_list_length (src_viewables) == 1)
|
|
{
|
|
if (drop_pos == GTK_TREE_VIEW_DROP_BEFORE)
|
|
{
|
|
if (dest_index == (src_index + 1))
|
|
return FALSE;
|
|
}
|
|
else if (drop_pos == GTK_TREE_VIEW_DROP_AFTER)
|
|
{
|
|
if (dest_index == (src_index - 1))
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
if (return_drag_action)
|
|
{
|
|
if (! g_type_is_a (G_TYPE_FROM_INSTANCE (src_viewable),
|
|
pika_container_get_children_type (container)))
|
|
*return_drag_action = GDK_ACTION_COPY;
|
|
}
|
|
}
|
|
|
|
if (return_drop_pos)
|
|
*return_drop_pos = drop_pos;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
pika_container_tree_view_real_drop_viewables (PikaContainerTreeView *tree_view,
|
|
GList *src_viewables,
|
|
PikaViewable *dest_viewable,
|
|
GtkTreeViewDropPosition drop_pos)
|
|
{
|
|
PikaContainerView *view = PIKA_CONTAINER_VIEW (tree_view);
|
|
PikaContainer *src_container;
|
|
PikaContainer *dest_container;
|
|
GList *iter;
|
|
gint dest_index = 0;
|
|
|
|
g_return_if_fail (g_list_length (src_viewables) > 0);
|
|
|
|
src_viewables = g_list_reverse (src_viewables);
|
|
for (iter = src_viewables; iter; iter = iter->next)
|
|
{
|
|
PikaViewable *src_viewable = iter->data;
|
|
|
|
if ((drop_pos == GTK_TREE_VIEW_DROP_INTO_OR_AFTER ||
|
|
drop_pos == GTK_TREE_VIEW_DROP_INTO_OR_BEFORE) &&
|
|
pika_viewable_get_children (dest_viewable))
|
|
{
|
|
dest_container = pika_viewable_get_children (dest_viewable);
|
|
dest_viewable = NULL;
|
|
drop_pos = GTK_TREE_VIEW_DROP_BEFORE;
|
|
}
|
|
else if (pika_viewable_get_parent (dest_viewable))
|
|
{
|
|
dest_container = pika_viewable_get_children (pika_viewable_get_parent (dest_viewable));
|
|
}
|
|
else
|
|
{
|
|
dest_container = pika_container_view_get_container (view);
|
|
}
|
|
|
|
if (dest_viewable)
|
|
{
|
|
dest_index = pika_container_get_child_index (dest_container,
|
|
PIKA_OBJECT (dest_viewable));
|
|
}
|
|
|
|
if (pika_viewable_get_parent (src_viewable))
|
|
{
|
|
src_container = pika_viewable_get_children (
|
|
pika_viewable_get_parent (src_viewable));
|
|
}
|
|
else
|
|
{
|
|
src_container = pika_container_view_get_container (view);
|
|
}
|
|
|
|
if (src_container == dest_container)
|
|
{
|
|
gint src_index;
|
|
|
|
src_index = pika_container_get_child_index (src_container,
|
|
PIKA_OBJECT (src_viewable));
|
|
|
|
switch (drop_pos)
|
|
{
|
|
case GTK_TREE_VIEW_DROP_AFTER:
|
|
case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
|
|
if (src_index > dest_index)
|
|
dest_index++;
|
|
break;
|
|
|
|
case GTK_TREE_VIEW_DROP_BEFORE:
|
|
case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
|
|
if (src_index < dest_index)
|
|
dest_index--;
|
|
break;
|
|
}
|
|
|
|
pika_container_reorder (src_container,
|
|
PIKA_OBJECT (src_viewable), dest_index);
|
|
}
|
|
else
|
|
{
|
|
switch (drop_pos)
|
|
{
|
|
case GTK_TREE_VIEW_DROP_AFTER:
|
|
case GTK_TREE_VIEW_DROP_INTO_OR_AFTER:
|
|
dest_index++;
|
|
break;
|
|
|
|
case GTK_TREE_VIEW_DROP_BEFORE:
|
|
case GTK_TREE_VIEW_DROP_INTO_OR_BEFORE:
|
|
break;
|
|
}
|
|
|
|
g_object_ref (src_viewable);
|
|
|
|
pika_container_remove (src_container, PIKA_OBJECT (src_viewable));
|
|
pika_container_insert (dest_container, PIKA_OBJECT (src_viewable),
|
|
dest_index);
|
|
|
|
g_object_unref (src_viewable);
|
|
}
|
|
}
|
|
}
|