mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2024-11-10 19:00:08 +00:00
d2c95a1b13
It was used by all surfaces to track 'is-mapped', but still part of the GdkToplevelState, and is now replaced with a separate boolean in the GdkSurface structure. It also caused issues when a widget was unmapped, and due to that unmapped a popover which hid its corresponding surface. When this surface was hidden, it emitted a state change event, which would then go back into GTK and queue a resize on popover widget, which would travel back down to the widget that was originally unmapped, causing confusino when doing future allocations. To summarize, one should not hide widgets during allocation, and to avoid this, make this new is-mapped boolean asynchronous when hiding a surface, meaning the notification event for the changed mapped state will be emitted in an idle callback. This avoids the above described reentry issue.
1686 lines
48 KiB
C
1686 lines
48 KiB
C
/* GDK - The GIMP Drawing Kit
|
|
* Copyright (C) 1995-2007 Peter Mattis, Spencer Kimball,
|
|
* Josh MacDonald, Ryan Lortie
|
|
*
|
|
* 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 "gdksurface-broadway.h"
|
|
|
|
#include "gdkbroadwaydisplay.h"
|
|
#include "gdkdeviceprivate.h"
|
|
#include "gdkdisplay-broadway.h"
|
|
#include "gdkdevice-broadway.h"
|
|
#include "gdkdisplay.h"
|
|
#include "gdkdragsurfaceprivate.h"
|
|
#include "gdkeventsource.h"
|
|
#include "gdkframeclockidleprivate.h"
|
|
#include "gdkinternals.h"
|
|
#include "gdkpopupprivate.h"
|
|
#include "gdkprivate-broadway.h"
|
|
#include "gdksurfaceprivate.h"
|
|
#include "gdktextureprivate.h"
|
|
#include "gdktoplevelprivate.h"
|
|
|
|
#include <graphene.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
/* Forward declarations */
|
|
static void gdk_broadway_surface_finalize (GObject *object);
|
|
|
|
G_DEFINE_TYPE (GdkBroadwaySurface, gdk_broadway_surface, GDK_TYPE_SURFACE)
|
|
|
|
GType gdk_broadway_toplevel_get_type (void) G_GNUC_CONST;
|
|
GType gdk_broadway_popup_get_type (void) G_GNUC_CONST;
|
|
GType gdk_broadway_drag_surface_get_type (void) G_GNUC_CONST;
|
|
|
|
#define GDK_TYPE_BROADWAY_TOPLEVEL (gdk_broadway_toplevel_get_type ())
|
|
#define GDK_TYPE_BROADWAY_POPUP (gdk_broadway_popup_get_type ())
|
|
#define GDK_TYPE_BROADWAY_DRAG_SURFACE (gdk_broadway_drag_surface_get_type ())
|
|
|
|
/* We need to flush in an idle rather than AFTER_PAINT, as the clock
|
|
is frozen during e.g. surface resizes so the paint will not happen
|
|
and the surface resize request is never flushed. */
|
|
static void
|
|
queue_flush (GdkSurface *surface)
|
|
{
|
|
gdk_broadway_display_flush_in_idle (gdk_surface_get_display (surface));
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_init (GdkBroadwaySurface *impl)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_finalize (GObject *object)
|
|
{
|
|
GdkBroadwaySurface *impl;
|
|
GdkBroadwayDisplay *broadway_display;
|
|
|
|
g_return_if_fail (GDK_IS_BROADWAY_SURFACE (object));
|
|
|
|
impl = GDK_BROADWAY_SURFACE (object);
|
|
|
|
_gdk_broadway_surface_grab_check_destroy (GDK_SURFACE (impl));
|
|
|
|
broadway_display = GDK_BROADWAY_DISPLAY (gdk_surface_get_display (GDK_SURFACE (impl)));
|
|
|
|
g_hash_table_remove (broadway_display->id_ht, GINT_TO_POINTER (impl->id));
|
|
|
|
if (impl->cursor)
|
|
g_object_unref (impl->cursor);
|
|
|
|
broadway_display->toplevels = g_list_remove (broadway_display->toplevels, impl);
|
|
|
|
G_OBJECT_CLASS (gdk_broadway_surface_parent_class)->finalize (object);
|
|
}
|
|
|
|
static gboolean
|
|
thaw_updates_cb (GdkSurface *surface)
|
|
{
|
|
if (!GDK_SURFACE_DESTROYED (surface))
|
|
gdk_surface_thaw_updates (surface);
|
|
g_object_unref (surface);
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
void
|
|
_gdk_broadway_roundtrip_notify (GdkSurface *surface,
|
|
guint32 tag,
|
|
gboolean local_reply)
|
|
{
|
|
GdkBroadwaySurface *impl = GDK_BROADWAY_SURFACE (surface);
|
|
GdkFrameClock *clock = gdk_surface_get_frame_clock (surface);
|
|
GdkFrameTimings *timings;
|
|
|
|
timings = gdk_frame_clock_get_timings (clock, impl->pending_frame_counter);
|
|
impl->pending_frame_counter = 0;
|
|
|
|
/* If there is no remote web client, rate limit update to once a second */
|
|
if (local_reply)
|
|
g_timeout_add_seconds (1, (GSourceFunc)thaw_updates_cb, g_object_ref (surface));
|
|
else
|
|
gdk_surface_thaw_updates (surface);
|
|
|
|
if (timings)
|
|
{
|
|
timings->refresh_interval = 33333; /* default to 1/30th of a second */
|
|
// This isn't quite right, since we've done a roundtrip back too, can we do better?
|
|
timings->presentation_time = g_get_monotonic_time ();
|
|
timings->complete = TRUE;
|
|
|
|
|
|
#ifdef G_ENABLE_DEBUG
|
|
if ((_gdk_debug_flags & GDK_DEBUG_FRAMES) != 0)
|
|
_gdk_frame_clock_debug_print_timings (clock, timings);
|
|
|
|
if (GDK_PROFILER_IS_RUNNING)
|
|
_gdk_frame_clock_add_timings_to_profiler (clock, timings);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
static void
|
|
on_frame_clock_after_paint (GdkFrameClock *clock,
|
|
GdkSurface *surface)
|
|
{
|
|
GdkDisplay *display = gdk_surface_get_display (surface);
|
|
GdkBroadwaySurface *impl = GDK_BROADWAY_SURFACE (surface);
|
|
GdkBroadwayDisplay *broadway_display;
|
|
|
|
impl->pending_frame_counter = gdk_frame_clock_get_frame_counter (clock);
|
|
gdk_surface_freeze_updates (surface);
|
|
|
|
broadway_display = GDK_BROADWAY_DISPLAY (display);
|
|
|
|
_gdk_broadway_server_roundtrip (broadway_display->server, impl->id, _gdk_display_get_next_serial (display));
|
|
|
|
gdk_display_flush (display);
|
|
}
|
|
|
|
static void
|
|
on_frame_clock_before_paint (GdkFrameClock *clock,
|
|
GdkSurface *surface)
|
|
{
|
|
GdkFrameTimings *timings = gdk_frame_clock_get_current_timings (clock);
|
|
gint64 presentation_time;
|
|
gint64 refresh_interval;
|
|
|
|
if (surface->update_freeze_count > 0)
|
|
return;
|
|
|
|
gdk_frame_clock_get_refresh_info (clock,
|
|
timings->frame_time,
|
|
&refresh_interval, &presentation_time);
|
|
if (presentation_time != 0)
|
|
{
|
|
timings->predicted_presentation_time = presentation_time + refresh_interval;
|
|
}
|
|
else
|
|
{
|
|
timings->predicted_presentation_time = timings->frame_time + refresh_interval / 2 + refresh_interval;
|
|
}
|
|
}
|
|
|
|
static void
|
|
connect_frame_clock (GdkSurface *surface)
|
|
{
|
|
GdkFrameClock *frame_clock = gdk_surface_get_frame_clock (surface);
|
|
|
|
g_signal_connect (frame_clock, "before-paint",
|
|
G_CALLBACK (on_frame_clock_before_paint), surface);
|
|
g_signal_connect (frame_clock, "after-paint",
|
|
G_CALLBACK (on_frame_clock_after_paint), surface);
|
|
}
|
|
|
|
static void
|
|
disconnect_frame_clock (GdkSurface *surface)
|
|
{
|
|
GdkFrameClock *frame_clock = gdk_surface_get_frame_clock (surface);
|
|
|
|
g_signal_handlers_disconnect_by_func (frame_clock,
|
|
on_frame_clock_before_paint, surface);
|
|
g_signal_handlers_disconnect_by_func (frame_clock,
|
|
on_frame_clock_after_paint, surface);
|
|
}
|
|
|
|
GdkSurface *
|
|
_gdk_broadway_display_create_surface (GdkDisplay *display,
|
|
GdkSurfaceType surface_type,
|
|
GdkSurface *parent,
|
|
int x,
|
|
int y,
|
|
int width,
|
|
int height)
|
|
{
|
|
GdkBroadwayDisplay *broadway_display;
|
|
GdkFrameClock *frame_clock;
|
|
GdkSurface *surface;
|
|
GdkBroadwaySurface *impl;
|
|
GType type;
|
|
|
|
if (parent)
|
|
frame_clock = g_object_ref (gdk_surface_get_frame_clock (parent));
|
|
else
|
|
frame_clock = _gdk_frame_clock_idle_new ();
|
|
|
|
switch (surface_type)
|
|
{
|
|
case GDK_SURFACE_TOPLEVEL:
|
|
type = GDK_TYPE_BROADWAY_TOPLEVEL;
|
|
break;
|
|
case GDK_SURFACE_POPUP:
|
|
type = GDK_TYPE_BROADWAY_POPUP;
|
|
break;
|
|
case GDK_SURFACE_TEMP:
|
|
type = GDK_TYPE_BROADWAY_DRAG_SURFACE;
|
|
break;
|
|
default:
|
|
g_assert_not_reached ();
|
|
break;
|
|
}
|
|
|
|
surface = g_object_new (type,
|
|
"display", display,
|
|
"frame-clock", frame_clock,
|
|
NULL);
|
|
|
|
g_object_unref (frame_clock);
|
|
|
|
surface->parent = parent;
|
|
surface->x = x;
|
|
surface->y = y;
|
|
surface->width = width;
|
|
surface->height = height;
|
|
|
|
broadway_display = GDK_BROADWAY_DISPLAY (display);
|
|
|
|
impl = GDK_BROADWAY_SURFACE (surface);
|
|
impl->root_x = x;
|
|
impl->root_y = y;
|
|
if (parent)
|
|
{
|
|
impl->root_x += GDK_BROADWAY_SURFACE (parent)->root_x;
|
|
impl->root_y += GDK_BROADWAY_SURFACE (parent)->root_y;
|
|
}
|
|
|
|
impl->id = _gdk_broadway_server_new_surface (broadway_display->server,
|
|
impl->root_x,
|
|
impl->root_y,
|
|
surface->width,
|
|
surface->height);
|
|
g_hash_table_insert (broadway_display->id_ht, GINT_TO_POINTER(impl->id), surface);
|
|
|
|
g_object_ref (surface);
|
|
|
|
if (!surface->parent)
|
|
broadway_display->toplevels = g_list_prepend (broadway_display->toplevels, impl);
|
|
|
|
connect_frame_clock (surface);
|
|
|
|
/* We treat the real parent as a default transient for to get stacking right */
|
|
if (parent)
|
|
{
|
|
impl->transient_for = GDK_BROADWAY_SURFACE (parent)->id;
|
|
_gdk_broadway_server_surface_set_transient_for (broadway_display->server, impl->id, impl->transient_for);
|
|
}
|
|
|
|
return surface;
|
|
}
|
|
|
|
static cairo_surface_t *
|
|
gdk_broadway_surface_ref_cairo_surface (GdkSurface *surface)
|
|
{
|
|
if (GDK_IS_BROADWAY_SURFACE (surface) &&
|
|
GDK_SURFACE_DESTROYED (surface))
|
|
return NULL;
|
|
|
|
return cairo_image_surface_create (CAIRO_FORMAT_ARGB32, 1, 1);
|
|
}
|
|
|
|
static void
|
|
_gdk_broadway_surface_destroy (GdkSurface *surface,
|
|
gboolean foreign_destroy)
|
|
{
|
|
GdkBroadwaySurface *impl;
|
|
GdkBroadwayDisplay *broadway_display;
|
|
|
|
g_return_if_fail (GDK_IS_SURFACE (surface));
|
|
|
|
impl = GDK_BROADWAY_SURFACE (surface);
|
|
|
|
disconnect_frame_clock (surface);
|
|
|
|
if (impl->node_data)
|
|
g_array_unref (impl->node_data);
|
|
if (impl->node_data_textures)
|
|
g_ptr_array_unref (impl->node_data_textures);
|
|
|
|
_gdk_broadway_surface_grab_check_destroy (surface);
|
|
|
|
broadway_display = GDK_BROADWAY_DISPLAY (gdk_surface_get_display (surface));
|
|
g_hash_table_remove (broadway_display->id_ht, GINT_TO_POINTER (impl->id));
|
|
|
|
_gdk_broadway_server_destroy_surface (broadway_display->server, impl->id);
|
|
}
|
|
|
|
void
|
|
gdk_broadway_surface_set_nodes (GdkSurface *surface,
|
|
GArray *nodes,
|
|
GPtrArray *node_textures)
|
|
{
|
|
GdkBroadwaySurface *impl;
|
|
GdkBroadwayDisplay *broadway_display;
|
|
|
|
g_return_if_fail (GDK_IS_SURFACE (surface));
|
|
|
|
impl = GDK_BROADWAY_SURFACE (surface);
|
|
|
|
broadway_display = GDK_BROADWAY_DISPLAY (gdk_surface_get_display (surface));
|
|
|
|
if (nodes)
|
|
g_array_ref (nodes);
|
|
if (impl->node_data)
|
|
g_array_unref (impl->node_data);
|
|
impl->node_data = nodes;
|
|
|
|
if (node_textures)
|
|
g_ptr_array_ref (node_textures);
|
|
if (impl->node_data_textures)
|
|
g_ptr_array_unref (impl->node_data_textures);
|
|
impl->node_data_textures = node_textures;
|
|
|
|
gdk_broadway_server_surface_set_nodes (broadway_display->server, impl->id, impl->node_data);
|
|
}
|
|
|
|
/* This function is called when the XSurface is really gone.
|
|
*/
|
|
static void
|
|
gdk_broadway_surface_destroy_notify (GdkSurface *surface)
|
|
{
|
|
if (!GDK_SURFACE_DESTROYED (surface))
|
|
_gdk_surface_destroy (surface, TRUE);
|
|
|
|
g_object_unref (surface);
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_show (GdkSurface *surface,
|
|
gboolean already_mapped)
|
|
{
|
|
GdkBroadwaySurface *impl;
|
|
GdkBroadwayDisplay *broadway_display;
|
|
|
|
impl = GDK_BROADWAY_SURFACE (surface);
|
|
impl->visible = TRUE;
|
|
|
|
/* FIXME: update state ? */
|
|
|
|
broadway_display = GDK_BROADWAY_DISPLAY (gdk_surface_get_display (surface));
|
|
if (_gdk_broadway_server_surface_show (broadway_display->server, impl->id))
|
|
queue_flush (surface);
|
|
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_hide (GdkSurface *surface)
|
|
{
|
|
GdkBroadwaySurface *impl;
|
|
GdkBroadwayDisplay *broadway_display;
|
|
|
|
impl = GDK_BROADWAY_SURFACE (surface);
|
|
impl->visible = FALSE;
|
|
|
|
/* FIXME: update state ? */
|
|
|
|
broadway_display = GDK_BROADWAY_DISPLAY (gdk_surface_get_display (surface));
|
|
|
|
_gdk_broadway_surface_grab_check_unmap (surface,
|
|
_gdk_broadway_server_get_next_serial (broadway_display->server));
|
|
|
|
if (_gdk_broadway_server_surface_hide (broadway_display->server, impl->id))
|
|
queue_flush (surface);
|
|
|
|
_gdk_surface_clear_update_area (surface);
|
|
}
|
|
|
|
static int
|
|
gdk_broadway_surface_get_scale_factor (GdkSurface *surface)
|
|
{
|
|
GdkBroadwayDisplay *broadway_display;
|
|
|
|
if (GDK_SURFACE_DESTROYED (surface))
|
|
return 1;
|
|
|
|
broadway_display = GDK_BROADWAY_DISPLAY (gdk_surface_get_display (surface));
|
|
|
|
return broadway_display->scale_factor;
|
|
}
|
|
|
|
static void
|
|
sync_child_root_pos (GdkSurface *parent)
|
|
{
|
|
GdkBroadwaySurface *parent_impl = GDK_BROADWAY_SURFACE (parent);
|
|
GdkBroadwayDisplay *broadway_display;
|
|
GList *l;
|
|
|
|
broadway_display = GDK_BROADWAY_DISPLAY (gdk_surface_get_display (parent));
|
|
|
|
for (l = parent->children; l; l = l->next)
|
|
{
|
|
GdkBroadwaySurface *child_impl = l->data;
|
|
GdkSurface *child = GDK_SURFACE (child_impl);
|
|
int root_x, root_y;
|
|
|
|
root_x = child->x + parent_impl->root_x;
|
|
root_y = child->y + parent_impl->root_y;
|
|
|
|
if (root_x != child_impl->root_x ||
|
|
root_y != child_impl->root_y)
|
|
{
|
|
child_impl->root_x = root_x;
|
|
child_impl->root_y = root_y;
|
|
|
|
_gdk_broadway_server_surface_move_resize (broadway_display->server,
|
|
child_impl->id,
|
|
TRUE,
|
|
child_impl->root_x, child_impl->root_y,
|
|
child->width, child->height);
|
|
sync_child_root_pos (child);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* x, y is relative to parent */
|
|
static void
|
|
gdk_broadway_surface_move_resize_internal (GdkSurface *surface,
|
|
gboolean with_move,
|
|
int x,
|
|
int y,
|
|
int width,
|
|
int height)
|
|
{
|
|
GdkBroadwaySurface *impl = GDK_BROADWAY_SURFACE (surface);
|
|
GdkBroadwayDisplay *broadway_display;
|
|
gboolean size_changed;
|
|
|
|
if (with_move)
|
|
{
|
|
surface->x = x;
|
|
surface->y = y;
|
|
impl->root_x = x;
|
|
impl->root_y = y;
|
|
if (surface->parent)
|
|
{
|
|
GdkBroadwaySurface *parent_impl = GDK_BROADWAY_SURFACE (surface->parent);
|
|
impl->root_x += parent_impl->root_x;
|
|
impl->root_y += parent_impl->root_y;
|
|
}
|
|
}
|
|
|
|
size_changed = FALSE;
|
|
|
|
broadway_display = GDK_BROADWAY_DISPLAY (gdk_surface_get_display (surface));
|
|
|
|
if (width > 0 || height > 0)
|
|
{
|
|
if (width < 1)
|
|
width = 1;
|
|
|
|
if (height < 1)
|
|
height = 1;
|
|
|
|
if (width != surface->width ||
|
|
height != surface->height)
|
|
{
|
|
size_changed = TRUE;
|
|
|
|
/* Resize clears the content */
|
|
impl->dirty = TRUE;
|
|
impl->last_synced = FALSE;
|
|
|
|
surface->width = width;
|
|
surface->height = height;
|
|
}
|
|
}
|
|
|
|
_gdk_broadway_server_surface_move_resize (broadway_display->server,
|
|
impl->id,
|
|
with_move,
|
|
impl->root_x, impl->root_y,
|
|
surface->width, surface->height);
|
|
sync_child_root_pos (surface);
|
|
|
|
queue_flush (surface);
|
|
if (size_changed)
|
|
{
|
|
surface->resize_count++;
|
|
_gdk_surface_update_size (surface);
|
|
}
|
|
}
|
|
|
|
void
|
|
gdk_broadway_surface_move_resize (GdkSurface *surface,
|
|
int x,
|
|
int y,
|
|
int width,
|
|
int height)
|
|
{
|
|
gdk_broadway_surface_move_resize_internal (surface, TRUE,
|
|
x, y,
|
|
width, height);
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_toplevel_resize (GdkSurface *surface,
|
|
int width,
|
|
int height)
|
|
{
|
|
gdk_broadway_surface_move_resize_internal (surface, FALSE,
|
|
0, 0,
|
|
width, height);
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_move (GdkSurface *surface,
|
|
int x,
|
|
int y)
|
|
{
|
|
gdk_broadway_surface_move_resize_internal (surface, TRUE, x, y, -1, -1);
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_layout_popup (GdkSurface *surface,
|
|
int width,
|
|
int height,
|
|
GdkPopupLayout *layout)
|
|
{
|
|
GdkBroadwaySurface *impl = GDK_BROADWAY_SURFACE (surface);
|
|
GdkMonitor *monitor;
|
|
GdkRectangle bounds;
|
|
GdkRectangle final_rect;
|
|
int x, y;
|
|
|
|
monitor = gdk_surface_get_layout_monitor (surface, layout,
|
|
gdk_monitor_get_geometry);
|
|
gdk_monitor_get_geometry (monitor, &bounds);
|
|
|
|
gdk_surface_layout_popup_helper (surface,
|
|
width,
|
|
height,
|
|
impl->shadow_left,
|
|
impl->shadow_right,
|
|
impl->shadow_top,
|
|
impl->shadow_bottom,
|
|
monitor,
|
|
&bounds,
|
|
layout,
|
|
&final_rect);
|
|
|
|
x = final_rect.x;
|
|
y = final_rect.y;
|
|
|
|
if (final_rect.width != surface->width ||
|
|
final_rect.height != surface->height)
|
|
{
|
|
gdk_broadway_surface_move_resize (surface,
|
|
x,
|
|
y,
|
|
final_rect.width,
|
|
final_rect.height);
|
|
}
|
|
else
|
|
{
|
|
gdk_broadway_surface_move (surface, x, y);
|
|
}
|
|
}
|
|
|
|
static void
|
|
show_popup (GdkSurface *surface)
|
|
{
|
|
gdk_surface_set_is_mapped (surface, TRUE);
|
|
gdk_broadway_surface_show (surface, FALSE);
|
|
gdk_surface_invalidate_rect (surface, NULL);
|
|
}
|
|
|
|
static void
|
|
show_grabbing_popup (GdkSeat *seat,
|
|
GdkSurface *surface,
|
|
gpointer user_data)
|
|
{
|
|
show_popup (surface);
|
|
}
|
|
|
|
static gboolean
|
|
gdk_broadway_surface_present_popup (GdkSurface *surface,
|
|
int width,
|
|
int height,
|
|
GdkPopupLayout *layout)
|
|
{
|
|
gdk_broadway_surface_layout_popup (surface, width, height, layout);
|
|
|
|
if (GDK_SURFACE_IS_MAPPED (surface))
|
|
return TRUE;
|
|
|
|
if (surface->autohide)
|
|
{
|
|
gdk_seat_grab (gdk_display_get_default_seat (surface->display),
|
|
surface,
|
|
GDK_SEAT_CAPABILITY_ALL,
|
|
TRUE,
|
|
NULL, NULL,
|
|
show_grabbing_popup, NULL);
|
|
}
|
|
else
|
|
{
|
|
show_popup (surface);
|
|
}
|
|
|
|
return GDK_SURFACE_IS_MAPPED (surface);
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_focus (GdkSurface *surface,
|
|
guint32 timestamp)
|
|
{
|
|
GdkBroadwaySurface *impl;
|
|
GdkBroadwayDisplay *broadway_display;
|
|
|
|
g_return_if_fail (GDK_IS_SURFACE (surface));
|
|
|
|
if (GDK_SURFACE_DESTROYED (surface))
|
|
return;
|
|
|
|
impl = GDK_BROADWAY_SURFACE (surface);
|
|
broadway_display = GDK_BROADWAY_DISPLAY (gdk_surface_get_display (surface));
|
|
_gdk_broadway_server_surface_focus (broadway_display->server,
|
|
impl->id);
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_set_geometry_hints (GdkSurface *surface,
|
|
const GdkGeometry *geometry,
|
|
GdkSurfaceHints geom_mask)
|
|
{
|
|
GdkBroadwaySurface *impl;
|
|
|
|
impl = GDK_BROADWAY_SURFACE (surface);
|
|
|
|
impl->geometry_hints = *geometry;
|
|
impl->geometry_hints_mask = geom_mask;
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_set_title (GdkSurface *surface,
|
|
const char *title)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_set_startup_id (GdkSurface *surface,
|
|
const char *startup_id)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_set_transient_for (GdkSurface *surface,
|
|
GdkSurface *parent)
|
|
{
|
|
GdkBroadwayDisplay *display;
|
|
GdkBroadwaySurface *impl;
|
|
int parent_id;
|
|
|
|
impl = GDK_BROADWAY_SURFACE (surface);
|
|
|
|
/* We treat the real parent as a default transient for to get stacking right */
|
|
if (parent == NULL)
|
|
parent = surface->parent;
|
|
|
|
parent_id = 0;
|
|
if (parent)
|
|
parent_id = GDK_BROADWAY_SURFACE (parent)->id;
|
|
|
|
impl->transient_for = parent_id;
|
|
|
|
display = GDK_BROADWAY_DISPLAY (gdk_surface_get_display (surface));
|
|
_gdk_broadway_server_surface_set_transient_for (display->server, impl->id, impl->transient_for);
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_get_geometry (GdkSurface *surface,
|
|
int *x,
|
|
int *y,
|
|
int *width,
|
|
int *height)
|
|
{
|
|
g_return_if_fail (GDK_IS_SURFACE (surface));
|
|
|
|
/* TODO: These should really roundtrip to the client to get the current data */
|
|
|
|
if (x)
|
|
*x = surface->x;
|
|
if (y)
|
|
*y = surface->y;
|
|
if (width)
|
|
*width = surface->width;
|
|
if (height)
|
|
*height = surface->height;
|
|
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_get_root_coords (GdkSurface *surface,
|
|
int x,
|
|
int y,
|
|
int *root_x,
|
|
int *root_y)
|
|
{
|
|
GdkBroadwaySurface *impl;
|
|
|
|
impl = GDK_BROADWAY_SURFACE (surface);
|
|
|
|
if (root_x)
|
|
*root_x = x + impl->root_x;
|
|
if (root_y)
|
|
*root_y = y + impl->root_y;
|
|
}
|
|
|
|
static gboolean
|
|
gdk_broadway_surface_get_device_state (GdkSurface *surface,
|
|
GdkDevice *device,
|
|
double *x,
|
|
double *y,
|
|
GdkModifierType *mask)
|
|
{
|
|
g_return_val_if_fail (surface == NULL || GDK_IS_SURFACE (surface), FALSE);
|
|
|
|
if (GDK_SURFACE_DESTROYED (surface))
|
|
return FALSE;
|
|
|
|
gdk_broadway_device_query_state (device, surface, x, y, mask);
|
|
|
|
return *x >= 0 && *y >= 0 && *x < surface->width && *y < surface->height;
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_set_input_region (GdkSurface *surface,
|
|
cairo_region_t *shape_region)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_minimize (GdkSurface *surface)
|
|
{
|
|
if (GDK_SURFACE_DESTROYED (surface))
|
|
return;
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_unminimize (GdkSurface *surface)
|
|
{
|
|
if (GDK_SURFACE_DESTROYED (surface))
|
|
return;
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_maximize (GdkSurface *surface)
|
|
{
|
|
GdkBroadwaySurface *impl;
|
|
GdkDisplay *display;
|
|
GdkRectangle geom;
|
|
|
|
if (GDK_SURFACE_DESTROYED (surface))
|
|
return;
|
|
|
|
impl = GDK_BROADWAY_SURFACE (surface);
|
|
|
|
if (impl->maximized)
|
|
return;
|
|
|
|
impl->maximized = TRUE;
|
|
|
|
gdk_synthesize_surface_state (surface, 0, GDK_TOPLEVEL_STATE_MAXIMIZED);
|
|
|
|
impl->pre_maximize_x = surface->x;
|
|
impl->pre_maximize_y = surface->y;
|
|
impl->pre_maximize_width = surface->width;
|
|
impl->pre_maximize_height = surface->height;
|
|
|
|
display = gdk_surface_get_display (surface);
|
|
gdk_monitor_get_geometry (GDK_BROADWAY_DISPLAY (display)->monitor, &geom);
|
|
|
|
gdk_broadway_surface_move_resize (surface,
|
|
geom.x, geom.y,
|
|
geom.width, geom.height);
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_unmaximize (GdkSurface *surface)
|
|
{
|
|
GdkBroadwaySurface *impl;
|
|
|
|
if (GDK_SURFACE_DESTROYED (surface))
|
|
return;
|
|
|
|
impl = GDK_BROADWAY_SURFACE (surface);
|
|
|
|
if (!impl->maximized)
|
|
return;
|
|
|
|
impl->maximized = FALSE;
|
|
|
|
gdk_synthesize_surface_state (surface, GDK_TOPLEVEL_STATE_MAXIMIZED, 0);
|
|
|
|
gdk_broadway_surface_move_resize (surface,
|
|
impl->pre_maximize_x,
|
|
impl->pre_maximize_y,
|
|
impl->pre_maximize_width,
|
|
impl->pre_maximize_height);
|
|
}
|
|
|
|
typedef struct _MoveResizeData MoveResizeData;
|
|
|
|
struct _MoveResizeData
|
|
{
|
|
GdkDisplay *display;
|
|
|
|
GdkSurface *moveresize_surface;
|
|
GdkSurface *moveresize_emulation_surface;
|
|
gboolean is_resize;
|
|
GdkSurfaceEdge resize_edge;
|
|
int moveresize_button;
|
|
int moveresize_x;
|
|
int moveresize_y;
|
|
int moveresize_orig_x;
|
|
int moveresize_orig_y;
|
|
int moveresize_orig_width;
|
|
int moveresize_orig_height;
|
|
long moveresize_process_time;
|
|
GdkSurfaceHints moveresize_geom_mask;
|
|
GdkGeometry moveresize_geometry;
|
|
BroadwayInputMsg *moveresize_pending_event;
|
|
};
|
|
|
|
static MoveResizeData *
|
|
get_move_resize_data (GdkDisplay *display,
|
|
gboolean create)
|
|
{
|
|
GdkBroadwayDisplay *broadway_display;
|
|
MoveResizeData *mv_resize;
|
|
|
|
broadway_display = GDK_BROADWAY_DISPLAY (display);
|
|
|
|
mv_resize = broadway_display->move_resize_data;
|
|
|
|
if (!mv_resize && create)
|
|
{
|
|
mv_resize = g_new0 (MoveResizeData, 1);
|
|
mv_resize->display = display;
|
|
|
|
broadway_display->move_resize_data = mv_resize;
|
|
}
|
|
|
|
return mv_resize;
|
|
}
|
|
|
|
static void
|
|
update_pos (MoveResizeData *mv_resize,
|
|
int new_root_x,
|
|
int new_root_y)
|
|
{
|
|
int dx, dy;
|
|
|
|
dx = new_root_x - mv_resize->moveresize_x;
|
|
dy = new_root_y - mv_resize->moveresize_y;
|
|
|
|
if (mv_resize->is_resize)
|
|
{
|
|
int x, y, w, h;
|
|
|
|
x = mv_resize->moveresize_orig_x;
|
|
y = mv_resize->moveresize_orig_y;
|
|
|
|
w = mv_resize->moveresize_orig_width;
|
|
h = mv_resize->moveresize_orig_height;
|
|
|
|
switch (mv_resize->resize_edge)
|
|
{
|
|
case GDK_SURFACE_EDGE_NORTH_WEST:
|
|
x += dx;
|
|
y += dy;
|
|
w -= dx;
|
|
h -= dy;
|
|
break;
|
|
case GDK_SURFACE_EDGE_NORTH:
|
|
y += dy;
|
|
h -= dy;
|
|
break;
|
|
case GDK_SURFACE_EDGE_NORTH_EAST:
|
|
y += dy;
|
|
h -= dy;
|
|
w += dx;
|
|
break;
|
|
case GDK_SURFACE_EDGE_SOUTH_WEST:
|
|
h += dy;
|
|
x += dx;
|
|
w -= dx;
|
|
break;
|
|
case GDK_SURFACE_EDGE_SOUTH_EAST:
|
|
w += dx;
|
|
h += dy;
|
|
break;
|
|
case GDK_SURFACE_EDGE_SOUTH:
|
|
h += dy;
|
|
break;
|
|
case GDK_SURFACE_EDGE_EAST:
|
|
w += dx;
|
|
break;
|
|
case GDK_SURFACE_EDGE_WEST:
|
|
x += dx;
|
|
w -= dx;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
x = MAX (x, 0);
|
|
y = MAX (y, 0);
|
|
w = MAX (w, 1);
|
|
h = MAX (h, 1);
|
|
|
|
if (mv_resize->moveresize_geom_mask)
|
|
{
|
|
gdk_surface_constrain_size (&mv_resize->moveresize_geometry,
|
|
mv_resize->moveresize_geom_mask,
|
|
w, h, &w, &h);
|
|
}
|
|
|
|
gdk_broadway_surface_move_resize (mv_resize->moveresize_surface,
|
|
x, y, w, h);
|
|
}
|
|
else
|
|
{
|
|
int x, y;
|
|
|
|
x = mv_resize->moveresize_orig_x + dx;
|
|
y = mv_resize->moveresize_orig_y + dy;
|
|
|
|
gdk_broadway_surface_move (mv_resize->moveresize_surface, x, y);
|
|
}
|
|
}
|
|
|
|
static void
|
|
finish_drag (MoveResizeData *mv_resize)
|
|
{
|
|
gdk_surface_destroy (mv_resize->moveresize_emulation_surface);
|
|
mv_resize->moveresize_emulation_surface = NULL;
|
|
g_object_unref (mv_resize->moveresize_surface);
|
|
mv_resize->moveresize_surface = NULL;
|
|
g_clear_pointer (&mv_resize->moveresize_pending_event, g_free);
|
|
}
|
|
|
|
static gboolean
|
|
moveresize_lookahead (GdkDisplay *display,
|
|
MoveResizeData *mv_resize,
|
|
BroadwayInputMsg *event)
|
|
{
|
|
GdkBroadwayDisplay *broadway_display;
|
|
|
|
broadway_display = GDK_BROADWAY_DISPLAY (display);
|
|
|
|
return !_gdk_broadway_server_lookahead_event (broadway_display->server, "mb");
|
|
}
|
|
|
|
gboolean
|
|
_gdk_broadway_moveresize_handle_event (GdkDisplay *display,
|
|
BroadwayInputMsg *event)
|
|
{
|
|
guint button_mask = 0;
|
|
MoveResizeData *mv_resize = get_move_resize_data (display, FALSE);
|
|
|
|
if (!mv_resize || !mv_resize->moveresize_surface)
|
|
return FALSE;
|
|
|
|
button_mask = GDK_BUTTON1_MASK << (mv_resize->moveresize_button - 1);
|
|
|
|
switch (event->base.type)
|
|
{
|
|
case BROADWAY_EVENT_TOUCH:
|
|
if (event->touch.touch_type == 2) /* END */
|
|
{
|
|
update_pos (mv_resize,
|
|
event->touch.root_x,
|
|
event->touch.root_y);
|
|
|
|
finish_drag (mv_resize);
|
|
}
|
|
else if (event->touch.touch_type == 1) /* UPDATE */
|
|
{
|
|
if (mv_resize->moveresize_surface->resize_count > 0)
|
|
{
|
|
if (mv_resize->moveresize_pending_event)
|
|
*mv_resize->moveresize_pending_event = *event;
|
|
else
|
|
mv_resize->moveresize_pending_event =
|
|
g_memdup (event, sizeof (BroadwayInputMsg));
|
|
|
|
break;
|
|
}
|
|
update_pos (mv_resize,
|
|
event->touch.root_x,
|
|
event->touch.root_y);
|
|
}
|
|
|
|
break;
|
|
|
|
case BROADWAY_EVENT_POINTER_MOVE:
|
|
if (mv_resize->moveresize_surface->resize_count > 0)
|
|
{
|
|
if (mv_resize->moveresize_pending_event)
|
|
*mv_resize->moveresize_pending_event = *event;
|
|
else
|
|
mv_resize->moveresize_pending_event =
|
|
g_memdup (event, sizeof (BroadwayInputMsg));
|
|
|
|
break;
|
|
}
|
|
if (!moveresize_lookahead (display, mv_resize, event))
|
|
break;
|
|
|
|
update_pos (mv_resize,
|
|
event->pointer.root_x,
|
|
event->pointer.root_y);
|
|
|
|
/* 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->pointer.state & button_mask) == 0)
|
|
finish_drag (mv_resize);
|
|
break;
|
|
|
|
case BROADWAY_EVENT_BUTTON_RELEASE:
|
|
update_pos (mv_resize,
|
|
event->pointer.root_x,
|
|
event->pointer.root_y);
|
|
|
|
if (event->button.button == mv_resize->moveresize_button)
|
|
finish_drag (mv_resize);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
_gdk_broadway_moveresize_configure_done (GdkDisplay *display,
|
|
GdkSurface *surface)
|
|
{
|
|
BroadwayInputMsg *tmp_event;
|
|
MoveResizeData *mv_resize = get_move_resize_data (display, FALSE);
|
|
|
|
if (!mv_resize || surface != mv_resize->moveresize_surface)
|
|
return FALSE;
|
|
|
|
if (mv_resize->moveresize_pending_event)
|
|
{
|
|
tmp_event = mv_resize->moveresize_pending_event;
|
|
mv_resize->moveresize_pending_event = NULL;
|
|
_gdk_broadway_moveresize_handle_event (display, tmp_event);
|
|
g_free (tmp_event);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
create_moveresize_surface (MoveResizeData *mv_resize,
|
|
guint32 timestamp)
|
|
{
|
|
GdkGrabStatus status;
|
|
GdkSeat *seat;
|
|
GdkDevice *pointer;
|
|
|
|
g_assert (mv_resize->moveresize_emulation_surface == NULL);
|
|
|
|
mv_resize->moveresize_emulation_surface =
|
|
_gdk_broadway_display_create_surface (mv_resize->display,
|
|
GDK_SURFACE_TEMP,
|
|
NULL,
|
|
-100, -100, 1, 1);
|
|
|
|
gdk_broadway_surface_show (mv_resize->moveresize_emulation_surface, FALSE);
|
|
|
|
seat = gdk_display_get_default_seat (mv_resize->display);
|
|
pointer = gdk_seat_get_pointer (seat);
|
|
|
|
G_GNUC_BEGIN_IGNORE_DEPRECATIONS;
|
|
status = gdk_device_grab (pointer,
|
|
mv_resize->moveresize_emulation_surface,
|
|
FALSE,
|
|
GDK_BUTTON_RELEASE_MASK |
|
|
GDK_POINTER_MOTION_MASK,
|
|
NULL,
|
|
timestamp);
|
|
G_GNUC_END_IGNORE_DEPRECATIONS;
|
|
|
|
if (status != GDK_GRAB_SUCCESS)
|
|
{
|
|
/* If this fails, some other client has grabbed the surface
|
|
* already.
|
|
*/
|
|
finish_drag (mv_resize);
|
|
}
|
|
|
|
mv_resize->moveresize_process_time = 0;
|
|
}
|
|
|
|
static void
|
|
calculate_unmoving_origin (MoveResizeData *mv_resize)
|
|
{
|
|
gdk_surface_get_geometry (mv_resize->moveresize_surface,
|
|
&mv_resize->moveresize_orig_x,
|
|
&mv_resize->moveresize_orig_y,
|
|
NULL, NULL);
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_toplevel_begin_resize (GdkToplevel *toplevel,
|
|
GdkSurfaceEdge edge,
|
|
GdkDevice *device,
|
|
int button,
|
|
double x,
|
|
double y,
|
|
guint32 timestamp)
|
|
{
|
|
GdkSurface *surface = GDK_SURFACE (toplevel);
|
|
GdkBroadwaySurface *impl = GDK_BROADWAY_SURFACE (surface);
|
|
MoveResizeData *mv_resize;
|
|
|
|
if (GDK_SURFACE_DESTROYED (surface))
|
|
return;
|
|
|
|
if (impl->maximized)
|
|
return;
|
|
|
|
mv_resize = get_move_resize_data (gdk_surface_get_display (surface), TRUE);
|
|
|
|
if (mv_resize->moveresize_surface != NULL)
|
|
return; /* already a drag operation in progress */
|
|
|
|
mv_resize->is_resize = TRUE;
|
|
mv_resize->moveresize_button = button;
|
|
mv_resize->resize_edge = edge;
|
|
mv_resize->moveresize_x = x + surface->x;
|
|
mv_resize->moveresize_y = y + surface->y;
|
|
mv_resize->moveresize_surface = g_object_ref (surface);
|
|
|
|
mv_resize->moveresize_orig_width = gdk_surface_get_width (surface);
|
|
mv_resize->moveresize_orig_height = gdk_surface_get_height (surface);
|
|
|
|
mv_resize->moveresize_geom_mask = impl->geometry_hints_mask;
|
|
mv_resize->moveresize_geometry = impl->geometry_hints;
|
|
|
|
calculate_unmoving_origin (mv_resize);
|
|
|
|
create_moveresize_surface (mv_resize, timestamp);
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_toplevel_begin_move (GdkToplevel *toplevel,
|
|
GdkDevice *device,
|
|
int button,
|
|
double x,
|
|
double y,
|
|
guint32 timestamp)
|
|
{
|
|
GdkSurface *surface = GDK_SURFACE (toplevel);
|
|
GdkBroadwaySurface *impl = GDK_BROADWAY_SURFACE (surface);
|
|
MoveResizeData *mv_resize;
|
|
|
|
if (GDK_SURFACE_DESTROYED (surface))
|
|
return;
|
|
|
|
if (impl->maximized)
|
|
return;
|
|
|
|
mv_resize = get_move_resize_data (gdk_surface_get_display (surface), TRUE);
|
|
|
|
if (mv_resize->moveresize_surface != NULL)
|
|
return; /* already a drag operation in progress */
|
|
|
|
mv_resize->is_resize = FALSE;
|
|
mv_resize->moveresize_button = button;
|
|
mv_resize->moveresize_x = x + surface->x;
|
|
mv_resize->moveresize_y = y + surface->y;
|
|
mv_resize->moveresize_surface = g_object_ref (surface);
|
|
|
|
mv_resize->moveresize_orig_width = gdk_surface_get_width (surface);
|
|
mv_resize->moveresize_orig_height = gdk_surface_get_height (surface);
|
|
|
|
mv_resize->moveresize_geom_mask = impl->geometry_hints_mask;
|
|
mv_resize->moveresize_geometry = impl->geometry_hints;
|
|
|
|
calculate_unmoving_origin (mv_resize);
|
|
|
|
create_moveresize_surface (mv_resize, timestamp);
|
|
}
|
|
|
|
static gboolean
|
|
gdk_broadway_surface_beep (GdkSurface *surface)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_surface_class_init (GdkBroadwaySurfaceClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
GdkSurfaceClass *impl_class = GDK_SURFACE_CLASS (klass);
|
|
|
|
object_class->finalize = gdk_broadway_surface_finalize;
|
|
|
|
impl_class->ref_cairo_surface = gdk_broadway_surface_ref_cairo_surface;
|
|
impl_class->hide = gdk_broadway_surface_hide;
|
|
impl_class->get_geometry = gdk_broadway_surface_get_geometry;
|
|
impl_class->get_root_coords = gdk_broadway_surface_get_root_coords;
|
|
impl_class->get_device_state = gdk_broadway_surface_get_device_state;
|
|
impl_class->set_input_region = gdk_broadway_surface_set_input_region;
|
|
impl_class->destroy = _gdk_broadway_surface_destroy;
|
|
impl_class->beep = gdk_broadway_surface_beep;
|
|
impl_class->destroy_notify = gdk_broadway_surface_destroy_notify;
|
|
impl_class->drag_begin = _gdk_broadway_surface_drag_begin;
|
|
impl_class->get_scale_factor = gdk_broadway_surface_get_scale_factor;
|
|
}
|
|
|
|
#define LAST_PROP 1
|
|
|
|
typedef struct
|
|
{
|
|
GdkBroadwaySurface parent_instance;
|
|
} GdkBroadwayPopup;
|
|
|
|
typedef struct
|
|
{
|
|
GdkBroadwaySurfaceClass parent_class;
|
|
} GdkBroadwayPopupClass;
|
|
|
|
static void gdk_broadway_popup_iface_init (GdkPopupInterface *iface);
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GdkBroadwayPopup, gdk_broadway_popup, GDK_TYPE_BROADWAY_SURFACE,
|
|
G_IMPLEMENT_INTERFACE (GDK_TYPE_POPUP,
|
|
gdk_broadway_popup_iface_init))
|
|
|
|
static void
|
|
gdk_broadway_popup_init (GdkBroadwayPopup *popup)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_popup_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GdkSurface *surface = GDK_SURFACE (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case LAST_PROP + GDK_POPUP_PROP_PARENT:
|
|
g_value_set_object (value, surface->parent);
|
|
break;
|
|
|
|
case LAST_PROP + GDK_POPUP_PROP_AUTOHIDE:
|
|
g_value_set_boolean (value, surface->autohide);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_popup_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GdkSurface *surface = GDK_SURFACE (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case LAST_PROP + GDK_POPUP_PROP_PARENT:
|
|
surface->parent = g_value_dup_object (value);
|
|
if (surface->parent != NULL)
|
|
surface->parent->children = g_list_prepend (surface->parent->children, surface);
|
|
break;
|
|
|
|
case LAST_PROP + GDK_POPUP_PROP_AUTOHIDE:
|
|
surface->autohide = g_value_get_boolean (value);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_popup_class_init (GdkBroadwayPopupClass *class)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
|
|
object_class->get_property = gdk_broadway_popup_get_property;
|
|
object_class->set_property = gdk_broadway_popup_set_property;
|
|
|
|
gdk_popup_install_properties (object_class, 1);
|
|
}
|
|
|
|
static gboolean
|
|
gdk_broadway_popup_present (GdkPopup *popup,
|
|
int width,
|
|
int height,
|
|
GdkPopupLayout *layout)
|
|
{
|
|
return gdk_broadway_surface_present_popup (GDK_SURFACE (popup), width, height, layout);
|
|
}
|
|
|
|
static GdkGravity
|
|
gdk_broadway_popup_get_surface_anchor (GdkPopup *popup)
|
|
{
|
|
return GDK_SURFACE (popup)->popup.surface_anchor;
|
|
}
|
|
|
|
static GdkGravity
|
|
gdk_broadway_popup_get_rect_anchor (GdkPopup *popup)
|
|
{
|
|
return GDK_SURFACE (popup)->popup.rect_anchor;
|
|
}
|
|
|
|
static int
|
|
gdk_broadway_popup_get_position_x (GdkPopup *popup)
|
|
{
|
|
return GDK_SURFACE (popup)->x;
|
|
}
|
|
|
|
static int
|
|
gdk_broadway_popup_get_position_y (GdkPopup *popup)
|
|
{
|
|
return GDK_SURFACE (popup)->y;
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_popup_iface_init (GdkPopupInterface *iface)
|
|
{
|
|
iface->present = gdk_broadway_popup_present;
|
|
iface->get_surface_anchor = gdk_broadway_popup_get_surface_anchor;
|
|
iface->get_rect_anchor = gdk_broadway_popup_get_rect_anchor;
|
|
iface->get_position_x = gdk_broadway_popup_get_position_x;
|
|
iface->get_position_y = gdk_broadway_popup_get_position_y;
|
|
}
|
|
|
|
typedef struct
|
|
{
|
|
GdkBroadwaySurface parent_instance;
|
|
} GdkBroadwayToplevel;
|
|
|
|
typedef struct
|
|
{
|
|
GdkBroadwaySurfaceClass parent_class;
|
|
} GdkBroadwayToplevelClass;
|
|
|
|
static void gdk_broadway_toplevel_iface_init (GdkToplevelInterface *iface);
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GdkBroadwayToplevel, gdk_broadway_toplevel, GDK_TYPE_BROADWAY_SURFACE,
|
|
G_IMPLEMENT_INTERFACE (GDK_TYPE_TOPLEVEL,
|
|
gdk_broadway_toplevel_iface_init))
|
|
|
|
static void
|
|
gdk_broadway_toplevel_init (GdkBroadwayToplevel *toplevel)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_toplevel_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GdkSurface *surface = GDK_SURFACE (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_TITLE:
|
|
gdk_broadway_surface_set_title (surface, g_value_get_string (value));
|
|
g_object_notify_by_pspec (G_OBJECT (surface), pspec);
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_STARTUP_ID:
|
|
gdk_broadway_surface_set_startup_id (surface, g_value_get_string (value));
|
|
g_object_notify_by_pspec (G_OBJECT (surface), pspec);
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_TRANSIENT_FOR:
|
|
gdk_broadway_surface_set_transient_for (surface, g_value_get_object (value));
|
|
g_object_notify_by_pspec (G_OBJECT (surface), pspec);
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_MODAL:
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_ICON_LIST:
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_DECORATED:
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_DELETABLE:
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_SHORTCUTS_INHIBITED:
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_toplevel_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
GdkSurface *surface = GDK_SURFACE (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_STATE:
|
|
g_value_set_flags (value, surface->state);
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_TITLE:
|
|
g_value_set_string (value, "");
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_STARTUP_ID:
|
|
g_value_set_string (value, "");
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_TRANSIENT_FOR:
|
|
g_value_set_object (value, surface->transient_for);
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_ICON_LIST:
|
|
g_value_set_pointer (value, NULL);
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_DECORATED:
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_DELETABLE:
|
|
break;
|
|
|
|
case LAST_PROP + GDK_TOPLEVEL_PROP_SHORTCUTS_INHIBITED:
|
|
g_value_set_boolean (value, surface->shortcuts_inhibited);
|
|
break;
|
|
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_toplevel_class_init (GdkBroadwayToplevelClass *class)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
|
|
|
object_class->get_property = gdk_broadway_toplevel_get_property;
|
|
object_class->set_property = gdk_broadway_toplevel_set_property;
|
|
|
|
gdk_toplevel_install_properties (object_class, 1);
|
|
}
|
|
|
|
static void
|
|
show_surface (GdkSurface *surface)
|
|
{
|
|
gboolean was_mapped;
|
|
|
|
if (surface->destroyed)
|
|
return;
|
|
|
|
was_mapped = GDK_SURFACE_IS_MAPPED (surface);
|
|
|
|
if (!was_mapped)
|
|
gdk_surface_set_is_mapped (surface, TRUE);
|
|
|
|
gdk_broadway_surface_show (surface, FALSE);
|
|
|
|
if (!was_mapped)
|
|
gdk_surface_invalidate_rect (surface, NULL);
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_toplevel_present (GdkToplevel *toplevel,
|
|
GdkToplevelLayout *layout)
|
|
{
|
|
GdkSurface *surface = GDK_SURFACE (toplevel);
|
|
GdkBroadwaySurface *impl = GDK_BROADWAY_SURFACE (surface);
|
|
GdkDisplay *display = gdk_surface_get_display (surface);
|
|
GdkMonitor *monitor;
|
|
GdkToplevelSize size;
|
|
int bounds_width, bounds_height;
|
|
int width, height;
|
|
GdkGeometry geometry;
|
|
GdkSurfaceHints mask;
|
|
|
|
gdk_broadway_surface_unminimize (surface);
|
|
|
|
monitor = gdk_display_get_monitor_at_surface (display, surface);
|
|
if (monitor)
|
|
{
|
|
GdkRectangle monitor_geometry;
|
|
|
|
gdk_monitor_get_geometry (monitor, &monitor_geometry);
|
|
bounds_width = monitor_geometry.width;
|
|
bounds_height = monitor_geometry.height;
|
|
}
|
|
else
|
|
{
|
|
bounds_width = G_MAXINT;
|
|
bounds_height = G_MAXINT;
|
|
}
|
|
|
|
gdk_toplevel_size_init (&size, bounds_width, bounds_height);
|
|
gdk_toplevel_notify_compute_size (toplevel, &size);
|
|
g_warn_if_fail (size.width > 0);
|
|
g_warn_if_fail (size.height > 0);
|
|
width = size.width;
|
|
height = size.height;
|
|
|
|
if (gdk_toplevel_layout_get_resizable (layout))
|
|
{
|
|
geometry.min_width = size.min_width;
|
|
geometry.min_height = size.min_height;
|
|
mask = GDK_HINT_MIN_SIZE;
|
|
}
|
|
else
|
|
{
|
|
geometry.max_width = geometry.min_width = width;
|
|
geometry.max_height = geometry.min_height = height;
|
|
mask = GDK_HINT_MIN_SIZE | GDK_HINT_MAX_SIZE;
|
|
}
|
|
gdk_broadway_surface_set_geometry_hints (surface, &geometry, mask);
|
|
gdk_surface_constrain_size (&geometry, mask, width, height, &width, &height);
|
|
gdk_broadway_surface_toplevel_resize (surface, width, height);
|
|
|
|
if (gdk_toplevel_layout_get_maximized (layout))
|
|
gdk_broadway_surface_maximize (surface);
|
|
else
|
|
gdk_broadway_surface_unmaximize (surface);
|
|
|
|
if (size.shadow.is_valid)
|
|
{
|
|
impl->shadow_left = size.shadow.left;
|
|
impl->shadow_right = size.shadow.right;
|
|
impl->shadow_top = size.shadow.top;
|
|
impl->shadow_bottom = size.shadow.bottom;
|
|
}
|
|
|
|
show_surface (surface);
|
|
}
|
|
|
|
static gboolean
|
|
gdk_broadway_toplevel_minimize (GdkToplevel *toplevel)
|
|
{
|
|
gdk_broadway_surface_minimize (GDK_SURFACE (toplevel));
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
gdk_broadway_toplevel_lower (GdkToplevel *toplevel)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_toplevel_focus (GdkToplevel *toplevel,
|
|
guint32 timestamp)
|
|
{
|
|
gdk_broadway_surface_focus (GDK_SURFACE (toplevel), timestamp);
|
|
}
|
|
|
|
static gboolean
|
|
gdk_broadway_toplevel_show_window_menu (GdkToplevel *toplevel,
|
|
GdkEvent *event)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_toplevel_iface_init (GdkToplevelInterface *iface)
|
|
{
|
|
iface->present = gdk_broadway_toplevel_present;
|
|
iface->minimize = gdk_broadway_toplevel_minimize;
|
|
iface->lower = gdk_broadway_toplevel_lower;
|
|
iface->focus = gdk_broadway_toplevel_focus;
|
|
iface->show_window_menu = gdk_broadway_toplevel_show_window_menu;
|
|
iface->begin_resize = gdk_broadway_toplevel_begin_resize;
|
|
iface->begin_move = gdk_broadway_toplevel_begin_move;
|
|
}
|
|
|
|
typedef struct
|
|
{
|
|
GdkBroadwaySurface parent_instance;
|
|
} GdkBroadwayDragSurface;
|
|
|
|
typedef struct
|
|
{
|
|
GdkBroadwaySurfaceClass parent_class;
|
|
} GdkBroadwayDragSurfaceClass;
|
|
|
|
static void gdk_broadway_drag_surface_iface_init (GdkDragSurfaceInterface *iface);
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (GdkBroadwayDragSurface, gdk_broadway_drag_surface, GDK_TYPE_BROADWAY_SURFACE,
|
|
G_IMPLEMENT_INTERFACE (GDK_TYPE_DRAG_SURFACE,
|
|
gdk_broadway_drag_surface_iface_init))
|
|
|
|
static void
|
|
gdk_broadway_drag_surface_init (GdkBroadwayDragSurface *surface)
|
|
{
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_drag_surface_class_init (GdkBroadwayDragSurfaceClass *class)
|
|
{
|
|
}
|
|
|
|
static gboolean
|
|
gdk_broadway_drag_surface_present (GdkDragSurface *drag_surface,
|
|
int width,
|
|
int height)
|
|
{
|
|
GdkSurface *surface = GDK_SURFACE (drag_surface);
|
|
|
|
gdk_broadway_surface_toplevel_resize (surface, width, height);
|
|
show_surface (surface);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
gdk_broadway_drag_surface_iface_init (GdkDragSurfaceInterface *iface)
|
|
{
|
|
iface->present = gdk_broadway_drag_surface_present;
|
|
}
|