/* 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 . */ /* * 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 "gdkdisplay.h" #include "gdksurfaceprivate.h" #include "gdkdisplay-broadway.h" #include "gdkprivate-broadway.h" #include "gdkinternals.h" #include "gdkdeviceprivate.h" #include "gdkeventsource.h" #include "gdktextureprivate.h" #include "gdkframeclockidleprivate.h" #include #include #include /* Forward declarations */ static void gdk_broadway_surface_finalize (GObject *object); #define SURFACE_IS_TOPLEVEL(surface) TRUE G_DEFINE_TYPE (GdkBroadwaySurface, gdk_broadway_surface, GDK_TYPE_SURFACE) /* 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) { 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 rountrip 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; frame_clock = _gdk_frame_clock_idle_new (); surface = g_object_new (GDK_TYPE_BROADWAY_SURFACE, "surface-type", surface_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->id = _gdk_broadway_server_new_surface (broadway_display->server, surface->x, surface->y, surface->width, surface->height, surface->surface_type == GDK_SURFACE_TEMP); g_hash_table_insert (broadway_display->id_ht, GINT_TO_POINTER(impl->id), surface); if (!surface->parent) broadway_display->toplevels = g_list_prepend (broadway_display->toplevels, impl); connect_frame_clock (surface); 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 void gdk_broadway_surface_withdraw (GdkSurface *surface) { gdk_broadway_surface_hide (surface); } static void gdk_broadway_surface_move_resize_internal (GdkSurface *surface, gboolean with_move, gint x, gint y, gint width, gint height) { GdkBroadwaySurface *impl = GDK_BROADWAY_SURFACE (surface); GdkBroadwayDisplay *broadway_display; gboolean size_changed; 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; } } if (surface->parent) { impl->offset_x = x - surface->parent->x; impl->offset_y = y - surface->parent->y; } _gdk_broadway_server_surface_move_resize (broadway_display->server, impl->id, with_move, x, y, surface->width, surface->height); queue_flush (surface); if (size_changed) { surface->resize_count++; _gdk_surface_update_size (surface); } } void gdk_broadway_surface_move_resize (GdkSurface *surface, gint x, gint y, gint width, gint height) { gdk_broadway_surface_move_resize_internal (surface, TRUE, x, y, width, height); } static void gdk_broadway_surface_toplevel_resize (GdkSurface *surface, gint width, gint height) { gdk_broadway_surface_move_resize_internal (surface, FALSE, 0, 0, width, height); } static void gdk_broadway_surface_move (GdkSurface *surface, gint x, gint y) { gdk_broadway_surface_move_resize_internal (surface, TRUE, x, y, -1, -1); } static void gdk_broadway_surface_moved_to_rect (GdkSurface *surface, GdkRectangle final_rect) { GdkSurface *toplevel; int x, y; if (surface->surface_type == GDK_SURFACE_POPUP) toplevel = surface->parent; else toplevel = surface->transient_for; gdk_surface_get_origin (toplevel, &x, &y); 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 gdk_broadway_surface_move_to_rect (GdkSurface *surface, const GdkRectangle *rect, GdkGravity rect_anchor, GdkGravity surface_anchor, GdkAnchorHints anchor_hints, gint rect_anchor_dx, gint rect_anchor_dy) { gdk_surface_move_to_rect_helper (surface, rect, rect_anchor, surface_anchor, anchor_hints, rect_anchor_dx, rect_anchor_dy, gdk_broadway_surface_moved_to_rect); } static void gdk_broadway_surface_raise (GdkSurface *surface) { } static void gdk_broadway_surface_restack_toplevel (GdkSurface *surface, GdkSurface *sibling, gboolean above) { } static void gdk_broadway_surface_lower (GdkSurface *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) || !surface->accept_focus) 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_type_hint (GdkSurface *surface, GdkSurfaceTypeHint hint) { } static GdkSurfaceTypeHint gdk_broadway_surface_get_type_hint (GdkSurface *surface) { return GDK_SURFACE_TYPE_HINT_NORMAL; } static void gdk_broadway_surface_set_modal_hint (GdkSurface *surface, gboolean modal) { } 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 gchar *title) { } static void gdk_broadway_surface_set_startup_id (GdkSurface *surface, const gchar *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); 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, gint *x, gint *y, gint *width, gint *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, gint x, gint y, gint *root_x, gint *root_y) { if (root_x) *root_x = x + surface->x; if (root_y) *root_y = y + surface->y; } static gboolean gdk_broadway_surface_get_device_state (GdkSurface *surface, GdkDevice *device, gdouble *x, gdouble *y, GdkModifierType *mask) { GdkSurface *child; g_return_val_if_fail (surface == NULL || GDK_IS_SURFACE (surface), FALSE); if (GDK_SURFACE_DESTROYED (surface)) return FALSE; GDK_DEVICE_GET_CLASS (device)->query_state (device, surface, &child, NULL, NULL, x, y, mask); return child != NULL; } static void gdk_broadway_surface_input_shape_combine_region (GdkSurface *surface, const cairo_region_t *shape_region, gint offset_x, gint offset_y) { } static void gdk_broadway_surface_set_accept_focus (GdkSurface *surface, gboolean accept_focus) { accept_focus = accept_focus != FALSE; if (surface->accept_focus != accept_focus) { surface->accept_focus = accept_focus; } } static void gdk_broadway_surface_set_focus_on_map (GdkSurface *surface, gboolean focus_on_map) { focus_on_map = focus_on_map != FALSE; if (surface->focus_on_map != focus_on_map) { surface->focus_on_map = focus_on_map; } } static void gdk_broadway_surface_set_icon_list (GdkSurface *surface, GList *surfaces) { } static void gdk_broadway_surface_set_icon_name (GdkSurface *surface, const gchar *name) { if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return; g_object_set_qdata (G_OBJECT (surface), g_quark_from_static_string ("gdk-icon-name-set"), GUINT_TO_POINTER (name != NULL)); } static void gdk_broadway_surface_minimize (GdkSurface *surface) { if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return; } static void gdk_broadway_surface_unminimize (GdkSurface *surface) { if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return; } static void gdk_broadway_surface_stick (GdkSurface *surface) { if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return; } static void gdk_broadway_surface_unstick (GdkSurface *surface) { if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return; } static void gdk_broadway_surface_maximize (GdkSurface *surface) { GdkBroadwaySurface *impl; GdkDisplay *display; GdkMonitor *monitor; GdkRectangle geom; if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return; impl = GDK_BROADWAY_SURFACE (surface); if (impl->maximized) return; impl->maximized = TRUE; gdk_synthesize_surface_state (surface, 0, GDK_SURFACE_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); monitor = gdk_display_get_primary_monitor (display); gdk_monitor_get_geometry (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) || !SURFACE_IS_TOPLEVEL (surface)) return; impl = GDK_BROADWAY_SURFACE (surface); if (!impl->maximized) return; impl->maximized = FALSE; gdk_synthesize_surface_state (surface, GDK_SURFACE_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); } static void gdk_broadway_surface_fullscreen (GdkSurface *surface) { if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return; } static void gdk_broadway_surface_unfullscreen (GdkSurface *surface) { if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return; } static void gdk_broadway_surface_set_keep_above (GdkSurface *surface, gboolean setting) { g_return_if_fail (GDK_IS_SURFACE (surface)); if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return; } static void gdk_broadway_surface_set_keep_below (GdkSurface *surface, gboolean setting) { g_return_if_fail (GDK_IS_SURFACE (surface)); if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return; } static void gdk_broadway_surface_set_decorations (GdkSurface *surface, GdkWMDecoration decorations) { if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return; } static gboolean gdk_broadway_surface_get_decorations (GdkSurface *surface, GdkWMDecoration *decorations) { gboolean result = FALSE; if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return FALSE; return result; } static void gdk_broadway_surface_set_functions (GdkSurface *surface, GdkWMFunction functions) { g_return_if_fail (GDK_IS_SURFACE (surface)); if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return; } void gdk_broadway_surface_update_popups (GdkSurface *parent) { GList *l; for (l = parent ->children; l; l = l->next) { GdkBroadwaySurface *popup_impl = l->data; GdkSurface *popup = GDK_SURFACE (popup_impl); int new_x = parent->x + popup_impl->offset_x; int new_y = parent->y + popup_impl->offset_y; if (new_x != popup->x || new_y != popup->y) gdk_broadway_surface_move_resize (popup, new_x, new_y, popup->width, popup->height); gdk_broadway_surface_restack_toplevel (popup, parent, TRUE); } } typedef struct _MoveResizeData MoveResizeData; struct _MoveResizeData { GdkDisplay *display; GdkSurface *moveresize_surface; GdkSurface *moveresize_emulation_surface; gboolean is_resize; GdkSurfaceEdge resize_edge; gint moveresize_button; gint moveresize_x; gint moveresize_y; gint moveresize_orig_x; gint moveresize_orig_y; gint moveresize_orig_width; gint 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, gint new_root_x, gint new_root_y) { gint dx, dy; dx = new_root_x - mv_resize->moveresize_x; dy = new_root_y - mv_resize->moveresize_y; if (mv_resize->is_resize) { gint 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 { gint 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; GdkRectangle rect = { -100, -100, 1, 1 }; g_assert (mv_resize->moveresize_emulation_surface == NULL); mv_resize->moveresize_emulation_surface = gdk_surface_new_temp (mv_resize->display, &rect); gdk_surface_show (mv_resize->moveresize_emulation_surface); 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, GDK_OWNERSHIP_APPLICATION, 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) { gint x, y, width, height; if (mv_resize->moveresize_geom_mask & GDK_HINT_WIN_GRAVITY && mv_resize->moveresize_geometry.win_gravity == GDK_GRAVITY_STATIC) { gdk_surface_get_origin (mv_resize->moveresize_surface, &mv_resize->moveresize_orig_x, &mv_resize->moveresize_orig_y); } else { gdk_surface_get_geometry (mv_resize->moveresize_surface, &x, &y, &width, &height); switch (mv_resize->moveresize_geometry.win_gravity) { case GDK_GRAVITY_NORTH_WEST: mv_resize->moveresize_orig_x = x; mv_resize->moveresize_orig_y = y; break; case GDK_GRAVITY_NORTH: mv_resize->moveresize_orig_x = x + width / 2; mv_resize->moveresize_orig_y = y; break; case GDK_GRAVITY_NORTH_EAST: mv_resize->moveresize_orig_x = x = width; mv_resize->moveresize_orig_y = y; break; case GDK_GRAVITY_WEST: mv_resize->moveresize_orig_x = x; mv_resize->moveresize_orig_y = y + height / 2; break; case GDK_GRAVITY_CENTER: mv_resize->moveresize_orig_x = x + width / 2; mv_resize->moveresize_orig_y = y + height / 2; break; case GDK_GRAVITY_EAST: mv_resize->moveresize_orig_x = x + width; mv_resize->moveresize_orig_y = y + height / 2; break; case GDK_GRAVITY_SOUTH_WEST: mv_resize->moveresize_orig_x = x + width; mv_resize->moveresize_orig_y = y + height; break; case GDK_GRAVITY_SOUTH: mv_resize->moveresize_orig_x = x + width / 2; mv_resize->moveresize_orig_y = y + height; break; case GDK_GRAVITY_SOUTH_EAST: mv_resize->moveresize_orig_x = x; mv_resize->moveresize_orig_y = y + height; break; case GDK_GRAVITY_STATIC: default: mv_resize->moveresize_orig_x = x; mv_resize->moveresize_orig_y = y; break; } } } static void gdk_broadway_surface_begin_resize_drag (GdkSurface *surface, GdkSurfaceEdge edge, GdkDevice *device, gint button, gint x, gint y, guint32 timestamp) { MoveResizeData *mv_resize; GdkBroadwaySurface *impl; impl = GDK_BROADWAY_SURFACE (surface); if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (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_surface_begin_move_drag (GdkSurface *surface, GdkDevice *device, gint button, gint x, gint y, guint32 timestamp) { MoveResizeData *mv_resize; GdkBroadwaySurface *impl; impl = GDK_BROADWAY_SURFACE (surface); if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (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_set_opacity (GdkSurface *surface, gdouble opacity) { g_return_if_fail (GDK_IS_SURFACE (surface)); if (GDK_SURFACE_DESTROYED (surface) || !SURFACE_IS_TOPLEVEL (surface)) return; if (opacity < 0) opacity = 0; else if (opacity > 1) opacity = 1; } guint32 gdk_broadway_get_last_seen_time (GdkSurface *surface) { GdkDisplay *display; display = gdk_surface_get_display (surface); return _gdk_broadway_server_get_last_seen_time (GDK_BROADWAY_DISPLAY (display)->server); } 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->show = gdk_broadway_surface_show; impl_class->hide = gdk_broadway_surface_hide; impl_class->withdraw = gdk_broadway_surface_withdraw; impl_class->raise = gdk_broadway_surface_raise; impl_class->lower = gdk_broadway_surface_lower; impl_class->restack_toplevel = gdk_broadway_surface_restack_toplevel; impl_class->toplevel_resize = gdk_broadway_surface_toplevel_resize; impl_class->move_to_rect = gdk_broadway_surface_move_to_rect; 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->input_shape_combine_region = gdk_broadway_surface_input_shape_combine_region; impl_class->destroy = _gdk_broadway_surface_destroy; impl_class->beep = gdk_broadway_surface_beep; impl_class->focus = gdk_broadway_surface_focus; impl_class->set_type_hint = gdk_broadway_surface_set_type_hint; impl_class->get_type_hint = gdk_broadway_surface_get_type_hint; impl_class->set_modal_hint = gdk_broadway_surface_set_modal_hint; impl_class->set_geometry_hints = gdk_broadway_surface_set_geometry_hints; impl_class->set_title = gdk_broadway_surface_set_title; impl_class->set_startup_id = gdk_broadway_surface_set_startup_id; impl_class->set_transient_for = gdk_broadway_surface_set_transient_for; impl_class->set_accept_focus = gdk_broadway_surface_set_accept_focus; impl_class->set_focus_on_map = gdk_broadway_surface_set_focus_on_map; impl_class->set_icon_list = gdk_broadway_surface_set_icon_list; impl_class->set_icon_name = gdk_broadway_surface_set_icon_name; impl_class->minimize = gdk_broadway_surface_minimize; impl_class->unminimize = gdk_broadway_surface_unminimize; impl_class->stick = gdk_broadway_surface_stick; impl_class->unstick = gdk_broadway_surface_unstick; impl_class->maximize = gdk_broadway_surface_maximize; impl_class->unmaximize = gdk_broadway_surface_unmaximize; impl_class->fullscreen = gdk_broadway_surface_fullscreen; impl_class->unfullscreen = gdk_broadway_surface_unfullscreen; impl_class->set_keep_above = gdk_broadway_surface_set_keep_above; impl_class->set_keep_below = gdk_broadway_surface_set_keep_below; impl_class->set_decorations = gdk_broadway_surface_set_decorations; impl_class->get_decorations = gdk_broadway_surface_get_decorations; impl_class->set_functions = gdk_broadway_surface_set_functions; impl_class->begin_resize_drag = gdk_broadway_surface_begin_resize_drag; impl_class->begin_move_drag = gdk_broadway_surface_begin_move_drag; impl_class->set_opacity = gdk_broadway_surface_set_opacity; impl_class->destroy_notify = gdk_broadway_surface_destroy_notify; impl_class->register_dnd = _gdk_broadway_surface_register_dnd; impl_class->drag_begin = _gdk_broadway_surface_drag_begin; }