mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-16 15:14:17 +00:00
5bf28a3869
We can't really support these on e.g. wayland anyway, and we're trying to get rid of subwindow at totally in the long term, so lets drop this. It allows us to drop a lot of complexity.
585 lines
15 KiB
C
585 lines
15 KiB
C
/* GDK - The GIMP Drawing Kit
|
||
* Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
|
||
*
|
||
* 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/>.
|
||
*/
|
||
|
||
/*
|
||
* Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
|
||
* file for a list of people on the GTK+ Team. See the ChangeLog
|
||
* files for a list of changes. These files are distributed with
|
||
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
||
*/
|
||
|
||
#include "config.h"
|
||
|
||
#include "gdkproperty.h"
|
||
#include "gdkmain.h"
|
||
#include "gdkprivate.h"
|
||
#include "gdkinternals.h"
|
||
#include "gdkselection.h"
|
||
#include "gdkprivate-x11.h"
|
||
#include "gdkdisplay-x11.h"
|
||
#include "gdkscreen-x11.h"
|
||
|
||
#include <X11/Xlib.h>
|
||
#include <X11/Xatom.h>
|
||
#include <string.h>
|
||
|
||
#define N_PREDEFINED_ATOMS 69
|
||
|
||
#define ATOM_TO_INDEX(atom) (GPOINTER_TO_UINT(atom))
|
||
#define INDEX_TO_ATOM(atom) ((GdkAtom)GUINT_TO_POINTER(atom))
|
||
|
||
static void
|
||
insert_atom_pair (GdkDisplay *display,
|
||
GdkAtom virtual_atom,
|
||
Atom xatom)
|
||
{
|
||
GdkX11Display *display_x11 = GDK_X11_DISPLAY (display);
|
||
|
||
if (!display_x11->atom_from_virtual)
|
||
{
|
||
display_x11->atom_from_virtual = g_hash_table_new (g_direct_hash, NULL);
|
||
display_x11->atom_to_virtual = g_hash_table_new (g_direct_hash, NULL);
|
||
}
|
||
|
||
g_hash_table_insert (display_x11->atom_from_virtual,
|
||
GDK_ATOM_TO_POINTER (virtual_atom),
|
||
GUINT_TO_POINTER (xatom));
|
||
g_hash_table_insert (display_x11->atom_to_virtual,
|
||
GUINT_TO_POINTER (xatom),
|
||
GDK_ATOM_TO_POINTER (virtual_atom));
|
||
}
|
||
|
||
static Atom
|
||
lookup_cached_xatom (GdkDisplay *display,
|
||
GdkAtom atom)
|
||
{
|
||
GdkX11Display *display_x11 = GDK_X11_DISPLAY (display);
|
||
|
||
if (ATOM_TO_INDEX (atom) < N_PREDEFINED_ATOMS)
|
||
return ATOM_TO_INDEX (atom);
|
||
|
||
if (display_x11->atom_from_virtual)
|
||
return GPOINTER_TO_UINT (g_hash_table_lookup (display_x11->atom_from_virtual,
|
||
GDK_ATOM_TO_POINTER (atom)));
|
||
|
||
return None;
|
||
}
|
||
|
||
/**
|
||
* gdk_x11_atom_to_xatom_for_display:
|
||
* @display: (type GdkX11Display): A #GdkDisplay
|
||
* @atom: A #GdkAtom, or %GDK_NONE
|
||
*
|
||
* Converts from a #GdkAtom to the X atom for a #GdkDisplay
|
||
* with the same string value. The special value %GDK_NONE
|
||
* is converted to %None.
|
||
*
|
||
* Returns: the X atom corresponding to @atom, or %None
|
||
*
|
||
* Since: 2.2
|
||
**/
|
||
Atom
|
||
gdk_x11_atom_to_xatom_for_display (GdkDisplay *display,
|
||
GdkAtom atom)
|
||
{
|
||
Atom xatom = None;
|
||
|
||
g_return_val_if_fail (GDK_IS_DISPLAY (display), None);
|
||
|
||
if (atom == GDK_NONE)
|
||
return None;
|
||
|
||
if (gdk_display_is_closed (display))
|
||
return None;
|
||
|
||
xatom = lookup_cached_xatom (display, atom);
|
||
|
||
if (!xatom)
|
||
{
|
||
char *name = gdk_atom_name (atom);
|
||
|
||
xatom = XInternAtom (GDK_DISPLAY_XDISPLAY (display), name, FALSE);
|
||
insert_atom_pair (display, atom, xatom);
|
||
|
||
g_free (name);
|
||
}
|
||
|
||
return xatom;
|
||
}
|
||
|
||
void
|
||
_gdk_x11_precache_atoms (GdkDisplay *display,
|
||
const gchar * const *atom_names,
|
||
gint n_atoms)
|
||
{
|
||
Atom *xatoms;
|
||
GdkAtom *atoms;
|
||
const gchar **xatom_names;
|
||
gint n_xatoms;
|
||
gint i;
|
||
|
||
xatoms = g_new (Atom, n_atoms);
|
||
xatom_names = g_new (const gchar *, n_atoms);
|
||
atoms = g_new (GdkAtom, n_atoms);
|
||
|
||
n_xatoms = 0;
|
||
for (i = 0; i < n_atoms; i++)
|
||
{
|
||
GdkAtom atom = gdk_atom_intern_static_string (atom_names[i]);
|
||
if (lookup_cached_xatom (display, atom) == None)
|
||
{
|
||
atoms[n_xatoms] = atom;
|
||
xatom_names[n_xatoms] = atom_names[i];
|
||
n_xatoms++;
|
||
}
|
||
}
|
||
|
||
if (n_xatoms)
|
||
XInternAtoms (GDK_DISPLAY_XDISPLAY (display),
|
||
(char **)xatom_names, n_xatoms, False, xatoms);
|
||
|
||
for (i = 0; i < n_xatoms; i++)
|
||
insert_atom_pair (display, atoms[i], xatoms[i]);
|
||
|
||
g_free (xatoms);
|
||
g_free (xatom_names);
|
||
g_free (atoms);
|
||
}
|
||
|
||
/**
|
||
* gdk_x11_atom_to_xatom:
|
||
* @atom: A #GdkAtom
|
||
*
|
||
* Converts from a #GdkAtom to the X atom for the default GDK display
|
||
* with the same string value.
|
||
*
|
||
* Returns: the X atom corresponding to @atom.
|
||
**/
|
||
Atom
|
||
gdk_x11_atom_to_xatom (GdkAtom atom)
|
||
{
|
||
return gdk_x11_atom_to_xatom_for_display (gdk_display_get_default (), atom);
|
||
}
|
||
|
||
/**
|
||
* gdk_x11_xatom_to_atom_for_display:
|
||
* @display: (type GdkX11Display): A #GdkDisplay
|
||
* @xatom: an X atom
|
||
*
|
||
* Convert from an X atom for a #GdkDisplay to the corresponding
|
||
* #GdkAtom.
|
||
*
|
||
* Returns: (transfer none): the corresponding #GdkAtom.
|
||
*
|
||
* Since: 2.2
|
||
**/
|
||
GdkAtom
|
||
gdk_x11_xatom_to_atom_for_display (GdkDisplay *display,
|
||
Atom xatom)
|
||
{
|
||
GdkX11Display *display_x11;
|
||
GdkAtom virtual_atom = GDK_NONE;
|
||
|
||
g_return_val_if_fail (GDK_IS_DISPLAY (display), GDK_NONE);
|
||
|
||
if (xatom == None)
|
||
return GDK_NONE;
|
||
|
||
if (gdk_display_is_closed (display))
|
||
return GDK_NONE;
|
||
|
||
display_x11 = GDK_X11_DISPLAY (display);
|
||
|
||
if (xatom < N_PREDEFINED_ATOMS)
|
||
return INDEX_TO_ATOM (xatom);
|
||
|
||
if (display_x11->atom_to_virtual)
|
||
virtual_atom = GDK_POINTER_TO_ATOM (g_hash_table_lookup (display_x11->atom_to_virtual,
|
||
GUINT_TO_POINTER (xatom)));
|
||
|
||
if (!virtual_atom)
|
||
{
|
||
/* If this atom doesn't exist, we'll die with an X error unless
|
||
* we take precautions
|
||
*/
|
||
char *name;
|
||
gdk_x11_display_error_trap_push (display);
|
||
name = XGetAtomName (GDK_DISPLAY_XDISPLAY (display), xatom);
|
||
if (gdk_x11_display_error_trap_pop (display))
|
||
{
|
||
g_warning (G_STRLOC " invalid X atom: %ld", xatom);
|
||
}
|
||
else
|
||
{
|
||
virtual_atom = gdk_atom_intern (name, FALSE);
|
||
XFree (name);
|
||
|
||
insert_atom_pair (display, virtual_atom, xatom);
|
||
}
|
||
}
|
||
|
||
return virtual_atom;
|
||
}
|
||
|
||
/**
|
||
* gdk_x11_xatom_to_atom:
|
||
* @xatom: an X atom for the default GDK display
|
||
*
|
||
* Convert from an X atom for the default display to the corresponding
|
||
* #GdkAtom.
|
||
*
|
||
* Returns: (transfer none): the corresponding G#dkAtom.
|
||
**/
|
||
GdkAtom
|
||
gdk_x11_xatom_to_atom (Atom xatom)
|
||
{
|
||
return gdk_x11_xatom_to_atom_for_display (gdk_display_get_default (), xatom);
|
||
}
|
||
|
||
/**
|
||
* gdk_x11_get_xatom_by_name_for_display:
|
||
* @display: (type GdkX11Display): a #GdkDisplay
|
||
* @atom_name: a string
|
||
*
|
||
* Returns the X atom for a #GdkDisplay corresponding to @atom_name.
|
||
* This function caches the result, so if called repeatedly it is much
|
||
* faster than XInternAtom(), which is a round trip to the server each time.
|
||
*
|
||
* Returns: a X atom for a #GdkDisplay
|
||
*
|
||
* Since: 2.2
|
||
**/
|
||
Atom
|
||
gdk_x11_get_xatom_by_name_for_display (GdkDisplay *display,
|
||
const gchar *atom_name)
|
||
{
|
||
g_return_val_if_fail (GDK_IS_DISPLAY (display), None);
|
||
return gdk_x11_atom_to_xatom_for_display (display,
|
||
gdk_atom_intern (atom_name, FALSE));
|
||
}
|
||
|
||
Atom
|
||
_gdk_x11_get_xatom_for_display_printf (GdkDisplay *display,
|
||
const gchar *format,
|
||
...)
|
||
{
|
||
va_list args;
|
||
char *atom_name;
|
||
Atom atom;
|
||
|
||
va_start (args, format);
|
||
atom_name = g_strdup_vprintf (format, args);
|
||
va_end (args);
|
||
|
||
atom = gdk_x11_get_xatom_by_name_for_display (display, atom_name);
|
||
|
||
g_free (atom_name);
|
||
|
||
return atom;
|
||
}
|
||
|
||
/**
|
||
* gdk_x11_get_xatom_by_name:
|
||
* @atom_name: a string
|
||
*
|
||
* Returns the X atom for GDK’s default display corresponding to @atom_name.
|
||
* This function caches the result, so if called repeatedly it is much
|
||
* faster than XInternAtom(), which is a round trip to the server each time.
|
||
*
|
||
* Returns: a X atom for GDK’s default display.
|
||
**/
|
||
Atom
|
||
gdk_x11_get_xatom_by_name (const gchar *atom_name)
|
||
{
|
||
return gdk_x11_get_xatom_by_name_for_display (gdk_display_get_default (),
|
||
atom_name);
|
||
}
|
||
|
||
/**
|
||
* gdk_x11_get_xatom_name_for_display:
|
||
* @display: (type GdkX11Display): the #GdkDisplay where @xatom is defined
|
||
* @xatom: an X atom
|
||
*
|
||
* Returns the name of an X atom for its display. This
|
||
* function is meant mainly for debugging, so for convenience, unlike
|
||
* XAtomName() and gdk_atom_name(), the result doesn’t need to
|
||
* be freed.
|
||
*
|
||
* Returns: name of the X atom; this string is owned by GDK,
|
||
* so it shouldn’t be modifed or freed.
|
||
*
|
||
* Since: 2.2
|
||
**/
|
||
const gchar *
|
||
gdk_x11_get_xatom_name_for_display (GdkDisplay *display,
|
||
Atom xatom)
|
||
{
|
||
g_return_val_if_fail (GDK_IS_DISPLAY (display), NULL);
|
||
|
||
return _gdk_atom_name_const (gdk_x11_xatom_to_atom_for_display (display, xatom));
|
||
}
|
||
|
||
/**
|
||
* gdk_x11_get_xatom_name:
|
||
* @xatom: an X atom for GDK’s default display
|
||
*
|
||
* Returns the name of an X atom for GDK’s default display. This
|
||
* function is meant mainly for debugging, so for convenience, unlike
|
||
* XAtomName() and gdk_atom_name(), the result
|
||
* doesn’t need to be freed. Also, this function will never return %NULL,
|
||
* even if @xatom is invalid.
|
||
*
|
||
* Returns: name of the X atom; this string is owned by GTK+,
|
||
* so it shouldn’t be modifed or freed.
|
||
**/
|
||
const gchar *
|
||
gdk_x11_get_xatom_name (Atom xatom)
|
||
{
|
||
return _gdk_atom_name_const (gdk_x11_xatom_to_atom (xatom));
|
||
}
|
||
|
||
gboolean
|
||
_gdk_x11_window_get_property (GdkWindow *window,
|
||
GdkAtom property,
|
||
GdkAtom type,
|
||
gulong offset,
|
||
gulong length,
|
||
gint pdelete,
|
||
GdkAtom *actual_property_type,
|
||
gint *actual_format_type,
|
||
gint *actual_length,
|
||
guchar **data)
|
||
{
|
||
GdkDisplay *display;
|
||
Atom ret_prop_type;
|
||
gint ret_format;
|
||
gulong ret_nitems;
|
||
gulong ret_bytes_after;
|
||
gulong get_length;
|
||
gulong ret_length;
|
||
guchar *ret_data;
|
||
Atom xproperty;
|
||
Atom xtype;
|
||
int res;
|
||
|
||
g_return_val_if_fail (!window || GDK_WINDOW_IS_X11 (window), FALSE);
|
||
|
||
if (!window)
|
||
{
|
||
GdkScreen *screen = gdk_screen_get_default ();
|
||
window = gdk_screen_get_root_window (screen);
|
||
}
|
||
else if (!GDK_WINDOW_IS_X11 (window))
|
||
return FALSE;
|
||
|
||
if (GDK_WINDOW_DESTROYED (window))
|
||
return FALSE;
|
||
|
||
display = gdk_window_get_display (window);
|
||
xproperty = gdk_x11_atom_to_xatom_for_display (display, property);
|
||
if (type == GDK_NONE)
|
||
xtype = AnyPropertyType;
|
||
else
|
||
xtype = gdk_x11_atom_to_xatom_for_display (display, type);
|
||
|
||
ret_data = NULL;
|
||
|
||
/*
|
||
* Round up length to next 4 byte value. Some code is in the (bad?)
|
||
* habit of passing G_MAXLONG as the length argument, causing an
|
||
* overflow to negative on the add. In this case, we clamp the
|
||
* value to G_MAXLONG.
|
||
*/
|
||
get_length = length + 3;
|
||
if (get_length > G_MAXLONG)
|
||
get_length = G_MAXLONG;
|
||
|
||
/* To fail, either the user passed 0 or G_MAXULONG */
|
||
get_length = get_length / 4;
|
||
if (get_length == 0)
|
||
{
|
||
g_warning ("gdk_propery-get(): invalid length 0");
|
||
return FALSE;
|
||
}
|
||
|
||
res = XGetWindowProperty (GDK_DISPLAY_XDISPLAY (display),
|
||
GDK_WINDOW_XID (window), xproperty,
|
||
offset, get_length, pdelete,
|
||
xtype, &ret_prop_type, &ret_format,
|
||
&ret_nitems, &ret_bytes_after,
|
||
&ret_data);
|
||
|
||
if (res != Success || (ret_prop_type == None && ret_format == 0))
|
||
{
|
||
return FALSE;
|
||
}
|
||
|
||
if (actual_property_type)
|
||
*actual_property_type = gdk_x11_xatom_to_atom_for_display (display, ret_prop_type);
|
||
if (actual_format_type)
|
||
*actual_format_type = ret_format;
|
||
|
||
if ((xtype != AnyPropertyType) && (ret_prop_type != xtype))
|
||
{
|
||
XFree (ret_data);
|
||
g_warning ("Couldn't match property type %s to %s\n",
|
||
gdk_x11_get_xatom_name_for_display (display, ret_prop_type),
|
||
gdk_x11_get_xatom_name_for_display (display, xtype));
|
||
return FALSE;
|
||
}
|
||
|
||
/* FIXME: ignoring bytes_after could have very bad effects */
|
||
|
||
if (data)
|
||
{
|
||
if (ret_prop_type == XA_ATOM ||
|
||
ret_prop_type == gdk_x11_get_xatom_by_name_for_display (display, "ATOM_PAIR"))
|
||
{
|
||
/*
|
||
* data is an array of X atom, we need to convert it
|
||
* to an array of GDK Atoms
|
||
*/
|
||
gint i;
|
||
GdkAtom *ret_atoms = g_new (GdkAtom, ret_nitems);
|
||
Atom *xatoms = (Atom *)ret_data;
|
||
|
||
*data = (guchar *)ret_atoms;
|
||
|
||
for (i = 0; i < ret_nitems; i++)
|
||
ret_atoms[i] = gdk_x11_xatom_to_atom_for_display (display, xatoms[i]);
|
||
|
||
if (actual_length)
|
||
*actual_length = ret_nitems * sizeof (GdkAtom);
|
||
}
|
||
else
|
||
{
|
||
switch (ret_format)
|
||
{
|
||
case 8:
|
||
ret_length = ret_nitems;
|
||
break;
|
||
case 16:
|
||
ret_length = sizeof(short) * ret_nitems;
|
||
break;
|
||
case 32:
|
||
ret_length = sizeof(long) * ret_nitems;
|
||
break;
|
||
default:
|
||
g_warning ("unknown property return format: %d", ret_format);
|
||
XFree (ret_data);
|
||
return FALSE;
|
||
}
|
||
|
||
*data = g_new (guchar, ret_length);
|
||
memcpy (*data, ret_data, ret_length);
|
||
if (actual_length)
|
||
*actual_length = ret_length;
|
||
}
|
||
}
|
||
|
||
XFree (ret_data);
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
void
|
||
_gdk_x11_window_change_property (GdkWindow *window,
|
||
GdkAtom property,
|
||
GdkAtom type,
|
||
gint format,
|
||
GdkPropMode mode,
|
||
const guchar *data,
|
||
gint nelements)
|
||
{
|
||
GdkDisplay *display;
|
||
Window xwindow;
|
||
Atom xproperty;
|
||
Atom xtype;
|
||
|
||
g_return_if_fail (!window || GDK_WINDOW_IS_X11 (window));
|
||
|
||
if (!window)
|
||
{
|
||
GdkScreen *screen;
|
||
|
||
screen = gdk_screen_get_default ();
|
||
window = gdk_screen_get_root_window (screen);
|
||
}
|
||
else if (!GDK_WINDOW_IS_X11 (window))
|
||
return;
|
||
|
||
if (GDK_WINDOW_DESTROYED (window))
|
||
return;
|
||
|
||
if (!gdk_window_has_native (window))
|
||
{
|
||
g_warning ("Can't change property on non-native window");
|
||
return;
|
||
}
|
||
|
||
display = gdk_window_get_display (window);
|
||
xproperty = gdk_x11_atom_to_xatom_for_display (display, property);
|
||
xtype = gdk_x11_atom_to_xatom_for_display (display, type);
|
||
xwindow = GDK_WINDOW_XID (window);
|
||
|
||
if (xtype == XA_ATOM ||
|
||
xtype == gdk_x11_get_xatom_by_name_for_display (display, "ATOM_PAIR"))
|
||
{
|
||
/*
|
||
* data is an array of GdkAtom, we need to convert it
|
||
* to an array of X Atoms
|
||
*/
|
||
gint i;
|
||
GdkAtom *atoms = (GdkAtom*) data;
|
||
Atom *xatoms;
|
||
|
||
xatoms = g_new (Atom, nelements);
|
||
for (i = 0; i < nelements; i++)
|
||
xatoms[i] = gdk_x11_atom_to_xatom_for_display (display, atoms[i]);
|
||
|
||
XChangeProperty (GDK_DISPLAY_XDISPLAY (display), xwindow,
|
||
xproperty, xtype,
|
||
format, mode, (guchar *)xatoms, nelements);
|
||
g_free (xatoms);
|
||
}
|
||
else
|
||
XChangeProperty (GDK_DISPLAY_XDISPLAY (display), xwindow, xproperty,
|
||
xtype, format, mode, (guchar *)data, nelements);
|
||
}
|
||
|
||
void
|
||
_gdk_x11_window_delete_property (GdkWindow *window,
|
||
GdkAtom property)
|
||
{
|
||
g_return_if_fail (!window || GDK_WINDOW_IS_X11 (window));
|
||
|
||
if (!window)
|
||
{
|
||
GdkScreen *screen = gdk_screen_get_default ();
|
||
window = gdk_screen_get_root_window (screen);
|
||
}
|
||
else if (!GDK_WINDOW_IS_X11 (window))
|
||
return;
|
||
|
||
if (GDK_WINDOW_DESTROYED (window))
|
||
return;
|
||
|
||
XDeleteProperty (GDK_WINDOW_XDISPLAY (window), GDK_WINDOW_XID (window),
|
||
gdk_x11_atom_to_xatom_for_display (GDK_WINDOW_DISPLAY (window),
|
||
property));
|
||
}
|