mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2025-01-07 03:10:07 +00:00
f6929cfef8
The Wayland backend manages a set of fake root coordinate spaces, where each non-relative positioned toplevel (i.e. not popups, popovers, tooltips etc) make up the basis of separate fake root coordinate spaces. This means that the Wayland backend doesn't have the abilitiy get a proper root coordinate when querying on a non-toplevel GdkWindow. To avoid this issue, first find the toplevel, while translating the anchor rect coordinates so that they are in the toplevel window coordinate space. Then use this toplevel to translate the coordinates to root window coordinate space. https://bugzilla.gnome.org/show_bug.cgi?id=769402
348 lines
10 KiB
C
348 lines
10 KiB
C
/* GDK - The GIMP Drawing Kit
|
|
* Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/*
|
|
* Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
|
|
* file for a list of people on the GTK+ Team. See the ChangeLog
|
|
* files for a list of changes. These files are distributed with
|
|
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "gdkwindowimpl.h"
|
|
|
|
#include "gdkinternals.h"
|
|
|
|
|
|
G_DEFINE_TYPE (GdkWindowImpl, gdk_window_impl, G_TYPE_OBJECT);
|
|
|
|
static gboolean
|
|
gdk_window_impl_beep (GdkWindow *window)
|
|
{
|
|
/* FALSE means windows can't beep, so the display will be
|
|
* made to beep instead. */
|
|
return FALSE;
|
|
}
|
|
|
|
static GdkDisplay *
|
|
get_display_for_window (GdkWindow *primary,
|
|
GdkWindow *secondary)
|
|
{
|
|
GdkDisplay *display = gdk_window_get_display (primary);
|
|
|
|
if (display)
|
|
return display;
|
|
|
|
display = gdk_window_get_display (secondary);
|
|
|
|
if (display)
|
|
return display;
|
|
|
|
g_warning ("no display for window, using default");
|
|
return gdk_display_get_default ();
|
|
}
|
|
|
|
static GdkMonitor *
|
|
get_monitor_for_rect (GdkDisplay *display,
|
|
const GdkRectangle *rect)
|
|
{
|
|
gint biggest_area = G_MININT;
|
|
GdkMonitor *best_monitor = NULL;
|
|
GdkMonitor *monitor;
|
|
GdkRectangle workarea;
|
|
GdkRectangle intersection;
|
|
gint x;
|
|
gint y;
|
|
gint i;
|
|
|
|
for (i = 0; i < gdk_display_get_n_monitors (display); i++)
|
|
{
|
|
monitor = gdk_display_get_monitor (display, i);
|
|
gdk_monitor_get_workarea (monitor, &workarea);
|
|
|
|
if (gdk_rectangle_intersect (&workarea, rect, &intersection))
|
|
{
|
|
if (intersection.width * intersection.height > biggest_area)
|
|
{
|
|
biggest_area = intersection.width * intersection.height;
|
|
best_monitor = monitor;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (best_monitor)
|
|
return best_monitor;
|
|
|
|
x = rect->x + rect->width / 2;
|
|
y = rect->y + rect->height / 2;
|
|
|
|
return gdk_display_get_monitor_at_point (display, x, y);
|
|
}
|
|
|
|
static gint
|
|
get_anchor_x_sign (GdkGravity anchor)
|
|
{
|
|
switch (anchor)
|
|
{
|
|
case GDK_GRAVITY_STATIC:
|
|
case GDK_GRAVITY_NORTH_WEST:
|
|
case GDK_GRAVITY_WEST:
|
|
case GDK_GRAVITY_SOUTH_WEST:
|
|
return -1;
|
|
|
|
default:
|
|
case GDK_GRAVITY_NORTH:
|
|
case GDK_GRAVITY_CENTER:
|
|
case GDK_GRAVITY_SOUTH:
|
|
return 0;
|
|
|
|
case GDK_GRAVITY_NORTH_EAST:
|
|
case GDK_GRAVITY_EAST:
|
|
case GDK_GRAVITY_SOUTH_EAST:
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
static gint
|
|
get_anchor_y_sign (GdkGravity anchor)
|
|
{
|
|
switch (anchor)
|
|
{
|
|
case GDK_GRAVITY_STATIC:
|
|
case GDK_GRAVITY_NORTH_WEST:
|
|
case GDK_GRAVITY_NORTH:
|
|
case GDK_GRAVITY_NORTH_EAST:
|
|
return -1;
|
|
|
|
default:
|
|
case GDK_GRAVITY_WEST:
|
|
case GDK_GRAVITY_CENTER:
|
|
case GDK_GRAVITY_EAST:
|
|
return 0;
|
|
|
|
case GDK_GRAVITY_SOUTH_WEST:
|
|
case GDK_GRAVITY_SOUTH:
|
|
case GDK_GRAVITY_SOUTH_EAST:
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
static gint
|
|
maybe_flip_position (gint bounds_pos,
|
|
gint bounds_size,
|
|
gint rect_pos,
|
|
gint rect_size,
|
|
gint window_size,
|
|
gint rect_sign,
|
|
gint window_sign,
|
|
gint offset,
|
|
gboolean flip,
|
|
gboolean *flipped)
|
|
{
|
|
gint primary;
|
|
gint secondary;
|
|
|
|
*flipped = FALSE;
|
|
primary = rect_pos + (1 + rect_sign) * rect_size / 2 + offset - (1 + window_sign) * window_size / 2;
|
|
|
|
if (!flip || (primary >= bounds_pos && primary + window_size <= bounds_pos + bounds_size))
|
|
return primary;
|
|
|
|
*flipped = TRUE;
|
|
secondary = rect_pos + (1 - rect_sign) * rect_size / 2 - offset - (1 - window_sign) * window_size / 2;
|
|
|
|
if (secondary >= bounds_pos && secondary + window_size <= bounds_pos + bounds_size)
|
|
return secondary;
|
|
|
|
*flipped = FALSE;
|
|
return primary;
|
|
}
|
|
|
|
static GdkWindow *
|
|
traverse_to_toplevel (GdkWindow *window,
|
|
gint x,
|
|
gint y,
|
|
gint *toplevel_x,
|
|
gint *toplevel_y)
|
|
{
|
|
GdkWindow *parent;
|
|
gdouble xf = x;
|
|
gdouble yf = y;
|
|
|
|
while ((parent = gdk_window_get_effective_parent (window)) != NULL &&
|
|
(gdk_window_get_window_type (parent) != GDK_WINDOW_ROOT))
|
|
{
|
|
gdk_window_coords_to_parent (window, xf, yf, &xf, &yf);
|
|
window = parent;
|
|
}
|
|
|
|
*toplevel_x = (gint) xf;
|
|
*toplevel_y = (gint) yf;
|
|
return window;
|
|
}
|
|
|
|
static void
|
|
gdk_window_impl_move_to_rect (GdkWindow *window,
|
|
const GdkRectangle *rect,
|
|
GdkGravity rect_anchor,
|
|
GdkGravity window_anchor,
|
|
GdkAnchorHints anchor_hints,
|
|
gint rect_anchor_dx,
|
|
gint rect_anchor_dy)
|
|
{
|
|
GdkWindow *transient_for_toplevel;
|
|
GdkDisplay *display;
|
|
GdkMonitor *monitor;
|
|
GdkRectangle bounds;
|
|
GdkRectangle root_rect = *rect;
|
|
GdkRectangle flipped_rect;
|
|
GdkRectangle final_rect;
|
|
gboolean flipped_x;
|
|
gboolean flipped_y;
|
|
|
|
/*
|
|
* First translate the anchor rect to toplevel coordinates. This is needed
|
|
* because not all backends will be able to get root coordinates for
|
|
* non-toplevel windows.
|
|
*/
|
|
transient_for_toplevel = traverse_to_toplevel (window->transient_for,
|
|
root_rect.x,
|
|
root_rect.y,
|
|
&root_rect.x,
|
|
&root_rect.y);
|
|
|
|
gdk_window_get_root_coords (transient_for_toplevel,
|
|
root_rect.x,
|
|
root_rect.y,
|
|
&root_rect.x,
|
|
&root_rect.y);
|
|
|
|
display = get_display_for_window (window, window->transient_for);
|
|
monitor = get_monitor_for_rect (display, &root_rect);
|
|
gdk_monitor_get_workarea (monitor, &bounds);
|
|
|
|
flipped_rect.width = window->width - window->shadow_left - window->shadow_right;
|
|
flipped_rect.height = window->height - window->shadow_top - window->shadow_bottom;
|
|
flipped_rect.x = maybe_flip_position (bounds.x,
|
|
bounds.width,
|
|
root_rect.x,
|
|
root_rect.width,
|
|
flipped_rect.width,
|
|
get_anchor_x_sign (rect_anchor),
|
|
get_anchor_x_sign (window_anchor),
|
|
rect_anchor_dx,
|
|
anchor_hints & GDK_ANCHOR_FLIP_X,
|
|
&flipped_x);
|
|
flipped_rect.y = maybe_flip_position (bounds.y,
|
|
bounds.height,
|
|
root_rect.y,
|
|
root_rect.height,
|
|
flipped_rect.height,
|
|
get_anchor_y_sign (rect_anchor),
|
|
get_anchor_y_sign (window_anchor),
|
|
rect_anchor_dy,
|
|
anchor_hints & GDK_ANCHOR_FLIP_Y,
|
|
&flipped_y);
|
|
|
|
final_rect = flipped_rect;
|
|
|
|
if (anchor_hints & GDK_ANCHOR_SLIDE_X)
|
|
{
|
|
if (final_rect.x + final_rect.width > bounds.x + bounds.width)
|
|
final_rect.x = bounds.x + bounds.width - final_rect.width;
|
|
|
|
if (final_rect.x < bounds.x)
|
|
final_rect.x = bounds.x;
|
|
}
|
|
|
|
if (anchor_hints & GDK_ANCHOR_SLIDE_Y)
|
|
{
|
|
if (final_rect.y + final_rect.height > bounds.y + bounds.height)
|
|
final_rect.y = bounds.y + bounds.height - final_rect.height;
|
|
|
|
if (final_rect.y < bounds.y)
|
|
final_rect.y = bounds.y;
|
|
}
|
|
|
|
if (anchor_hints & GDK_ANCHOR_RESIZE_X)
|
|
{
|
|
if (final_rect.x < bounds.x)
|
|
{
|
|
final_rect.width -= bounds.x - final_rect.x;
|
|
final_rect.x = bounds.x;
|
|
}
|
|
|
|
if (final_rect.x + final_rect.width > bounds.x + bounds.width)
|
|
final_rect.width = bounds.x + bounds.width - final_rect.x;
|
|
}
|
|
|
|
if (anchor_hints & GDK_ANCHOR_RESIZE_Y)
|
|
{
|
|
if (final_rect.y < bounds.y)
|
|
{
|
|
final_rect.height -= bounds.y - final_rect.y;
|
|
final_rect.y = bounds.y;
|
|
}
|
|
|
|
if (final_rect.y + final_rect.height > bounds.y + bounds.height)
|
|
final_rect.height = bounds.y + bounds.height - final_rect.y;
|
|
}
|
|
|
|
flipped_rect.x -= window->shadow_left;
|
|
flipped_rect.y -= window->shadow_top;
|
|
flipped_rect.width += window->shadow_left + window->shadow_right;
|
|
flipped_rect.height += window->shadow_top + window->shadow_bottom;
|
|
|
|
final_rect.x -= window->shadow_left;
|
|
final_rect.y -= window->shadow_top;
|
|
final_rect.width += window->shadow_left + window->shadow_right;
|
|
final_rect.height += window->shadow_top + window->shadow_bottom;
|
|
|
|
if (final_rect.width != window->width || final_rect.height != window->height)
|
|
gdk_window_move_resize (window, final_rect.x, final_rect.y, final_rect.width, final_rect.height);
|
|
else
|
|
gdk_window_move (window, final_rect.x, final_rect.y);
|
|
|
|
g_signal_emit_by_name (window,
|
|
"moved-to-rect",
|
|
&flipped_rect,
|
|
&final_rect,
|
|
flipped_x,
|
|
flipped_y);
|
|
}
|
|
|
|
static void
|
|
gdk_window_impl_process_updates_recurse (GdkWindow *window,
|
|
cairo_region_t *region)
|
|
{
|
|
_gdk_window_process_updates_recurse (window, region);
|
|
}
|
|
|
|
static void
|
|
gdk_window_impl_class_init (GdkWindowImplClass *impl_class)
|
|
{
|
|
impl_class->beep = gdk_window_impl_beep;
|
|
impl_class->move_to_rect = gdk_window_impl_move_to_rect;
|
|
impl_class->process_updates_recurse = gdk_window_impl_process_updates_recurse;
|
|
}
|
|
|
|
static void
|
|
gdk_window_impl_init (GdkWindowImpl *impl)
|
|
{
|
|
}
|