/* 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 "gdkwindow-broadway.h" #include "gdkscreen-broadway.h" #include "gdkwindow.h" #include "gdkwindowimpl.h" #include "gdkdisplay-broadway.h" #include "gdkprivate-broadway.h" #include "gdkinternals.h" #include "gdkdeviceprivate.h" #include "gdkeventsource.h" #include #include #include #include #include /* Forward declarations */ static void gdk_window_broadway_set_background (GdkWindow *window, cairo_pattern_t *pattern); static void gdk_window_impl_broadway_finalize (GObject *object); static const cairo_user_data_key_t gdk_broadway_cairo_key; #define WINDOW_IS_TOPLEVEL_OR_FOREIGN(window) \ (GDK_WINDOW_TYPE (window) != GDK_WINDOW_CHILD && \ GDK_WINDOW_TYPE (window) != GDK_WINDOW_OFFSCREEN) #define WINDOW_IS_TOPLEVEL(window) \ (GDK_WINDOW_TYPE (window) != GDK_WINDOW_CHILD && \ GDK_WINDOW_TYPE (window) != GDK_WINDOW_FOREIGN && \ GDK_WINDOW_TYPE (window) != GDK_WINDOW_OFFSCREEN) struct _GdkBroadwayWindow { GdkWindow parent; }; struct _GdkBroadwayWindowClass { GdkWindowClass parent_class; }; G_DEFINE_TYPE (GdkBroadwayWindow, gdk_broadway_window, GDK_TYPE_WINDOW) static void gdk_broadway_window_class_init (GdkBroadwayWindowClass *broadway_window_class) { } static void gdk_broadway_window_init (GdkBroadwayWindow *broadway_window) { } G_DEFINE_TYPE (GdkWindowImplBroadway, gdk_window_impl_broadway, GDK_TYPE_WINDOW_IMPL) static void diff_surfaces (cairo_surface_t *surface, cairo_surface_t *old_surface) { guint8 *data, *old_data; guint32 *line, *old_line; int w, h, stride, old_stride; int x, y; data = cairo_image_surface_get_data (surface); old_data = cairo_image_surface_get_data (old_surface); w = cairo_image_surface_get_width (surface); h = cairo_image_surface_get_height (surface); stride = cairo_image_surface_get_stride (surface); old_stride = cairo_image_surface_get_stride (old_surface); for (y = 0; y < h; y++) { line = (guint32 *)data; old_line = (guint32 *)old_data; for (x = 0; x < w; x++) { if ((*line & 0xffffff) == (*old_line & 0xffffff)) *old_line = 0; else *old_line = *line | 0xff000000; line ++; old_line ++; } data += stride; old_data += old_stride; } } static guint dirty_flush_id = 0; static void window_data_send (BroadwayOutput *output, GdkWindowImplBroadway *impl) { cairo_t *cr; if (impl->surface == NULL) return; if (impl->last_synced) { diff_surfaces (impl->surface, impl->last_surface); broadway_output_put_rgba (output, impl->id, 0, 0, cairo_image_surface_get_width (impl->last_surface), cairo_image_surface_get_height (impl->last_surface), cairo_image_surface_get_stride (impl->last_surface), cairo_image_surface_get_data (impl->last_surface)); } else { impl->last_synced = TRUE; broadway_output_put_rgb (output, impl->id, 0, 0, cairo_image_surface_get_width (impl->surface), cairo_image_surface_get_height (impl->surface), cairo_image_surface_get_stride (impl->surface), cairo_image_surface_get_data (impl->surface)); } broadway_output_surface_flush (output, impl->id); cr = cairo_create (impl->last_surface); cairo_set_source_surface (cr, impl->surface, 0, 0); cairo_paint (cr); cairo_destroy (cr); } static gboolean dirty_flush_idle (gpointer data) { GList *l; GdkBroadwayDisplay *display; BroadwayOutput *output; dirty_flush_id = 0; display = GDK_BROADWAY_DISPLAY (gdk_display_get_default ()); output = display->output; if (output == NULL) return FALSE; for (l = display->toplevels; l != NULL; l = l->next) { GdkWindowImplBroadway *impl = l->data; if (impl->dirty) { impl->dirty = FALSE; window_data_send (display->output, impl); } } gdk_display_flush (GDK_DISPLAY (display)); return FALSE; } static void queue_dirty_flush (GdkBroadwayDisplay *display) { if (dirty_flush_id == 0 && display->output != NULL) dirty_flush_id = gdk_threads_add_idle (dirty_flush_idle, NULL); } void _gdk_broadway_resync_windows (void) { GdkBroadwayDisplay *display; GList *l; dirty_flush_id = 0; display = GDK_BROADWAY_DISPLAY (gdk_display_get_default ()); /* First create all windows */ for (l = display->toplevels; l != NULL; l = l->next) { GdkWindowImplBroadway *impl = l->data; GdkWindow *window; window = impl->wrapper; if (impl->id == 0) continue; /* Skip root */ impl->dirty = FALSE; impl->last_synced = FALSE; broadway_output_new_surface (display->output, impl->id, window->x, window->y, window->width, window->height, window->window_type == GDK_WINDOW_TEMP); } /* Then do everything that may reference other windows */ for (l = display->toplevels; l != NULL; l = l->next) { GdkWindowImplBroadway *impl = l->data; if (impl->id == 0) continue; /* Skip root */ if (impl->transient_for) broadway_output_set_transient_for (display->output, impl->id, impl->transient_for); /* Can't check GDK_WINDOW_IS_MAPPED here, because that doesn't correctly handle withdrawn windows like menus */ if (impl->visible) { broadway_output_show_surface (display->output, impl->id); window_data_send (display->output, impl); } } gdk_display_flush (GDK_DISPLAY (display)); } static void gdk_window_impl_broadway_init (GdkWindowImplBroadway *impl) { impl->toplevel_window_type = -1; impl->device_cursor = g_hash_table_new_full (NULL, NULL, NULL, (GDestroyNotify) g_object_unref); } static void gdk_window_impl_broadway_finalize (GObject *object) { GdkWindow *wrapper; GdkWindowImplBroadway *impl; GdkBroadwayDisplay *broadway_display; g_return_if_fail (GDK_IS_WINDOW_IMPL_BROADWAY (object)); impl = GDK_WINDOW_IMPL_BROADWAY (object); wrapper = impl->wrapper; _gdk_broadway_window_grab_check_destroy (wrapper); broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (impl->wrapper)); if (broadway_display->mouse_in_toplevel == GDK_WINDOW (wrapper)) { /* TODO: Send leave + enter event, update cursors, etc */ broadway_display->mouse_in_toplevel = NULL; } g_hash_table_remove (broadway_display->id_ht, GINT_TO_POINTER(impl->id)); if (impl->cursor) g_object_unref (impl->cursor); g_hash_table_destroy (impl->device_cursor); broadway_display->toplevels = g_list_remove (broadway_display->toplevels, impl); G_OBJECT_CLASS (gdk_window_impl_broadway_parent_class)->finalize (object); } void _gdk_broadway_screen_init_root_window (GdkScreen * screen) { GdkWindow *window; GdkWindowImplBroadway *impl; GdkBroadwayScreen *broadway_screen; broadway_screen = GDK_BROADWAY_SCREEN (screen); g_assert (broadway_screen->root_window == NULL); broadway_screen->root_window = g_object_new (GDK_TYPE_BROADWAY_WINDOW, NULL); window = broadway_screen->root_window; window->impl = g_object_new (GDK_TYPE_WINDOW_IMPL_BROADWAY, NULL); window->impl_window = window; window->visual = gdk_screen_get_system_visual (screen); impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); impl->screen = screen; impl->wrapper = window; window->window_type = GDK_WINDOW_ROOT; window->depth = 24; window->x = 0; window->y = 0; window->abs_x = 0; window->abs_y = 0; window->width = gdk_screen_get_width (screen); window->height = gdk_screen_get_height (screen); window->viewable = TRUE; _gdk_window_update_size (broadway_screen->root_window); } void _gdk_broadway_display_create_window_impl (GdkDisplay *display, GdkWindow *window, GdkWindow *real_parent, GdkScreen *screen, GdkEventMask event_mask, GdkWindowAttr *attributes, gint attributes_mask) { GdkWindowImplBroadway *impl; GdkBroadwayDisplay *broadway_display; static int current_id = 1; /* 0 is the root window */ broadway_display = GDK_BROADWAY_DISPLAY (display); impl = g_object_new (GDK_TYPE_WINDOW_IMPL_BROADWAY, NULL); window->impl = (GdkWindowImpl *)impl; impl->id = current_id++; g_hash_table_insert (broadway_display->id_ht, GINT_TO_POINTER(impl->id), window); impl->wrapper = window; impl->screen = screen; g_assert (window->window_type == GDK_WINDOW_TOPLEVEL || window->window_type == GDK_WINDOW_TEMP); g_assert (GDK_WINDOW_TYPE (window->parent) == GDK_WINDOW_ROOT); broadway_display->toplevels = g_list_prepend (broadway_display->toplevels, impl); if (broadway_display->output) broadway_output_new_surface (broadway_display->output, impl->id, window->x, window->y, window->width, window->height, window->window_type == GDK_WINDOW_TEMP); } void _gdk_broadway_window_resize_surface (GdkWindow *window) { GdkWindowImplBroadway *impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); cairo_surface_t *old, *last_old; if (impl->surface) { old = impl->surface; last_old = impl->last_surface; impl->surface = cairo_image_surface_create (CAIRO_FORMAT_RGB24, gdk_window_get_width (impl->wrapper), gdk_window_get_height (impl->wrapper)); impl->last_surface = cairo_image_surface_create (CAIRO_FORMAT_RGB24, gdk_window_get_width (impl->wrapper), gdk_window_get_height (impl->wrapper)); cairo_surface_destroy (old); cairo_surface_destroy (last_old); } if (impl->ref_surface) { cairo_surface_set_user_data (impl->ref_surface, &gdk_broadway_cairo_key, NULL, NULL); impl->ref_surface = NULL; } gdk_window_invalidate_rect (window, NULL, TRUE); } static void ref_surface_destroyed (void *data) { GdkWindowImplBroadway *impl = data; impl->ref_surface = NULL; } static cairo_surface_t * gdk_window_broadway_ref_cairo_surface (GdkWindow *window) { GdkWindowImplBroadway *impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); cairo_t *cr; int w, h; if (GDK_IS_WINDOW_IMPL_BROADWAY (window) && GDK_WINDOW_DESTROYED (impl->wrapper)) return NULL; w = gdk_window_get_width (impl->wrapper); h = gdk_window_get_height (impl->wrapper); /* Create actual backing store if missing */ if (!impl->surface) { impl->surface = cairo_image_surface_create (CAIRO_FORMAT_RGB24, w, h); impl->last_surface = cairo_image_surface_create (CAIRO_FORMAT_RGB24, w, h); cr = cairo_create (impl->surface); cairo_set_source_rgb (cr, 0.0, 0.0, 0.0); cairo_rectangle (cr, 0, 0, w, h); cairo_fill (cr); cairo_destroy (cr); cr = cairo_create (impl->last_surface); cairo_set_source_rgb (cr, 0.0, 0.0, 0.0); cairo_rectangle (cr, 0, 0, w, h); cairo_fill (cr); cairo_destroy (cr); } /* Create a destroyable surface referencing the real one */ if (!impl->ref_surface) { impl->ref_surface = cairo_surface_create_for_rectangle (impl->surface, 0, 0, w, h); if (impl->ref_surface) cairo_surface_set_user_data (impl->ref_surface, &gdk_broadway_cairo_key, impl, ref_surface_destroyed); } else cairo_surface_reference (impl->ref_surface); return impl->ref_surface; } static void _gdk_broadway_window_destroy (GdkWindow *window, gboolean recursing, gboolean foreign_destroy) { GdkWindowImplBroadway *impl; GdkBroadwayDisplay *broadway_display; g_return_if_fail (GDK_IS_WINDOW (window)); impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); _gdk_broadway_selection_window_destroyed (window); _gdk_broadway_window_grab_check_destroy (window); if (impl->ref_surface) { cairo_surface_finish (impl->ref_surface); cairo_surface_set_user_data (impl->ref_surface, &gdk_broadway_cairo_key, NULL, NULL); } if (impl->surface) { cairo_surface_destroy (impl->surface); impl->surface = NULL; cairo_surface_destroy (impl->last_surface); impl->last_surface = NULL; } broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (window)); g_hash_table_remove (broadway_display->id_ht, GINT_TO_POINTER(impl->id)); if (broadway_display->output) broadway_output_destroy_surface (broadway_display->output, impl->id); } static cairo_surface_t * gdk_window_broadway_resize_cairo_surface (GdkWindow *window, cairo_surface_t *surface, gint width, gint height) { /* Image surfaces cannot be resized */ cairo_surface_destroy (surface); return NULL; } static void gdk_broadway_window_destroy_foreign (GdkWindow *window) { } /* This function is called when the XWindow is really gone. */ static void gdk_broadway_window_destroy_notify (GdkWindow *window) { if (!GDK_WINDOW_DESTROYED (window)) { if (GDK_WINDOW_TYPE(window) != GDK_WINDOW_FOREIGN) g_warning ("GdkWindow %p unexpectedly destroyed", window); _gdk_window_destroy (window, TRUE); } g_object_unref (window); } static void gdk_window_broadway_show (GdkWindow *window, gboolean already_mapped) { GdkWindowImplBroadway *impl; GdkBroadwayDisplay *broadway_display; impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); impl->visible = TRUE; if (window->event_mask & GDK_STRUCTURE_MASK) _gdk_make_event (GDK_WINDOW (window), GDK_MAP, NULL, FALSE); if (window->parent && window->parent->event_mask & GDK_SUBSTRUCTURE_MASK) _gdk_make_event (GDK_WINDOW (window), GDK_MAP, NULL, FALSE); broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (window)); if (broadway_display->output) { broadway_output_show_surface (broadway_display->output, impl->id); queue_dirty_flush (broadway_display); } } static void gdk_window_broadway_hide (GdkWindow *window) { GdkWindowImplBroadway *impl; GdkBroadwayDisplay *broadway_display; impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); impl->visible = FALSE; if (window->event_mask & GDK_STRUCTURE_MASK) _gdk_make_event (GDK_WINDOW (window), GDK_UNMAP, NULL, FALSE); if (window->parent && window->parent->event_mask & GDK_SUBSTRUCTURE_MASK) _gdk_make_event (GDK_WINDOW (window), GDK_UNMAP, NULL, FALSE); broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (window)); if (broadway_display->output) { broadway_output_hide_surface (broadway_display->output, impl->id); queue_dirty_flush (broadway_display); } if (broadway_display->mouse_in_toplevel == window) { /* TODO: Send leave + enter event, update cursors, etc */ broadway_display->mouse_in_toplevel = NULL; } _gdk_window_clear_update_area (window); } static void gdk_window_broadway_withdraw (GdkWindow *window) { gdk_window_broadway_hide (window); } static void gdk_window_broadway_move_resize (GdkWindow *window, gboolean with_move, gint x, gint y, gint width, gint height) { GdkWindowImplBroadway *impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); GdkBroadwayDisplay *broadway_display; gboolean changed, size_changed;; gboolean with_resize; size_changed = changed = FALSE; broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (window)); if (with_move) { changed = TRUE; window->x = x; window->y = y; } with_resize = FALSE; if (width > 0 || height > 0) { with_resize = TRUE; if (width < 1) width = 1; if (height < 1) height = 1; if (width != window->width || height != window->height) { changed = TRUE; size_changed = TRUE; /* Resize clears the content */ impl->dirty = TRUE; impl->last_synced = FALSE; window->width = width; window->height = height; _gdk_broadway_window_resize_surface (window); } } if (changed) { GdkEvent *event; GList *node; if (broadway_display->output != NULL) { broadway_output_move_resize_surface (broadway_display->output, impl->id, with_move, window->x, window->y, with_resize, window->width, window->height); queue_dirty_flush (broadway_display); if (size_changed) window->resize_count++; } event = gdk_event_new (GDK_CONFIGURE); event->configure.window = g_object_ref (window); event->configure.x = window->x; event->configure.y = window->y; event->configure.width = window->width; event->configure.height = window->height; gdk_event_set_device (event, GDK_DISPLAY_OBJECT (broadway_display)->core_pointer); node = _gdk_event_queue_append (GDK_DISPLAY_OBJECT (broadway_display), event); _gdk_windowing_got_event (GDK_DISPLAY_OBJECT (broadway_display), node, event, _gdk_display_get_next_serial (GDK_DISPLAY (broadway_display)) - 1); } } static gboolean gdk_window_broadway_reparent (GdkWindow *window, GdkWindow *new_parent, gint x, gint y) { return FALSE; } static void gdk_window_broadway_raise (GdkWindow *window) { } static void gdk_window_broadway_restack_under (GdkWindow *window, GList *native_siblings /* in requested order, first is bottom-most */) { } static void gdk_window_broadway_restack_toplevel (GdkWindow *window, GdkWindow *sibling, gboolean above) { } static void gdk_window_broadway_lower (GdkWindow *window) { } static void gdk_broadway_window_focus (GdkWindow *window, guint32 timestamp) { } static void gdk_broadway_window_set_type_hint (GdkWindow *window, GdkWindowTypeHint hint) { } static GdkWindowTypeHint gdk_broadway_window_get_type_hint (GdkWindow *window) { return GDK_WINDOW_TYPE_HINT_NORMAL; } static void gdk_broadway_window_set_modal_hint (GdkWindow *window, gboolean modal) { } static void gdk_broadway_window_set_skip_taskbar_hint (GdkWindow *window, gboolean skips_taskbar) { } static void gdk_broadway_window_set_skip_pager_hint (GdkWindow *window, gboolean skips_pager) { } static void gdk_broadway_window_set_urgency_hint (GdkWindow *window, gboolean urgent) { } static void gdk_broadway_window_set_geometry_hints (GdkWindow *window, const GdkGeometry *geometry, GdkWindowHints geom_mask) { } static void gdk_broadway_window_set_title (GdkWindow *window, const gchar *title) { } static void gdk_broadway_window_set_role (GdkWindow *window, const gchar *role) { } static void gdk_broadway_window_set_startup_id (GdkWindow *window, const gchar *startup_id) { } static void gdk_broadway_window_set_transient_for (GdkWindow *window, GdkWindow *parent) { GdkBroadwayDisplay *display; GdkWindowImplBroadway *impl; int parent_id; impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); parent_id = 0; if (parent) parent_id = GDK_WINDOW_IMPL_BROADWAY (parent->impl)->id; impl->transient_for = parent_id; display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (impl->wrapper)); if (display->output) { broadway_output_set_transient_for (display->output, impl->id, impl->transient_for); gdk_display_flush (GDK_DISPLAY (display)); } } static void gdk_window_broadway_set_background (GdkWindow *window, cairo_pattern_t *pattern) { return; } static void gdk_window_broadway_set_device_cursor (GdkWindow *window, GdkDevice *device, GdkCursor *cursor) { GdkWindowImplBroadway *impl; g_return_if_fail (GDK_IS_WINDOW (window)); g_return_if_fail (GDK_IS_DEVICE (device)); impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); if (!cursor) g_hash_table_remove (impl->device_cursor, device); else { _gdk_broadway_cursor_update_theme (cursor); g_hash_table_replace (impl->device_cursor, device, gdk_cursor_ref (cursor)); } if (!GDK_WINDOW_DESTROYED (window)) GDK_DEVICE_GET_CLASS (device)->set_window_cursor (device, window, cursor); } GdkCursor * _gdk_broadway_window_get_cursor (GdkWindow *window) { GdkWindowImplBroadway *impl; g_return_val_if_fail (GDK_IS_WINDOW (window), NULL); impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); return impl->cursor; } static void gdk_window_broadway_get_geometry (GdkWindow *window, gint *x, gint *y, gint *width, gint *height) { GdkWindowImplBroadway *impl; g_return_if_fail (GDK_IS_WINDOW (window)); impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); /* TODO: These should really roundtrip to the client to get the current data */ if (x) *x = impl->wrapper->x; if (y) *y = impl->wrapper->y; if (width) *width = impl->wrapper->width; if (height) *height = impl->wrapper->height; } static gint gdk_window_broadway_get_root_coords (GdkWindow *window, gint x, gint y, gint *root_x, gint *root_y) { GdkWindowImplBroadway *impl; impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); if (root_x) *root_x = x + impl->wrapper->x; if (root_y) *root_y = y + impl->wrapper->y; return 1; } static void gdk_broadway_window_get_root_origin (GdkWindow *window, gint *x, gint *y) { GdkWindowImplBroadway *impl; impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); /* TODO: This should take wm frame into account */ if (x) *x = impl->wrapper->x; if (y) *y = impl->wrapper->x; } static void gdk_broadway_window_get_frame_extents (GdkWindow *window, GdkRectangle *rect) { g_return_if_fail (rect != NULL); /* TODO: This should take wm frame into account */ rect->x = window->x; rect->y = window->y; rect->width = window->width; rect->height = window->height; } static gboolean gdk_window_broadway_get_device_state (GdkWindow *window, GdkDevice *device, gint *x, gint *y, GdkModifierType *mask) { GdkWindow *child; g_return_val_if_fail (window == NULL || GDK_IS_WINDOW (window), FALSE); if (GDK_WINDOW_DESTROYED (window)) return FALSE; GDK_DEVICE_GET_CLASS (device)->query_state (device, window, NULL, &child, NULL, NULL, x, y, mask); return child != NULL; } static GdkEventMask gdk_window_broadway_get_events (GdkWindow *window) { if (GDK_WINDOW_DESTROYED (window)) return 0; return 0; } static void gdk_window_broadway_set_events (GdkWindow *window, GdkEventMask event_mask) { if (!GDK_WINDOW_DESTROYED (window)) { } } static void gdk_window_broadway_shape_combine_region (GdkWindow *window, const cairo_region_t *shape_region, gint offset_x, gint offset_y) { } static void gdk_window_broadway_input_shape_combine_region (GdkWindow *window, const cairo_region_t *shape_region, gint offset_x, gint offset_y) { } static void gdk_broadway_window_set_override_redirect (GdkWindow *window, gboolean override_redirect) { } static void gdk_broadway_window_set_accept_focus (GdkWindow *window, gboolean accept_focus) { accept_focus = accept_focus != FALSE; if (window->accept_focus != accept_focus) { window->accept_focus = accept_focus; } } static void gdk_broadway_window_set_focus_on_map (GdkWindow *window, gboolean focus_on_map) { focus_on_map = focus_on_map != FALSE; if (window->focus_on_map != focus_on_map) { window->focus_on_map = focus_on_map; } } static void gdk_broadway_window_set_icon_list (GdkWindow *window, GList *pixbufs) { } static void gdk_broadway_window_set_icon_name (GdkWindow *window, const gchar *name) { if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; g_object_set_qdata (G_OBJECT (window), g_quark_from_static_string ("gdk-icon-name-set"), GUINT_TO_POINTER (name != NULL)); } static void gdk_broadway_window_iconify (GdkWindow *window) { if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; } static void gdk_broadway_window_deiconify (GdkWindow *window) { if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; } static void gdk_broadway_window_stick (GdkWindow *window) { if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; } static void gdk_broadway_window_unstick (GdkWindow *window) { if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; } static void gdk_broadway_window_maximize (GdkWindow *window) { if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; } static void gdk_broadway_window_unmaximize (GdkWindow *window) { if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; } static void gdk_broadway_window_fullscreen (GdkWindow *window) { if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; } static void gdk_broadway_window_unfullscreen (GdkWindow *window) { if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; } static void gdk_broadway_window_set_keep_above (GdkWindow *window, gboolean setting) { g_return_if_fail (GDK_IS_WINDOW (window)); if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; } static void gdk_broadway_window_set_keep_below (GdkWindow *window, gboolean setting) { g_return_if_fail (GDK_IS_WINDOW (window)); if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; } static GdkWindow * gdk_broadway_window_get_group (GdkWindow *window) { if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL (window)) return NULL; return window; } static void gdk_broadway_window_set_group (GdkWindow *window, GdkWindow *leader) { } static void gdk_broadway_window_set_decorations (GdkWindow *window, GdkWMDecoration decorations) { if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; } static gboolean gdk_broadway_window_get_decorations (GdkWindow *window, GdkWMDecoration *decorations) { gboolean result = FALSE; if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return FALSE; return result; } static void gdk_broadway_window_set_functions (GdkWindow *window, GdkWMFunction functions) { g_return_if_fail (GDK_IS_WINDOW (window)); if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; } static cairo_region_t * gdk_broadway_window_get_shape (GdkWindow *window) { return NULL; } static cairo_region_t * gdk_broadway_window_get_input_shape (GdkWindow *window) { return NULL; } static gboolean gdk_window_broadway_set_static_gravities (GdkWindow *window, gboolean use_static) { return TRUE; } typedef struct _MoveResizeData MoveResizeData; struct _MoveResizeData { GdkDisplay *display; GdkWindow *moveresize_window; GdkWindow *moveresize_emulation_window; gboolean is_resize; GdkWindowEdge 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; BroadwayInputMsg *moveresize_pending_event; }; static MoveResizeData * get_move_resize_data (GdkDisplay *display, gboolean create) { MoveResizeData *mv_resize; static GQuark move_resize_quark = 0; if (!move_resize_quark) move_resize_quark = g_quark_from_static_string ("gdk-window-moveresize"); mv_resize = g_object_get_qdata (G_OBJECT (display), move_resize_quark); if (!mv_resize && create) { mv_resize = g_new0 (MoveResizeData, 1); mv_resize->display = display; g_object_set_qdata (G_OBJECT (display), move_resize_quark, 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_WINDOW_EDGE_NORTH_WEST: x += dx; y += dy; w -= dx; h -= dy; break; case GDK_WINDOW_EDGE_NORTH: y += dy; h -= dy; break; case GDK_WINDOW_EDGE_NORTH_EAST: y += dy; h -= dy; w += dx; break; case GDK_WINDOW_EDGE_SOUTH_WEST: h += dy; x += dx; w -= dx; break; case GDK_WINDOW_EDGE_SOUTH_EAST: w += dx; h += dy; break; case GDK_WINDOW_EDGE_SOUTH: h += dy; break; case GDK_WINDOW_EDGE_EAST: w += dx; break; case GDK_WINDOW_EDGE_WEST: x += dx; w -= dx; break; } x = MAX (x, 0); y = MAX (y, 0); w = MAX (w, 1); h = MAX (h, 1); gdk_window_move_resize (mv_resize->moveresize_window, x, y, w, h); } else { gint x, y; x = mv_resize->moveresize_orig_x + dx; y = mv_resize->moveresize_orig_y + dy; gdk_window_move (mv_resize->moveresize_window, x, y); } } static void finish_drag (MoveResizeData *mv_resize) { gdk_window_destroy (mv_resize->moveresize_emulation_window); mv_resize->moveresize_emulation_window = NULL; g_object_unref (mv_resize->moveresize_window); mv_resize->moveresize_window = NULL; if (mv_resize->moveresize_pending_event) { g_free (mv_resize->moveresize_pending_event); mv_resize->moveresize_pending_event = NULL; } } static gboolean moveresize_lookahead (GdkDisplay *display, MoveResizeData *mv_resize, BroadwayInputMsg *event) { GdkBroadwayDisplay *broadway_display; BroadwayInputMsg *message; GList *l; broadway_display = GDK_BROADWAY_DISPLAY (display); for (l = broadway_display->input_messages; l != NULL; l = l->next) { message = l->data; if (message->base.type == 'm') return FALSE; if (message->base.type == 'b') return FALSE; } return TRUE; } 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_window) return FALSE; button_mask = GDK_BUTTON1_MASK << (mv_resize->moveresize_button - 1); switch (event->base.type) { case 'm': if (mv_resize->moveresize_window->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 'B': 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; } return TRUE; } gboolean _gdk_broadway_moveresize_configure_done (GdkDisplay *display, GdkWindow *window) { BroadwayInputMsg *tmp_event; MoveResizeData *mv_resize = get_move_resize_data (display, FALSE); if (!mv_resize || window != mv_resize->moveresize_window) 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_window (MoveResizeData *mv_resize, guint32 timestamp) { GdkWindowAttr attributes; gint attributes_mask; GdkGrabStatus status; g_assert (mv_resize->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; mv_resize->moveresize_emulation_window = gdk_window_new (gdk_screen_get_root_window (gdk_display_get_default_screen (mv_resize->display)), &attributes, attributes_mask); gdk_window_show (mv_resize->moveresize_emulation_window); status = gdk_pointer_grab (mv_resize->moveresize_emulation_window, FALSE, GDK_BUTTON_RELEASE_MASK | GDK_POINTER_MOTION_MASK, NULL, NULL, timestamp); if (status != GDK_GRAB_SUCCESS) { /* If this fails, some other client has grabbed the window * already. */ finish_drag (mv_resize); } mv_resize->moveresize_process_time = 0; } static void gdk_broadway_window_begin_resize_drag (GdkWindow *window, GdkWindowEdge edge, GdkDevice *device, gint button, gint root_x, gint root_y, guint32 timestamp) { GdkBroadwayDisplay *broadway_display; MoveResizeData *mv_resize; if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window)) return; /* We need a connection to be able to get mouse events, if not, punt */ broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (window)); if (!broadway_display->output) return; mv_resize = get_move_resize_data (GDK_WINDOW_DISPLAY (window), TRUE); mv_resize->is_resize = TRUE; mv_resize->moveresize_button = button; mv_resize->resize_edge = edge; mv_resize->moveresize_x = root_x; mv_resize->moveresize_y = root_y; mv_resize->moveresize_window = g_object_ref (window); gdk_window_get_origin (mv_resize->moveresize_window, &mv_resize->moveresize_orig_x, &mv_resize->moveresize_orig_y); mv_resize->moveresize_orig_width = gdk_window_get_width (window); mv_resize->moveresize_orig_height = gdk_window_get_height (window); create_moveresize_window (mv_resize, timestamp); } static void gdk_broadway_window_begin_move_drag (GdkWindow *window, GdkDevice *device, gint button, gint root_x, gint root_y, guint32 timestamp) { if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL (window)) return; } static void gdk_broadway_window_enable_synchronized_configure (GdkWindow *window) { if (!GDK_IS_WINDOW_IMPL_BROADWAY (window->impl)) return; } static void gdk_broadway_window_configure_finished (GdkWindow *window) { if (!WINDOW_IS_TOPLEVEL (window)) return; } static gboolean gdk_broadway_window_beep (GdkWindow *window) { return FALSE; } static void gdk_broadway_window_set_opacity (GdkWindow *window, gdouble opacity) { g_return_if_fail (GDK_IS_WINDOW (window)); if (GDK_WINDOW_DESTROYED (window) || !WINDOW_IS_TOPLEVEL (window)) return; if (opacity < 0) opacity = 0; else if (opacity > 1) opacity = 1; } static void gdk_broadway_window_set_composited (GdkWindow *window, gboolean composited) { } static void gdk_broadway_window_process_updates_recurse (GdkWindow *window, cairo_region_t *region) { GdkWindowImplBroadway *impl; _gdk_window_process_updates_recurse (window, region); impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); impl->dirty = TRUE; queue_dirty_flush (GDK_BROADWAY_DISPLAY (gdk_window_get_display (window))); } void _gdk_broadway_display_before_process_all_updates (GdkDisplay *display) { } void _gdk_broadway_display_after_process_all_updates (GdkDisplay *display) { } gboolean _gdk_broadway_window_queue_antiexpose (GdkWindow *window, cairo_region_t *area) { return TRUE; } static void copy_region (cairo_surface_t *surface, cairo_region_t *area, gint dx, gint dy) { cairo_t *cr; cr = cairo_create (surface); gdk_cairo_region (cr, area); cairo_clip (cr); /* NB: This is a self-copy and Cairo doesn't support that yet. * So we do a litle trick. */ cairo_push_group (cr); cairo_set_source_surface (cr, surface, dx, dy); cairo_paint (cr); cairo_pop_group_to_source (cr); cairo_paint (cr); cairo_destroy (cr); } void _gdk_broadway_window_translate (GdkWindow *window, cairo_region_t *area, gint dx, gint dy) { GdkWindowImplBroadway *impl; GdkBroadwayDisplay *broadway_display; int n_rects, i; BroadwayRect *rects; cairo_rectangle_int_t rect; impl = GDK_WINDOW_IMPL_BROADWAY (window->impl); if (impl->surface) { copy_region (impl->surface, area, dx, dy); broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (window)); if (GDK_WINDOW_IMPL_BROADWAY (impl)->last_synced && broadway_display->output) { copy_region (impl->last_surface, area, dx, dy); n_rects = cairo_region_num_rectangles (area); rects = g_new (BroadwayRect, n_rects); for (i = 0; i < n_rects; i++) { cairo_region_get_rectangle (area, i, &rect); rects[i].x = rect.x; rects[i].y = rect.y; rects[i].width = rect.width; rects[i].height = rect.height; } broadway_output_copy_rectangles (broadway_display->output, GDK_WINDOW_IMPL_BROADWAY (impl)->id, rects, n_rects, dx, dy); queue_dirty_flush (broadway_display); g_free (rects); } } } guint32 gdk_broadway_get_last_seen_time (GdkWindow *window) { GdkDisplay *display; display = gdk_window_get_display (window); _gdk_broadway_display_consume_all_input (display); return (guint32) GDK_BROADWAY_DISPLAY (display)->last_seen_time; } static void gdk_window_impl_broadway_class_init (GdkWindowImplBroadwayClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); GdkWindowImplClass *impl_class = GDK_WINDOW_IMPL_CLASS (klass); object_class->finalize = gdk_window_impl_broadway_finalize; impl_class->ref_cairo_surface = gdk_window_broadway_ref_cairo_surface; impl_class->show = gdk_window_broadway_show; impl_class->hide = gdk_window_broadway_hide; impl_class->withdraw = gdk_window_broadway_withdraw; impl_class->set_events = gdk_window_broadway_set_events; impl_class->get_events = gdk_window_broadway_get_events; impl_class->raise = gdk_window_broadway_raise; impl_class->lower = gdk_window_broadway_lower; impl_class->restack_under = gdk_window_broadway_restack_under; impl_class->restack_toplevel = gdk_window_broadway_restack_toplevel; impl_class->move_resize = gdk_window_broadway_move_resize; impl_class->set_background = gdk_window_broadway_set_background; impl_class->reparent = gdk_window_broadway_reparent; impl_class->set_device_cursor = gdk_window_broadway_set_device_cursor; impl_class->get_geometry = gdk_window_broadway_get_geometry; impl_class->get_root_coords = gdk_window_broadway_get_root_coords; impl_class->get_device_state = gdk_window_broadway_get_device_state; impl_class->shape_combine_region = gdk_window_broadway_shape_combine_region; impl_class->input_shape_combine_region = gdk_window_broadway_input_shape_combine_region; impl_class->set_static_gravities = gdk_window_broadway_set_static_gravities; impl_class->queue_antiexpose = _gdk_broadway_window_queue_antiexpose; impl_class->translate = _gdk_broadway_window_translate; impl_class->destroy = _gdk_broadway_window_destroy; impl_class->destroy_foreign = gdk_broadway_window_destroy_foreign; impl_class->resize_cairo_surface = gdk_window_broadway_resize_cairo_surface; impl_class->get_shape = gdk_broadway_window_get_shape; impl_class->get_input_shape = gdk_broadway_window_get_input_shape; impl_class->beep = gdk_broadway_window_beep; impl_class->focus = gdk_broadway_window_focus; impl_class->set_type_hint = gdk_broadway_window_set_type_hint; impl_class->get_type_hint = gdk_broadway_window_get_type_hint; impl_class->set_modal_hint = gdk_broadway_window_set_modal_hint; impl_class->set_skip_taskbar_hint = gdk_broadway_window_set_skip_taskbar_hint; impl_class->set_skip_pager_hint = gdk_broadway_window_set_skip_pager_hint; impl_class->set_urgency_hint = gdk_broadway_window_set_urgency_hint; impl_class->set_geometry_hints = gdk_broadway_window_set_geometry_hints; impl_class->set_title = gdk_broadway_window_set_title; impl_class->set_role = gdk_broadway_window_set_role; impl_class->set_startup_id = gdk_broadway_window_set_startup_id; impl_class->set_transient_for = gdk_broadway_window_set_transient_for; impl_class->get_root_origin = gdk_broadway_window_get_root_origin; impl_class->get_frame_extents = gdk_broadway_window_get_frame_extents; impl_class->set_override_redirect = gdk_broadway_window_set_override_redirect; impl_class->set_accept_focus = gdk_broadway_window_set_accept_focus; impl_class->set_focus_on_map = gdk_broadway_window_set_focus_on_map; impl_class->set_icon_list = gdk_broadway_window_set_icon_list; impl_class->set_icon_name = gdk_broadway_window_set_icon_name; impl_class->iconify = gdk_broadway_window_iconify; impl_class->deiconify = gdk_broadway_window_deiconify; impl_class->stick = gdk_broadway_window_stick; impl_class->unstick = gdk_broadway_window_unstick; impl_class->maximize = gdk_broadway_window_maximize; impl_class->unmaximize = gdk_broadway_window_unmaximize; impl_class->fullscreen = gdk_broadway_window_fullscreen; impl_class->unfullscreen = gdk_broadway_window_unfullscreen; impl_class->set_keep_above = gdk_broadway_window_set_keep_above; impl_class->set_keep_below = gdk_broadway_window_set_keep_below; impl_class->get_group = gdk_broadway_window_get_group; impl_class->set_group = gdk_broadway_window_set_group; impl_class->set_decorations = gdk_broadway_window_set_decorations; impl_class->get_decorations = gdk_broadway_window_get_decorations; impl_class->set_functions = gdk_broadway_window_set_functions; impl_class->set_functions = gdk_broadway_window_set_functions; impl_class->begin_resize_drag = gdk_broadway_window_begin_resize_drag; impl_class->begin_move_drag = gdk_broadway_window_begin_move_drag; impl_class->enable_synchronized_configure = gdk_broadway_window_enable_synchronized_configure; impl_class->configure_finished = gdk_broadway_window_configure_finished; impl_class->set_opacity = gdk_broadway_window_set_opacity; impl_class->set_composited = gdk_broadway_window_set_composited; impl_class->destroy_notify = gdk_broadway_window_destroy_notify; impl_class->register_dnd = _gdk_broadway_window_register_dnd; impl_class->drag_begin = _gdk_broadway_window_drag_begin; impl_class->process_updates_recurse = gdk_broadway_window_process_updates_recurse; impl_class->sync_rendering = _gdk_broadway_window_sync_rendering; impl_class->simulate_key = _gdk_broadway_window_simulate_key; impl_class->simulate_button = _gdk_broadway_window_simulate_button; impl_class->get_property = _gdk_broadway_window_get_property; impl_class->change_property = _gdk_broadway_window_change_property; impl_class->delete_property = _gdk_broadway_window_delete_property; impl_class->get_drag_protocol = _gdk_broadway_window_get_drag_protocol; }