/* * Copyright © 2010 Intel Corporation * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library. If not, see . */ #include "config.h" #include "gdkdndprivate.h" #include "gdkinternals.h" #include "gdkproperty.h" #include "gdkprivate-wayland.h" #include "gdkcontentformats.h" #include "gdkdisplay-wayland.h" #include "gdkintl.h" #include "gdkseat-wayland.h" #include "gdkdeviceprivate.h" #include #include #include #include #define GDK_TYPE_WAYLAND_DRAG_CONTEXT (gdk_wayland_drag_context_get_type ()) #define GDK_WAYLAND_DRAG_CONTEXT(object) (G_TYPE_CHECK_INSTANCE_CAST ((object), GDK_TYPE_WAYLAND_DRAG_CONTEXT, GdkWaylandDragContext)) #define GDK_WAYLAND_DRAG_CONTEXT_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GDK_TYPE_WAYLAND_DRAG_CONTEXT, GdkWaylandDragContextClass)) #define GDK_IS_WAYLAND_DRAG_CONTEXT(object) (G_TYPE_CHECK_INSTANCE_TYPE ((object), GDK_TYPE_WAYLAND_DRAG_CONTEXT)) #define GDK_IS_WAYLAND_DRAG_CONTEXT_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GDK_TYPE_WAYLAND_DRAG_CONTEXT)) #define GDK_WAYLAND_DRAG_CONTEXT_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GDK_TYPE_WAYLAND_DRAG_CONTEXT, GdkWaylandDragContextClass)) typedef struct _GdkWaylandDragContext GdkWaylandDragContext; typedef struct _GdkWaylandDragContextClass GdkWaylandDragContextClass; struct _GdkWaylandDragContext { GdkDragContext context; GdkWindow *dnd_window; struct wl_surface *dnd_surface; struct wl_data_source *data_source; GdkDragAction selected_action; uint32_t serial; gdouble x; gdouble y; gint hot_x; gint hot_y; }; struct _GdkWaylandDragContextClass { GdkDragContextClass parent_class; }; static GList *contexts; GType gdk_wayland_drag_context_get_type (void); G_DEFINE_TYPE (GdkWaylandDragContext, gdk_wayland_drag_context, GDK_TYPE_DRAG_CONTEXT) static void gdk_wayland_drag_context_finalize (GObject *object) { GdkWaylandDragContext *wayland_context = GDK_WAYLAND_DRAG_CONTEXT (object); GdkDragContext *context = GDK_DRAG_CONTEXT (object); GdkWindow *dnd_window; contexts = g_list_remove (contexts, context); if (context->is_source) { gdk_drag_context_set_cursor (context, NULL); } if (wayland_context->data_source) wl_data_source_destroy (wayland_context->data_source); dnd_window = wayland_context->dnd_window; G_OBJECT_CLASS (gdk_wayland_drag_context_parent_class)->finalize (object); if (dnd_window) gdk_window_destroy (dnd_window); } void _gdk_wayland_drag_context_emit_event (GdkDragContext *context, GdkEventType type, guint32 time_) { GdkWindow *window; GdkEvent *event; switch ((guint) type) { case GDK_DRAG_ENTER: case GDK_DRAG_LEAVE: case GDK_DRAG_MOTION: case GDK_DRAG_STATUS: case GDK_DROP_START: case GDK_DROP_FINISHED: break; default: return; } if (context->is_source) window = gdk_drag_context_get_source_window (context); else window = gdk_drag_context_get_dest_window (context); event = gdk_event_new (type); event->any.window = g_object_ref (window); event->dnd.context = g_object_ref (context); event->dnd.time = time_; event->dnd.x_root = GDK_WAYLAND_DRAG_CONTEXT (context)->x; event->dnd.y_root = GDK_WAYLAND_DRAG_CONTEXT (context)->y; gdk_event_set_device (event, gdk_drag_context_get_device (context)); gdk_display_put_event (gdk_window_get_display (window), event); g_object_unref (event); } static GdkWindow * gdk_wayland_drag_context_find_window (GdkDragContext *context, GdkWindow *drag_window, gint x_root, gint y_root, GdkDragProtocol *protocol) { GdkDevice *device; GdkWindow *window; device = gdk_drag_context_get_device (context); window = gdk_device_get_window_at_position (device, NULL, NULL); if (window) { window = gdk_window_get_toplevel (window); *protocol = GDK_DRAG_PROTO_WAYLAND; return g_object_ref (window); } return NULL; } static inline uint32_t gdk_to_wl_actions (GdkDragAction action) { uint32_t dnd_actions = 0; if (action & (GDK_ACTION_COPY | GDK_ACTION_LINK | GDK_ACTION_PRIVATE)) dnd_actions |= WL_DATA_DEVICE_MANAGER_DND_ACTION_COPY; if (action & GDK_ACTION_MOVE) dnd_actions |= WL_DATA_DEVICE_MANAGER_DND_ACTION_MOVE; if (action & GDK_ACTION_ASK) dnd_actions |= WL_DATA_DEVICE_MANAGER_DND_ACTION_ASK; return dnd_actions; } void gdk_wayland_drag_context_set_action (GdkDragContext *context, GdkDragAction action) { context->suggested_action = context->action = action; } static gboolean gdk_wayland_drag_context_drag_motion (GdkDragContext *context, GdkWindow *dest_window, GdkDragProtocol protocol, gint x_root, gint y_root, GdkDragAction suggested_action, GdkDragAction possible_actions, guint32 time) { if (context->dest_window != dest_window) { context->dest_window = dest_window ? g_object_ref (dest_window) : NULL; _gdk_wayland_drag_context_set_coords (context, x_root, y_root); _gdk_wayland_drag_context_emit_event (context, GDK_DRAG_STATUS, time); } gdk_wayland_drag_context_set_action (context, suggested_action); return context->dest_window != NULL; } static void gdk_wayland_drag_context_drag_abort (GdkDragContext *context, guint32 time) { } static void gdk_wayland_drag_context_drag_drop (GdkDragContext *context, guint32 time) { } /* Destination side */ static void gdk_wayland_drop_context_set_status (GdkDragContext *context, gboolean accepted) { GdkWaylandDragContext *context_wayland; GdkDisplay *display; struct wl_data_offer *wl_offer; if (!context->dest_window) return; context_wayland = GDK_WAYLAND_DRAG_CONTEXT (context); display = gdk_device_get_display (gdk_drag_context_get_device (context)); wl_offer = gdk_wayland_selection_get_offer (display, gdk_drag_get_selection (context)); if (!wl_offer) return; if (accepted) { const char *const *mimetypes; gsize i, n_mimetypes; mimetypes = gdk_content_formats_get_mime_types (context->formats, &n_mimetypes); for (i = 0; i < n_mimetypes; i++) { if (mimetypes[i] != gdk_atom_intern_static_string ("DELETE")) break; } if (i < n_mimetypes) { wl_data_offer_accept (wl_offer, context_wayland->serial, mimetypes[i]); return; } } wl_data_offer_accept (wl_offer, context_wayland->serial, NULL); } static void gdk_wayland_drag_context_commit_status (GdkDragContext *context) { GdkWaylandDragContext *wayland_context; GdkDisplay *display; uint32_t dnd_actions; wayland_context = GDK_WAYLAND_DRAG_CONTEXT (context); display = gdk_device_get_display (gdk_drag_context_get_device (context)); dnd_actions = gdk_to_wl_actions (wayland_context->selected_action); gdk_wayland_selection_set_current_offer_actions (display, dnd_actions); gdk_wayland_drop_context_set_status (context, wayland_context->selected_action != 0); } static void gdk_wayland_drag_context_drag_status (GdkDragContext *context, GdkDragAction action, guint32 time_) { GdkWaylandDragContext *wayland_context; wayland_context = GDK_WAYLAND_DRAG_CONTEXT (context); wayland_context->selected_action = action; } static void gdk_wayland_drag_context_drop_reply (GdkDragContext *context, gboolean accepted, guint32 time_) { if (!accepted) gdk_wayland_drop_context_set_status (context, accepted); } static void gdk_wayland_drag_context_drop_finish (GdkDragContext *context, gboolean success, guint32 time) { GdkDisplay *display = gdk_device_get_display (gdk_drag_context_get_device (context)); GdkWaylandDisplay *display_wayland = GDK_WAYLAND_DISPLAY (display); GdkWaylandDragContext *wayland_context; struct wl_data_offer *wl_offer; GdkAtom selection; wayland_context = GDK_WAYLAND_DRAG_CONTEXT (context); selection = gdk_drag_get_selection (context); wl_offer = gdk_wayland_selection_get_offer (display, selection); if (wl_offer && success && wayland_context->selected_action && wayland_context->selected_action != GDK_ACTION_ASK) { gdk_wayland_drag_context_commit_status (context); if (display_wayland->data_device_manager_version >= WL_DATA_OFFER_FINISH_SINCE_VERSION) wl_data_offer_finish (wl_offer); } gdk_wayland_selection_set_offer (display, selection, NULL); } static void gdk_wayland_drag_context_read_async (GdkDragContext *context, GdkContentFormats *formats, int io_priority, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data) { GdkDisplay *display; GdkContentFormats *dnd_formats; GInputStream *stream; struct wl_data_offer *offer; const char *mime_type; int pipe_fd[2]; GError *error = NULL; GTask *task; display = gdk_drag_context_get_display (context), task = g_task_new (context, cancellable, callback, user_data); g_task_set_priority (task, io_priority); g_task_set_source_tag (task, gdk_wayland_drag_context_read_async); GDK_NOTE (DND, char *s = gdk_content_formats_to_string (formats); g_printerr ("%p: read for %s\n", context, s); g_free (s); ); dnd_formats = gdk_wayland_selection_get_targets (display, gdk_drag_get_selection (context)); mime_type = gdk_content_formats_match_mime_type (formats, dnd_formats); if (mime_type == NULL) { g_task_return_new_error (task, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED, _("No compatible transfer format found")); return; } /* offer formats should be empty if we have no offer */ offer = gdk_wayland_selection_get_offer (display, gdk_drag_get_selection (context)); g_assert (offer); g_task_set_task_data (task, (gpointer) mime_type, NULL); if (!g_unix_open_pipe (pipe_fd, FD_CLOEXEC, &error)) { g_task_return_error (task, error); return; } wl_data_offer_accept (offer, _gdk_wayland_display_get_serial (GDK_WAYLAND_DISPLAY (display)), mime_type); wl_data_offer_receive (offer, mime_type, pipe_fd[1]); stream = g_unix_input_stream_new (pipe_fd[0], TRUE); close (pipe_fd[1]); g_task_return_pointer (task, stream, g_object_unref); } static GInputStream * gdk_wayland_drag_context_read_finish (GdkDragContext *context, const char **out_mime_type, GAsyncResult *result, GError **error) { GTask *task; g_return_val_if_fail (g_task_is_valid (result, G_OBJECT (context)), NULL); task = G_TASK (result); g_return_val_if_fail (g_task_get_source_tag (task) == gdk_wayland_drag_context_read_async, NULL); if (out_mime_type) *out_mime_type = g_task_get_task_data (task); return g_task_propagate_pointer (task, error); } static gboolean gdk_wayland_drag_context_drop_status (GdkDragContext *context) { return FALSE; } static GdkAtom gdk_wayland_drag_context_get_selection (GdkDragContext *context) { return gdk_atom_intern_static_string ("GdkWaylandSelection"); } static void gdk_wayland_drag_context_init (GdkWaylandDragContext *context_wayland) { GdkDragContext *context; context = GDK_DRAG_CONTEXT (context_wayland); contexts = g_list_prepend (contexts, context); context->action = GDK_ACTION_COPY; context->suggested_action = GDK_ACTION_COPY; context->actions = GDK_ACTION_COPY | GDK_ACTION_MOVE; } static GdkWindow * gdk_wayland_drag_context_get_drag_window (GdkDragContext *context) { return GDK_WAYLAND_DRAG_CONTEXT (context)->dnd_window; } static void gdk_wayland_drag_context_set_hotspot (GdkDragContext *context, gint hot_x, gint hot_y) { GdkWaylandDragContext *context_wayland = GDK_WAYLAND_DRAG_CONTEXT (context); gint prev_hot_x = context_wayland->hot_x; gint prev_hot_y = context_wayland->hot_y; const GdkRectangle damage_rect = { .width = 1, .height = 1 }; context_wayland->hot_x = hot_x; context_wayland->hot_y = hot_y; if (prev_hot_x == hot_x && prev_hot_y == hot_y) return; _gdk_wayland_window_offset_next_wl_buffer (context_wayland->dnd_window, -hot_x, -hot_y); gdk_window_invalidate_rect (context_wayland->dnd_window, &damage_rect, FALSE); } static void gdk_wayland_drag_context_set_cursor (GdkDragContext *context, GdkCursor *cursor) { GdkDevice *device = gdk_drag_context_get_device (context); gdk_wayland_seat_set_global_cursor (gdk_device_get_seat (device), cursor); } static void gdk_wayland_drag_context_action_changed (GdkDragContext *context, GdkDragAction action) { GdkCursor *cursor; cursor = gdk_drag_get_cursor (context, action); gdk_drag_context_set_cursor (context, cursor); } static void gdk_wayland_drag_context_drop_performed (GdkDragContext *context, guint32 time_) { gdk_drag_context_set_cursor (context, NULL); } static void gdk_wayland_drag_context_cancel (GdkDragContext *context, GdkDragCancelReason reason) { gdk_drag_context_set_cursor (context, NULL); } static void gdk_wayland_drag_context_drop_done (GdkDragContext *context, gboolean success) { GdkWaylandDragContext *context_wayland = GDK_WAYLAND_DRAG_CONTEXT (context); if (success) { if (context_wayland->dnd_window) gdk_window_hide (context_wayland->dnd_window); } } static void gdk_wayland_drag_context_class_init (GdkWaylandDragContextClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); GdkDragContextClass *context_class = GDK_DRAG_CONTEXT_CLASS (klass); object_class->finalize = gdk_wayland_drag_context_finalize; context_class->find_window = gdk_wayland_drag_context_find_window; context_class->drag_status = gdk_wayland_drag_context_drag_status; context_class->drag_motion = gdk_wayland_drag_context_drag_motion; context_class->drag_abort = gdk_wayland_drag_context_drag_abort; context_class->drag_drop = gdk_wayland_drag_context_drag_drop; context_class->drop_reply = gdk_wayland_drag_context_drop_reply; context_class->drop_finish = gdk_wayland_drag_context_drop_finish; context_class->drop_finish = gdk_wayland_drag_context_drop_finish; context_class->read_async = gdk_wayland_drag_context_read_async; context_class->read_finish = gdk_wayland_drag_context_read_finish; context_class->drop_status = gdk_wayland_drag_context_drop_status; context_class->get_selection = gdk_wayland_drag_context_get_selection; context_class->get_drag_window = gdk_wayland_drag_context_get_drag_window; context_class->set_hotspot = gdk_wayland_drag_context_set_hotspot; context_class->drop_done = gdk_wayland_drag_context_drop_done; context_class->set_cursor = gdk_wayland_drag_context_set_cursor; context_class->action_changed = gdk_wayland_drag_context_action_changed; context_class->drop_performed = gdk_wayland_drag_context_drop_performed; context_class->cancel = gdk_wayland_drag_context_cancel; context_class->commit_drag_status = gdk_wayland_drag_context_commit_status; } void _gdk_wayland_window_register_dnd (GdkWindow *window) { } static GdkWindow * create_dnd_window (GdkDisplay *display) { GdkWindow *window; window = gdk_window_new_popup (display, &(GdkRectangle) { 0, 0, 100, 100 }); gdk_window_set_type_hint (window, GDK_WINDOW_TYPE_HINT_DND); return window; } GdkDragContext * _gdk_wayland_window_drag_begin (GdkWindow *window, GdkDevice *device, GdkContentProvider *content, GdkDragAction actions, gint dx, gint dy) { GdkWaylandDragContext *context_wayland; GdkDragContext *context; GdkWaylandDisplay *display_wayland; GdkWindow *toplevel; const char *const *mimetypes; gsize i, n_mimetypes; display_wayland = GDK_WAYLAND_DISPLAY (gdk_device_get_display (device)); context_wayland = g_object_new (GDK_TYPE_WAYLAND_DRAG_CONTEXT, "display", display_wayland, "content", content, NULL); context = GDK_DRAG_CONTEXT (context_wayland); context->source_window = g_object_ref (window); context->is_source = TRUE; gdk_drag_context_set_device (context, device); context_wayland->dnd_window = create_dnd_window (gdk_window_get_display (window)); context_wayland->dnd_surface = gdk_wayland_window_get_wl_surface (context_wayland->dnd_window); context_wayland->data_source = gdk_wayland_selection_get_data_source (window, gdk_wayland_drag_context_get_selection (context)); mimetypes = gdk_content_formats_get_mime_types (context->formats, &n_mimetypes); for (i = 0; i < n_mimetypes; i++) { wl_data_source_offer (context_wayland->data_source, mimetypes[i]); } toplevel = _gdk_device_window_at_position (device, NULL, NULL, NULL, TRUE); if (display_wayland->data_device_manager_version >= WL_DATA_SOURCE_SET_ACTIONS_SINCE_VERSION) { wl_data_source_set_actions (context_wayland->data_source, gdk_to_wl_actions (actions)); } wl_data_device_start_drag (gdk_wayland_device_get_data_device (device), context_wayland->data_source, gdk_wayland_window_get_wl_surface (toplevel), context_wayland->dnd_surface, _gdk_wayland_display_get_serial (display_wayland)); gdk_seat_ungrab (gdk_device_get_seat (device)); return context; } GdkDragContext * _gdk_wayland_drop_context_new (GdkDisplay *display, struct wl_data_device *data_device) { GdkWaylandDragContext *context_wayland; GdkDragContext *context; context_wayland = g_object_new (GDK_TYPE_WAYLAND_DRAG_CONTEXT, "display", display, NULL); context = GDK_DRAG_CONTEXT (context_wayland); context->is_source = FALSE; context->formats = gdk_content_formats_new (NULL, 0); return context; } void gdk_wayland_drop_context_update_targets (GdkDragContext *context) { GdkDisplay *display; GdkDevice *device; device = gdk_drag_context_get_device (context); display = gdk_device_get_display (device); gdk_content_formats_unref (context->formats); context->formats = gdk_wayland_selection_get_targets (display, gdk_drag_get_selection (context)); if (context->formats) gdk_content_formats_ref (context->formats); else context->formats = gdk_content_formats_new (NULL, 0); } void _gdk_wayland_drag_context_set_coords (GdkDragContext *context, gdouble x, gdouble y) { GdkWaylandDragContext *context_wayland; context_wayland = GDK_WAYLAND_DRAG_CONTEXT (context); context_wayland->x = x; context_wayland->y = y; } void _gdk_wayland_drag_context_set_source_window (GdkDragContext *context, GdkWindow *window) { if (context->source_window) g_object_unref (context->source_window); context->source_window = window ? g_object_ref (window) : NULL; } void _gdk_wayland_drag_context_set_dest_window (GdkDragContext *context, GdkWindow *dest_window, uint32_t serial) { if (context->dest_window) g_object_unref (context->dest_window); context->dest_window = dest_window ? g_object_ref (dest_window) : NULL; GDK_WAYLAND_DRAG_CONTEXT (context)->serial = serial; gdk_wayland_drop_context_update_targets (context); } GdkDragContext * gdk_wayland_drag_context_lookup_by_data_source (struct wl_data_source *source) { GList *l; for (l = contexts; l; l = l->next) { GdkWaylandDragContext *wayland_context = l->data; if (wayland_context->data_source == source) return l->data; } return NULL; } GdkDragContext * gdk_wayland_drag_context_lookup_by_source_window (GdkWindow *window) { GList *l; for (l = contexts; l; l = l->next) { if (window == gdk_drag_context_get_source_window (l->data)) return l->data; } return NULL; } struct wl_data_source * gdk_wayland_drag_context_get_data_source (GdkDragContext *context) { return GDK_WAYLAND_DRAG_CONTEXT (context)->data_source; }