623 lines
17 KiB
C
623 lines
17 KiB
C
/* GTK - The GIMP Toolkit
|
|
* Copyright 2019 Matthias Clasen
|
|
*
|
|
* 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, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gtkdragicon.h"
|
|
|
|
#include "gtkwidgetprivate.h"
|
|
#include "gtkcssstyleprivate.h"
|
|
#include "gtkcsstypesprivate.h"
|
|
#include "gtknativeprivate.h"
|
|
#include "gtkpicture.h"
|
|
#include "gtkcssboxesimplprivate.h"
|
|
#include "gtkcssnumbervalueprivate.h"
|
|
|
|
#include "gdk/gdksurfaceprivate.h"
|
|
#include "gdk/gdkdragsurfacesize.h"
|
|
|
|
/* for the drag icons */
|
|
#include "gtkcolorswatchprivate.h"
|
|
#include "gtkimage.h"
|
|
#include "gtklabel.h"
|
|
#include "gtkrendernodepaintableprivate.h"
|
|
#include "gtktextutilprivate.h"
|
|
|
|
|
|
/**
|
|
* GtkDragIcon:
|
|
*
|
|
* `GtkDragIcon` is a `GtkRoot` implementation for drag icons.
|
|
*
|
|
* A drag icon moves with the pointer during a Drag-and-Drop operation
|
|
* and is destroyed when the drag ends.
|
|
*
|
|
* To set up a drag icon and associate it with an ongoing drag operation,
|
|
* use [func@Gtk.DragIcon.get_for_drag] to get the icon for a drag. You can
|
|
* then use it like any other widget and use [method@Gtk.DragIcon.set_child]
|
|
* to set whatever widget should be used for the drag icon.
|
|
*
|
|
* Keep in mind that drag icons do not allow user input.
|
|
*/
|
|
struct _GtkDragIcon
|
|
{
|
|
GtkWidget parent_instance;
|
|
|
|
GdkSurface *surface;
|
|
GskRenderer *renderer;
|
|
GtkWidget *child;
|
|
};
|
|
|
|
struct _GtkDragIconClass
|
|
{
|
|
GtkWidgetClass parent_class;
|
|
};
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_CHILD,
|
|
|
|
LAST_ARG
|
|
};
|
|
|
|
static GParamSpec *properties[LAST_ARG] = { NULL, };
|
|
|
|
static void gtk_drag_icon_root_init (GtkRootInterface *iface);
|
|
static void gtk_drag_icon_native_init (GtkNativeInterface *iface);
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GtkDragIcon, gtk_drag_icon, GTK_TYPE_WIDGET,
|
|
G_IMPLEMENT_INTERFACE (GTK_TYPE_NATIVE,
|
|
gtk_drag_icon_native_init)
|
|
G_IMPLEMENT_INTERFACE (GTK_TYPE_ROOT,
|
|
gtk_drag_icon_root_init))
|
|
|
|
static GdkDisplay *
|
|
gtk_drag_icon_root_get_display (GtkRoot *self)
|
|
{
|
|
GtkDragIcon *icon = GTK_DRAG_ICON (self);
|
|
|
|
if (icon->surface)
|
|
return gdk_surface_get_display (icon->surface);
|
|
|
|
return gdk_display_get_default ();
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_root_init (GtkRootInterface *iface)
|
|
{
|
|
iface->get_display = gtk_drag_icon_root_get_display;
|
|
}
|
|
|
|
static GdkSurface *
|
|
gtk_drag_icon_native_get_surface (GtkNative *native)
|
|
{
|
|
GtkDragIcon *icon = GTK_DRAG_ICON (native);
|
|
|
|
return icon->surface;
|
|
}
|
|
|
|
static GskRenderer *
|
|
gtk_drag_icon_native_get_renderer (GtkNative *native)
|
|
{
|
|
GtkDragIcon *icon = GTK_DRAG_ICON (native);
|
|
|
|
return icon->renderer;
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_native_get_surface_transform (GtkNative *native,
|
|
double *x,
|
|
double *y)
|
|
{
|
|
GtkCssBoxes css_boxes;
|
|
const graphene_rect_t *margin_rect;
|
|
|
|
gtk_css_boxes_init (&css_boxes, GTK_WIDGET (native));
|
|
margin_rect = gtk_css_boxes_get_margin_rect (&css_boxes);
|
|
|
|
*x = - margin_rect->origin.x;
|
|
*y = - margin_rect->origin.y;
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_move_resize (GtkDragIcon *icon)
|
|
{
|
|
GtkRequisition req;
|
|
|
|
if (icon->surface)
|
|
{
|
|
gtk_widget_get_preferred_size (GTK_WIDGET (icon), NULL, &req);
|
|
gdk_drag_surface_present (GDK_DRAG_SURFACE (icon->surface),
|
|
MAX (1, req.width),
|
|
MAX (1, req.height));
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_present (GtkDragIcon *icon)
|
|
{
|
|
GtkWidget *widget = GTK_WIDGET (icon);
|
|
|
|
if (!_gtk_widget_get_alloc_needed (widget))
|
|
gtk_widget_ensure_allocate (widget);
|
|
else if (gtk_widget_get_visible (widget))
|
|
gtk_drag_icon_move_resize (icon);
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_native_layout (GtkNative *native,
|
|
int width,
|
|
int height)
|
|
{
|
|
gtk_widget_allocate (GTK_WIDGET (native), width, height, -1, NULL);
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_native_init (GtkNativeInterface *iface)
|
|
{
|
|
iface->get_surface = gtk_drag_icon_native_get_surface;
|
|
iface->get_renderer = gtk_drag_icon_native_get_renderer;
|
|
iface->get_surface_transform = gtk_drag_icon_native_get_surface_transform;
|
|
iface->layout = gtk_drag_icon_native_layout;
|
|
}
|
|
|
|
static gboolean
|
|
surface_render (GdkSurface *surface,
|
|
cairo_region_t *region,
|
|
GtkWidget *widget)
|
|
{
|
|
gtk_widget_render (widget, surface, region);
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
surface_compute_size (GdkDragSurface *surface,
|
|
GdkDragSurfaceSize *size,
|
|
GtkWidget *widget)
|
|
{
|
|
GtkRequisition nat_size;
|
|
gtk_widget_get_preferred_size (widget, NULL, &nat_size);
|
|
gdk_drag_surface_size_set_size (size, nat_size.width, nat_size.height);
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_realize (GtkWidget *widget)
|
|
{
|
|
GtkDragIcon *icon = GTK_DRAG_ICON (widget);
|
|
|
|
g_warn_if_fail (icon->surface != NULL);
|
|
|
|
gdk_surface_set_widget (icon->surface, widget);
|
|
|
|
g_signal_connect (icon->surface, "render", G_CALLBACK (surface_render), widget);
|
|
g_signal_connect (icon->surface, "compute-size", G_CALLBACK (surface_compute_size), widget);
|
|
|
|
GTK_WIDGET_CLASS (gtk_drag_icon_parent_class)->realize (widget);
|
|
|
|
icon->renderer = gsk_renderer_new_for_surface (icon->surface);
|
|
|
|
gtk_native_realize (GTK_NATIVE (icon));
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_unrealize (GtkWidget *widget)
|
|
{
|
|
GtkDragIcon *icon = GTK_DRAG_ICON (widget);
|
|
|
|
gtk_native_unrealize (GTK_NATIVE (icon));
|
|
|
|
GTK_WIDGET_CLASS (gtk_drag_icon_parent_class)->unrealize (widget);
|
|
|
|
gsk_renderer_unrealize (icon->renderer);
|
|
g_clear_object (&icon->renderer);
|
|
|
|
if (icon->surface)
|
|
{
|
|
g_signal_handlers_disconnect_by_func (icon->surface, surface_render, widget);
|
|
g_signal_handlers_disconnect_by_func (icon->surface, surface_compute_size, widget);
|
|
gdk_surface_set_widget (icon->surface, NULL);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_map (GtkWidget *widget)
|
|
{
|
|
GtkDragIcon *icon = GTK_DRAG_ICON (widget);
|
|
|
|
gtk_drag_icon_move_resize (icon);
|
|
|
|
GTK_WIDGET_CLASS (gtk_drag_icon_parent_class)->map (widget);
|
|
|
|
if (icon->child && gtk_widget_get_visible (icon->child))
|
|
gtk_widget_map (icon->child);
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_unmap (GtkWidget *widget)
|
|
{
|
|
GtkDragIcon *icon = GTK_DRAG_ICON (widget);
|
|
|
|
g_warn_if_fail (icon->surface != NULL);
|
|
GTK_WIDGET_CLASS (gtk_drag_icon_parent_class)->unmap (widget);
|
|
if (icon->surface)
|
|
gdk_surface_hide (icon->surface);
|
|
|
|
if (icon->child)
|
|
gtk_widget_unmap (icon->child);
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_measure (GtkWidget *widget,
|
|
GtkOrientation orientation,
|
|
int for_size,
|
|
int *minimum,
|
|
int *natural,
|
|
int *minimum_baseline,
|
|
int *natural_baseline)
|
|
{
|
|
GtkDragIcon *icon = GTK_DRAG_ICON (widget);
|
|
|
|
if (icon->child)
|
|
gtk_widget_measure (icon->child,
|
|
orientation, for_size,
|
|
minimum, natural,
|
|
minimum_baseline, natural_baseline);
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_size_allocate (GtkWidget *widget,
|
|
int width,
|
|
int height,
|
|
int baseline)
|
|
{
|
|
GtkDragIcon *icon = GTK_DRAG_ICON (widget);
|
|
|
|
if (icon->child)
|
|
gtk_widget_allocate (icon->child, width, height, baseline, NULL);
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_show (GtkWidget *widget)
|
|
{
|
|
_gtk_widget_set_visible_flag (widget, TRUE);
|
|
gtk_css_node_validate (gtk_widget_get_css_node (widget));
|
|
gtk_widget_realize (widget);
|
|
gtk_drag_icon_present (GTK_DRAG_ICON (widget));
|
|
gtk_widget_map (widget);
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_hide (GtkWidget *widget)
|
|
{
|
|
_gtk_widget_set_visible_flag (widget, FALSE);
|
|
gtk_widget_unmap (widget);
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_dispose (GObject *object)
|
|
{
|
|
GtkDragIcon *icon = GTK_DRAG_ICON (object);
|
|
|
|
g_clear_pointer (&icon->child, gtk_widget_unparent);
|
|
|
|
G_OBJECT_CLASS (gtk_drag_icon_parent_class)->dispose (object);
|
|
|
|
g_clear_object (&icon->surface);
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GtkDragIcon *self = GTK_DRAG_ICON (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_CHILD:
|
|
g_value_set_object (value, self->child);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GtkDragIcon *self = GTK_DRAG_ICON (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_CHILD:
|
|
gtk_drag_icon_set_child (self, g_value_get_object (value));
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_class_init (GtkDragIconClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
object_class->dispose = gtk_drag_icon_dispose;
|
|
object_class->get_property = gtk_drag_icon_get_property;
|
|
object_class->set_property = gtk_drag_icon_set_property;
|
|
|
|
widget_class->realize = gtk_drag_icon_realize;
|
|
widget_class->unrealize = gtk_drag_icon_unrealize;
|
|
widget_class->map = gtk_drag_icon_map;
|
|
widget_class->unmap = gtk_drag_icon_unmap;
|
|
widget_class->measure = gtk_drag_icon_measure;
|
|
widget_class->size_allocate = gtk_drag_icon_size_allocate;
|
|
widget_class->show = gtk_drag_icon_show;
|
|
widget_class->hide = gtk_drag_icon_hide;
|
|
|
|
/**
|
|
* GtkDragIcon:child: (attributes org.gtk.Property.get=gtk_drag_icon_get_child org.gtk.Property.set=gtk_drag_icon_set_child)
|
|
*
|
|
* The widget to display as drag icon.
|
|
*/
|
|
properties[PROP_CHILD] =
|
|
g_param_spec_object ("child", NULL, NULL,
|
|
GTK_TYPE_WIDGET,
|
|
G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY | G_PARAM_STATIC_STRINGS);
|
|
|
|
g_object_class_install_properties (object_class, LAST_ARG, properties);
|
|
|
|
gtk_widget_class_set_css_name (widget_class, "dnd");
|
|
}
|
|
|
|
static void
|
|
gtk_drag_icon_init (GtkDragIcon *self)
|
|
{
|
|
gtk_widget_set_can_target (GTK_WIDGET (self), FALSE);
|
|
}
|
|
|
|
/**
|
|
* gtk_drag_icon_get_for_drag:
|
|
* @drag: a `GdkDrag`
|
|
*
|
|
* Gets the `GtkDragIcon` in use with @drag.
|
|
*
|
|
* If no drag icon exists yet, a new one will be created
|
|
* and shown.
|
|
*
|
|
* Returns: (transfer none): the `GtkDragIcon`
|
|
*/
|
|
GtkWidget *
|
|
gtk_drag_icon_get_for_drag (GdkDrag *drag)
|
|
{
|
|
static GQuark drag_icon_quark = 0;
|
|
GtkWidget *self;
|
|
|
|
g_return_val_if_fail (GDK_IS_DRAG (drag), NULL);
|
|
|
|
if (G_UNLIKELY (drag_icon_quark == 0))
|
|
drag_icon_quark = g_quark_from_static_string ("-gtk-drag-icon");
|
|
|
|
self = g_object_get_qdata (G_OBJECT (drag), drag_icon_quark);
|
|
if (self == NULL)
|
|
{
|
|
self = g_object_new (GTK_TYPE_DRAG_ICON, NULL);
|
|
|
|
GTK_DRAG_ICON (self)->surface = g_object_ref (gdk_drag_get_drag_surface (drag));
|
|
|
|
g_object_set_qdata_full (G_OBJECT (drag), drag_icon_quark, g_object_ref_sink (self), g_object_unref);
|
|
|
|
if (GTK_DRAG_ICON (self)->child != NULL)
|
|
gtk_widget_set_visible (self, TRUE);
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
/**
|
|
* gtk_drag_icon_set_from_paintable:
|
|
* @drag: a `GdkDrag`
|
|
* @paintable: a `GdkPaintable` to display
|
|
* @hot_x: X coordinate of the hotspot
|
|
* @hot_y: Y coordinate of the hotspot
|
|
*
|
|
* Creates a `GtkDragIcon` that shows @paintable, and associates
|
|
* it with the drag operation.
|
|
*
|
|
* The hotspot position on the paintable is aligned with the
|
|
* hotspot of the cursor.
|
|
*/
|
|
void
|
|
gtk_drag_icon_set_from_paintable (GdkDrag *drag,
|
|
GdkPaintable *paintable,
|
|
int hot_x,
|
|
int hot_y)
|
|
{
|
|
GtkWidget *icon;
|
|
GtkWidget *picture;
|
|
|
|
gdk_drag_set_hotspot (drag, hot_x, hot_y);
|
|
|
|
icon = gtk_drag_icon_get_for_drag (drag);
|
|
|
|
picture = gtk_picture_new_for_paintable (paintable);
|
|
gtk_picture_set_can_shrink (GTK_PICTURE (picture), FALSE);
|
|
gtk_drag_icon_set_child (GTK_DRAG_ICON (icon), picture);
|
|
}
|
|
|
|
/**
|
|
* gtk_drag_icon_set_child: (attributes org.gtk.Method.set_property=child)
|
|
* @self: a `GtkDragIcon`
|
|
* @child: (nullable): a `GtkWidget`
|
|
*
|
|
* Sets the widget to display as the drag icon.
|
|
*/
|
|
void
|
|
gtk_drag_icon_set_child (GtkDragIcon *self,
|
|
GtkWidget *child)
|
|
{
|
|
g_return_if_fail (GTK_IS_DRAG_ICON (self));
|
|
g_return_if_fail (child == NULL || gtk_widget_get_parent (child) == NULL);
|
|
|
|
if (self->child == child)
|
|
return;
|
|
|
|
if (self->child)
|
|
gtk_widget_unparent (self->child);
|
|
|
|
self->child = child;
|
|
|
|
if (self->child)
|
|
{
|
|
gtk_widget_set_parent (self->child, GTK_WIDGET (self));
|
|
gtk_widget_set_visible (GTK_WIDGET (self), TRUE);
|
|
}
|
|
|
|
g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_CHILD]);
|
|
}
|
|
|
|
/**
|
|
* gtk_drag_icon_get_child: (attributes org.gtk.Method.get_property=child)
|
|
* @self: a `GtkDragIcon`
|
|
*
|
|
* Gets the widget currently used as drag icon.
|
|
*
|
|
* Returns: (nullable) (transfer none): The drag icon
|
|
**/
|
|
GtkWidget *
|
|
gtk_drag_icon_get_child (GtkDragIcon *self)
|
|
{
|
|
g_return_val_if_fail (GTK_IS_DRAG_ICON (self), NULL);
|
|
|
|
return self->child;
|
|
}
|
|
|
|
/**
|
|
* gtk_drag_icon_create_widget_for_value:
|
|
* @value: a `GValue`
|
|
*
|
|
* Creates a widget that can be used as a drag icon for the given
|
|
* @value.
|
|
*
|
|
* Supported types include strings, `GdkRGBA` and `GtkTextBuffer`.
|
|
* If GTK does not know how to create a widget for a given value,
|
|
* it will return %NULL.
|
|
*
|
|
* This method is used to set the default drag icon on drag-and-drop
|
|
* operations started by `GtkDragSource`, so you don't need to set
|
|
* a drag icon using this function there.
|
|
*
|
|
* Returns: (nullable) (transfer full): A new `GtkWidget`
|
|
* for displaying @value as a drag icon.
|
|
*/
|
|
GtkWidget *
|
|
gtk_drag_icon_create_widget_for_value (const GValue *value)
|
|
{
|
|
g_return_val_if_fail (G_IS_VALUE (value), NULL);
|
|
|
|
if (G_VALUE_HOLDS (value, G_TYPE_STRING))
|
|
{
|
|
return gtk_label_new (g_value_get_string (value));
|
|
}
|
|
else if (G_VALUE_HOLDS (value, GDK_TYPE_PAINTABLE))
|
|
{
|
|
GtkWidget *image;
|
|
|
|
image = gtk_image_new_from_paintable (g_value_get_object (value));
|
|
gtk_widget_add_css_class (image, "large-icons");
|
|
|
|
return image;
|
|
}
|
|
else if (G_VALUE_HOLDS (value, GDK_TYPE_RGBA))
|
|
{
|
|
GtkWidget *swatch;
|
|
|
|
swatch = gtk_color_swatch_new ();
|
|
gtk_color_swatch_set_can_drag (GTK_COLOR_SWATCH (swatch), FALSE);
|
|
gtk_color_swatch_set_can_drop (GTK_COLOR_SWATCH (swatch), FALSE);
|
|
gtk_color_swatch_set_rgba (GTK_COLOR_SWATCH (swatch), g_value_get_boxed (value));
|
|
|
|
return swatch;
|
|
}
|
|
else if (G_VALUE_HOLDS (value, G_TYPE_FILE))
|
|
{
|
|
GFileInfo *info;
|
|
GtkWidget *image;
|
|
|
|
info = g_file_query_info (G_FILE (g_value_get_object (value)), "standard::icon", 0, NULL, NULL);
|
|
if (!info)
|
|
return NULL;
|
|
|
|
image = gtk_image_new_from_gicon (g_file_info_get_icon (info));
|
|
gtk_widget_add_css_class (image, "large-icons");
|
|
g_object_unref (info);
|
|
|
|
return image;
|
|
}
|
|
else if (G_VALUE_HOLDS (value, GTK_TYPE_TEXT_BUFFER))
|
|
{
|
|
GtkTextBuffer *buffer = g_value_get_object (value);
|
|
GtkTextIter start, end;
|
|
GdkPaintable *paintable;
|
|
GtkWidget *picture;
|
|
|
|
if (buffer == NULL || !gtk_text_buffer_get_selection_bounds (buffer, &start, &end))
|
|
return NULL;
|
|
|
|
picture = gtk_picture_new ();
|
|
paintable = gtk_text_util_create_rich_drag_icon (picture, buffer, &start, &end);
|
|
gtk_picture_set_paintable (GTK_PICTURE (picture), paintable);
|
|
gtk_picture_set_can_shrink (GTK_PICTURE (picture), FALSE);
|
|
g_object_unref (paintable);
|
|
|
|
return picture;
|
|
}
|
|
else if (G_VALUE_HOLDS (value, GSK_TYPE_RENDER_NODE))
|
|
{
|
|
GskRenderNode *node;
|
|
GdkPaintable *paintable;
|
|
graphene_rect_t bounds;
|
|
GtkWidget *image;
|
|
|
|
node = gsk_value_get_render_node (value);
|
|
if (node == NULL)
|
|
return NULL;
|
|
|
|
gsk_render_node_get_bounds (node, &bounds);
|
|
paintable = gtk_render_node_paintable_new (node, &bounds);
|
|
image = gtk_image_new_from_paintable (paintable);
|
|
gtk_image_set_icon_size (GTK_IMAGE (image), GTK_ICON_SIZE_LARGE);
|
|
g_object_unref (paintable);
|
|
|
|
return image;
|
|
}
|
|
else
|
|
{
|
|
return NULL;
|
|
}
|
|
}
|
|
|