2018-11-19 17:17:25 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 Red Hat
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* 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, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
|
|
* 02111-1307, USA.
|
|
|
|
*
|
|
|
|
* Author: Carlos Garnacho <carlosg@gnome.org>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <gdk/gdkx.h>
|
|
|
|
|
2020-04-08 11:37:19 +00:00
|
|
|
#include "core/meta-selection-private.h"
|
2020-04-20 14:49:24 +00:00
|
|
|
#include "meta/meta-selection-source-memory.h"
|
2018-11-19 17:17:25 +00:00
|
|
|
#include "x11/meta-selection-source-x11-private.h"
|
|
|
|
#include "x11/meta-x11-selection-output-stream-private.h"
|
|
|
|
#include "x11/meta-x11-selection-private.h"
|
|
|
|
|
2020-01-13 13:18:00 +00:00
|
|
|
#define UTF8_STRING_MIMETYPE "text/plain;charset=utf-8"
|
|
|
|
#define STRING_MIMETYPE "text/plain"
|
|
|
|
|
2018-11-19 17:17:25 +00:00
|
|
|
static gboolean
|
|
|
|
atom_to_selection_type (Display *xdisplay,
|
|
|
|
Atom selection,
|
|
|
|
MetaSelectionType *selection_type)
|
|
|
|
{
|
|
|
|
if (selection == XInternAtom (xdisplay, "PRIMARY", False))
|
|
|
|
*selection_type = META_SELECTION_PRIMARY;
|
|
|
|
else if (selection == XInternAtom (xdisplay, "CLIPBOARD", False))
|
|
|
|
*selection_type = META_SELECTION_CLIPBOARD;
|
|
|
|
else if (selection == XInternAtom (xdisplay, "XdndSelection", False))
|
|
|
|
*selection_type = META_SELECTION_DND;
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Atom
|
|
|
|
selection_to_atom (MetaSelectionType type,
|
|
|
|
Display *xdisplay)
|
|
|
|
{
|
|
|
|
Atom atom;
|
|
|
|
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case META_SELECTION_PRIMARY:
|
|
|
|
atom = XInternAtom (xdisplay, "PRIMARY", False);
|
|
|
|
break;
|
|
|
|
case META_SELECTION_CLIPBOARD:
|
|
|
|
atom = XInternAtom (xdisplay, "CLIPBOARD", False);
|
|
|
|
break;
|
|
|
|
case META_SELECTION_DND:
|
|
|
|
atom = XInternAtom (xdisplay, "XdndSelection", False);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
g_warn_if_reached ();
|
|
|
|
atom = None;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return atom;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GBytes *
|
|
|
|
mimetypes_to_bytes (GList *mimetypes,
|
|
|
|
Display *xdisplay)
|
|
|
|
{
|
2020-07-24 19:20:58 +00:00
|
|
|
GArray *atoms = g_array_new (FALSE, FALSE, sizeof (Atom));
|
2018-11-19 17:17:25 +00:00
|
|
|
GList *l;
|
2020-07-24 19:20:58 +00:00
|
|
|
char *mimetype;
|
|
|
|
Atom atom;
|
|
|
|
gboolean utf8_string_found = FALSE, utf8_string_mimetype_found = FALSE;
|
|
|
|
gboolean string_found = FALSE, string_mimetype_found = FALSE;
|
|
|
|
GBytes *bytes;
|
2018-11-19 17:17:25 +00:00
|
|
|
|
|
|
|
for (l = mimetypes; l; l = l->next)
|
2020-07-24 19:20:58 +00:00
|
|
|
{
|
|
|
|
mimetype = l->data;
|
|
|
|
atom = XInternAtom (xdisplay, mimetype, False);
|
|
|
|
g_array_append_val (atoms, atom);
|
|
|
|
utf8_string_mimetype_found |= strcmp (mimetype, UTF8_STRING_MIMETYPE) == 0;
|
|
|
|
utf8_string_found |= strcmp (mimetype, "UTF8_STRING") == 0;
|
|
|
|
string_mimetype_found |= strcmp (mimetype, STRING_MIMETYPE) == 0;
|
|
|
|
string_found |= strcmp (mimetype, "STRING") == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Some X11 clients can only handle STRING/UTF8_STRING but not the
|
|
|
|
* corresponding mimetypes. */
|
|
|
|
if (utf8_string_mimetype_found && !utf8_string_found)
|
|
|
|
{
|
|
|
|
atom = XInternAtom (xdisplay, "UTF8_STRING", False);
|
|
|
|
g_array_append_val (atoms, atom);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (string_mimetype_found && !string_found)
|
|
|
|
{
|
|
|
|
atom = XInternAtom (xdisplay, "STRING", False);
|
|
|
|
g_array_append_val (atoms, atom);
|
|
|
|
}
|
|
|
|
|
|
|
|
atom = XInternAtom (xdisplay, "TARGETS", False);
|
|
|
|
g_array_append_val (atoms, atom);
|
|
|
|
|
|
|
|
atom = XInternAtom (xdisplay, "TIMESTAMP", False);
|
|
|
|
g_array_append_val (atoms, atom);
|
2018-11-19 17:17:25 +00:00
|
|
|
|
2020-07-24 19:20:58 +00:00
|
|
|
bytes = g_bytes_new_take (atoms->data, atoms->len * sizeof (Atom));
|
|
|
|
g_array_free (atoms, FALSE);
|
2018-11-19 17:17:25 +00:00
|
|
|
|
2020-07-24 19:20:58 +00:00
|
|
|
return bytes;
|
2018-11-19 17:17:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
send_selection_notify (XSelectionRequestEvent *request_event,
|
|
|
|
gboolean accepted)
|
|
|
|
{
|
|
|
|
Display *xdisplay = GDK_DISPLAY_XDISPLAY (gdk_display_get_default ());
|
|
|
|
XSelectionEvent event;
|
|
|
|
|
|
|
|
memset(&event, 0, sizeof (XSelectionEvent));
|
|
|
|
event.type = SelectionNotify;
|
|
|
|
event.time = request_event->time;
|
|
|
|
event.requestor = request_event->requestor;
|
|
|
|
event.selection = request_event->selection;
|
|
|
|
event.target = request_event->target;
|
|
|
|
event.property = accepted ? request_event->property : None;
|
|
|
|
|
|
|
|
XSendEvent (xdisplay, request_event->requestor,
|
|
|
|
False, NoEventMask, (XEvent *) &event);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
write_mimetypes_cb (GOutputStream *stream,
|
|
|
|
GAsyncResult *res,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
g_output_stream_write_bytes_finish (stream, res, &error);
|
|
|
|
g_output_stream_close (stream, NULL, NULL);
|
2020-01-05 01:20:18 +00:00
|
|
|
g_object_unref (stream);
|
2018-11-19 17:17:25 +00:00
|
|
|
|
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
g_warning ("Could not fetch selection mimetypes: %s\n", error->message);
|
|
|
|
g_error_free (error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
transfer_cb (MetaSelection *selection,
|
|
|
|
GAsyncResult *res,
|
|
|
|
GOutputStream *output)
|
|
|
|
{
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
if (!meta_selection_transfer_finish (selection, res, &error))
|
|
|
|
{
|
|
|
|
g_warning ("Error writing data to X11 selection: %s", error->message);
|
|
|
|
g_error_free (error);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_output_stream_close (output, NULL, NULL);
|
|
|
|
g_object_unref (output);
|
|
|
|
}
|
|
|
|
|
2020-01-13 13:18:00 +00:00
|
|
|
static char *
|
|
|
|
meta_x11_selection_find_target (MetaX11Display *x11_display,
|
|
|
|
MetaSelection *selection,
|
|
|
|
MetaSelectionType selection_type,
|
|
|
|
Atom selection_atom)
|
|
|
|
{
|
|
|
|
GList* mimetypes = NULL;
|
|
|
|
const gchar *atom_name;
|
|
|
|
char *retval;
|
|
|
|
|
|
|
|
mimetypes = meta_selection_get_mimetypes (selection, selection_type);
|
|
|
|
atom_name = gdk_x11_get_xatom_name (selection_atom);
|
|
|
|
|
|
|
|
if (g_list_find_custom (mimetypes, atom_name, (GCompareFunc) g_strcmp0))
|
|
|
|
{
|
|
|
|
retval = g_strdup (atom_name);
|
|
|
|
}
|
|
|
|
else if (strcmp (atom_name, "UTF8_STRING") == 0 &&
|
|
|
|
g_list_find_custom (mimetypes, UTF8_STRING_MIMETYPE,
|
|
|
|
(GCompareFunc) g_strcmp0))
|
|
|
|
{
|
|
|
|
retval = g_strdup (UTF8_STRING_MIMETYPE);
|
|
|
|
}
|
|
|
|
else if (strcmp (atom_name, "STRING") == 0 &&
|
|
|
|
g_list_find_custom (mimetypes, STRING_MIMETYPE,
|
|
|
|
(GCompareFunc) g_strcmp0))
|
|
|
|
{
|
|
|
|
retval = g_strdup (STRING_MIMETYPE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
retval = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_list_free_full (mimetypes, g_free);
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2018-11-19 17:17:25 +00:00
|
|
|
static gboolean
|
|
|
|
meta_x11_selection_handle_selection_request (MetaX11Display *x11_display,
|
|
|
|
XEvent *xevent)
|
|
|
|
{
|
|
|
|
XSelectionRequestEvent *event = (XSelectionRequestEvent *) xevent;
|
|
|
|
MetaSelectionType selection_type;
|
|
|
|
MetaSelection *selection;
|
|
|
|
GOutputStream *output;
|
|
|
|
GList *mimetypes;
|
|
|
|
|
|
|
|
if (!atom_to_selection_type (x11_display->xdisplay, event->selection, &selection_type))
|
|
|
|
return FALSE;
|
|
|
|
if (x11_display->selection.xwindow != event->owner)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
selection = meta_display_get_selection (meta_get_display ());
|
|
|
|
|
|
|
|
if (event->target == gdk_x11_get_xatom_by_name ("TARGETS"))
|
|
|
|
{
|
|
|
|
GBytes *bytes;
|
|
|
|
|
|
|
|
mimetypes = meta_selection_get_mimetypes (selection, selection_type);
|
|
|
|
|
|
|
|
if (!mimetypes)
|
|
|
|
{
|
|
|
|
send_selection_notify (event, FALSE);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
output = meta_x11_selection_output_stream_new (x11_display, event->requestor,
|
|
|
|
gdk_x11_get_xatom_name (event->selection),
|
|
|
|
gdk_x11_get_xatom_name (event->target),
|
|
|
|
gdk_x11_get_xatom_name (event->property),
|
|
|
|
"ATOM", 32, event->time);
|
|
|
|
|
|
|
|
bytes = mimetypes_to_bytes (mimetypes, x11_display->xdisplay);
|
|
|
|
g_list_free_full (mimetypes, g_free);
|
|
|
|
|
|
|
|
g_output_stream_write_bytes_async (output,
|
|
|
|
bytes,
|
|
|
|
G_PRIORITY_DEFAULT,
|
|
|
|
NULL,
|
|
|
|
(GAsyncReadyCallback) write_mimetypes_cb,
|
|
|
|
output);
|
|
|
|
g_bytes_unref (bytes);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else if (event->target == gdk_x11_get_xatom_by_name ("DELETE"))
|
|
|
|
{
|
|
|
|
/* DnD only, this is just handled through other means on our non-x11
|
|
|
|
* sources, so just go with it.
|
|
|
|
*/
|
|
|
|
send_selection_notify (event, TRUE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-01-13 13:18:00 +00:00
|
|
|
g_autofree char *target = NULL;
|
2018-11-19 17:17:25 +00:00
|
|
|
|
2020-01-13 13:18:00 +00:00
|
|
|
target = meta_x11_selection_find_target (x11_display, selection,
|
|
|
|
selection_type, event->target);
|
2018-11-19 17:17:25 +00:00
|
|
|
|
2020-01-13 13:18:00 +00:00
|
|
|
if (target != NULL)
|
2018-11-19 17:17:25 +00:00
|
|
|
{
|
|
|
|
output = meta_x11_selection_output_stream_new (x11_display,
|
|
|
|
event->requestor,
|
|
|
|
gdk_x11_get_xatom_name (event->selection),
|
|
|
|
gdk_x11_get_xatom_name (event->target),
|
|
|
|
gdk_x11_get_xatom_name (event->property),
|
|
|
|
gdk_x11_get_xatom_name (event->target),
|
|
|
|
8, event->time);
|
|
|
|
|
|
|
|
meta_selection_transfer_async (selection,
|
|
|
|
selection_type,
|
2020-01-13 13:18:00 +00:00
|
|
|
target,
|
2018-11-19 17:17:25 +00:00
|
|
|
-1,
|
|
|
|
output,
|
|
|
|
NULL,
|
|
|
|
(GAsyncReadyCallback) transfer_cb,
|
|
|
|
output);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else
|
2020-01-13 13:18:00 +00:00
|
|
|
{
|
|
|
|
send_selection_notify (event, FALSE);
|
|
|
|
}
|
2018-11-19 17:17:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
MetaX11Display *x11_display;
|
|
|
|
MetaSelection *selection;
|
|
|
|
MetaSelectionType selection_type;
|
|
|
|
} SourceNewData;
|
|
|
|
|
|
|
|
static void
|
|
|
|
source_new_cb (GObject *object,
|
|
|
|
GAsyncResult *res,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
MetaSelectionSource *source;
|
|
|
|
SourceNewData *data = user_data;
|
|
|
|
MetaSelection *selection = data->selection;
|
|
|
|
MetaSelectionType selection_type = data->selection_type;
|
|
|
|
MetaX11Display *x11_display = data->x11_display;
|
2019-05-13 10:44:30 +00:00
|
|
|
g_autoptr (GError) error = NULL;
|
2018-11-19 17:17:25 +00:00
|
|
|
|
|
|
|
source = meta_selection_source_x11_new_finish (res, &error);
|
|
|
|
if (source)
|
|
|
|
{
|
|
|
|
g_set_object (&x11_display->selection.owners[selection_type], source);
|
2020-04-20 14:47:49 +00:00
|
|
|
meta_selection_set_owner (selection, selection_type, source);
|
2020-01-11 02:31:02 +00:00
|
|
|
g_object_unref (source);
|
2018-11-19 17:17:25 +00:00
|
|
|
}
|
2019-05-13 10:44:30 +00:00
|
|
|
else if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
2018-11-19 17:17:25 +00:00
|
|
|
{
|
|
|
|
g_warning ("Could not create selection source for X11: %s",
|
|
|
|
error->message);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (data);
|
|
|
|
}
|
|
|
|
|
2020-07-14 17:02:06 +00:00
|
|
|
static gboolean
|
|
|
|
unset_clipboard_owner (gpointer data)
|
|
|
|
{
|
|
|
|
MetaDisplay *display = meta_get_display ();
|
|
|
|
MetaSelection *selection = meta_display_get_selection (display);
|
|
|
|
MetaX11Display *x11_display = meta_display_get_x11_display (display);
|
|
|
|
|
|
|
|
meta_selection_unset_owner (selection, META_SELECTION_CLIPBOARD,
|
|
|
|
x11_display->selection.owners[META_SELECTION_CLIPBOARD]);
|
|
|
|
g_clear_object (&x11_display->selection.owners[META_SELECTION_CLIPBOARD]);
|
|
|
|
|
|
|
|
x11_display->selection.timeout_id = 0;
|
|
|
|
|
|
|
|
return G_SOURCE_REMOVE;
|
|
|
|
}
|
|
|
|
|
2018-11-19 17:17:25 +00:00
|
|
|
static gboolean
|
|
|
|
meta_x11_selection_handle_xfixes_selection_notify (MetaX11Display *x11_display,
|
|
|
|
XEvent *xevent)
|
|
|
|
{
|
|
|
|
XFixesSelectionNotifyEvent *event = (XFixesSelectionNotifyEvent *) xevent;
|
|
|
|
Display *xdisplay = x11_display->xdisplay;
|
|
|
|
MetaSelectionType selection_type;
|
|
|
|
MetaSelection *selection;
|
|
|
|
|
|
|
|
if (!atom_to_selection_type (xdisplay, event->selection, &selection_type))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
selection = meta_display_get_selection (meta_get_display ());
|
|
|
|
|
2020-07-14 17:02:06 +00:00
|
|
|
if (selection_type == META_SELECTION_CLIPBOARD)
|
|
|
|
g_clear_handle_id (&x11_display->selection.timeout_id, g_source_remove);
|
|
|
|
|
2018-11-19 17:17:25 +00:00
|
|
|
if (x11_display->selection.cancellables[selection_type])
|
|
|
|
{
|
|
|
|
g_cancellable_cancel (x11_display->selection.cancellables[selection_type]);
|
|
|
|
g_clear_object (&x11_display->selection.cancellables[selection_type]);
|
|
|
|
}
|
|
|
|
|
|
|
|
x11_display->selection.cancellables[selection_type] = g_cancellable_new ();
|
|
|
|
|
2020-04-20 14:49:24 +00:00
|
|
|
if (event->owner == None && x11_display->selection.owners[selection_type])
|
2018-11-19 17:17:25 +00:00
|
|
|
{
|
2020-07-03 11:47:38 +00:00
|
|
|
if (event->subtype == XFixesSetSelectionOwnerNotify)
|
2020-04-20 14:49:24 +00:00
|
|
|
{
|
|
|
|
MetaSelectionSource *source;
|
|
|
|
|
|
|
|
/* Replace with an empty owner */
|
|
|
|
source = g_object_new (META_TYPE_SELECTION_SOURCE_MEMORY, NULL);
|
|
|
|
g_set_object (&x11_display->selection.owners[selection_type], source);
|
|
|
|
meta_selection_set_owner (selection, selection_type, source);
|
|
|
|
g_object_unref (source);
|
|
|
|
}
|
2020-07-14 17:02:06 +00:00
|
|
|
else if (event->subtype == XFixesSelectionWindowDestroyNotify &&
|
|
|
|
selection_type == META_SELECTION_CLIPBOARD)
|
|
|
|
{
|
|
|
|
/* Selection window might have gotten destroyed as part of application
|
|
|
|
* shutdown. Trigger restoring clipboard, but wait a bit, because some
|
|
|
|
* clients, like wine, destroy the old window immediately before a new
|
|
|
|
* selection. Restoring the clipboard in this case would overwrite the
|
|
|
|
* new selection, so this will be cancelled when a new selection
|
|
|
|
* arrives. */
|
|
|
|
x11_display->selection.timeout_id = g_timeout_add (10,
|
|
|
|
unset_clipboard_owner,
|
|
|
|
NULL);
|
|
|
|
}
|
2020-04-20 14:49:24 +00:00
|
|
|
else
|
2018-11-19 17:17:25 +00:00
|
|
|
{
|
|
|
|
/* An X client went away, clear the selection */
|
|
|
|
meta_selection_unset_owner (selection, selection_type,
|
|
|
|
x11_display->selection.owners[selection_type]);
|
2020-04-17 12:13:50 +00:00
|
|
|
g_clear_object (&x11_display->selection.owners[selection_type]);
|
2018-11-19 17:17:25 +00:00
|
|
|
}
|
|
|
|
}
|
2020-04-20 14:49:24 +00:00
|
|
|
else if (event->owner != None && event->owner != x11_display->selection.xwindow)
|
2018-11-19 17:17:25 +00:00
|
|
|
{
|
|
|
|
SourceNewData *data;
|
|
|
|
|
|
|
|
data = g_new (SourceNewData, 1);
|
|
|
|
data->x11_display = x11_display;
|
|
|
|
data->selection = selection;
|
|
|
|
data->selection_type = selection_type;
|
|
|
|
|
|
|
|
meta_selection_source_x11_new_async (x11_display,
|
|
|
|
event->owner,
|
|
|
|
event->timestamp,
|
|
|
|
event->selection,
|
|
|
|
x11_display->selection.cancellables[selection_type],
|
|
|
|
source_new_cb,
|
|
|
|
data);
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_x11_selection_handle_event (MetaX11Display *x11_display,
|
|
|
|
XEvent *xevent)
|
|
|
|
{
|
|
|
|
if (xevent->type == SelectionRequest)
|
|
|
|
return meta_x11_selection_handle_selection_request (x11_display, xevent);
|
|
|
|
else if (xevent->type - x11_display->xfixes_event_base == XFixesSelectionNotify)
|
|
|
|
return meta_x11_selection_handle_xfixes_selection_notify (x11_display, xevent);
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2020-04-08 11:37:19 +00:00
|
|
|
notify_selection_owner (MetaX11Display *x11_display,
|
|
|
|
MetaSelectionType selection_type,
|
|
|
|
MetaSelectionSource *new_owner)
|
2018-11-19 17:17:25 +00:00
|
|
|
{
|
|
|
|
Display *xdisplay = x11_display->xdisplay;
|
|
|
|
|
2020-04-20 14:47:49 +00:00
|
|
|
if (new_owner && new_owner != x11_display->selection.owners[selection_type])
|
2018-11-19 17:17:25 +00:00
|
|
|
{
|
|
|
|
if (x11_display->selection.cancellables[selection_type])
|
|
|
|
{
|
|
|
|
g_cancellable_cancel (x11_display->selection.cancellables[selection_type]);
|
|
|
|
g_clear_object (&x11_display->selection.cancellables[selection_type]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If the owner is non-X11, claim the selection on our selection
|
|
|
|
* window, so X11 apps can interface with it.
|
|
|
|
*/
|
|
|
|
XSetSelectionOwner (xdisplay,
|
|
|
|
selection_to_atom (selection_type, xdisplay),
|
|
|
|
x11_display->selection.xwindow,
|
|
|
|
META_CURRENT_TIME);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_x11_selection_init (MetaX11Display *x11_display)
|
|
|
|
{
|
|
|
|
XSetWindowAttributes attributes = { 0 };
|
|
|
|
MetaDisplay *display = meta_get_display ();
|
2020-04-08 11:37:19 +00:00
|
|
|
MetaSelection *selection;
|
2018-11-19 17:17:25 +00:00
|
|
|
guint mask, i;
|
|
|
|
|
|
|
|
attributes.event_mask = PropertyChangeMask | SubstructureNotifyMask;
|
|
|
|
attributes.override_redirect = True;
|
|
|
|
|
2020-07-14 17:02:06 +00:00
|
|
|
x11_display->selection.timeout_id = 0;
|
2018-11-19 17:17:25 +00:00
|
|
|
x11_display->selection.xwindow =
|
|
|
|
XCreateWindow (x11_display->xdisplay,
|
|
|
|
x11_display->xroot,
|
|
|
|
-1, -1, 1, 1,
|
|
|
|
0, /* border width */
|
|
|
|
0, /* depth */
|
|
|
|
InputOnly, /* class */
|
|
|
|
CopyFromParent, /* visual */
|
|
|
|
CWEventMask | CWOverrideRedirect,
|
|
|
|
&attributes);
|
|
|
|
|
|
|
|
mask = XFixesSetSelectionOwnerNotifyMask |
|
|
|
|
XFixesSelectionWindowDestroyNotifyMask |
|
|
|
|
XFixesSelectionClientCloseNotifyMask;
|
|
|
|
|
2020-04-08 11:37:19 +00:00
|
|
|
selection = meta_display_get_selection (display);
|
|
|
|
|
2018-11-19 17:17:25 +00:00
|
|
|
for (i = 0; i < META_N_SELECTION_TYPES; i++)
|
|
|
|
{
|
2020-04-08 11:37:19 +00:00
|
|
|
MetaSelectionSource *owner;
|
|
|
|
|
2018-11-19 17:17:25 +00:00
|
|
|
XFixesSelectSelectionInput (x11_display->xdisplay,
|
|
|
|
x11_display->selection.xwindow,
|
|
|
|
selection_to_atom (i, x11_display->xdisplay),
|
|
|
|
mask);
|
2020-04-08 11:37:19 +00:00
|
|
|
owner = meta_selection_get_current_owner (selection, i);
|
|
|
|
notify_selection_owner (x11_display, i, owner);
|
2018-11-19 17:17:25 +00:00
|
|
|
}
|
|
|
|
|
2020-04-08 11:37:19 +00:00
|
|
|
g_signal_connect_swapped (selection,
|
|
|
|
"owner-changed",
|
|
|
|
G_CALLBACK (notify_selection_owner),
|
|
|
|
x11_display);
|
2018-11-19 17:17:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_x11_selection_shutdown (MetaX11Display *x11_display)
|
|
|
|
{
|
|
|
|
MetaDisplay *display = meta_get_display ();
|
2020-01-30 08:19:39 +00:00
|
|
|
guint i;
|
2018-11-19 17:17:25 +00:00
|
|
|
|
|
|
|
g_signal_handlers_disconnect_by_func (meta_display_get_selection (display),
|
2020-04-08 11:37:19 +00:00
|
|
|
notify_selection_owner,
|
2018-11-19 17:17:25 +00:00
|
|
|
x11_display);
|
2020-01-30 08:19:39 +00:00
|
|
|
|
|
|
|
for (i = 0; i < META_N_SELECTION_TYPES; i++)
|
|
|
|
{
|
|
|
|
g_clear_object (&x11_display->selection.owners[i]);
|
|
|
|
if (x11_display->selection.cancellables[i])
|
|
|
|
{
|
|
|
|
g_cancellable_cancel (x11_display->selection.cancellables[i]);
|
|
|
|
g_clear_object (&x11_display->selection.cancellables[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-19 17:17:25 +00:00
|
|
|
if (x11_display->selection.xwindow != None)
|
|
|
|
{
|
|
|
|
XDestroyWindow (x11_display->xdisplay, x11_display->selection.xwindow);
|
|
|
|
x11_display->selection.xwindow = None;
|
|
|
|
}
|
2020-07-14 17:02:06 +00:00
|
|
|
|
|
|
|
g_clear_handle_id (&x11_display->selection.timeout_id, g_source_remove);
|
2018-11-19 17:17:25 +00:00
|
|
|
}
|