mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-04 09:40:19 +00:00
a5d1bc41ec
Wed May 2 17:26:22 2001 Owen Taylor <otaylor@redhat.com> * gdk/x11/gdkevents-x11.c gdk/x11/gdkwindow-x11.c: Always trap errors around calls to XSetInputFocus since we have no way of knowing reliably whether we are viewable or not. (#53947)
3818 lines
95 KiB
C
3818 lines
95 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, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
/*
|
|
* 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 <X11/Xlib.h>
|
|
#include <X11/Xutil.h>
|
|
#include <X11/Xatom.h>
|
|
#include <netinet/in.h>
|
|
#include "gdk.h"
|
|
#include "config.h"
|
|
|
|
#include "gdkwindow.h"
|
|
#include "gdkinputprivate.h"
|
|
#include "gdkprivate-x11.h"
|
|
#include "gdkregion.h"
|
|
#include "gdkinternals.h"
|
|
#include "MwmUtil.h"
|
|
#include "gdkwindow-x11.h"
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
|
|
#ifdef HAVE_SHAPE_EXT
|
|
#include <X11/extensions/shape.h>
|
|
#endif
|
|
|
|
const int gdk_event_mask_table[21] =
|
|
{
|
|
ExposureMask,
|
|
PointerMotionMask,
|
|
PointerMotionHintMask,
|
|
ButtonMotionMask,
|
|
Button1MotionMask,
|
|
Button2MotionMask,
|
|
Button3MotionMask,
|
|
ButtonPressMask,
|
|
ButtonReleaseMask,
|
|
KeyPressMask,
|
|
KeyReleaseMask,
|
|
EnterWindowMask,
|
|
LeaveWindowMask,
|
|
FocusChangeMask,
|
|
StructureNotifyMask,
|
|
PropertyChangeMask,
|
|
VisibilityChangeMask,
|
|
0, /* PROXIMITY_IN */
|
|
0, /* PROXIMTY_OUT */
|
|
SubstructureNotifyMask,
|
|
ButtonPressMask /* SCROLL; on X mouse wheel events is treated as mouse button 4/5 */
|
|
};
|
|
const int gdk_nevent_masks = sizeof (gdk_event_mask_table) / sizeof (int);
|
|
|
|
/* Forward declarations */
|
|
static gboolean gdk_window_gravity_works (void);
|
|
static void gdk_window_set_static_win_gravity (GdkWindow *window,
|
|
gboolean on);
|
|
static gboolean gdk_window_have_shape_ext (void);
|
|
static gboolean gdk_window_icon_name_set (GdkWindow *window);
|
|
|
|
static GdkColormap* gdk_window_impl_x11_get_colormap (GdkDrawable *drawable);
|
|
static void gdk_window_impl_x11_set_colormap (GdkDrawable *drawable,
|
|
GdkColormap *cmap);
|
|
static void gdk_window_impl_x11_get_size (GdkDrawable *drawable,
|
|
gint *width,
|
|
gint *height);
|
|
static GdkRegion* gdk_window_impl_x11_get_visible_region (GdkDrawable *drawable);
|
|
static void gdk_window_impl_x11_init (GdkWindowImplX11 *window);
|
|
static void gdk_window_impl_x11_class_init (GdkWindowImplX11Class *klass);
|
|
static void gdk_window_impl_x11_finalize (GObject *object);
|
|
|
|
static gpointer parent_class = NULL;
|
|
|
|
GType
|
|
gdk_window_impl_x11_get_type (void)
|
|
{
|
|
static GType object_type = 0;
|
|
|
|
if (!object_type)
|
|
{
|
|
static const GTypeInfo object_info =
|
|
{
|
|
sizeof (GdkWindowImplX11Class),
|
|
(GBaseInitFunc) NULL,
|
|
(GBaseFinalizeFunc) NULL,
|
|
(GClassInitFunc) gdk_window_impl_x11_class_init,
|
|
NULL, /* class_finalize */
|
|
NULL, /* class_data */
|
|
sizeof (GdkWindowImplX11),
|
|
0, /* n_preallocs */
|
|
(GInstanceInitFunc) gdk_window_impl_x11_init,
|
|
};
|
|
|
|
object_type = g_type_register_static (GDK_TYPE_DRAWABLE_IMPL_X11,
|
|
"GdkWindowImplX11",
|
|
&object_info, 0);
|
|
}
|
|
|
|
return object_type;
|
|
}
|
|
|
|
GType
|
|
_gdk_window_impl_get_type (void)
|
|
{
|
|
return gdk_window_impl_x11_get_type ();
|
|
}
|
|
|
|
static void
|
|
gdk_window_impl_x11_init (GdkWindowImplX11 *impl)
|
|
{
|
|
impl->width = 1;
|
|
impl->height = 1;
|
|
}
|
|
|
|
static void
|
|
gdk_window_impl_x11_class_init (GdkWindowImplX11Class *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GdkDrawableClass *drawable_class = GDK_DRAWABLE_CLASS (klass);
|
|
|
|
parent_class = g_type_class_peek_parent (klass);
|
|
|
|
object_class->finalize = gdk_window_impl_x11_finalize;
|
|
|
|
drawable_class->set_colormap = gdk_window_impl_x11_set_colormap;
|
|
drawable_class->get_colormap = gdk_window_impl_x11_get_colormap;
|
|
drawable_class->get_size = gdk_window_impl_x11_get_size;
|
|
|
|
/* Visible and clip regions are the same */
|
|
drawable_class->get_clip_region = gdk_window_impl_x11_get_visible_region;
|
|
drawable_class->get_visible_region = gdk_window_impl_x11_get_visible_region;
|
|
}
|
|
|
|
static void
|
|
gdk_window_impl_x11_finalize (GObject *object)
|
|
{
|
|
GdkWindowObject *wrapper;
|
|
GdkDrawableImplX11 *draw_impl;
|
|
GdkWindowImplX11 *window_impl;
|
|
|
|
g_return_if_fail (GDK_IS_WINDOW_IMPL_X11 (object));
|
|
|
|
draw_impl = GDK_DRAWABLE_IMPL_X11 (object);
|
|
window_impl = GDK_WINDOW_IMPL_X11 (object);
|
|
|
|
wrapper = (GdkWindowObject*) draw_impl->wrapper;
|
|
|
|
if (!GDK_WINDOW_DESTROYED (wrapper))
|
|
{
|
|
gdk_xid_table_remove (draw_impl->xid);
|
|
if (window_impl->focus_window)
|
|
gdk_xid_table_remove (window_impl->focus_window);
|
|
}
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static GdkColormap*
|
|
gdk_window_impl_x11_get_colormap (GdkDrawable *drawable)
|
|
{
|
|
GdkDrawableImplX11 *drawable_impl;
|
|
GdkWindowImplX11 *window_impl;
|
|
|
|
g_return_val_if_fail (GDK_IS_WINDOW_IMPL_X11 (drawable), NULL);
|
|
|
|
drawable_impl = GDK_DRAWABLE_IMPL_X11 (drawable);
|
|
window_impl = GDK_WINDOW_IMPL_X11 (drawable);
|
|
|
|
if (!((GdkWindowObject *) drawable_impl->wrapper)->input_only &&
|
|
drawable_impl->colormap == NULL)
|
|
{
|
|
XWindowAttributes window_attributes;
|
|
|
|
XGetWindowAttributes (drawable_impl->xdisplay,
|
|
drawable_impl->xid,
|
|
&window_attributes);
|
|
drawable_impl->colormap =
|
|
gdk_colormap_lookup (window_attributes.colormap);
|
|
}
|
|
|
|
return drawable_impl->colormap;
|
|
}
|
|
|
|
static void
|
|
gdk_window_impl_x11_set_colormap (GdkDrawable *drawable,
|
|
GdkColormap *cmap)
|
|
{
|
|
GdkWindowImplX11 *impl;
|
|
GdkDrawableImplX11 *draw_impl;
|
|
|
|
g_return_if_fail (GDK_IS_WINDOW_IMPL_X11 (drawable));
|
|
|
|
impl = GDK_WINDOW_IMPL_X11 (drawable);
|
|
draw_impl = GDK_DRAWABLE_IMPL_X11 (drawable);
|
|
|
|
/* chain up */
|
|
GDK_DRAWABLE_CLASS (parent_class)->set_colormap (drawable, cmap);
|
|
|
|
if (cmap)
|
|
{
|
|
XSetWindowColormap (draw_impl->xdisplay,
|
|
draw_impl->xid,
|
|
GDK_COLORMAP_XCOLORMAP (cmap));
|
|
|
|
if (((GdkWindowObject*)draw_impl->wrapper)->window_type !=
|
|
GDK_WINDOW_TOPLEVEL)
|
|
gdk_window_add_colormap_windows (GDK_WINDOW (draw_impl->wrapper));
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
gdk_window_impl_x11_get_size (GdkDrawable *drawable,
|
|
gint *width,
|
|
gint *height)
|
|
{
|
|
g_return_if_fail (GDK_IS_WINDOW_IMPL_X11 (drawable));
|
|
|
|
if (width)
|
|
*width = GDK_WINDOW_IMPL_X11 (drawable)->width;
|
|
if (height)
|
|
*height = GDK_WINDOW_IMPL_X11 (drawable)->height;
|
|
}
|
|
|
|
static GdkRegion*
|
|
gdk_window_impl_x11_get_visible_region (GdkDrawable *drawable)
|
|
{
|
|
GdkWindowImplX11 *impl = GDK_WINDOW_IMPL_X11 (drawable);
|
|
GdkRectangle result_rect;
|
|
|
|
result_rect.x = 0;
|
|
result_rect.y = 0;
|
|
result_rect.width = impl->width;
|
|
result_rect.height = impl->height;
|
|
|
|
gdk_rectangle_intersect (&result_rect, &impl->position_info.clip_rect, &result_rect);
|
|
|
|
return gdk_region_rectangle (&result_rect);
|
|
}
|
|
|
|
void
|
|
_gdk_windowing_window_init (void)
|
|
{
|
|
GdkWindowObject *private;
|
|
GdkWindowImplX11 *impl;
|
|
GdkDrawableImplX11 *draw_impl;
|
|
XWindowAttributes xattributes;
|
|
unsigned int width;
|
|
unsigned int height;
|
|
unsigned int border_width;
|
|
unsigned int depth;
|
|
int x, y;
|
|
|
|
g_assert (gdk_parent_root == NULL);
|
|
|
|
XGetGeometry (gdk_display, gdk_root_window, &gdk_root_window,
|
|
&x, &y, &width, &height, &border_width, &depth);
|
|
XGetWindowAttributes (gdk_display, gdk_root_window, &xattributes);
|
|
|
|
gdk_parent_root = g_object_new (GDK_TYPE_WINDOW, NULL);
|
|
private = (GdkWindowObject *)gdk_parent_root;
|
|
impl = GDK_WINDOW_IMPL_X11 (private->impl);
|
|
draw_impl = GDK_DRAWABLE_IMPL_X11 (private->impl);
|
|
|
|
draw_impl->xdisplay = gdk_display;
|
|
draw_impl->xid = gdk_root_window;
|
|
draw_impl->wrapper = GDK_DRAWABLE (private);
|
|
|
|
private->window_type = GDK_WINDOW_ROOT;
|
|
private->depth = depth;
|
|
impl->width = width;
|
|
impl->height = height;
|
|
|
|
gdk_xid_table_insert (&gdk_root_window, gdk_parent_root);
|
|
}
|
|
|
|
static GdkAtom wm_client_leader_atom = GDK_NONE;
|
|
|
|
GdkWindow*
|
|
gdk_window_new (GdkWindow *parent,
|
|
GdkWindowAttr *attributes,
|
|
gint attributes_mask)
|
|
{
|
|
GdkWindow *window;
|
|
GdkWindowObject *private;
|
|
GdkWindowObject *parent_private;
|
|
GdkWindowImplX11 *impl;
|
|
GdkDrawableImplX11 *draw_impl;
|
|
|
|
GdkVisual *visual;
|
|
Window xparent;
|
|
Visual *xvisual;
|
|
Display *xdisplay;
|
|
Window xid;
|
|
|
|
XSetWindowAttributes xattributes;
|
|
long xattributes_mask;
|
|
XSizeHints size_hints;
|
|
XWMHints wm_hints;
|
|
XClassHint *class_hint;
|
|
int x, y, depth;
|
|
|
|
unsigned int class;
|
|
char *title;
|
|
int i;
|
|
|
|
g_return_val_if_fail (attributes != NULL, NULL);
|
|
|
|
if (!parent)
|
|
parent = gdk_parent_root;
|
|
|
|
g_return_val_if_fail (GDK_IS_WINDOW (parent), NULL);
|
|
|
|
parent_private = (GdkWindowObject*) parent;
|
|
if (GDK_WINDOW_DESTROYED (parent))
|
|
return NULL;
|
|
|
|
xparent = GDK_WINDOW_XID (parent);
|
|
|
|
window = g_object_new (GDK_TYPE_WINDOW, NULL);
|
|
private = (GdkWindowObject *)window;
|
|
impl = GDK_WINDOW_IMPL_X11 (private->impl);
|
|
draw_impl = GDK_DRAWABLE_IMPL_X11 (private->impl);
|
|
draw_impl->wrapper = GDK_DRAWABLE (window);
|
|
|
|
xdisplay = draw_impl->xdisplay = GDK_WINDOW_XDISPLAY (parent);
|
|
|
|
private->parent = (GdkWindowObject *)parent;
|
|
|
|
xattributes_mask = 0;
|
|
|
|
if (attributes_mask & GDK_WA_X)
|
|
x = attributes->x;
|
|
else
|
|
x = 0;
|
|
|
|
if (attributes_mask & GDK_WA_Y)
|
|
y = attributes->y;
|
|
else
|
|
y = 0;
|
|
|
|
private->x = x;
|
|
private->y = y;
|
|
impl->width = (attributes->width > 1) ? (attributes->width) : (1);
|
|
impl->height = (attributes->height > 1) ? (attributes->height) : (1);
|
|
private->window_type = attributes->window_type;
|
|
|
|
_gdk_window_init_position (GDK_WINDOW (private));
|
|
if (impl->position_info.big)
|
|
private->guffaw_gravity = TRUE;
|
|
|
|
if (attributes_mask & GDK_WA_VISUAL)
|
|
visual = attributes->visual;
|
|
else
|
|
visual = gdk_visual_get_system ();
|
|
xvisual = ((GdkVisualPrivate*) visual)->xvisual;
|
|
|
|
xattributes.event_mask = StructureNotifyMask;
|
|
for (i = 0; i < gdk_nevent_masks; i++)
|
|
{
|
|
if (attributes->event_mask & (1 << (i + 1)))
|
|
xattributes.event_mask |= gdk_event_mask_table[i];
|
|
}
|
|
|
|
if (xattributes.event_mask)
|
|
xattributes_mask |= CWEventMask;
|
|
|
|
if (attributes_mask & GDK_WA_NOREDIR)
|
|
{
|
|
xattributes.override_redirect =
|
|
(attributes->override_redirect == FALSE)?False:True;
|
|
xattributes_mask |= CWOverrideRedirect;
|
|
}
|
|
else
|
|
xattributes.override_redirect = False;
|
|
|
|
if (parent_private && parent_private->guffaw_gravity)
|
|
{
|
|
xattributes.win_gravity = StaticGravity;
|
|
xattributes_mask |= CWWinGravity;
|
|
}
|
|
|
|
if (attributes->wclass == GDK_INPUT_OUTPUT)
|
|
{
|
|
class = InputOutput;
|
|
depth = visual->depth;
|
|
|
|
private->input_only = FALSE;
|
|
private->depth = depth;
|
|
|
|
if (attributes_mask & GDK_WA_COLORMAP)
|
|
{
|
|
draw_impl->colormap = attributes->colormap;
|
|
gdk_colormap_ref (attributes->colormap);
|
|
}
|
|
else
|
|
{
|
|
if ((((GdkVisualPrivate*)gdk_visual_get_system ())->xvisual) == xvisual)
|
|
{
|
|
draw_impl->colormap =
|
|
gdk_colormap_get_system ();
|
|
gdk_colormap_ref (draw_impl->colormap);
|
|
}
|
|
else
|
|
{
|
|
draw_impl->colormap =
|
|
gdk_colormap_new (visual, FALSE);
|
|
}
|
|
}
|
|
|
|
private->bg_color.pixel = BlackPixel (gdk_display, gdk_screen);
|
|
xattributes.background_pixel = private->bg_color.pixel;
|
|
|
|
private->bg_pixmap = NULL;
|
|
|
|
xattributes.border_pixel = BlackPixel (gdk_display, gdk_screen);
|
|
xattributes_mask |= CWBorderPixel | CWBackPixel;
|
|
|
|
if (private->guffaw_gravity)
|
|
xattributes.bit_gravity = StaticGravity;
|
|
else
|
|
xattributes.bit_gravity = NorthWestGravity;
|
|
|
|
xattributes_mask |= CWBitGravity;
|
|
|
|
switch (private->window_type)
|
|
{
|
|
case GDK_WINDOW_TOPLEVEL:
|
|
xattributes.colormap = GDK_COLORMAP_XCOLORMAP (draw_impl->colormap);
|
|
xattributes_mask |= CWColormap;
|
|
|
|
xparent = gdk_root_window;
|
|
break;
|
|
|
|
case GDK_WINDOW_CHILD:
|
|
xattributes.colormap = GDK_COLORMAP_XCOLORMAP (draw_impl->colormap);
|
|
xattributes_mask |= CWColormap;
|
|
break;
|
|
|
|
case GDK_WINDOW_DIALOG:
|
|
xattributes.colormap = GDK_COLORMAP_XCOLORMAP (draw_impl->colormap);
|
|
xattributes_mask |= CWColormap;
|
|
|
|
xparent = gdk_root_window;
|
|
break;
|
|
|
|
case GDK_WINDOW_TEMP:
|
|
xattributes.colormap = GDK_COLORMAP_XCOLORMAP (draw_impl->colormap);
|
|
xattributes_mask |= CWColormap;
|
|
|
|
xparent = gdk_root_window;
|
|
|
|
xattributes.save_under = True;
|
|
xattributes.override_redirect = True;
|
|
xattributes.cursor = None;
|
|
xattributes_mask |= CWSaveUnder | CWOverrideRedirect;
|
|
break;
|
|
case GDK_WINDOW_ROOT:
|
|
g_error ("cannot make windows of type GDK_WINDOW_ROOT");
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
depth = 0;
|
|
private->depth = 0;
|
|
class = InputOnly;
|
|
private->input_only = TRUE;
|
|
draw_impl->colormap = gdk_colormap_get_system ();
|
|
gdk_colormap_ref (draw_impl->colormap);
|
|
}
|
|
|
|
xid = draw_impl->xid = XCreateWindow (xdisplay, xparent,
|
|
impl->position_info.x, impl->position_info.y,
|
|
impl->position_info.width, impl->position_info.height,
|
|
0, depth, class, xvisual,
|
|
xattributes_mask, &xattributes);
|
|
|
|
gdk_drawable_ref (window);
|
|
gdk_xid_table_insert (&draw_impl->xid, window);
|
|
|
|
gdk_window_set_cursor (window, ((attributes_mask & GDK_WA_CURSOR) ?
|
|
(attributes->cursor) :
|
|
NULL));
|
|
|
|
if (parent_private)
|
|
parent_private->children = g_list_prepend (parent_private->children, window);
|
|
|
|
switch (GDK_WINDOW_TYPE (private))
|
|
{
|
|
case GDK_WINDOW_DIALOG:
|
|
XSetTransientForHint (xdisplay, xid, xparent);
|
|
case GDK_WINDOW_TOPLEVEL:
|
|
case GDK_WINDOW_TEMP:
|
|
XSetWMProtocols (xdisplay, xid,
|
|
gdk_wm_window_protocols, 3);
|
|
break;
|
|
case GDK_WINDOW_CHILD:
|
|
if ((attributes->wclass == GDK_INPUT_OUTPUT) &&
|
|
(draw_impl->colormap != gdk_colormap_get_system ()) &&
|
|
(draw_impl->colormap != gdk_window_get_colormap (gdk_window_get_toplevel (window))))
|
|
{
|
|
GDK_NOTE (MISC, g_message ("adding colormap window\n"));
|
|
gdk_window_add_colormap_windows (window);
|
|
}
|
|
|
|
return window;
|
|
default:
|
|
|
|
return window;
|
|
}
|
|
|
|
if (class != InputOnly)
|
|
{
|
|
/* The focus window is off the visible area, and serves to receive key
|
|
* press events so they don't get sent to child windows.
|
|
*/
|
|
impl->focus_window = XCreateSimpleWindow (xdisplay, xid,
|
|
-1, -1, 1, 1, 0,
|
|
xattributes.background_pixel,
|
|
xattributes.background_pixel);
|
|
/* FIXME: probably better to actually track the requested event mask for the toplevel
|
|
*/
|
|
XSelectInput (xdisplay, impl->focus_window,
|
|
KeyPressMask | KeyReleaseMask | FocusChangeMask);
|
|
|
|
XMapWindow (xdisplay, impl->focus_window);
|
|
gdk_xid_table_insert (&impl->focus_window, window);
|
|
}
|
|
|
|
size_hints.flags = PSize;
|
|
size_hints.width = impl->width;
|
|
size_hints.height = impl->height;
|
|
|
|
wm_hints.flags = StateHint | WindowGroupHint;
|
|
wm_hints.window_group = gdk_leader_window;
|
|
wm_hints.input = True;
|
|
wm_hints.initial_state = NormalState;
|
|
|
|
/* FIXME: Is there any point in doing this? Do any WM's pay
|
|
* attention to PSize, and even if they do, is this the
|
|
* correct value???
|
|
*/
|
|
XSetWMNormalHints (xdisplay, xid, &size_hints);
|
|
|
|
XSetWMHints (xdisplay, xid, &wm_hints);
|
|
|
|
if (!wm_client_leader_atom)
|
|
wm_client_leader_atom = gdk_atom_intern ("WM_CLIENT_LEADER", FALSE);
|
|
|
|
XChangeProperty (xdisplay, xid,
|
|
wm_client_leader_atom,
|
|
XA_WINDOW, 32, PropModeReplace,
|
|
(guchar*) &gdk_leader_window, 1);
|
|
|
|
if (attributes_mask & GDK_WA_TITLE)
|
|
title = attributes->title;
|
|
else
|
|
title = g_get_prgname ();
|
|
|
|
gdk_window_set_title (window, title);
|
|
|
|
if (attributes_mask & GDK_WA_WMCLASS)
|
|
{
|
|
class_hint = XAllocClassHint ();
|
|
class_hint->res_name = attributes->wmclass_name;
|
|
class_hint->res_class = attributes->wmclass_class;
|
|
XSetClassHint (xdisplay, xid, class_hint);
|
|
XFree (class_hint);
|
|
}
|
|
|
|
return window;
|
|
}
|
|
|
|
GdkWindow *
|
|
gdk_window_foreign_new (GdkNativeWindow anid)
|
|
{
|
|
GdkWindow *window;
|
|
GdkWindowObject *private;
|
|
GdkWindowObject *parent_private;
|
|
GdkWindowImplX11 *impl;
|
|
GdkDrawableImplX11 *draw_impl;
|
|
XWindowAttributes attrs;
|
|
Window root, parent;
|
|
Window *children = NULL;
|
|
guint nchildren;
|
|
gboolean result;
|
|
|
|
gdk_error_trap_push ();
|
|
result = XGetWindowAttributes (gdk_display, anid, &attrs);
|
|
if (gdk_error_trap_pop () || !result)
|
|
return NULL;
|
|
|
|
/* FIXME: This is pretty expensive. Maybe the caller should supply
|
|
* the parent */
|
|
gdk_error_trap_push ();
|
|
result = XQueryTree (gdk_display, anid, &root, &parent, &children, &nchildren);
|
|
if (gdk_error_trap_pop () || !result)
|
|
return NULL;
|
|
|
|
if (children)
|
|
XFree (children);
|
|
|
|
window = g_object_new (GDK_TYPE_WINDOW, NULL);
|
|
private = (GdkWindowObject *)window;
|
|
impl = GDK_WINDOW_IMPL_X11 (private->impl);
|
|
draw_impl = GDK_DRAWABLE_IMPL_X11 (private->impl);
|
|
draw_impl->wrapper = GDK_DRAWABLE (window);
|
|
|
|
private->parent = gdk_xid_table_lookup (parent);
|
|
|
|
parent_private = (GdkWindowObject *)private->parent;
|
|
|
|
if (parent_private)
|
|
parent_private->children = g_list_prepend (parent_private->children, window);
|
|
|
|
draw_impl->xid = anid;
|
|
draw_impl->xdisplay = gdk_display;
|
|
|
|
private->x = attrs.x;
|
|
private->y = attrs.y;
|
|
impl->width = attrs.width;
|
|
impl->height = attrs.height;
|
|
private->window_type = GDK_WINDOW_FOREIGN;
|
|
private->destroyed = FALSE;
|
|
|
|
if (attrs.map_state == IsUnmapped)
|
|
private->state = GDK_WINDOW_STATE_WITHDRAWN;
|
|
else
|
|
private->state = 0;
|
|
|
|
private->depth = attrs.depth;
|
|
|
|
_gdk_window_init_position (GDK_WINDOW (private));
|
|
|
|
gdk_drawable_ref (window);
|
|
gdk_xid_table_insert (&GDK_WINDOW_XID (window), window);
|
|
|
|
return window;
|
|
}
|
|
|
|
void
|
|
_gdk_windowing_window_destroy (GdkWindow *window,
|
|
gboolean recursing,
|
|
gboolean foreign_destroy)
|
|
{
|
|
GdkWindowObject *private = (GdkWindowObject *)window;
|
|
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
_gdk_selection_window_destroyed (window);
|
|
|
|
if (private->extension_events != 0)
|
|
gdk_input_window_destroy (window);
|
|
|
|
if (private->window_type == GDK_WINDOW_FOREIGN)
|
|
{
|
|
if (!foreign_destroy && (private->parent != NULL))
|
|
{
|
|
/* It's somebody else's window, but in our heirarchy,
|
|
* so reparent it to the root window, and then send
|
|
* it a delete event, as if we were a WM
|
|
*/
|
|
XClientMessageEvent xevent;
|
|
|
|
gdk_error_trap_push ();
|
|
gdk_window_hide (window);
|
|
gdk_window_reparent (window, NULL, 0, 0);
|
|
|
|
xevent.type = ClientMessage;
|
|
xevent.window = GDK_WINDOW_XID (window);
|
|
xevent.message_type = gdk_wm_protocols;
|
|
xevent.format = 32;
|
|
xevent.data.l[0] = gdk_wm_delete_window;
|
|
xevent.data.l[1] = CurrentTime;
|
|
|
|
XSendEvent (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
False, 0, (XEvent *)&xevent);
|
|
gdk_flush ();
|
|
gdk_error_trap_pop ();
|
|
}
|
|
}
|
|
else if (!recursing && !foreign_destroy)
|
|
XDestroyWindow (GDK_WINDOW_XDISPLAY (window), GDK_WINDOW_XID (window));
|
|
}
|
|
|
|
/* This function is called when the XWindow is really gone.
|
|
*/
|
|
void
|
|
gdk_window_destroy_notify (GdkWindow *window)
|
|
{
|
|
GdkWindowImplX11 *window_impl;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
|
|
window_impl = GDK_WINDOW_IMPL_X11 (((GdkWindowObject *)window)->impl);
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
{
|
|
if (GDK_WINDOW_TYPE(window) != GDK_WINDOW_FOREIGN)
|
|
g_warning ("GdkWindow %#lx unexpectedly destroyed", GDK_WINDOW_XID (window));
|
|
|
|
_gdk_window_destroy (window, TRUE);
|
|
}
|
|
|
|
gdk_xid_table_remove (GDK_WINDOW_XID (window));
|
|
if (window_impl->focus_window)
|
|
gdk_xid_table_remove (window_impl->focus_window);
|
|
|
|
gdk_drawable_unref (window);
|
|
}
|
|
|
|
static void
|
|
set_initial_hints (GdkWindow *window)
|
|
{
|
|
GdkWindowObject *private;
|
|
GdkAtom atoms[5];
|
|
gint i;
|
|
|
|
private = (GdkWindowObject*) window;
|
|
|
|
if (private->state & GDK_WINDOW_STATE_ICONIFIED)
|
|
{
|
|
XWMHints *wm_hints;
|
|
|
|
wm_hints = XGetWMHints (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window));
|
|
if (!wm_hints)
|
|
wm_hints = XAllocWMHints ();
|
|
|
|
wm_hints->flags |= StateHint;
|
|
wm_hints->initial_state = IconicState;
|
|
|
|
XSetWMHints (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window), wm_hints);
|
|
XFree (wm_hints);
|
|
}
|
|
|
|
/* We set the spec hints regardless of whether the spec is supported,
|
|
* since it can't hurt and it's kind of expensive to check whether
|
|
* it's supported.
|
|
*/
|
|
|
|
i = 0;
|
|
|
|
if (private->state & GDK_WINDOW_STATE_MAXIMIZED)
|
|
{
|
|
atoms[i] = gdk_atom_intern ("_NET_WM_STATE_MAXIMIZED_VERT", FALSE);
|
|
++i;
|
|
atoms[i] = gdk_atom_intern ("_NET_WM_STATE_MAXIMIZED_HORZ", FALSE);
|
|
++i;
|
|
}
|
|
|
|
if (private->state & GDK_WINDOW_STATE_STICKY)
|
|
{
|
|
atoms[i] = gdk_atom_intern ("_NET_WM_STATE_STICKY", FALSE);
|
|
++i;
|
|
}
|
|
|
|
if (private->modal_hint)
|
|
{
|
|
atoms[i] = gdk_atom_intern ("_NET_WM_STATE_MODAL", FALSE);
|
|
++i;
|
|
}
|
|
|
|
if (i > 0)
|
|
{
|
|
XChangeProperty (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
gdk_atom_intern ("_NET_WM_STATE", FALSE),
|
|
XA_ATOM, 32, PropModeReplace,
|
|
(guchar*) atoms, i);
|
|
}
|
|
|
|
if (private->state & GDK_WINDOW_STATE_STICKY)
|
|
{
|
|
atoms[0] = 0xFFFFFFFF;
|
|
XChangeProperty (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
gdk_atom_intern ("_NET_WM_DESKTOP", FALSE),
|
|
XA_CARDINAL, 32, PropModeReplace,
|
|
(guchar*) atoms, 1);
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_show (GdkWindow *window)
|
|
{
|
|
GdkWindowObject *private;
|
|
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
private = (GdkWindowObject*) window;
|
|
if (!private->destroyed)
|
|
{
|
|
XRaiseWindow (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window));
|
|
|
|
if (!GDK_WINDOW_IS_MAPPED (window))
|
|
{
|
|
set_initial_hints (window);
|
|
|
|
gdk_synthesize_window_state (window,
|
|
GDK_WINDOW_STATE_WITHDRAWN,
|
|
0);
|
|
}
|
|
|
|
g_assert (GDK_WINDOW_IS_MAPPED (window));
|
|
|
|
if (GDK_WINDOW_IMPL_X11 (private->impl)->position_info.mapped)
|
|
XMapWindow (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window));
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_hide (GdkWindow *window)
|
|
{
|
|
GdkWindowObject *private;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
|
|
private = (GdkWindowObject*) window;
|
|
|
|
/* You can't simply unmap toplevel windows. */
|
|
switch (private->window_type)
|
|
{
|
|
case GDK_WINDOW_TOPLEVEL:
|
|
case GDK_WINDOW_DIALOG:
|
|
case GDK_WINDOW_TEMP: /* ? */
|
|
gdk_window_withdraw (window);
|
|
return;
|
|
break;
|
|
|
|
case GDK_WINDOW_FOREIGN:
|
|
case GDK_WINDOW_ROOT:
|
|
case GDK_WINDOW_CHILD:
|
|
break;
|
|
}
|
|
|
|
if (!private->destroyed)
|
|
{
|
|
if (GDK_WINDOW_IS_MAPPED (window))
|
|
gdk_synthesize_window_state (window,
|
|
0,
|
|
GDK_WINDOW_STATE_WITHDRAWN);
|
|
|
|
g_assert (!GDK_WINDOW_IS_MAPPED (window));
|
|
|
|
_gdk_window_clear_update_area (window);
|
|
|
|
XUnmapWindow (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window));
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_withdraw (GdkWindow *window)
|
|
{
|
|
GdkWindowObject *private;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
|
|
private = (GdkWindowObject*) window;
|
|
if (!private->destroyed)
|
|
{
|
|
if (GDK_WINDOW_IS_MAPPED (window))
|
|
gdk_synthesize_window_state (window,
|
|
0,
|
|
GDK_WINDOW_STATE_WITHDRAWN);
|
|
|
|
g_assert (!GDK_WINDOW_IS_MAPPED (window));
|
|
|
|
XWithdrawWindow (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window), 0);
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_move (GdkWindow *window,
|
|
gint x,
|
|
gint y)
|
|
{
|
|
GdkWindowObject *private = (GdkWindowObject *)window;
|
|
GdkWindowImplX11 *impl;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
impl = GDK_WINDOW_IMPL_X11 (private->impl);
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
{
|
|
if (GDK_WINDOW_TYPE (private) == GDK_WINDOW_CHILD)
|
|
_gdk_window_move_resize_child (window, x, y,
|
|
impl->width, impl->height);
|
|
else
|
|
{
|
|
XMoveWindow (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
x, y);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_resize (GdkWindow *window,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
GdkWindowObject *private;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (width < 1)
|
|
width = 1;
|
|
if (height < 1)
|
|
height = 1;
|
|
|
|
private = (GdkWindowObject*) window;
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
{
|
|
if (GDK_WINDOW_TYPE (private) == GDK_WINDOW_CHILD)
|
|
_gdk_window_move_resize_child (window, private->x, private->y,
|
|
width, height);
|
|
else
|
|
{
|
|
GdkWindowImplX11 *impl = GDK_WINDOW_IMPL_X11 (private->impl);
|
|
|
|
if (width != impl->width || height != impl->height)
|
|
private->resize_count += 1;
|
|
|
|
XResizeWindow (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
width, height);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_move_resize (GdkWindow *window,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
GdkWindowObject *private;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (width < 1)
|
|
width = 1;
|
|
if (height < 1)
|
|
height = 1;
|
|
|
|
private = (GdkWindowObject*) window;
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
{
|
|
if (GDK_WINDOW_TYPE (private) == GDK_WINDOW_CHILD)
|
|
_gdk_window_move_resize_child (window, x, y, width, height);
|
|
else
|
|
{
|
|
GdkWindowImplX11 *impl = GDK_WINDOW_IMPL_X11 (private->impl);
|
|
|
|
if (width != impl->width || height != impl->height)
|
|
private->resize_count += 1;
|
|
|
|
XMoveResizeWindow (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
x, y, width, height);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_reparent (GdkWindow *window,
|
|
GdkWindow *new_parent,
|
|
gint x,
|
|
gint y)
|
|
{
|
|
GdkWindowObject *window_private;
|
|
GdkWindowObject *parent_private;
|
|
GdkWindowObject *old_parent_private;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
g_return_if_fail (new_parent == NULL || GDK_IS_WINDOW (new_parent));
|
|
|
|
if (!new_parent)
|
|
new_parent = gdk_parent_root;
|
|
|
|
window_private = (GdkWindowObject*) window;
|
|
old_parent_private = (GdkWindowObject*)window_private->parent;
|
|
parent_private = (GdkWindowObject*) new_parent;
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window) && !GDK_WINDOW_DESTROYED (new_parent))
|
|
XReparentWindow (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
GDK_WINDOW_XID (new_parent),
|
|
x, y);
|
|
|
|
window_private->parent = (GdkWindowObject *)new_parent;
|
|
|
|
if (old_parent_private)
|
|
old_parent_private->children = g_list_remove (old_parent_private->children, window);
|
|
|
|
if ((old_parent_private &&
|
|
(!old_parent_private->guffaw_gravity != !parent_private->guffaw_gravity)) ||
|
|
(!old_parent_private && parent_private->guffaw_gravity))
|
|
gdk_window_set_static_win_gravity (window, parent_private->guffaw_gravity);
|
|
|
|
parent_private->children = g_list_prepend (parent_private->children, window);
|
|
_gdk_window_init_position (GDK_WINDOW (window_private));
|
|
}
|
|
|
|
void
|
|
_gdk_windowing_window_clear_area (GdkWindow *window,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
XClearArea (GDK_WINDOW_XDISPLAY (window), GDK_WINDOW_XID (window),
|
|
x, y, width, height, False);
|
|
}
|
|
|
|
void
|
|
_gdk_windowing_window_clear_area_e (GdkWindow *window,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
XClearArea (GDK_WINDOW_XDISPLAY (window), GDK_WINDOW_XID (window),
|
|
x, y, width, height, True);
|
|
}
|
|
|
|
void
|
|
gdk_window_raise (GdkWindow *window)
|
|
{
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
XRaiseWindow (GDK_WINDOW_XDISPLAY (window), GDK_WINDOW_XID (window));
|
|
}
|
|
|
|
void
|
|
gdk_window_lower (GdkWindow *window)
|
|
{
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
XLowerWindow (GDK_WINDOW_XDISPLAY (window), GDK_WINDOW_XID (window));
|
|
}
|
|
|
|
void
|
|
gdk_window_focus (GdkWindow *window,
|
|
guint32 timestamp)
|
|
{
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
if (gdk_net_wm_supports (gdk_atom_intern ("_NET_ACTIVE_WINDOW", FALSE)))
|
|
{
|
|
XEvent xev;
|
|
|
|
xev.xclient.type = ClientMessage;
|
|
xev.xclient.serial = 0;
|
|
xev.xclient.send_event = True;
|
|
xev.xclient.window = GDK_WINDOW_XWINDOW (window);
|
|
xev.xclient.display = gdk_display;
|
|
xev.xclient.message_type = gdk_atom_intern ("_NET_ACTIVE_WINDOW", FALSE);
|
|
xev.xclient.format = 32;
|
|
xev.xclient.data.l[0] = 0;
|
|
|
|
XSendEvent (gdk_display, gdk_root_window, False,
|
|
SubstructureRedirectMask | SubstructureNotifyMask,
|
|
&xev);
|
|
}
|
|
else
|
|
{
|
|
XRaiseWindow (GDK_WINDOW_XDISPLAY (window), GDK_WINDOW_XID (window));
|
|
|
|
/* There is no way of knowing reliably whether we are viewable so we need
|
|
* to trap errors so we don't cause a BadMatch.
|
|
*/
|
|
gdk_error_trap_push ();
|
|
XSetInputFocus (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XWINDOW (window),
|
|
RevertToNone,
|
|
timestamp);
|
|
XSync (GDK_WINDOW_XDISPLAY (window), False);
|
|
gdk_error_trap_pop ();
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_set_hints (GdkWindow *window,
|
|
gint x,
|
|
gint y,
|
|
gint min_width,
|
|
gint min_height,
|
|
gint max_width,
|
|
gint max_height,
|
|
gint flags)
|
|
{
|
|
XSizeHints size_hints;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
size_hints.flags = 0;
|
|
|
|
if (flags & GDK_HINT_POS)
|
|
{
|
|
size_hints.flags |= PPosition;
|
|
size_hints.x = x;
|
|
size_hints.y = y;
|
|
}
|
|
|
|
if (flags & GDK_HINT_MIN_SIZE)
|
|
{
|
|
size_hints.flags |= PMinSize;
|
|
size_hints.min_width = min_width;
|
|
size_hints.min_height = min_height;
|
|
}
|
|
|
|
if (flags & GDK_HINT_MAX_SIZE)
|
|
{
|
|
size_hints.flags |= PMaxSize;
|
|
size_hints.max_width = max_width;
|
|
size_hints.max_height = max_height;
|
|
}
|
|
|
|
/* FIXME: Would it be better to delete this property of
|
|
* flags == 0? It would save space on the server
|
|
*/
|
|
XSetWMNormalHints (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
&size_hints);
|
|
}
|
|
|
|
/**
|
|
* gdk_window_set_type_hint:
|
|
* @window: A #GdkWindow
|
|
* @hint: A hint of the function this window will have
|
|
*
|
|
* The application can use this call to provide a hint to the window
|
|
* manager about the functionality of a window. The window manager
|
|
* can use this information when determining the decoration and behaviour
|
|
* of the window.
|
|
*
|
|
* The hint must be set before the window is mapped.
|
|
**/
|
|
void
|
|
gdk_window_set_type_hint (GdkWindow *window,
|
|
GdkWindowTypeHint hint)
|
|
{
|
|
GdkAtom atom;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
switch (hint)
|
|
{
|
|
case GDK_WINDOW_TYPE_HINT_DIALOG:
|
|
atom = gdk_atom_intern ("_NET_WM_WINDOW_TYPE_DIALOG", FALSE);
|
|
break;
|
|
case GDK_WINDOW_TYPE_HINT_MENU:
|
|
atom = gdk_atom_intern ("_NET_WM_WINDOW_TYPE_MENU", FALSE);
|
|
break;
|
|
case GDK_WINDOW_TYPE_HINT_TOOLBAR:
|
|
atom = gdk_atom_intern ("_NET_WM_WINDOW_TYPE_TOOLBAR", FALSE);
|
|
break;
|
|
default:
|
|
g_warning ("Unknown hint %d passed to gdk_window_set_type_hint", hint);
|
|
/* Fall thru */
|
|
case GDK_WINDOW_TYPE_HINT_NORMAL:
|
|
atom = gdk_atom_intern ("_NET_WM_WINDOW_TYPE_NORMAL", FALSE);
|
|
break;
|
|
}
|
|
|
|
XChangeProperty (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
gdk_atom_intern ("_NET_WM_WINDOW_TYPE", FALSE),
|
|
XA_ATOM, 32, PropModeReplace,
|
|
(guchar *)&atom, 1);
|
|
}
|
|
|
|
|
|
static void
|
|
gdk_wmspec_change_state (gboolean add,
|
|
GdkWindow *window,
|
|
GdkAtom state1,
|
|
GdkAtom state2)
|
|
{
|
|
XEvent xev;
|
|
Atom op;
|
|
|
|
if (add)
|
|
op = gdk_atom_intern ("_NET_WM_STATE_ADD", FALSE);
|
|
else
|
|
op = gdk_atom_intern ("_NET_WM_STATE_REMOVE", FALSE);
|
|
|
|
xev.xclient.type = ClientMessage;
|
|
xev.xclient.serial = 0;
|
|
xev.xclient.send_event = True;
|
|
xev.xclient.display = gdk_display;
|
|
xev.xclient.window = GDK_WINDOW_XID (window);
|
|
xev.xclient.message_type = gdk_atom_intern ("_NET_WM_STATE", FALSE);
|
|
xev.xclient.format = 32;
|
|
xev.xclient.data.l[0] = op;
|
|
xev.xclient.data.l[1] = state1;
|
|
xev.xclient.data.l[2] = state2;
|
|
|
|
XSendEvent (gdk_display, gdk_root_window, False,
|
|
SubstructureRedirectMask | SubstructureNotifyMask,
|
|
&xev);
|
|
}
|
|
/**
|
|
* gdk_window_set_modal_hint:
|
|
* @window: A #GdkWindow
|
|
* @modal: TRUE if the window is modal, FALSE otherwise.
|
|
*
|
|
* The application can use this hint to tell the window manager
|
|
* that a certain window has modal behaviour. The window manager
|
|
* can use this information to handle modal windows in a special
|
|
* way.
|
|
*
|
|
* You should only use this on windows for which you have
|
|
* previously called #gdk_window_set_transient_for()
|
|
**/
|
|
void
|
|
gdk_window_set_modal_hint (GdkWindow *window,
|
|
gboolean modal)
|
|
{
|
|
GdkWindowObject *private;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
private = (GdkWindowObject*) window;
|
|
|
|
private->modal_hint = modal;
|
|
|
|
if (GDK_WINDOW_IS_MAPPED (window))
|
|
gdk_wmspec_change_state (modal, window,
|
|
gdk_atom_intern ("_NET_WM_STATE_MODAL", FALSE),
|
|
0);
|
|
}
|
|
|
|
void
|
|
gdk_window_set_geometry_hints (GdkWindow *window,
|
|
GdkGeometry *geometry,
|
|
GdkWindowHints geom_mask)
|
|
{
|
|
XSizeHints size_hints;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
size_hints.flags = 0;
|
|
|
|
if (geom_mask & GDK_HINT_POS)
|
|
{
|
|
size_hints.flags |= PPosition;
|
|
/* We need to initialize the following obsolete fields because KWM
|
|
* apparently uses these fields if they are non-zero.
|
|
* #@#!#!$!.
|
|
*/
|
|
size_hints.x = 0;
|
|
size_hints.y = 0;
|
|
}
|
|
|
|
if (geom_mask & GDK_HINT_MIN_SIZE)
|
|
{
|
|
size_hints.flags |= PMinSize;
|
|
size_hints.min_width = geometry->min_width;
|
|
size_hints.min_height = geometry->min_height;
|
|
}
|
|
|
|
if (geom_mask & GDK_HINT_MAX_SIZE)
|
|
{
|
|
size_hints.flags |= PMaxSize;
|
|
size_hints.max_width = MAX (geometry->max_width, 1);
|
|
size_hints.max_height = MAX (geometry->max_height, 1);
|
|
}
|
|
|
|
if (geom_mask & GDK_HINT_BASE_SIZE)
|
|
{
|
|
size_hints.flags |= PBaseSize;
|
|
size_hints.base_width = geometry->base_width;
|
|
size_hints.base_height = geometry->base_height;
|
|
}
|
|
|
|
if (geom_mask & GDK_HINT_RESIZE_INC)
|
|
{
|
|
size_hints.flags |= PResizeInc;
|
|
size_hints.width_inc = geometry->width_inc;
|
|
size_hints.height_inc = geometry->height_inc;
|
|
}
|
|
|
|
if (geom_mask & GDK_HINT_ASPECT)
|
|
{
|
|
size_hints.flags |= PAspect;
|
|
if (geometry->min_aspect <= 1)
|
|
{
|
|
size_hints.min_aspect.x = 65536 * geometry->min_aspect;
|
|
size_hints.min_aspect.y = 65536;
|
|
}
|
|
else
|
|
{
|
|
size_hints.min_aspect.x = 65536;
|
|
size_hints.min_aspect.y = 65536 / geometry->min_aspect;;
|
|
}
|
|
if (geometry->max_aspect <= 1)
|
|
{
|
|
size_hints.max_aspect.x = 65536 * geometry->max_aspect;
|
|
size_hints.max_aspect.y = 65536;
|
|
}
|
|
else
|
|
{
|
|
size_hints.max_aspect.x = 65536;
|
|
size_hints.max_aspect.y = 65536 / geometry->max_aspect;;
|
|
}
|
|
}
|
|
|
|
if (geom_mask & GDK_HINT_WIN_GRAVITY)
|
|
{
|
|
size_hints.flags |= PWinGravity;
|
|
size_hints.win_gravity = geometry->win_gravity;
|
|
}
|
|
|
|
/* FIXME: Would it be better to delete this property of
|
|
* geom_mask == 0? It would save space on the server
|
|
*/
|
|
XSetWMNormalHints (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
&size_hints);
|
|
}
|
|
|
|
static void
|
|
gdk_window_get_geometry_hints (GdkWindow *window,
|
|
GdkGeometry *geometry,
|
|
GdkWindowHints *geom_mask)
|
|
{
|
|
XSizeHints size_hints;
|
|
glong junk_size_mask = 0;
|
|
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
g_return_if_fail (geometry != NULL);
|
|
g_return_if_fail (geom_mask != NULL);
|
|
|
|
*geom_mask = 0;
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
if (!XGetWMNormalHints (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
&size_hints,
|
|
&junk_size_mask))
|
|
return;
|
|
|
|
if (size_hints.flags & PMinSize)
|
|
{
|
|
*geom_mask |= GDK_HINT_MIN_SIZE;
|
|
geometry->min_width = size_hints.min_width;
|
|
geometry->min_height = size_hints.min_height;
|
|
}
|
|
|
|
if (size_hints.flags & PMaxSize)
|
|
{
|
|
*geom_mask |= GDK_HINT_MAX_SIZE;
|
|
geometry->max_width = MAX (size_hints.max_width, 1);
|
|
geometry->max_height = MAX (size_hints.max_height, 1);
|
|
}
|
|
|
|
if (size_hints.flags & PResizeInc)
|
|
{
|
|
*geom_mask |= GDK_HINT_RESIZE_INC;
|
|
geometry->width_inc = size_hints.width_inc;
|
|
geometry->height_inc = size_hints.height_inc;
|
|
}
|
|
|
|
if (size_hints.flags & PAspect)
|
|
{
|
|
*geom_mask |= GDK_HINT_ASPECT;
|
|
|
|
geometry->min_aspect = (gdouble) size_hints.min_aspect.x / (gdouble) size_hints.min_aspect.y;
|
|
geometry->max_aspect = (gdouble) size_hints.max_aspect.x / (gdouble) size_hints.max_aspect.y;
|
|
}
|
|
|
|
if (size_hints.flags & PWinGravity)
|
|
{
|
|
*geom_mask |= GDK_HINT_WIN_GRAVITY;
|
|
geometry->win_gravity = size_hints.win_gravity;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
utf8_is_latin1 (const gchar *str)
|
|
{
|
|
const char *p = str;
|
|
|
|
while (*p)
|
|
{
|
|
gunichar ch = g_utf8_get_char (p);
|
|
|
|
if (ch >= 0xff)
|
|
return FALSE;
|
|
|
|
p = g_utf8_next_char (p);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* Set the property to @utf8_str as STRING if the @utf8_str is fully
|
|
* convertable to STRING, otherwise, set it as compound text
|
|
*/
|
|
static void
|
|
set_text_property (GdkWindow *window,
|
|
GdkAtom property,
|
|
const gchar *utf8_str)
|
|
{
|
|
guchar *prop_text = NULL;
|
|
GdkAtom prop_type;
|
|
gint prop_length;
|
|
gint prop_format;
|
|
|
|
if (utf8_is_latin1 (utf8_str))
|
|
{
|
|
prop_type = GDK_TARGET_STRING;
|
|
prop_text = gdk_utf8_to_string_target (utf8_str);
|
|
prop_length = strlen (prop_text);
|
|
prop_format = 8;
|
|
}
|
|
else
|
|
{
|
|
gdk_utf8_to_compound_text (utf8_str, &prop_type, &prop_format,
|
|
&prop_text, &prop_length);
|
|
}
|
|
|
|
if (prop_text)
|
|
{
|
|
XChangeProperty (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
property,
|
|
prop_type, prop_format,
|
|
PropModeReplace, prop_text,
|
|
prop_length);
|
|
|
|
g_free (prop_text);
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_set_title (GdkWindow *window,
|
|
const gchar *title)
|
|
{
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
XChangeProperty (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
gdk_atom_intern ("_NET_WM_NAME", FALSE),
|
|
gdk_atom_intern ("UTF8_STRING", FALSE), 8,
|
|
PropModeReplace, title,
|
|
strlen (title));
|
|
|
|
set_text_property (window, gdk_atom_intern ("WM_NAME", FALSE), title);
|
|
if (!gdk_window_icon_name_set (window))
|
|
{
|
|
XChangeProperty (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
gdk_atom_intern ("_NET_WM_ICON_NAME", FALSE),
|
|
gdk_atom_intern ("UTF8_STRING", FALSE), 8,
|
|
PropModeReplace, title,
|
|
strlen (title));
|
|
set_text_property (window, gdk_atom_intern ("WM_ICON_NAME", FALSE), title);
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_set_role (GdkWindow *window,
|
|
const gchar *role)
|
|
{
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
{
|
|
if (role)
|
|
XChangeProperty (GDK_WINDOW_XDISPLAY (window), GDK_WINDOW_XID (window),
|
|
gdk_atom_intern ("WM_WINDOW_ROLE", FALSE), XA_STRING,
|
|
8, PropModeReplace, role, strlen (role));
|
|
else
|
|
XDeleteProperty (GDK_WINDOW_XDISPLAY (window), GDK_WINDOW_XID (window),
|
|
gdk_atom_intern ("WM_WINDOW_ROLE", FALSE));
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_set_transient_for (GdkWindow *window,
|
|
GdkWindow *parent)
|
|
{
|
|
GdkWindowObject *private;
|
|
GdkWindowObject *parent_private;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
private = (GdkWindowObject*) window;
|
|
parent_private = (GdkWindowObject*) parent;
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window) && !GDK_WINDOW_DESTROYED (parent))
|
|
XSetTransientForHint (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
GDK_WINDOW_XID (parent));
|
|
}
|
|
|
|
void
|
|
gdk_window_set_background (GdkWindow *window,
|
|
GdkColor *color)
|
|
{
|
|
GdkWindowObject *private = (GdkWindowObject *)window;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
XSetWindowBackground (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window), color->pixel);
|
|
|
|
private->bg_color = *color;
|
|
|
|
if (private->bg_pixmap &&
|
|
private->bg_pixmap != GDK_PARENT_RELATIVE_BG &&
|
|
private->bg_pixmap != GDK_NO_BG)
|
|
{
|
|
gdk_pixmap_unref (private->bg_pixmap);
|
|
private->bg_pixmap = NULL;
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_set_back_pixmap (GdkWindow *window,
|
|
GdkPixmap *pixmap,
|
|
gboolean parent_relative)
|
|
{
|
|
GdkWindowObject *private = (GdkWindowObject *)window;
|
|
Pixmap xpixmap;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
g_return_if_fail (pixmap == NULL || !parent_relative);
|
|
|
|
if (private->bg_pixmap &&
|
|
private->bg_pixmap != GDK_PARENT_RELATIVE_BG &&
|
|
private->bg_pixmap != GDK_NO_BG)
|
|
gdk_pixmap_unref (private->bg_pixmap);
|
|
|
|
if (parent_relative)
|
|
{
|
|
xpixmap = ParentRelative;
|
|
private->bg_pixmap = GDK_PARENT_RELATIVE_BG;
|
|
}
|
|
else
|
|
{
|
|
if (pixmap)
|
|
{
|
|
gdk_pixmap_ref (pixmap);
|
|
private->bg_pixmap = pixmap;
|
|
xpixmap = GDK_PIXMAP_XID (pixmap);
|
|
}
|
|
else
|
|
{
|
|
xpixmap = None;
|
|
private->bg_pixmap = GDK_NO_BG;
|
|
}
|
|
}
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
XSetWindowBackgroundPixmap (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window), xpixmap);
|
|
}
|
|
|
|
void
|
|
gdk_window_set_cursor (GdkWindow *window,
|
|
GdkCursor *cursor)
|
|
{
|
|
GdkCursorPrivate *cursor_private;
|
|
Cursor xcursor;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
cursor_private = (GdkCursorPrivate*) cursor;
|
|
|
|
if (!cursor)
|
|
xcursor = None;
|
|
else
|
|
xcursor = cursor_private->xcursor;
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
XDefineCursor (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
xcursor);
|
|
}
|
|
|
|
void
|
|
gdk_window_get_geometry (GdkWindow *window,
|
|
gint *x,
|
|
gint *y,
|
|
gint *width,
|
|
gint *height,
|
|
gint *depth)
|
|
{
|
|
Window root;
|
|
gint tx;
|
|
gint ty;
|
|
guint twidth;
|
|
guint theight;
|
|
guint tborder_width;
|
|
guint tdepth;
|
|
|
|
g_return_if_fail (window == NULL || GDK_IS_WINDOW (window));
|
|
|
|
if (!window)
|
|
window = gdk_parent_root;
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
{
|
|
XGetGeometry (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
&root, &tx, &ty, &twidth, &theight, &tborder_width, &tdepth);
|
|
|
|
if (x)
|
|
*x = tx;
|
|
if (y)
|
|
*y = ty;
|
|
if (width)
|
|
*width = twidth;
|
|
if (height)
|
|
*height = theight;
|
|
if (depth)
|
|
*depth = tdepth;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* gdk_window_get_origin:
|
|
* @window: a #GdkWindow
|
|
* @x: return location for X coordinate
|
|
* @y: return location for Y coordinate
|
|
*
|
|
* Obtains the position of a window in root window coordinates.
|
|
* (Compare with gdk_window_get_position() and
|
|
* gdk_window_get_geometry() which return the position of a window
|
|
* relative to its parent window.)
|
|
*
|
|
* Return value: not meaningful, ignore
|
|
**/
|
|
gint
|
|
gdk_window_get_origin (GdkWindow *window,
|
|
gint *x,
|
|
gint *y)
|
|
{
|
|
gint return_val;
|
|
Window child;
|
|
gint tx = 0;
|
|
gint ty = 0;
|
|
|
|
g_return_val_if_fail (window != NULL, 0);
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
{
|
|
return_val = XTranslateCoordinates (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
gdk_root_window,
|
|
0, 0, &tx, &ty,
|
|
&child);
|
|
|
|
}
|
|
else
|
|
return_val = 0;
|
|
|
|
if (x)
|
|
*x = tx;
|
|
if (y)
|
|
*y = ty;
|
|
|
|
return return_val;
|
|
}
|
|
|
|
/**
|
|
* gdk_window_get_deskrelative_origin:
|
|
* @window: a #GdkWindow
|
|
* @x: return location for X coordinate
|
|
* @y: return location for Y coordinate
|
|
*
|
|
* This gets the origin of a #GdkWindow relative to
|
|
* an Enlightenment-window-manager desktop. As long as you don't
|
|
* assume that the user's desktop/workspace covers the entire
|
|
* root window (i.e. you don't assume that the desktop begins
|
|
* at root window coordinate 0,0) this function is not necessary.
|
|
* It's deprecated for that reason.
|
|
*
|
|
* Return value: not meaningful
|
|
**/
|
|
gboolean
|
|
gdk_window_get_deskrelative_origin (GdkWindow *window,
|
|
gint *x,
|
|
gint *y)
|
|
{
|
|
gboolean return_val = FALSE;
|
|
gint num_children, format_return;
|
|
Window win, *child, parent, root;
|
|
gint tx = 0;
|
|
gint ty = 0;
|
|
Atom type_return;
|
|
static Atom atom = 0;
|
|
gulong number_return, bytes_after_return;
|
|
guchar *data_return;
|
|
|
|
g_return_val_if_fail (window != NULL, 0);
|
|
g_return_val_if_fail (GDK_IS_WINDOW (window), 0);
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
{
|
|
if (!atom)
|
|
atom = gdk_atom_intern ("ENLIGHTENMENT_DESKTOP", FALSE);
|
|
win = GDK_WINDOW_XID (window);
|
|
|
|
while (XQueryTree (GDK_WINDOW_XDISPLAY (window), win, &root, &parent,
|
|
&child, (unsigned int *)&num_children))
|
|
{
|
|
if ((child) && (num_children > 0))
|
|
XFree (child);
|
|
|
|
if (!parent)
|
|
break;
|
|
else
|
|
win = parent;
|
|
|
|
if (win == root)
|
|
break;
|
|
|
|
data_return = NULL;
|
|
XGetWindowProperty (GDK_WINDOW_XDISPLAY (window), win, atom, 0, 0,
|
|
False, XA_CARDINAL, &type_return, &format_return,
|
|
&number_return, &bytes_after_return, &data_return);
|
|
if (type_return == XA_CARDINAL)
|
|
{
|
|
XFree (data_return);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return_val = XTranslateCoordinates (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
win,
|
|
0, 0, &tx, &ty,
|
|
&root);
|
|
if (x)
|
|
*x = tx;
|
|
if (y)
|
|
*y = ty;
|
|
}
|
|
|
|
|
|
return return_val;
|
|
}
|
|
|
|
/**
|
|
* gdk_window_get_root_origin:
|
|
* @window: a #GdkWindow
|
|
* @x: return location for X position of window frame
|
|
* @y: return location for Y position of window frame
|
|
*
|
|
* Obtains the top-left corner of the window manager frame in root
|
|
* window coordinates.
|
|
*
|
|
**/
|
|
void
|
|
gdk_window_get_root_origin (GdkWindow *window,
|
|
gint *x,
|
|
gint *y)
|
|
{
|
|
GdkRectangle rect;
|
|
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
gdk_window_get_frame_extents (window, &rect);
|
|
|
|
if (x)
|
|
*x = rect.x;
|
|
|
|
if (y)
|
|
*y = rect.y;
|
|
}
|
|
|
|
/**
|
|
* gdk_window_get_frame_extents:
|
|
* @window: a #GdkWindow
|
|
* @rect: rectangle to fill with bounding box of the window frame
|
|
*
|
|
* Obtains the bounding box of the window, including window manager
|
|
* titlebar/borders if any. The frame position is given in root window
|
|
* coordinates. To get the position of the window itself (rather than
|
|
* the frame) in root window coordinates, use gdk_window_get_origin().
|
|
*
|
|
**/
|
|
void
|
|
gdk_window_get_frame_extents (GdkWindow *window,
|
|
GdkRectangle *rect)
|
|
{
|
|
GdkWindowObject *private;
|
|
Window xwindow;
|
|
Window xparent;
|
|
Window root;
|
|
Window *children;
|
|
unsigned int nchildren;
|
|
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
g_return_if_fail (rect != NULL);
|
|
|
|
private = (GdkWindowObject*) window;
|
|
|
|
rect->x = 0;
|
|
rect->y = 0;
|
|
rect->width = 1;
|
|
rect->height = 1;
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
while (private->parent && ((GdkWindowObject*) private->parent)->parent)
|
|
private = (GdkWindowObject*) private->parent;
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
xparent = GDK_WINDOW_XID (window);
|
|
do
|
|
{
|
|
xwindow = xparent;
|
|
if (!XQueryTree (GDK_WINDOW_XDISPLAY (window), xwindow,
|
|
&root, &xparent,
|
|
&children, &nchildren))
|
|
return;
|
|
|
|
if (children)
|
|
XFree (children);
|
|
}
|
|
while (xparent != root);
|
|
|
|
if (xparent == root)
|
|
{
|
|
unsigned int ww, wh, wb, wd;
|
|
int wx, wy;
|
|
|
|
if (XGetGeometry (GDK_WINDOW_XDISPLAY (window), xwindow, &root, &wx, &wy, &ww, &wh, &wb, &wd))
|
|
{
|
|
rect->x = wx;
|
|
rect->y = wy;
|
|
rect->width = ww;
|
|
rect->height = wh;
|
|
}
|
|
}
|
|
}
|
|
|
|
GdkWindow*
|
|
gdk_window_get_pointer (GdkWindow *window,
|
|
gint *x,
|
|
gint *y,
|
|
GdkModifierType *mask)
|
|
{
|
|
GdkWindow *return_val;
|
|
Window root;
|
|
Window child;
|
|
int rootx, rooty;
|
|
int winx = 0;
|
|
int winy = 0;
|
|
unsigned int xmask = 0;
|
|
gint xoffset, yoffset;
|
|
|
|
g_return_val_if_fail (window == NULL || GDK_IS_WINDOW (window), NULL);
|
|
|
|
if (!window)
|
|
window = gdk_parent_root;
|
|
|
|
_gdk_windowing_window_get_offsets (window, &xoffset, &yoffset);
|
|
|
|
return_val = NULL;
|
|
if (!GDK_WINDOW_DESTROYED (window) &&
|
|
XQueryPointer (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
&root, &child, &rootx, &rooty, &winx, &winy, &xmask))
|
|
{
|
|
if (child)
|
|
return_val = gdk_window_lookup (child);
|
|
}
|
|
|
|
if (x)
|
|
*x = winx + xoffset;
|
|
if (y)
|
|
*y = winy + yoffset;
|
|
if (mask)
|
|
*mask = xmask;
|
|
|
|
return return_val;
|
|
}
|
|
|
|
GdkWindow*
|
|
gdk_window_at_pointer (gint *win_x,
|
|
gint *win_y)
|
|
{
|
|
GdkWindow *window;
|
|
Window root;
|
|
Window xwindow;
|
|
Window xwindow_last = 0;
|
|
Display *xdisplay;
|
|
int rootx = -1, rooty = -1;
|
|
int winx, winy;
|
|
unsigned int xmask;
|
|
|
|
xwindow = GDK_ROOT_WINDOW ();
|
|
xdisplay = GDK_DISPLAY ();
|
|
|
|
XGrabServer (xdisplay);
|
|
while (xwindow)
|
|
{
|
|
xwindow_last = xwindow;
|
|
XQueryPointer (xdisplay, xwindow,
|
|
&root, &xwindow,
|
|
&rootx, &rooty,
|
|
&winx, &winy,
|
|
&xmask);
|
|
}
|
|
XUngrabServer (xdisplay);
|
|
|
|
window = gdk_window_lookup (xwindow_last);
|
|
|
|
if (win_x)
|
|
*win_x = window ? winx : -1;
|
|
if (win_y)
|
|
*win_y = window ? winy : -1;
|
|
|
|
return window;
|
|
}
|
|
|
|
GdkEventMask
|
|
gdk_window_get_events (GdkWindow *window)
|
|
{
|
|
XWindowAttributes attrs;
|
|
GdkEventMask event_mask;
|
|
int i;
|
|
|
|
g_return_val_if_fail (window != NULL, 0);
|
|
g_return_val_if_fail (GDK_IS_WINDOW (window), 0);
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return 0;
|
|
else
|
|
{
|
|
XGetWindowAttributes (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
&attrs);
|
|
|
|
event_mask = 0;
|
|
for (i = 0; i < gdk_nevent_masks; i++)
|
|
{
|
|
if (attrs.your_event_mask & gdk_event_mask_table[i])
|
|
event_mask |= 1 << (i + 1);
|
|
}
|
|
|
|
return event_mask;
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_set_events (GdkWindow *window,
|
|
GdkEventMask event_mask)
|
|
{
|
|
long xevent_mask;
|
|
int i;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
{
|
|
xevent_mask = StructureNotifyMask;
|
|
for (i = 0; i < gdk_nevent_masks; i++)
|
|
{
|
|
if (event_mask & (1 << (i + 1)))
|
|
xevent_mask |= gdk_event_mask_table[i];
|
|
}
|
|
|
|
XSelectInput (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
xevent_mask);
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_add_colormap_windows (GdkWindow *window)
|
|
{
|
|
GdkWindow *toplevel;
|
|
Window *old_windows;
|
|
Window *new_windows;
|
|
int i, count;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
toplevel = gdk_window_get_toplevel (window);
|
|
if (GDK_WINDOW_DESTROYED (toplevel))
|
|
return;
|
|
|
|
old_windows = NULL;
|
|
if (!XGetWMColormapWindows (GDK_WINDOW_XDISPLAY (toplevel),
|
|
GDK_WINDOW_XID (toplevel),
|
|
&old_windows, &count))
|
|
{
|
|
count = 0;
|
|
}
|
|
|
|
for (i = 0; i < count; i++)
|
|
if (old_windows[i] == GDK_WINDOW_XID (window))
|
|
{
|
|
XFree (old_windows);
|
|
return;
|
|
}
|
|
|
|
new_windows = g_new (Window, count + 1);
|
|
|
|
for (i = 0; i < count; i++)
|
|
new_windows[i] = old_windows[i];
|
|
new_windows[count] = GDK_WINDOW_XID (window);
|
|
|
|
XSetWMColormapWindows (GDK_WINDOW_XDISPLAY (toplevel),
|
|
GDK_WINDOW_XID (toplevel),
|
|
new_windows, count + 1);
|
|
|
|
g_free (new_windows);
|
|
if (old_windows)
|
|
XFree (old_windows);
|
|
}
|
|
|
|
static gboolean
|
|
gdk_window_have_shape_ext (void)
|
|
{
|
|
enum { UNKNOWN, NO, YES };
|
|
static gint have_shape = UNKNOWN;
|
|
|
|
if (have_shape == UNKNOWN)
|
|
{
|
|
int ignore;
|
|
if (XQueryExtension (gdk_display, "SHAPE", &ignore, &ignore, &ignore))
|
|
have_shape = YES;
|
|
else
|
|
have_shape = NO;
|
|
}
|
|
|
|
return (have_shape == YES);
|
|
}
|
|
|
|
#define WARN_SHAPE_TOO_BIG() g_warning ("GdkWindow is too large to allow the use of shape masks or shape regions.")
|
|
|
|
/*
|
|
* This needs the X11 shape extension.
|
|
* If not available, shaped windows will look
|
|
* ugly, but programs still work. Stefan Wille
|
|
*/
|
|
void
|
|
gdk_window_shape_combine_mask (GdkWindow *window,
|
|
GdkBitmap *mask,
|
|
gint x, gint y)
|
|
{
|
|
Pixmap pixmap;
|
|
gint xoffset, yoffset;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
#ifdef HAVE_SHAPE_EXT
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
_gdk_windowing_window_get_offsets (window, &xoffset, &yoffset);
|
|
|
|
if (xoffset != 0 || yoffset != 0)
|
|
{
|
|
WARN_SHAPE_TOO_BIG ();
|
|
return;
|
|
}
|
|
|
|
if (gdk_window_have_shape_ext ())
|
|
{
|
|
if (mask)
|
|
{
|
|
pixmap = GDK_PIXMAP_XID (mask);
|
|
}
|
|
else
|
|
{
|
|
x = 0;
|
|
y = 0;
|
|
pixmap = None;
|
|
}
|
|
|
|
XShapeCombineMask (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
ShapeBounding,
|
|
x, y,
|
|
pixmap,
|
|
ShapeSet);
|
|
}
|
|
#endif /* HAVE_SHAPE_EXT */
|
|
}
|
|
|
|
void
|
|
gdk_window_shape_combine_region (GdkWindow *window,
|
|
GdkRegion *shape_region,
|
|
gint offset_x,
|
|
gint offset_y)
|
|
{
|
|
gint xoffset, yoffset;
|
|
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
#ifdef HAVE_SHAPE_EXT
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
_gdk_windowing_window_get_offsets (window, &xoffset, &yoffset);
|
|
|
|
if (xoffset != 0 || yoffset != 0)
|
|
{
|
|
WARN_SHAPE_TOO_BIG ();
|
|
return;
|
|
}
|
|
|
|
if (shape_region == NULL)
|
|
{
|
|
/* Use NULL mask to unset the shape */
|
|
gdk_window_shape_combine_mask (window, NULL, 0, 0);
|
|
return;
|
|
}
|
|
|
|
if (gdk_window_have_shape_ext ())
|
|
{
|
|
gint n_rects = 0;
|
|
XRectangle *xrects = NULL;
|
|
|
|
_gdk_region_get_xrectangles (shape_region,
|
|
0, 0,
|
|
&xrects, &n_rects);
|
|
|
|
XShapeCombineRectangles (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
ShapeBounding,
|
|
offset_x, offset_y,
|
|
xrects, n_rects,
|
|
ShapeSet,
|
|
YXBanded);
|
|
|
|
g_free (xrects);
|
|
}
|
|
#endif /* HAVE_SHAPE_EXT */
|
|
}
|
|
|
|
|
|
void
|
|
gdk_window_set_override_redirect (GdkWindow *window,
|
|
gboolean override_redirect)
|
|
{
|
|
XSetWindowAttributes attr;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
{
|
|
attr.override_redirect = (override_redirect == FALSE)?False:True;
|
|
XChangeWindowAttributes (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
CWOverrideRedirect,
|
|
&attr);
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* gdk_window_set_icon_list:
|
|
* @window: The #GdkWindow toplevel window to set the icon of.
|
|
* @pixbufs: A list of pixbufs, of different sizes.
|
|
* @Returns: TRUE if the icons were set, false otherwise
|
|
*
|
|
* Sets a list of icons for the window. One of these will be used
|
|
* to represent the window when it has been iconified. The icon is
|
|
* usually shown in an icon box or some sort of task bar. Which icon
|
|
* size is shown depends on the window manager. The window manager
|
|
* can scale the icon but setting several size icons can give better
|
|
* image quality since the window manager may only need to scale the
|
|
* icon by a small amount or not at all.
|
|
*
|
|
* On the X11 backend this call might fail if the window manager
|
|
* doesn't support the Extended Window Manager Hints. Then this
|
|
* function returns FALSE, and the application should fall back
|
|
* to #gdk_window_set_icon().
|
|
**/
|
|
gboolean
|
|
gdk_window_set_icon_list (GdkWindow *window,
|
|
GList *pixbufs)
|
|
{
|
|
guint *data;
|
|
guchar *pixels;
|
|
guint *p;
|
|
gint size;
|
|
GList *l;
|
|
GdkPixbuf *pixbuf;
|
|
gint width, height, stride;
|
|
gint x, y;
|
|
gint n_channels;
|
|
|
|
g_return_val_if_fail (window != NULL, FALSE);
|
|
g_return_val_if_fail (GDK_IS_WINDOW (window), FALSE);
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return FALSE;
|
|
|
|
if (!gdk_net_wm_supports (gdk_atom_intern ("_NET_WM_ICON", FALSE)))
|
|
return FALSE;
|
|
|
|
l = pixbufs;
|
|
size = 0;
|
|
|
|
while (l)
|
|
{
|
|
pixbuf = l->data;
|
|
g_return_val_if_fail (GDK_IS_PIXBUF (pixbuf), FALSE);
|
|
|
|
width = gdk_pixbuf_get_width (pixbuf);
|
|
height = gdk_pixbuf_get_height (pixbuf);
|
|
|
|
size += 2 + width * height;
|
|
|
|
l = g_list_next (l);
|
|
}
|
|
|
|
data = g_malloc (size*4);
|
|
|
|
l = pixbufs;
|
|
p = data;
|
|
while (l)
|
|
{
|
|
pixbuf = l->data;
|
|
|
|
width = gdk_pixbuf_get_width (pixbuf);
|
|
height = gdk_pixbuf_get_height (pixbuf);
|
|
stride = gdk_pixbuf_get_rowstride (pixbuf);
|
|
n_channels = gdk_pixbuf_get_n_channels (pixbuf);
|
|
|
|
*p++ = width;
|
|
*p++ = height;
|
|
|
|
pixels = gdk_pixbuf_get_pixels (pixbuf);
|
|
|
|
for (y = 0; y < height; y++)
|
|
{
|
|
for (x = 0; x < width; x++)
|
|
{
|
|
guchar r, g, b, a;
|
|
|
|
r = pixels[y*stride + x*n_channels + 0];
|
|
g = pixels[y*stride + x*n_channels + 1];
|
|
b = pixels[y*stride + x*n_channels + 2];
|
|
if (n_channels >= 4)
|
|
a = pixels[y*stride + x*n_channels + 3];
|
|
else
|
|
a = 255;
|
|
|
|
*p++ = a << 24 | r << 16 | g << 8 | b ;
|
|
}
|
|
}
|
|
|
|
l = g_list_next (l);
|
|
}
|
|
|
|
XChangeProperty (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
gdk_atom_intern ("_NET_WM_ICON", FALSE),
|
|
XA_CARDINAL, 32,
|
|
PropModeReplace,
|
|
(guchar*) data, size);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
gdk_window_set_icon (GdkWindow *window,
|
|
GdkWindow *icon_window,
|
|
GdkPixmap *pixmap,
|
|
GdkBitmap *mask)
|
|
{
|
|
XWMHints *wm_hints;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
wm_hints = XGetWMHints (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window));
|
|
if (!wm_hints)
|
|
wm_hints = XAllocWMHints ();
|
|
|
|
if (icon_window != NULL)
|
|
{
|
|
wm_hints->flags |= IconWindowHint;
|
|
wm_hints->icon_window = GDK_WINDOW_XID (icon_window);
|
|
}
|
|
|
|
if (pixmap != NULL)
|
|
{
|
|
wm_hints->flags |= IconPixmapHint;
|
|
wm_hints->icon_pixmap = GDK_PIXMAP_XID (pixmap);
|
|
}
|
|
|
|
if (mask != NULL)
|
|
{
|
|
wm_hints->flags |= IconMaskHint;
|
|
wm_hints->icon_mask = GDK_PIXMAP_XID (mask);
|
|
}
|
|
|
|
XSetWMHints (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window), wm_hints);
|
|
XFree (wm_hints);
|
|
}
|
|
|
|
static gboolean
|
|
gdk_window_icon_name_set (GdkWindow *window)
|
|
{
|
|
return GPOINTER_TO_UINT (g_object_get_qdata (G_OBJECT (window),
|
|
g_quark_from_static_string ("gdk-icon-name-set")));
|
|
}
|
|
|
|
void
|
|
gdk_window_set_icon_name (GdkWindow *window,
|
|
const gchar *name)
|
|
{
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
g_object_set_qdata (G_OBJECT (window), g_quark_from_static_string ("gdk-icon-name-set"),
|
|
GUINT_TO_POINTER (TRUE));
|
|
|
|
XChangeProperty (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
gdk_atom_intern ("_NET_WM_ICON_NAME", FALSE),
|
|
gdk_atom_intern ("UTF8_STRING", FALSE), 8,
|
|
PropModeReplace, name,
|
|
strlen (name));
|
|
set_text_property (window, gdk_atom_intern ("WM_ICON_NAME", FALSE), name);
|
|
}
|
|
|
|
void
|
|
gdk_window_iconify (GdkWindow *window)
|
|
{
|
|
Display *display;
|
|
GdkWindowObject *private;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
display = GDK_WINDOW_XDISPLAY (window);
|
|
|
|
private = (GdkWindowObject*) window;
|
|
|
|
if (GDK_WINDOW_IS_MAPPED (window))
|
|
{
|
|
XIconifyWindow (display, GDK_WINDOW_XWINDOW (window), DefaultScreen (display));
|
|
|
|
}
|
|
else
|
|
{
|
|
/* Flip our client side flag, the real work happens on map. */
|
|
gdk_synthesize_window_state (window,
|
|
0,
|
|
GDK_WINDOW_STATE_ICONIFIED);
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_deiconify (GdkWindow *window)
|
|
{
|
|
Display *display;
|
|
GdkWindowObject *private;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
display = GDK_WINDOW_XDISPLAY (window);
|
|
|
|
private = (GdkWindowObject*) window;
|
|
|
|
if (GDK_WINDOW_IS_MAPPED (window))
|
|
{
|
|
gdk_window_show (window);
|
|
}
|
|
else
|
|
{
|
|
/* Flip our client side flag, the real work happens on map. */
|
|
gdk_synthesize_window_state (window,
|
|
GDK_WINDOW_STATE_ICONIFIED,
|
|
0);
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_stick (GdkWindow *window)
|
|
{
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
if (GDK_WINDOW_IS_MAPPED (window))
|
|
{
|
|
/* "stick" means stick to all desktops _and_ do not scroll with the
|
|
* viewport. i.e. glue to the monitor glass in all cases.
|
|
*/
|
|
|
|
XEvent xev;
|
|
|
|
/* Request stick during viewport scroll */
|
|
gdk_wmspec_change_state (TRUE, window,
|
|
gdk_atom_intern ("_NET_WM_STATE_STICKY", FALSE),
|
|
0);
|
|
|
|
/* Request desktop 0xFFFFFFFF */
|
|
xev.xclient.type = ClientMessage;
|
|
xev.xclient.serial = 0;
|
|
xev.xclient.send_event = True;
|
|
xev.xclient.window = GDK_WINDOW_XWINDOW (window);
|
|
xev.xclient.display = gdk_display;
|
|
xev.xclient.message_type = gdk_atom_intern ("_NET_WM_DESKTOP", FALSE);
|
|
xev.xclient.format = 32;
|
|
|
|
xev.xclient.data.l[0] = 0xFFFFFFFF;
|
|
|
|
XSendEvent (gdk_display, gdk_root_window, False,
|
|
SubstructureRedirectMask | SubstructureNotifyMask,
|
|
&xev);
|
|
}
|
|
else
|
|
{
|
|
/* Flip our client side flag, the real work happens on map. */
|
|
gdk_synthesize_window_state (window,
|
|
0,
|
|
GDK_WINDOW_STATE_STICKY);
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_unstick (GdkWindow *window)
|
|
{
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
if (GDK_WINDOW_IS_MAPPED (window))
|
|
{
|
|
XEvent xev;
|
|
Atom type;
|
|
gint format;
|
|
gulong nitems;
|
|
gulong bytes_after;
|
|
gulong *current_desktop;
|
|
|
|
/* Request unstick from viewport */
|
|
gdk_wmspec_change_state (FALSE, window,
|
|
gdk_atom_intern ("_NET_WM_STATE_STICKY", FALSE),
|
|
0);
|
|
|
|
/* Get current desktop, then set it; this is a race, but not
|
|
* one that matters much in practice.
|
|
*/
|
|
XGetWindowProperty (gdk_display, gdk_root_window,
|
|
gdk_atom_intern ("_NET_CURRENT_DESKTOP", FALSE),
|
|
0, G_MAXLONG,
|
|
False, XA_CARDINAL, &type, &format, &nitems,
|
|
&bytes_after, (guchar **)¤t_desktop);
|
|
|
|
if (type == XA_CARDINAL)
|
|
{
|
|
xev.xclient.type = ClientMessage;
|
|
xev.xclient.serial = 0;
|
|
xev.xclient.send_event = True;
|
|
xev.xclient.window = GDK_WINDOW_XWINDOW (window);
|
|
xev.xclient.display = gdk_display;
|
|
xev.xclient.message_type = gdk_atom_intern ("_NET_WM_DESKTOP", FALSE);
|
|
xev.xclient.format = 32;
|
|
|
|
xev.xclient.data.l[0] = *current_desktop;
|
|
|
|
XSendEvent (gdk_display, gdk_root_window, False,
|
|
SubstructureRedirectMask | SubstructureNotifyMask,
|
|
&xev);
|
|
|
|
XFree (current_desktop);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Flip our client side flag, the real work happens on map. */
|
|
gdk_synthesize_window_state (window,
|
|
GDK_WINDOW_STATE_STICKY,
|
|
0);
|
|
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_window_maximize (GdkWindow *window)
|
|
{
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
if (GDK_WINDOW_IS_MAPPED (window))
|
|
gdk_wmspec_change_state (TRUE, window,
|
|
gdk_atom_intern ("_NET_WM_STATE_MAXIMIZED_VERT", FALSE),
|
|
gdk_atom_intern ("_NET_WM_STATE_MAXIMIZED_HORZ", FALSE));
|
|
else
|
|
gdk_synthesize_window_state (window,
|
|
0,
|
|
GDK_WINDOW_STATE_MAXIMIZED);
|
|
}
|
|
|
|
void
|
|
gdk_window_unmaximize (GdkWindow *window)
|
|
{
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
if (GDK_WINDOW_IS_MAPPED (window))
|
|
gdk_wmspec_change_state (FALSE, window,
|
|
gdk_atom_intern ("_NET_WM_STATE_MAXIMIZED_VERT", FALSE),
|
|
gdk_atom_intern ("_NET_WM_STATE_MAXIMIZED_HORZ", FALSE));
|
|
else
|
|
gdk_synthesize_window_state (window,
|
|
GDK_WINDOW_STATE_MAXIMIZED,
|
|
0);
|
|
}
|
|
|
|
void
|
|
gdk_window_set_group (GdkWindow *window,
|
|
GdkWindow *leader)
|
|
{
|
|
XWMHints *wm_hints;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
g_return_if_fail (leader != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (leader));
|
|
|
|
if (GDK_WINDOW_DESTROYED (window) || GDK_WINDOW_DESTROYED (leader))
|
|
return;
|
|
|
|
wm_hints = XGetWMHints (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window));
|
|
if (!wm_hints)
|
|
wm_hints = XAllocWMHints ();
|
|
|
|
wm_hints->flags |= WindowGroupHint;
|
|
wm_hints->window_group = GDK_WINDOW_XID (leader);
|
|
|
|
XSetWMHints (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window), wm_hints);
|
|
XFree (wm_hints);
|
|
}
|
|
|
|
static MotifWmHints *
|
|
gdk_window_get_mwm_hints (GdkWindow *window)
|
|
{
|
|
static Atom hints_atom = None;
|
|
MotifWmHints *hints;
|
|
Atom type;
|
|
gint format;
|
|
gulong nitems;
|
|
gulong bytes_after;
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return NULL;
|
|
|
|
if (!hints_atom)
|
|
hints_atom = XInternAtom (GDK_WINDOW_XDISPLAY (window),
|
|
_XA_MOTIF_WM_HINTS, FALSE);
|
|
|
|
XGetWindowProperty (GDK_WINDOW_XDISPLAY (window), GDK_WINDOW_XID (window),
|
|
hints_atom, 0, sizeof (MotifWmHints)/sizeof (long),
|
|
False, AnyPropertyType, &type, &format, &nitems,
|
|
&bytes_after, (guchar **)&hints);
|
|
|
|
if (type == None)
|
|
return NULL;
|
|
|
|
return hints;
|
|
}
|
|
|
|
static void
|
|
gdk_window_set_mwm_hints (GdkWindow *window,
|
|
MotifWmHints *new_hints)
|
|
{
|
|
static Atom hints_atom = None;
|
|
MotifWmHints *hints;
|
|
Atom type;
|
|
gint format;
|
|
gulong nitems;
|
|
gulong bytes_after;
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
if (!hints_atom)
|
|
hints_atom = XInternAtom (GDK_WINDOW_XDISPLAY (window),
|
|
_XA_MOTIF_WM_HINTS, FALSE);
|
|
|
|
XGetWindowProperty (GDK_WINDOW_XDISPLAY (window), GDK_WINDOW_XID (window),
|
|
hints_atom, 0, sizeof (MotifWmHints)/sizeof (long),
|
|
False, AnyPropertyType, &type, &format, &nitems,
|
|
&bytes_after, (guchar **)&hints);
|
|
|
|
if (type == None)
|
|
hints = new_hints;
|
|
else
|
|
{
|
|
if (new_hints->flags & MWM_HINTS_FUNCTIONS)
|
|
{
|
|
hints->flags |= MWM_HINTS_FUNCTIONS;
|
|
hints->functions = new_hints->functions;
|
|
}
|
|
if (new_hints->flags & MWM_HINTS_DECORATIONS)
|
|
{
|
|
hints->flags |= MWM_HINTS_DECORATIONS;
|
|
hints->decorations = new_hints->decorations;
|
|
}
|
|
}
|
|
|
|
XChangeProperty (GDK_WINDOW_XDISPLAY (window), GDK_WINDOW_XID (window),
|
|
hints_atom, hints_atom, 32, PropModeReplace,
|
|
(guchar *)hints, sizeof (MotifWmHints)/sizeof (long));
|
|
|
|
if (hints != new_hints)
|
|
XFree (hints);
|
|
}
|
|
|
|
void
|
|
gdk_window_set_decorations (GdkWindow *window,
|
|
GdkWMDecoration decorations)
|
|
{
|
|
MotifWmHints hints;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
hints.flags = MWM_HINTS_DECORATIONS;
|
|
hints.decorations = decorations;
|
|
|
|
gdk_window_set_mwm_hints (window, &hints);
|
|
}
|
|
|
|
/**
|
|
* gdk_window_get_decorations:
|
|
* @window: The #GdkWindow to get the decorations from
|
|
* @decorations: The window decorations will be written here
|
|
*
|
|
* Returns the decorations set on the GdkWindow with #gdk_window_set_decorations
|
|
* Returns: TRUE if the window has decorations set, FALSE otherwise.
|
|
**/
|
|
gboolean
|
|
gdk_window_get_decorations(GdkWindow *window,
|
|
GdkWMDecoration *decorations)
|
|
{
|
|
MotifWmHints *hints;
|
|
gboolean result = FALSE;
|
|
|
|
hints = gdk_window_get_mwm_hints (window);
|
|
|
|
if (hints)
|
|
{
|
|
if (hints->flags & MWM_HINTS_DECORATIONS)
|
|
{
|
|
*decorations = hints->decorations;
|
|
result = TRUE;
|
|
}
|
|
|
|
XFree (hints);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
void
|
|
gdk_window_set_functions (GdkWindow *window,
|
|
GdkWMFunction functions)
|
|
{
|
|
MotifWmHints hints;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
hints.flags = MWM_HINTS_FUNCTIONS;
|
|
hints.functions = functions;
|
|
|
|
gdk_window_set_mwm_hints (window, &hints);
|
|
}
|
|
|
|
#ifdef HAVE_SHAPE_EXT
|
|
|
|
/*
|
|
* propagate the shapes from all child windows of a GDK window to the parent
|
|
* window. Shamelessly ripped from Enlightenment's code
|
|
*
|
|
* - Raster
|
|
*/
|
|
struct _gdk_span
|
|
{
|
|
gint start;
|
|
gint end;
|
|
struct _gdk_span *next;
|
|
};
|
|
|
|
static void
|
|
gdk_add_to_span (struct _gdk_span **s,
|
|
gint x,
|
|
gint xx)
|
|
{
|
|
struct _gdk_span *ptr1, *ptr2, *noo, *ss;
|
|
gchar spanning;
|
|
|
|
ptr2 = NULL;
|
|
ptr1 = *s;
|
|
spanning = 0;
|
|
ss = NULL;
|
|
/* scan the spans for this line */
|
|
while (ptr1)
|
|
{
|
|
/* -- -> new span */
|
|
/* == -> existing span */
|
|
/* ## -> spans intersect */
|
|
/* if we are in the middle of spanning the span into the line */
|
|
if (spanning)
|
|
{
|
|
/* case: ---- ==== */
|
|
if (xx < ptr1->start - 1)
|
|
{
|
|
/* ends before next span - extend to here */
|
|
ss->end = xx;
|
|
return;
|
|
}
|
|
/* case: ----##=== */
|
|
else if (xx <= ptr1->end)
|
|
{
|
|
/* crosses into next span - delete next span and append */
|
|
ss->end = ptr1->end;
|
|
ss->next = ptr1->next;
|
|
g_free (ptr1);
|
|
return;
|
|
}
|
|
/* case: ---###--- */
|
|
else
|
|
{
|
|
/* overlaps next span - delete and keep checking */
|
|
ss->next = ptr1->next;
|
|
g_free (ptr1);
|
|
ptr1 = ss;
|
|
}
|
|
}
|
|
/* otherwise havent started spanning it in yet */
|
|
else
|
|
{
|
|
/* case: ---- ==== */
|
|
if (xx < ptr1->start - 1)
|
|
{
|
|
/* insert span here in list */
|
|
noo = g_malloc (sizeof (struct _gdk_span));
|
|
|
|
if (noo)
|
|
{
|
|
noo->start = x;
|
|
noo->end = xx;
|
|
noo->next = ptr1;
|
|
if (ptr2)
|
|
ptr2->next = noo;
|
|
else
|
|
*s = noo;
|
|
}
|
|
return;
|
|
}
|
|
/* case: ----##=== */
|
|
else if ((x < ptr1->start) && (xx <= ptr1->end))
|
|
{
|
|
/* expand this span to the left point of the new one */
|
|
ptr1->start = x;
|
|
return;
|
|
}
|
|
/* case: ===###=== */
|
|
else if ((x >= ptr1->start) && (xx <= ptr1->end))
|
|
{
|
|
/* throw the span away */
|
|
return;
|
|
}
|
|
/* case: ---###--- */
|
|
else if ((x < ptr1->start) && (xx > ptr1->end))
|
|
{
|
|
ss = ptr1;
|
|
spanning = 1;
|
|
ptr1->start = x;
|
|
ptr1->end = xx;
|
|
}
|
|
/* case: ===##---- */
|
|
else if ((x >= ptr1->start) && (x <= ptr1->end + 1) && (xx > ptr1->end))
|
|
{
|
|
ss = ptr1;
|
|
spanning = 1;
|
|
ptr1->end = xx;
|
|
}
|
|
/* case: ==== ---- */
|
|
/* case handled by next loop iteration - first case */
|
|
}
|
|
ptr2 = ptr1;
|
|
ptr1 = ptr1->next;
|
|
}
|
|
/* it started in the middle but spans beyond your current list */
|
|
if (spanning)
|
|
{
|
|
ptr2->end = xx;
|
|
return;
|
|
}
|
|
/* it does not start inside a span or in the middle, so add it to the end */
|
|
noo = g_malloc (sizeof (struct _gdk_span));
|
|
|
|
if (noo)
|
|
{
|
|
noo->start = x;
|
|
noo->end = xx;
|
|
if (ptr2)
|
|
{
|
|
noo->next = ptr2->next;
|
|
ptr2->next = noo;
|
|
}
|
|
else
|
|
{
|
|
noo->next = NULL;
|
|
*s = noo;
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
static void
|
|
gdk_add_rectangles (Display *disp,
|
|
Window win,
|
|
struct _gdk_span **spans,
|
|
gint basew,
|
|
gint baseh,
|
|
gint x,
|
|
gint y)
|
|
{
|
|
gint a, k;
|
|
gint x1, y1, x2, y2;
|
|
gint rn, ord;
|
|
XRectangle *rl;
|
|
|
|
rl = XShapeGetRectangles (disp, win, ShapeBounding, &rn, &ord);
|
|
if (rl)
|
|
{
|
|
/* go through all clip rects in this window's shape */
|
|
for (k = 0; k < rn; k++)
|
|
{
|
|
/* for each clip rect, add it to each line's spans */
|
|
x1 = x + rl[k].x;
|
|
x2 = x + rl[k].x + (rl[k].width - 1);
|
|
y1 = y + rl[k].y;
|
|
y2 = y + rl[k].y + (rl[k].height - 1);
|
|
if (x1 < 0)
|
|
x1 = 0;
|
|
if (y1 < 0)
|
|
y1 = 0;
|
|
if (x2 >= basew)
|
|
x2 = basew - 1;
|
|
if (y2 >= baseh)
|
|
y2 = baseh - 1;
|
|
for (a = y1; a <= y2; a++)
|
|
{
|
|
if ((x2 - x1) >= 0)
|
|
gdk_add_to_span (&spans[a], x1, x2);
|
|
}
|
|
}
|
|
XFree (rl);
|
|
}
|
|
}
|
|
|
|
static void
|
|
gdk_propagate_shapes (Display *disp,
|
|
Window win,
|
|
gboolean merge)
|
|
{
|
|
Window rt, par, *list = NULL;
|
|
gint i, j, num = 0, num_rects = 0;
|
|
gint x, y, contig;
|
|
guint w, h, d;
|
|
gint baseh, basew;
|
|
XRectangle *rects = NULL;
|
|
struct _gdk_span **spans = NULL, *ptr1, *ptr2, *ptr3;
|
|
XWindowAttributes xatt;
|
|
|
|
XGetGeometry (disp, win, &rt, &x, &y, &w, &h, &d, &d);
|
|
if (h <= 0)
|
|
return;
|
|
basew = w;
|
|
baseh = h;
|
|
spans = g_malloc (sizeof (struct _gdk_span *) * h);
|
|
|
|
for (i = 0; i < h; i++)
|
|
spans[i] = NULL;
|
|
XQueryTree (disp, win, &rt, &par, &list, (unsigned int *)&num);
|
|
if (list)
|
|
{
|
|
/* go through all child windows and create/insert spans */
|
|
for (i = 0; i < num; i++)
|
|
{
|
|
if (XGetWindowAttributes (disp, list[i], &xatt) && (xatt.map_state != IsUnmapped))
|
|
if (XGetGeometry (disp, list[i], &rt, &x, &y, &w, &h, &d, &d))
|
|
gdk_add_rectangles (disp, list[i], spans, basew, baseh, x, y);
|
|
}
|
|
if (merge)
|
|
gdk_add_rectangles (disp, win, spans, basew, baseh, x, y);
|
|
|
|
/* go through the spans list and build a list of rects */
|
|
rects = g_malloc (sizeof (XRectangle) * 256);
|
|
num_rects = 0;
|
|
for (i = 0; i < baseh; i++)
|
|
{
|
|
ptr1 = spans[i];
|
|
/* go through the line for all spans */
|
|
while (ptr1)
|
|
{
|
|
rects[num_rects].x = ptr1->start;
|
|
rects[num_rects].y = i;
|
|
rects[num_rects].width = ptr1->end - ptr1->start + 1;
|
|
rects[num_rects].height = 1;
|
|
j = i + 1;
|
|
/* if there are more lines */
|
|
contig = 1;
|
|
/* while contigous rects (same start/end coords) exist */
|
|
while ((contig) && (j < baseh))
|
|
{
|
|
/* search next line for spans matching this one */
|
|
contig = 0;
|
|
ptr2 = spans[j];
|
|
ptr3 = NULL;
|
|
while (ptr2)
|
|
{
|
|
/* if we have an exact span match set contig */
|
|
if ((ptr2->start == ptr1->start) &&
|
|
(ptr2->end == ptr1->end))
|
|
{
|
|
contig = 1;
|
|
/* remove the span - not needed */
|
|
if (ptr3)
|
|
{
|
|
ptr3->next = ptr2->next;
|
|
g_free (ptr2);
|
|
ptr2 = NULL;
|
|
}
|
|
else
|
|
{
|
|
spans[j] = ptr2->next;
|
|
g_free (ptr2);
|
|
ptr2 = NULL;
|
|
}
|
|
break;
|
|
}
|
|
/* gone past the span point no point looking */
|
|
else if (ptr2->start < ptr1->start)
|
|
break;
|
|
if (ptr2)
|
|
{
|
|
ptr3 = ptr2;
|
|
ptr2 = ptr2->next;
|
|
}
|
|
}
|
|
/* if a contiguous span was found increase the rect h */
|
|
if (contig)
|
|
{
|
|
rects[num_rects].height++;
|
|
j++;
|
|
}
|
|
}
|
|
/* up the rect count */
|
|
num_rects++;
|
|
/* every 256 new rects increase the rect array */
|
|
if ((num_rects % 256) == 0)
|
|
rects = g_realloc (rects, sizeof (XRectangle) * (num_rects + 256));
|
|
ptr1 = ptr1->next;
|
|
}
|
|
}
|
|
/* set the rects as the shape mask */
|
|
if (rects)
|
|
{
|
|
XShapeCombineRectangles (disp, win, ShapeBounding, 0, 0, rects, num_rects,
|
|
ShapeSet, YXSorted);
|
|
g_free (rects);
|
|
}
|
|
XFree (list);
|
|
}
|
|
/* free up all the spans we made */
|
|
for (i = 0; i < baseh; i++)
|
|
{
|
|
ptr1 = spans[i];
|
|
while (ptr1)
|
|
{
|
|
ptr2 = ptr1;
|
|
ptr1 = ptr1->next;
|
|
g_free (ptr2);
|
|
}
|
|
}
|
|
g_free (spans);
|
|
}
|
|
|
|
#endif /* HAVE_SHAPE_EXT */
|
|
|
|
void
|
|
gdk_window_set_child_shapes (GdkWindow *window)
|
|
{
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
#ifdef HAVE_SHAPE_EXT
|
|
if (!GDK_WINDOW_DESTROYED (window) &&
|
|
gdk_window_have_shape_ext ())
|
|
gdk_propagate_shapes (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window), FALSE);
|
|
#endif
|
|
}
|
|
|
|
void
|
|
gdk_window_merge_child_shapes (GdkWindow *window)
|
|
{
|
|
g_return_if_fail (window != NULL);
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
|
|
#ifdef HAVE_SHAPE_EXT
|
|
if (!GDK_WINDOW_DESTROYED (window) &&
|
|
gdk_window_have_shape_ext ())
|
|
gdk_propagate_shapes (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window), TRUE);
|
|
#endif
|
|
}
|
|
|
|
/* Support for windows that can be guffaw-scrolled
|
|
* (See http://www.gtk.org/~otaylor/whitepapers/guffaw-scrolling.txt)
|
|
*/
|
|
|
|
static gboolean
|
|
gdk_window_gravity_works (void)
|
|
{
|
|
enum { UNKNOWN, NO, YES };
|
|
static gint gravity_works = UNKNOWN;
|
|
|
|
if (gravity_works == UNKNOWN)
|
|
{
|
|
GdkWindowAttr attr;
|
|
GdkWindow *parent;
|
|
GdkWindow *child;
|
|
gint y;
|
|
|
|
/* This particular server apparently has a bug so that the test
|
|
* works but the actual code crashes it
|
|
*/
|
|
if ((!strcmp (XServerVendor (gdk_display), "Sun Microsystems, Inc.")) &&
|
|
(VendorRelease (gdk_display) == 3400))
|
|
{
|
|
gravity_works = NO;
|
|
return FALSE;
|
|
}
|
|
|
|
attr.window_type = GDK_WINDOW_TEMP;
|
|
attr.wclass = GDK_INPUT_OUTPUT;
|
|
attr.x = 0;
|
|
attr.y = 0;
|
|
attr.width = 100;
|
|
attr.height = 100;
|
|
attr.event_mask = 0;
|
|
|
|
parent = gdk_window_new (NULL, &attr, GDK_WA_X | GDK_WA_Y);
|
|
|
|
attr.window_type = GDK_WINDOW_CHILD;
|
|
child = gdk_window_new (parent, &attr, GDK_WA_X | GDK_WA_Y);
|
|
|
|
gdk_window_set_static_win_gravity (child, TRUE);
|
|
|
|
gdk_window_resize (parent, 100, 110);
|
|
gdk_window_move (parent, 0, -10);
|
|
gdk_window_move_resize (parent, 0, 0, 100, 100);
|
|
|
|
gdk_window_resize (parent, 100, 110);
|
|
gdk_window_move (parent, 0, -10);
|
|
gdk_window_move_resize (parent, 0, 0, 100, 100);
|
|
|
|
gdk_window_get_geometry (child, NULL, &y, NULL, NULL, NULL);
|
|
|
|
gdk_window_destroy (parent);
|
|
gdk_window_destroy (child);
|
|
|
|
gravity_works = ((y == -20) ? YES : NO);
|
|
}
|
|
|
|
return (gravity_works == YES);
|
|
}
|
|
|
|
static void
|
|
gdk_window_set_static_bit_gravity (GdkWindow *window, gboolean on)
|
|
{
|
|
XSetWindowAttributes xattributes;
|
|
guint xattributes_mask = 0;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
|
|
xattributes.bit_gravity = StaticGravity;
|
|
xattributes_mask |= CWBitGravity;
|
|
xattributes.bit_gravity = on ? StaticGravity : ForgetGravity;
|
|
XChangeWindowAttributes (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
CWBitGravity, &xattributes);
|
|
}
|
|
|
|
static void
|
|
gdk_window_set_static_win_gravity (GdkWindow *window, gboolean on)
|
|
{
|
|
XSetWindowAttributes xattributes;
|
|
|
|
g_return_if_fail (window != NULL);
|
|
|
|
xattributes.win_gravity = on ? StaticGravity : NorthWestGravity;
|
|
|
|
XChangeWindowAttributes (GDK_WINDOW_XDISPLAY (window),
|
|
GDK_WINDOW_XID (window),
|
|
CWWinGravity, &xattributes);
|
|
}
|
|
|
|
/*************************************************************
|
|
* gdk_window_set_static_gravities:
|
|
* Set the bit gravity of the given window to static,
|
|
* and flag it so all children get static subwindow
|
|
* gravity.
|
|
* arguments:
|
|
* window: window for which to set static gravity
|
|
* use_static: Whether to turn static gravity on or off.
|
|
* results:
|
|
* Does the XServer support static gravity?
|
|
*************************************************************/
|
|
|
|
gboolean
|
|
gdk_window_set_static_gravities (GdkWindow *window,
|
|
gboolean use_static)
|
|
{
|
|
GdkWindowObject *private = (GdkWindowObject *)window;
|
|
GList *tmp_list;
|
|
|
|
g_return_val_if_fail (window != NULL, FALSE);
|
|
g_return_val_if_fail (GDK_IS_WINDOW (window), FALSE);
|
|
|
|
if (!use_static == !private->guffaw_gravity)
|
|
return TRUE;
|
|
|
|
if (use_static && !gdk_window_gravity_works ())
|
|
return FALSE;
|
|
|
|
private->guffaw_gravity = use_static;
|
|
|
|
if (!GDK_WINDOW_DESTROYED (window))
|
|
{
|
|
gdk_window_set_static_bit_gravity (window, use_static);
|
|
|
|
tmp_list = private->children;
|
|
while (tmp_list)
|
|
{
|
|
gdk_window_set_static_win_gravity (window, use_static);
|
|
|
|
tmp_list = tmp_list->next;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* internal function created for and used by gdk_window_xid_at_coords */
|
|
Window
|
|
gdk_window_xid_at (Window base,
|
|
gint bx,
|
|
gint by,
|
|
gint x,
|
|
gint y,
|
|
GList *excludes,
|
|
gboolean excl_child)
|
|
{
|
|
Display *xdisplay;
|
|
Window *list = NULL;
|
|
Window child = 0, parent_win = 0, root_win = 0;
|
|
int i;
|
|
unsigned int ww, wh, wb, wd, num;
|
|
int wx, wy;
|
|
|
|
xdisplay = GDK_DISPLAY ();
|
|
if (!XGetGeometry (xdisplay, base, &root_win, &wx, &wy, &ww, &wh, &wb, &wd))
|
|
return 0;
|
|
wx += bx;
|
|
wy += by;
|
|
|
|
if (!((x >= wx) &&
|
|
(y >= wy) &&
|
|
(x < (int) (wx + ww)) &&
|
|
(y < (int) (wy + wh))))
|
|
return 0;
|
|
|
|
if (!XQueryTree (xdisplay, base, &root_win, &parent_win, &list, &num))
|
|
return base;
|
|
|
|
if (list)
|
|
{
|
|
for (i = num - 1; ; i--)
|
|
{
|
|
if ((!excl_child) || (!g_list_find (excludes, (gpointer *) list[i])))
|
|
{
|
|
if ((child = gdk_window_xid_at (list[i], wx, wy, x, y, excludes, excl_child)) != 0)
|
|
{
|
|
XFree (list);
|
|
return child;
|
|
}
|
|
}
|
|
if (!i)
|
|
break;
|
|
}
|
|
XFree (list);
|
|
}
|
|
return base;
|
|
}
|
|
|
|
/*
|
|
* The following fucntion by The Rasterman <raster@redhat.com>
|
|
* This function returns the X Window ID in which the x y location is in
|
|
* (x and y being relative to the root window), excluding any windows listed
|
|
* in the GList excludes (this is a list of X Window ID's - gpointer being
|
|
* the Window ID).
|
|
*
|
|
* This is primarily designed for internal gdk use - for DND for example
|
|
* when using a shaped icon window as the drag object - you exclude the
|
|
* X Window ID of the "icon" (perhaps more if excludes may be needed) and
|
|
* You can get back an X Window ID as to what X Window ID is infact under
|
|
* those X,Y co-ordinates.
|
|
*/
|
|
Window
|
|
gdk_window_xid_at_coords (gint x,
|
|
gint y,
|
|
GList *excludes,
|
|
gboolean excl_child)
|
|
{
|
|
GdkWindow *window;
|
|
Display *xdisplay;
|
|
Window *list = NULL;
|
|
Window root, child = 0, parent_win = 0, root_win = 0;
|
|
unsigned int num;
|
|
int i;
|
|
|
|
window = gdk_parent_root;
|
|
xdisplay = GDK_WINDOW_XDISPLAY (window);
|
|
root = GDK_WINDOW_XID (window);
|
|
num = g_list_length (excludes);
|
|
|
|
XGrabServer (xdisplay);
|
|
if (!XQueryTree (xdisplay, root, &root_win, &parent_win, &list, &num))
|
|
{
|
|
XUngrabServer (xdisplay);
|
|
return root;
|
|
}
|
|
if (list)
|
|
{
|
|
i = num - 1;
|
|
do
|
|
{
|
|
XWindowAttributes xwa;
|
|
|
|
XGetWindowAttributes (xdisplay, list [i], &xwa);
|
|
|
|
if (xwa.map_state != IsViewable)
|
|
continue;
|
|
|
|
if (excl_child && g_list_find (excludes, (gpointer *) list[i]))
|
|
continue;
|
|
|
|
if ((child = gdk_window_xid_at (list[i], 0, 0, x, y, excludes, excl_child)) == 0)
|
|
continue;
|
|
|
|
if (excludes)
|
|
{
|
|
if (!g_list_find (excludes, (gpointer *) child))
|
|
{
|
|
XFree (list);
|
|
XUngrabServer (xdisplay);
|
|
return child;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
XFree (list);
|
|
XUngrabServer (xdisplay);
|
|
return child;
|
|
}
|
|
} while (--i > 0);
|
|
XFree (list);
|
|
}
|
|
XUngrabServer (xdisplay);
|
|
return root;
|
|
}
|
|
|
|
static void
|
|
wmspec_moveresize (GdkWindow *window,
|
|
gint direction,
|
|
gint root_x,
|
|
gint root_y,
|
|
guint32 timestamp)
|
|
{
|
|
XEvent xev;
|
|
|
|
/* Release passive grab */
|
|
gdk_pointer_ungrab (timestamp);
|
|
|
|
xev.xclient.type = ClientMessage;
|
|
xev.xclient.serial = 0;
|
|
xev.xclient.send_event = True;
|
|
xev.xclient.display = gdk_display;
|
|
xev.xclient.window = GDK_WINDOW_XID (window);
|
|
xev.xclient.message_type = gdk_atom_intern ("_NET_WM_MOVERESIZE", FALSE);
|
|
xev.xclient.format = 32;
|
|
xev.xclient.data.l[0] = root_x;
|
|
xev.xclient.data.l[1] = root_y;
|
|
xev.xclient.data.l[2] = direction;
|
|
xev.xclient.data.l[3] = 0;
|
|
xev.xclient.data.l[4] = 0;
|
|
|
|
XSendEvent (gdk_display, gdk_root_window, False,
|
|
SubstructureRedirectMask | SubstructureNotifyMask,
|
|
&xev);
|
|
}
|
|
|
|
/* From the WM spec */
|
|
#define _NET_WM_MOVERESIZE_SIZE_TOPLEFT 0
|
|
#define _NET_WM_MOVERESIZE_SIZE_TOP 1
|
|
#define _NET_WM_MOVERESIZE_SIZE_TOPRIGHT 2
|
|
#define _NET_WM_MOVERESIZE_SIZE_RIGHT 3
|
|
#define _NET_WM_MOVERESIZE_SIZE_BOTTOMRIGHT 4
|
|
#define _NET_WM_MOVERESIZE_SIZE_BOTTOM 5
|
|
#define _NET_WM_MOVERESIZE_SIZE_BOTTOMLEFT 6
|
|
#define _NET_WM_MOVERESIZE_SIZE_LEFT 7
|
|
#define _NET_WM_MOVERESIZE_MOVE 8
|
|
|
|
static void
|
|
wmspec_resize_drag (GdkWindow *window,
|
|
GdkWindowEdge edge,
|
|
gint button,
|
|
gint root_x,
|
|
gint root_y,
|
|
guint32 timestamp)
|
|
{
|
|
gint direction;
|
|
|
|
/* Let the compiler turn a switch into a table, instead
|
|
* of doing the table manually, this way is easier to verify.
|
|
*/
|
|
switch (edge)
|
|
{
|
|
case GDK_WINDOW_EDGE_NORTH_WEST:
|
|
direction = _NET_WM_MOVERESIZE_SIZE_TOPLEFT;
|
|
break;
|
|
|
|
case GDK_WINDOW_EDGE_NORTH:
|
|
direction = _NET_WM_MOVERESIZE_SIZE_TOP;
|
|
break;
|
|
|
|
case GDK_WINDOW_EDGE_NORTH_EAST:
|
|
direction = _NET_WM_MOVERESIZE_SIZE_TOPRIGHT;
|
|
break;
|
|
|
|
case GDK_WINDOW_EDGE_WEST:
|
|
direction = _NET_WM_MOVERESIZE_SIZE_LEFT;
|
|
break;
|
|
|
|
case GDK_WINDOW_EDGE_EAST:
|
|
direction = _NET_WM_MOVERESIZE_SIZE_RIGHT;
|
|
break;
|
|
|
|
case GDK_WINDOW_EDGE_SOUTH_WEST:
|
|
direction = _NET_WM_MOVERESIZE_SIZE_BOTTOMLEFT;
|
|
break;
|
|
|
|
case GDK_WINDOW_EDGE_SOUTH:
|
|
direction = _NET_WM_MOVERESIZE_SIZE_BOTTOM;
|
|
break;
|
|
|
|
case GDK_WINDOW_EDGE_SOUTH_EAST:
|
|
direction = _NET_WM_MOVERESIZE_SIZE_BOTTOMRIGHT;
|
|
break;
|
|
|
|
default:
|
|
g_warning ("gdk_window_begin_resize_drag: bad resize edge %d!",
|
|
edge);
|
|
return;
|
|
break;
|
|
}
|
|
|
|
wmspec_moveresize (window, direction, root_x, root_y, timestamp);
|
|
}
|
|
|
|
/* This is global for use in gdkevents-x11.c */
|
|
GdkWindow *_gdk_moveresize_window;
|
|
|
|
static GdkWindow *moveresize_emulation_window = NULL;
|
|
static gboolean is_resize = FALSE;
|
|
static GdkWindowEdge resize_edge;
|
|
static gint moveresize_button;
|
|
static gint moveresize_x;
|
|
static gint moveresize_y;
|
|
static gint moveresize_orig_x;
|
|
static gint moveresize_orig_y;
|
|
static gint moveresize_orig_width;
|
|
static gint moveresize_orig_height;
|
|
static GdkWindowHints moveresize_geom_mask = 0;
|
|
static GdkGeometry moveresize_geometry;
|
|
static Time moveresize_process_time;
|
|
|
|
static XEvent *moveresize_pending_event;
|
|
|
|
static void
|
|
update_pos (gint new_root_x,
|
|
gint new_root_y)
|
|
{
|
|
gint dx, dy;
|
|
|
|
dx = new_root_x - moveresize_x;
|
|
dy = new_root_y - moveresize_y;
|
|
|
|
if (is_resize)
|
|
{
|
|
gint w, h;
|
|
|
|
w = moveresize_orig_width;
|
|
h = moveresize_orig_height;
|
|
|
|
switch (resize_edge)
|
|
{
|
|
case GDK_WINDOW_EDGE_SOUTH_EAST:
|
|
w += dx;
|
|
h += dy;
|
|
break;
|
|
}
|
|
|
|
w = MAX (w, 1);
|
|
h = MAX (h, 1);
|
|
|
|
if (moveresize_geom_mask)
|
|
{
|
|
gdk_window_constrain_size (&moveresize_geometry,
|
|
moveresize_geom_mask,
|
|
w, h,
|
|
&w, &h);
|
|
}
|
|
|
|
gdk_window_resize (_gdk_moveresize_window, w, h);
|
|
}
|
|
else
|
|
{
|
|
gint x, y;
|
|
|
|
x = moveresize_orig_x + dx;
|
|
y = moveresize_orig_y + dy;
|
|
|
|
gdk_window_move (_gdk_moveresize_window, x, y);
|
|
}
|
|
}
|
|
|
|
static void
|
|
finish_drag (void)
|
|
{
|
|
gdk_window_destroy (moveresize_emulation_window);
|
|
moveresize_emulation_window = NULL;
|
|
_gdk_moveresize_window = NULL;
|
|
|
|
if (moveresize_pending_event)
|
|
{
|
|
g_free (moveresize_pending_event);
|
|
moveresize_pending_event = NULL;
|
|
}
|
|
}
|
|
|
|
static int
|
|
lookahead_motion_predicate (Display *display,
|
|
XEvent *event,
|
|
XPointer arg)
|
|
{
|
|
gboolean *seen_release = (gboolean *)arg;
|
|
|
|
if (*seen_release)
|
|
return False;
|
|
|
|
switch (event->xany.type)
|
|
{
|
|
case ButtonRelease:
|
|
*seen_release = TRUE;
|
|
break;
|
|
case MotionNotify:
|
|
moveresize_process_time = event->xmotion.time;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return False;
|
|
}
|
|
|
|
static gboolean
|
|
moveresize_lookahead (XEvent *event)
|
|
{
|
|
XEvent tmp_event;
|
|
gboolean seen_release = FALSE;
|
|
|
|
if (moveresize_process_time)
|
|
{
|
|
if (event->xmotion.time == moveresize_process_time)
|
|
{
|
|
moveresize_process_time = 0;
|
|
return TRUE;
|
|
}
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
XCheckIfEvent (gdk_display, &tmp_event,
|
|
lookahead_motion_predicate, (XPointer)&seen_release);
|
|
|
|
return moveresize_process_time == 0;
|
|
}
|
|
|
|
void
|
|
_gdk_moveresize_handle_event (XEvent *event)
|
|
{
|
|
guint button_mask = 0;
|
|
GdkWindowObject *window_private = (GdkWindowObject *) _gdk_moveresize_window;
|
|
|
|
button_mask = GDK_BUTTON1_MASK << (moveresize_button - 1);
|
|
|
|
switch (event->xany.type)
|
|
{
|
|
case MotionNotify:
|
|
if (window_private->resize_count > 0)
|
|
{
|
|
if (moveresize_pending_event)
|
|
*moveresize_pending_event = *event;
|
|
else
|
|
moveresize_pending_event = g_memdup (event, sizeof (XEvent));
|
|
|
|
break;
|
|
}
|
|
if (!moveresize_lookahead (event))
|
|
break;
|
|
|
|
update_pos (event->xmotion.x_root,
|
|
event->xmotion.y_root);
|
|
|
|
/* This should never be triggered in normal cases, but in the
|
|
* case where the drag started without an implicit grab being
|
|
* in effect, we could miss the release if it occurs before
|
|
* we grab the pointer; this ensures that we will never
|
|
* get a permanently stuck grab.
|
|
*/
|
|
if ((event->xmotion.state & button_mask) == 0)
|
|
finish_drag ();
|
|
break;
|
|
|
|
case ButtonRelease:
|
|
update_pos (event->xbutton.x_root,
|
|
event->xbutton.y_root);
|
|
|
|
if (event->xbutton.button == moveresize_button)
|
|
finish_drag ();
|
|
break;
|
|
}
|
|
}
|
|
|
|
void
|
|
_gdk_moveresize_configure_done (void)
|
|
{
|
|
XEvent *tmp_event;
|
|
|
|
if (moveresize_pending_event)
|
|
{
|
|
tmp_event = moveresize_pending_event;
|
|
moveresize_pending_event = NULL;
|
|
_gdk_moveresize_handle_event (tmp_event);
|
|
g_free (tmp_event);
|
|
}
|
|
}
|
|
|
|
static void
|
|
create_moveresize_window (guint32 timestamp)
|
|
{
|
|
GdkWindowAttr attributes;
|
|
gint attributes_mask;
|
|
GdkGrabStatus status;
|
|
|
|
g_assert (moveresize_emulation_window == NULL);
|
|
|
|
attributes.x = -100;
|
|
attributes.y = -100;
|
|
attributes.width = 10;
|
|
attributes.height = 10;
|
|
attributes.window_type = GDK_WINDOW_TEMP;
|
|
attributes.wclass = GDK_INPUT_ONLY;
|
|
attributes.override_redirect = TRUE;
|
|
attributes.event_mask = 0;
|
|
|
|
attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_NOREDIR;
|
|
|
|
moveresize_emulation_window =
|
|
gdk_window_new (NULL, &attributes, attributes_mask);
|
|
|
|
gdk_window_show (moveresize_emulation_window);
|
|
|
|
status = gdk_pointer_grab (moveresize_emulation_window,
|
|
FALSE,
|
|
GDK_BUTTON_RELEASE_MASK |
|
|
GDK_POINTER_MOTION_MASK,
|
|
FALSE,
|
|
NULL,
|
|
timestamp);
|
|
|
|
if (status != GDK_GRAB_SUCCESS)
|
|
{
|
|
/* If this fails, some other client has grabbed the window
|
|
* already.
|
|
*/
|
|
gdk_window_destroy (moveresize_emulation_window);
|
|
moveresize_emulation_window = NULL;
|
|
}
|
|
|
|
moveresize_process_time = 0;
|
|
}
|
|
|
|
static void
|
|
emulate_resize_drag (GdkWindow *window,
|
|
GdkWindowEdge edge,
|
|
gint button,
|
|
gint root_x,
|
|
gint root_y,
|
|
guint32 timestamp)
|
|
{
|
|
is_resize = TRUE;
|
|
moveresize_button = button;
|
|
resize_edge = edge;
|
|
moveresize_x = root_x;
|
|
moveresize_y = root_y;
|
|
_gdk_moveresize_window = GDK_WINDOW (g_object_ref (G_OBJECT (window)));
|
|
|
|
gdk_window_get_size (window, &moveresize_orig_width, &moveresize_orig_height);
|
|
|
|
moveresize_geom_mask = 0;
|
|
gdk_window_get_geometry_hints (window,
|
|
&moveresize_geometry,
|
|
&moveresize_geom_mask);
|
|
|
|
create_moveresize_window (timestamp);
|
|
}
|
|
|
|
static void
|
|
emulate_move_drag (GdkWindow *window,
|
|
gint button,
|
|
gint root_x,
|
|
gint root_y,
|
|
guint32 timestamp)
|
|
{
|
|
is_resize = FALSE;
|
|
moveresize_button = button;
|
|
moveresize_x = root_x;
|
|
moveresize_y = root_y;
|
|
_gdk_moveresize_window = GDK_WINDOW (g_object_ref (G_OBJECT (window)));
|
|
|
|
gdk_window_get_deskrelative_origin (_gdk_moveresize_window,
|
|
&moveresize_orig_x,
|
|
&moveresize_orig_y);
|
|
|
|
create_moveresize_window (timestamp);
|
|
}
|
|
|
|
void
|
|
gdk_window_begin_resize_drag (GdkWindow *window,
|
|
GdkWindowEdge edge,
|
|
gint button,
|
|
gint root_x,
|
|
gint root_y,
|
|
guint32 timestamp)
|
|
{
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
g_return_if_fail (moveresize_emulation_window == NULL);
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
if (gdk_net_wm_supports (gdk_atom_intern ("_NET_WM_MOVERESIZE", FALSE)))
|
|
wmspec_resize_drag (window, edge, button, root_x, root_y, timestamp);
|
|
else
|
|
emulate_resize_drag (window, edge, button, root_x, root_y, timestamp);
|
|
}
|
|
|
|
void
|
|
gdk_window_begin_move_drag (GdkWindow *window,
|
|
gint button,
|
|
gint root_x,
|
|
gint root_y,
|
|
guint32 timestamp)
|
|
{
|
|
g_return_if_fail (GDK_IS_WINDOW (window));
|
|
g_return_if_fail (moveresize_emulation_window == NULL);
|
|
|
|
if (GDK_WINDOW_DESTROYED (window))
|
|
return;
|
|
|
|
if (gdk_net_wm_supports (gdk_atom_intern ("_NET_WM_MOVERESIZE", FALSE)))
|
|
wmspec_moveresize (window, _NET_WM_MOVERESIZE_MOVE,
|
|
root_x, root_y, timestamp);
|
|
else
|
|
emulate_move_drag (window, button, root_x, root_y, timestamp);
|
|
}
|
|
|