1 /* GDK - The GIMP Drawing Kit
2 * Copyright (C) 1995-2007 Peter Mattis, Spencer Kimball,
3 * Josh MacDonald, Ryan Lortie
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the
17 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18 * Boston, MA 02111-1307, USA.
22 * Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
23 * file for a list of people on the GTK+ Team. See the ChangeLog
24 * files for a list of changes. These files are distributed with
25 * GTK+ at ftp://ftp.gtk.org/pub/gtk/.
30 #include "gdkwindow-broadway.h"
31 #include "gdkscreen-broadway.h"
33 #include "gdkwindow.h"
34 #include "gdkwindowimpl.h"
35 #include "gdkdisplay-broadway.h"
36 #include "gdkprivate-broadway.h"
37 #include "gdkinternals.h"
38 #include "gdkdeviceprivate.h"
39 #include "gdkeventsource.h"
44 #include <netinet/in.h>
47 /* Forward declarations */
48 static void gdk_window_broadway_set_background (GdkWindow *window,
49 cairo_pattern_t *pattern);
51 static void gdk_window_impl_broadway_finalize (GObject *object);
53 static const cairo_user_data_key_t gdk_broadway_cairo_key;
55 #define WINDOW_IS_TOPLEVEL_OR_FOREIGN(window) \
56 (GDK_WINDOW_TYPE (window) != GDK_WINDOW_CHILD && \
57 GDK_WINDOW_TYPE (window) != GDK_WINDOW_OFFSCREEN)
59 #define WINDOW_IS_TOPLEVEL(window) \
60 (GDK_WINDOW_TYPE (window) != GDK_WINDOW_CHILD && \
61 GDK_WINDOW_TYPE (window) != GDK_WINDOW_FOREIGN && \
62 GDK_WINDOW_TYPE (window) != GDK_WINDOW_OFFSCREEN)
64 struct _GdkBroadwayWindow {
68 struct _GdkBroadwayWindowClass {
69 GdkWindowClass parent_class;
72 G_DEFINE_TYPE (GdkBroadwayWindow, gdk_broadway_window, GDK_TYPE_WINDOW)
75 gdk_broadway_window_class_init (GdkBroadwayWindowClass *broadway_window_class)
80 gdk_broadway_window_init (GdkBroadwayWindow *broadway_window)
84 G_DEFINE_TYPE (GdkWindowImplBroadway,
85 gdk_window_impl_broadway,
89 diff_surfaces (cairo_surface_t *surface,
90 cairo_surface_t *old_surface)
92 guint8 *data, *old_data;
93 guint32 *line, *old_line;
94 int w, h, stride, old_stride;
97 data = cairo_image_surface_get_data (surface);
98 old_data = cairo_image_surface_get_data (old_surface);
100 w = cairo_image_surface_get_width (surface);
101 h = cairo_image_surface_get_height (surface);
103 stride = cairo_image_surface_get_stride (surface);
104 old_stride = cairo_image_surface_get_stride (old_surface);
106 for (y = 0; y < h; y++)
108 line = (guint32 *)data;
109 old_line = (guint32 *)old_data;
111 for (x = 0; x < w; x++)
113 if ((*line & 0xffffff) == (*old_line & 0xffffff))
116 *old_line = *line | 0xff000000;
122 old_data += old_stride;
126 static guint dirty_flush_id = 0;
129 window_data_send (BroadwayOutput *output, GdkWindowImplBroadway *impl)
133 if (impl->surface == NULL)
136 if (impl->last_synced)
138 diff_surfaces (impl->surface,
140 broadway_output_put_rgba (output, impl->id, 0, 0,
141 cairo_image_surface_get_width (impl->last_surface),
142 cairo_image_surface_get_height (impl->last_surface),
143 cairo_image_surface_get_stride (impl->last_surface),
144 cairo_image_surface_get_data (impl->last_surface));
148 impl->last_synced = TRUE;
149 broadway_output_put_rgb (output, impl->id, 0, 0,
150 cairo_image_surface_get_width (impl->surface),
151 cairo_image_surface_get_height (impl->surface),
152 cairo_image_surface_get_stride (impl->surface),
153 cairo_image_surface_get_data (impl->surface));
156 broadway_output_surface_flush (output, impl->id);
158 cr = cairo_create (impl->last_surface);
159 cairo_set_source_surface (cr, impl->surface, 0, 0);
165 dirty_flush_idle (gpointer data)
168 GdkBroadwayDisplay *display;
169 BroadwayOutput *output;
173 display = GDK_BROADWAY_DISPLAY (gdk_display_get_default ());
174 output = display->output;
178 for (l = display->toplevels; l != NULL; l = l->next)
180 GdkWindowImplBroadway *impl = l->data;
185 window_data_send (display->output, impl);
189 gdk_display_flush (GDK_DISPLAY (display));
195 queue_dirty_flush (GdkBroadwayDisplay *display)
197 if (dirty_flush_id == 0 && display->output != NULL)
198 dirty_flush_id = gdk_threads_add_idle (dirty_flush_idle, NULL);
202 _gdk_broadway_resync_windows (void)
204 GdkBroadwayDisplay *display;
209 display = GDK_BROADWAY_DISPLAY (gdk_display_get_default ());
211 /* First create all windows */
212 for (l = display->toplevels; l != NULL; l = l->next)
214 GdkWindowImplBroadway *impl = l->data;
217 window = impl->wrapper;
220 continue; /* Skip root */
223 impl->last_synced = FALSE;
224 broadway_output_new_surface (display->output,
230 window->window_type == GDK_WINDOW_TEMP);
233 /* Then do everything that may reference other windows */
234 for (l = display->toplevels; l != NULL; l = l->next)
236 GdkWindowImplBroadway *impl = l->data;
239 continue; /* Skip root */
241 if (impl->transient_for)
242 broadway_output_set_transient_for (display->output, impl->id, impl->transient_for);
243 /* Can't check GDK_WINDOW_IS_MAPPED here, because that doesn't correctly handle
244 withdrawn windows like menus */
247 broadway_output_show_surface (display->output, impl->id);
248 window_data_send (display->output, impl);
252 gdk_display_flush (GDK_DISPLAY (display));
256 gdk_window_impl_broadway_init (GdkWindowImplBroadway *impl)
258 impl->toplevel_window_type = -1;
259 impl->device_cursor = g_hash_table_new_full (NULL, NULL, NULL,
260 (GDestroyNotify) g_object_unref);
264 gdk_window_impl_broadway_finalize (GObject *object)
267 GdkWindowImplBroadway *impl;
268 GdkBroadwayDisplay *broadway_display;
270 g_return_if_fail (GDK_IS_WINDOW_IMPL_BROADWAY (object));
272 impl = GDK_WINDOW_IMPL_BROADWAY (object);
274 wrapper = impl->wrapper;
276 _gdk_broadway_window_grab_check_destroy (wrapper);
278 broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (impl->wrapper));
280 if (broadway_display->mouse_in_toplevel == GDK_WINDOW (wrapper))
282 /* TODO: Send leave + enter event, update cursors, etc */
283 broadway_display->mouse_in_toplevel = NULL;
286 g_hash_table_remove (broadway_display->id_ht, GINT_TO_POINTER(impl->id));
289 g_object_unref (impl->cursor);
291 g_hash_table_destroy (impl->device_cursor);
293 broadway_display->toplevels = g_list_remove (broadway_display->toplevels, impl);
295 G_OBJECT_CLASS (gdk_window_impl_broadway_parent_class)->finalize (object);
299 _gdk_broadway_screen_init_root_window (GdkScreen * screen)
302 GdkWindowImplBroadway *impl;
303 GdkBroadwayScreen *broadway_screen;
305 broadway_screen = GDK_BROADWAY_SCREEN (screen);
307 g_assert (broadway_screen->root_window == NULL);
309 broadway_screen->root_window = g_object_new (GDK_TYPE_BROADWAY_WINDOW, NULL);
311 window = broadway_screen->root_window;
312 window->impl = g_object_new (GDK_TYPE_WINDOW_IMPL_BROADWAY, NULL);
313 window->impl_window = window;
314 window->visual = gdk_screen_get_system_visual (screen);
316 impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
318 impl->screen = screen;
319 impl->wrapper = window;
321 window->window_type = GDK_WINDOW_ROOT;
328 window->width = gdk_screen_get_width (screen);
329 window->height = gdk_screen_get_height (screen);
330 window->viewable = TRUE;
332 _gdk_window_update_size (broadway_screen->root_window);
336 _gdk_broadway_display_create_window_impl (GdkDisplay *display,
338 GdkWindow *real_parent,
340 GdkEventMask event_mask,
341 GdkWindowAttr *attributes,
342 gint attributes_mask)
344 GdkWindowImplBroadway *impl;
345 GdkBroadwayDisplay *broadway_display;
346 static int current_id = 1; /* 0 is the root window */
348 broadway_display = GDK_BROADWAY_DISPLAY (display);
350 impl = g_object_new (GDK_TYPE_WINDOW_IMPL_BROADWAY, NULL);
351 window->impl = (GdkWindowImpl *)impl;
352 impl->id = current_id++;
353 g_hash_table_insert (broadway_display->id_ht, GINT_TO_POINTER(impl->id), window);
354 impl->wrapper = window;
356 impl->screen = screen;
358 g_assert (window->window_type == GDK_WINDOW_TOPLEVEL ||
359 window->window_type == GDK_WINDOW_TEMP);
360 g_assert (GDK_WINDOW_TYPE (window->parent) == GDK_WINDOW_ROOT);
362 broadway_display->toplevels = g_list_prepend (broadway_display->toplevels, impl);
364 if (broadway_display->output)
365 broadway_output_new_surface (broadway_display->output,
371 window->window_type == GDK_WINDOW_TEMP);
375 _gdk_broadway_window_resize_surface (GdkWindow *window)
377 GdkWindowImplBroadway *impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
378 cairo_surface_t *old, *last_old;
383 last_old = impl->last_surface;
385 impl->surface = cairo_image_surface_create (CAIRO_FORMAT_RGB24,
386 gdk_window_get_width (impl->wrapper),
387 gdk_window_get_height (impl->wrapper));
388 impl->last_surface = cairo_image_surface_create (CAIRO_FORMAT_RGB24,
389 gdk_window_get_width (impl->wrapper),
390 gdk_window_get_height (impl->wrapper));
392 cairo_surface_destroy (old);
393 cairo_surface_destroy (last_old);
396 if (impl->ref_surface)
398 cairo_surface_set_user_data (impl->ref_surface, &gdk_broadway_cairo_key,
400 impl->ref_surface = NULL;
403 gdk_window_invalidate_rect (window, NULL, TRUE);
407 ref_surface_destroyed (void *data)
409 GdkWindowImplBroadway *impl = data;
411 impl->ref_surface = NULL;
414 static cairo_surface_t *
415 gdk_window_broadway_ref_cairo_surface (GdkWindow *window)
417 GdkWindowImplBroadway *impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
421 if (GDK_IS_WINDOW_IMPL_BROADWAY (window) &&
422 GDK_WINDOW_DESTROYED (impl->wrapper))
425 w = gdk_window_get_width (impl->wrapper);
426 h = gdk_window_get_height (impl->wrapper);
428 /* Create actual backing store if missing */
431 impl->surface = cairo_image_surface_create (CAIRO_FORMAT_RGB24, w, h);
432 impl->last_surface = cairo_image_surface_create (CAIRO_FORMAT_RGB24, w, h);
434 cr = cairo_create (impl->surface);
435 cairo_set_source_rgb (cr, 0.0, 0.0, 0.0);
436 cairo_rectangle (cr, 0, 0, w, h);
440 cr = cairo_create (impl->last_surface);
441 cairo_set_source_rgb (cr, 0.0, 0.0, 0.0);
442 cairo_rectangle (cr, 0, 0, w, h);
447 /* Create a destroyable surface referencing the real one */
448 if (!impl->ref_surface)
451 cairo_surface_create_for_rectangle (impl->surface,
454 if (impl->ref_surface)
455 cairo_surface_set_user_data (impl->ref_surface, &gdk_broadway_cairo_key,
456 impl, ref_surface_destroyed);
459 cairo_surface_reference (impl->ref_surface);
461 return impl->ref_surface;
465 _gdk_broadway_window_destroy (GdkWindow *window,
467 gboolean foreign_destroy)
469 GdkWindowImplBroadway *impl;
470 GdkBroadwayDisplay *broadway_display;
472 g_return_if_fail (GDK_IS_WINDOW (window));
474 impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
476 _gdk_broadway_selection_window_destroyed (window);
477 _gdk_broadway_window_grab_check_destroy (window);
479 if (impl->ref_surface)
481 cairo_surface_finish (impl->ref_surface);
482 cairo_surface_set_user_data (impl->ref_surface, &gdk_broadway_cairo_key,
488 cairo_surface_destroy (impl->surface);
489 impl->surface = NULL;
490 cairo_surface_destroy (impl->last_surface);
491 impl->last_surface = NULL;
494 broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (window));
495 g_hash_table_remove (broadway_display->id_ht, GINT_TO_POINTER(impl->id));
497 if (broadway_display->output)
498 broadway_output_destroy_surface (broadway_display->output,
502 static cairo_surface_t *
503 gdk_window_broadway_resize_cairo_surface (GdkWindow *window,
504 cairo_surface_t *surface,
508 /* Image surfaces cannot be resized */
509 cairo_surface_destroy (surface);
515 gdk_broadway_window_destroy_foreign (GdkWindow *window)
519 /* This function is called when the XWindow is really gone.
522 gdk_broadway_window_destroy_notify (GdkWindow *window)
524 if (!GDK_WINDOW_DESTROYED (window))
526 if (GDK_WINDOW_TYPE(window) != GDK_WINDOW_FOREIGN)
527 g_warning ("GdkWindow %p unexpectedly destroyed", window);
529 _gdk_window_destroy (window, TRUE);
532 g_object_unref (window);
536 gdk_window_broadway_show (GdkWindow *window, gboolean already_mapped)
538 GdkWindowImplBroadway *impl;
539 GdkBroadwayDisplay *broadway_display;
541 impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
542 impl->visible = TRUE;
544 if (window->event_mask & GDK_STRUCTURE_MASK)
545 _gdk_make_event (GDK_WINDOW (window), GDK_MAP, NULL, FALSE);
547 if (window->parent && window->parent->event_mask & GDK_SUBSTRUCTURE_MASK)
548 _gdk_make_event (GDK_WINDOW (window), GDK_MAP, NULL, FALSE);
550 broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (window));
551 if (broadway_display->output)
553 broadway_output_show_surface (broadway_display->output, impl->id);
554 queue_dirty_flush (broadway_display);
559 gdk_window_broadway_hide (GdkWindow *window)
561 GdkWindowImplBroadway *impl;
562 GdkBroadwayDisplay *broadway_display;
564 impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
565 impl->visible = FALSE;
567 if (window->event_mask & GDK_STRUCTURE_MASK)
568 _gdk_make_event (GDK_WINDOW (window), GDK_UNMAP, NULL, FALSE);
570 if (window->parent && window->parent->event_mask & GDK_SUBSTRUCTURE_MASK)
571 _gdk_make_event (GDK_WINDOW (window), GDK_UNMAP, NULL, FALSE);
573 broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (window));
574 if (broadway_display->output)
576 broadway_output_hide_surface (broadway_display->output, impl->id);
577 queue_dirty_flush (broadway_display);
580 if (broadway_display->mouse_in_toplevel == window)
582 /* TODO: Send leave + enter event, update cursors, etc */
583 broadway_display->mouse_in_toplevel = NULL;
586 _gdk_window_clear_update_area (window);
590 gdk_window_broadway_withdraw (GdkWindow *window)
592 gdk_window_broadway_hide (window);
596 gdk_window_broadway_move_resize (GdkWindow *window,
603 GdkWindowImplBroadway *impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
604 GdkBroadwayDisplay *broadway_display;
605 gboolean changed, size_changed;;
606 gboolean with_resize;
608 size_changed = changed = FALSE;
610 broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (window));
619 if (width > 0 || height > 0)
628 if (width != window->width ||
629 height != window->height)
634 /* Resize clears the content */
636 impl->last_synced = FALSE;
638 window->width = width;
639 window->height = height;
640 _gdk_broadway_window_resize_surface (window);
649 if (broadway_display->output != NULL)
651 broadway_output_move_resize_surface (broadway_display->output,
653 with_move, window->x, window->y,
654 with_resize, window->width, window->height);
655 queue_dirty_flush (broadway_display);
657 window->resize_count++;
660 event = gdk_event_new (GDK_CONFIGURE);
661 event->configure.window = g_object_ref (window);
662 event->configure.x = window->x;
663 event->configure.y = window->y;
664 event->configure.width = window->width;
665 event->configure.height = window->height;
667 gdk_event_set_device (event, GDK_DISPLAY_OBJECT (broadway_display)->core_pointer);
669 node = _gdk_event_queue_append (GDK_DISPLAY_OBJECT (broadway_display), event);
670 _gdk_windowing_got_event (GDK_DISPLAY_OBJECT (broadway_display), node, event,
671 _gdk_display_get_next_serial (GDK_DISPLAY (broadway_display)) - 1);
676 gdk_window_broadway_reparent (GdkWindow *window,
677 GdkWindow *new_parent,
685 gdk_window_broadway_raise (GdkWindow *window)
690 gdk_window_broadway_restack_under (GdkWindow *window,
691 GList *native_siblings /* in requested order, first is bottom-most */)
696 gdk_window_broadway_restack_toplevel (GdkWindow *window,
703 gdk_window_broadway_lower (GdkWindow *window)
709 gdk_broadway_window_focus (GdkWindow *window,
715 gdk_broadway_window_set_type_hint (GdkWindow *window,
716 GdkWindowTypeHint hint)
720 static GdkWindowTypeHint
721 gdk_broadway_window_get_type_hint (GdkWindow *window)
723 return GDK_WINDOW_TYPE_HINT_NORMAL;
727 gdk_broadway_window_set_modal_hint (GdkWindow *window,
733 gdk_broadway_window_set_skip_taskbar_hint (GdkWindow *window,
734 gboolean skips_taskbar)
739 gdk_broadway_window_set_skip_pager_hint (GdkWindow *window,
740 gboolean skips_pager)
745 gdk_broadway_window_set_urgency_hint (GdkWindow *window,
751 gdk_broadway_window_set_geometry_hints (GdkWindow *window,
752 const GdkGeometry *geometry,
753 GdkWindowHints geom_mask)
758 gdk_broadway_window_set_title (GdkWindow *window,
764 gdk_broadway_window_set_role (GdkWindow *window,
770 gdk_broadway_window_set_startup_id (GdkWindow *window,
771 const gchar *startup_id)
776 gdk_broadway_window_set_transient_for (GdkWindow *window,
779 GdkBroadwayDisplay *display;
780 GdkWindowImplBroadway *impl;
783 impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
787 parent_id = GDK_WINDOW_IMPL_BROADWAY (parent->impl)->id;
789 impl->transient_for = parent_id;
791 display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (impl->wrapper));
794 broadway_output_set_transient_for (display->output, impl->id, impl->transient_for);
795 gdk_display_flush (GDK_DISPLAY (display));
800 gdk_window_broadway_set_background (GdkWindow *window,
801 cairo_pattern_t *pattern)
807 gdk_window_broadway_set_device_cursor (GdkWindow *window,
811 GdkWindowImplBroadway *impl;
813 g_return_if_fail (GDK_IS_WINDOW (window));
814 g_return_if_fail (GDK_IS_DEVICE (device));
816 impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
819 g_hash_table_remove (impl->device_cursor, device);
822 _gdk_broadway_cursor_update_theme (cursor);
823 g_hash_table_replace (impl->device_cursor,
824 device, gdk_cursor_ref (cursor));
827 if (!GDK_WINDOW_DESTROYED (window))
828 GDK_DEVICE_GET_CLASS (device)->set_window_cursor (device, window, cursor);
832 _gdk_broadway_window_get_cursor (GdkWindow *window)
834 GdkWindowImplBroadway *impl;
836 g_return_val_if_fail (GDK_IS_WINDOW (window), NULL);
838 impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
844 gdk_window_broadway_get_geometry (GdkWindow *window,
850 GdkWindowImplBroadway *impl;
852 g_return_if_fail (GDK_IS_WINDOW (window));
854 impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
856 /* TODO: These should really roundtrip to the client to get the current data */
859 *x = impl->wrapper->x;
861 *y = impl->wrapper->y;
863 *width = impl->wrapper->width;
865 *height = impl->wrapper->height;
870 gdk_window_broadway_get_root_coords (GdkWindow *window,
876 GdkWindowImplBroadway *impl;
878 impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
881 *root_x = x + impl->wrapper->x;
883 *root_y = y + impl->wrapper->y;
889 gdk_broadway_window_get_root_origin (GdkWindow *window,
893 GdkWindowImplBroadway *impl;
895 impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
897 /* TODO: This should take wm frame into account */
900 *x = impl->wrapper->x;
903 *y = impl->wrapper->x;
907 gdk_broadway_window_get_frame_extents (GdkWindow *window,
910 g_return_if_fail (rect != NULL);
912 /* TODO: This should take wm frame into account */
916 rect->width = window->width;
917 rect->height = window->height;
921 gdk_window_broadway_get_device_state (GdkWindow *window,
925 GdkModifierType *mask)
929 g_return_val_if_fail (window == NULL || GDK_IS_WINDOW (window), FALSE);
931 if (GDK_WINDOW_DESTROYED (window))
934 GDK_DEVICE_GET_CLASS (device)->query_state (device, window,
938 return child != NULL;
942 gdk_window_broadway_get_events (GdkWindow *window)
944 if (GDK_WINDOW_DESTROYED (window))
951 gdk_window_broadway_set_events (GdkWindow *window,
952 GdkEventMask event_mask)
954 if (!GDK_WINDOW_DESTROYED (window))
960 gdk_window_broadway_shape_combine_region (GdkWindow *window,
961 const cairo_region_t *shape_region,
968 gdk_window_broadway_input_shape_combine_region (GdkWindow *window,
969 const cairo_region_t *shape_region,
977 gdk_broadway_window_set_override_redirect (GdkWindow *window,
978 gboolean override_redirect)
983 gdk_broadway_window_set_accept_focus (GdkWindow *window,
984 gboolean accept_focus)
986 accept_focus = accept_focus != FALSE;
988 if (window->accept_focus != accept_focus)
990 window->accept_focus = accept_focus;
995 gdk_broadway_window_set_focus_on_map (GdkWindow *window,
996 gboolean focus_on_map)
998 focus_on_map = focus_on_map != FALSE;
1000 if (window->focus_on_map != focus_on_map)
1002 window->focus_on_map = focus_on_map;
1008 gdk_broadway_window_set_icon_list (GdkWindow *window,
1014 gdk_broadway_window_set_icon_name (GdkWindow *window,
1017 if (GDK_WINDOW_DESTROYED (window) ||
1018 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1021 g_object_set_qdata (G_OBJECT (window), g_quark_from_static_string ("gdk-icon-name-set"),
1022 GUINT_TO_POINTER (name != NULL));
1026 gdk_broadway_window_iconify (GdkWindow *window)
1028 if (GDK_WINDOW_DESTROYED (window) ||
1029 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1034 gdk_broadway_window_deiconify (GdkWindow *window)
1036 if (GDK_WINDOW_DESTROYED (window) ||
1037 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1042 gdk_broadway_window_stick (GdkWindow *window)
1044 if (GDK_WINDOW_DESTROYED (window) ||
1045 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1051 gdk_broadway_window_unstick (GdkWindow *window)
1053 if (GDK_WINDOW_DESTROYED (window) ||
1054 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1060 gdk_broadway_window_maximize (GdkWindow *window)
1062 if (GDK_WINDOW_DESTROYED (window) ||
1063 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1069 gdk_broadway_window_unmaximize (GdkWindow *window)
1071 if (GDK_WINDOW_DESTROYED (window) ||
1072 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1078 gdk_broadway_window_fullscreen (GdkWindow *window)
1080 if (GDK_WINDOW_DESTROYED (window) ||
1081 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1087 gdk_broadway_window_unfullscreen (GdkWindow *window)
1089 if (GDK_WINDOW_DESTROYED (window) ||
1090 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1096 gdk_broadway_window_set_keep_above (GdkWindow *window,
1099 g_return_if_fail (GDK_IS_WINDOW (window));
1101 if (GDK_WINDOW_DESTROYED (window) ||
1102 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1108 gdk_broadway_window_set_keep_below (GdkWindow *window, gboolean setting)
1110 g_return_if_fail (GDK_IS_WINDOW (window));
1112 if (GDK_WINDOW_DESTROYED (window) ||
1113 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1119 gdk_broadway_window_get_group (GdkWindow *window)
1121 if (GDK_WINDOW_DESTROYED (window) ||
1122 !WINDOW_IS_TOPLEVEL (window))
1129 gdk_broadway_window_set_group (GdkWindow *window,
1135 gdk_broadway_window_set_decorations (GdkWindow *window,
1136 GdkWMDecoration decorations)
1138 if (GDK_WINDOW_DESTROYED (window) ||
1139 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1145 gdk_broadway_window_get_decorations (GdkWindow *window,
1146 GdkWMDecoration *decorations)
1148 gboolean result = FALSE;
1150 if (GDK_WINDOW_DESTROYED (window) ||
1151 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1158 gdk_broadway_window_set_functions (GdkWindow *window,
1159 GdkWMFunction functions)
1161 g_return_if_fail (GDK_IS_WINDOW (window));
1163 if (GDK_WINDOW_DESTROYED (window) ||
1164 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1168 static cairo_region_t *
1169 gdk_broadway_window_get_shape (GdkWindow *window)
1174 static cairo_region_t *
1175 gdk_broadway_window_get_input_shape (GdkWindow *window)
1182 gdk_window_broadway_set_static_gravities (GdkWindow *window,
1183 gboolean use_static)
1188 typedef struct _MoveResizeData MoveResizeData;
1190 struct _MoveResizeData
1192 GdkDisplay *display;
1194 GdkWindow *moveresize_window;
1195 GdkWindow *moveresize_emulation_window;
1197 GdkWindowEdge resize_edge;
1198 gint moveresize_button;
1201 gint moveresize_orig_x;
1202 gint moveresize_orig_y;
1203 gint moveresize_orig_width;
1204 gint moveresize_orig_height;
1205 long moveresize_process_time;
1206 BroadwayInputMsg *moveresize_pending_event;
1209 static MoveResizeData *
1210 get_move_resize_data (GdkDisplay *display,
1213 MoveResizeData *mv_resize;
1214 static GQuark move_resize_quark = 0;
1216 if (!move_resize_quark)
1217 move_resize_quark = g_quark_from_static_string ("gdk-window-moveresize");
1219 mv_resize = g_object_get_qdata (G_OBJECT (display), move_resize_quark);
1221 if (!mv_resize && create)
1223 mv_resize = g_new0 (MoveResizeData, 1);
1224 mv_resize->display = display;
1226 g_object_set_qdata (G_OBJECT (display), move_resize_quark, mv_resize);
1233 update_pos (MoveResizeData *mv_resize,
1239 dx = new_root_x - mv_resize->moveresize_x;
1240 dy = new_root_y - mv_resize->moveresize_y;
1242 if (mv_resize->is_resize)
1246 x = mv_resize->moveresize_orig_x;
1247 y = mv_resize->moveresize_orig_y;
1249 w = mv_resize->moveresize_orig_width;
1250 h = mv_resize->moveresize_orig_height;
1252 switch (mv_resize->resize_edge)
1254 case GDK_WINDOW_EDGE_NORTH_WEST:
1260 case GDK_WINDOW_EDGE_NORTH:
1264 case GDK_WINDOW_EDGE_NORTH_EAST:
1269 case GDK_WINDOW_EDGE_SOUTH_WEST:
1274 case GDK_WINDOW_EDGE_SOUTH_EAST:
1278 case GDK_WINDOW_EDGE_SOUTH:
1281 case GDK_WINDOW_EDGE_EAST:
1284 case GDK_WINDOW_EDGE_WEST:
1295 gdk_window_move_resize (mv_resize->moveresize_window, x, y, w, h);
1301 x = mv_resize->moveresize_orig_x + dx;
1302 y = mv_resize->moveresize_orig_y + dy;
1304 gdk_window_move (mv_resize->moveresize_window, x, y);
1309 finish_drag (MoveResizeData *mv_resize)
1311 gdk_window_destroy (mv_resize->moveresize_emulation_window);
1312 mv_resize->moveresize_emulation_window = NULL;
1313 g_object_unref (mv_resize->moveresize_window);
1314 mv_resize->moveresize_window = NULL;
1316 if (mv_resize->moveresize_pending_event)
1318 g_free (mv_resize->moveresize_pending_event);
1319 mv_resize->moveresize_pending_event = NULL;
1324 moveresize_lookahead (GdkDisplay *display,
1325 MoveResizeData *mv_resize,
1326 BroadwayInputMsg *event)
1328 GdkBroadwayDisplay *broadway_display;
1329 BroadwayInputMsg *message;
1332 broadway_display = GDK_BROADWAY_DISPLAY (display);
1333 for (l = broadway_display->input_messages; l != NULL; l = l->next)
1336 if (message->base.type == 'm')
1338 if (message->base.type == 'b')
1346 _gdk_broadway_moveresize_handle_event (GdkDisplay *display,
1347 BroadwayInputMsg *event)
1349 guint button_mask = 0;
1350 MoveResizeData *mv_resize = get_move_resize_data (display, FALSE);
1352 if (!mv_resize || !mv_resize->moveresize_window)
1355 button_mask = GDK_BUTTON1_MASK << (mv_resize->moveresize_button - 1);
1357 switch (event->base.type)
1360 if (mv_resize->moveresize_window->resize_count > 0)
1362 if (mv_resize->moveresize_pending_event)
1363 *mv_resize->moveresize_pending_event = *event;
1365 mv_resize->moveresize_pending_event =
1366 g_memdup (event, sizeof (BroadwayInputMsg));
1370 if (!moveresize_lookahead (display, mv_resize, event))
1373 update_pos (mv_resize,
1374 event->pointer.root_x,
1375 event->pointer.root_y);
1377 /* This should never be triggered in normal cases, but in the
1378 * case where the drag started without an implicit grab being
1379 * in effect, we could miss the release if it occurs before
1380 * we grab the pointer; this ensures that we will never
1381 * get a permanently stuck grab.
1383 if ((event->pointer.state & button_mask) == 0)
1384 finish_drag (mv_resize);
1388 update_pos (mv_resize,
1389 event->pointer.root_x,
1390 event->pointer.root_y);
1392 if (event->button.button == mv_resize->moveresize_button)
1393 finish_drag (mv_resize);
1400 _gdk_broadway_moveresize_configure_done (GdkDisplay *display,
1403 BroadwayInputMsg *tmp_event;
1404 MoveResizeData *mv_resize = get_move_resize_data (display, FALSE);
1406 if (!mv_resize || window != mv_resize->moveresize_window)
1409 if (mv_resize->moveresize_pending_event)
1411 tmp_event = mv_resize->moveresize_pending_event;
1412 mv_resize->moveresize_pending_event = NULL;
1413 _gdk_broadway_moveresize_handle_event (display, tmp_event);
1421 create_moveresize_window (MoveResizeData *mv_resize,
1424 GdkWindowAttr attributes;
1425 gint attributes_mask;
1426 GdkGrabStatus status;
1428 g_assert (mv_resize->moveresize_emulation_window == NULL);
1430 attributes.x = -100;
1431 attributes.y = -100;
1432 attributes.width = 10;
1433 attributes.height = 10;
1434 attributes.window_type = GDK_WINDOW_TEMP;
1435 attributes.wclass = GDK_INPUT_ONLY;
1436 attributes.override_redirect = TRUE;
1437 attributes.event_mask = 0;
1439 attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_NOREDIR;
1441 mv_resize->moveresize_emulation_window =
1442 gdk_window_new (gdk_screen_get_root_window (gdk_display_get_default_screen (mv_resize->display)),
1446 gdk_window_show (mv_resize->moveresize_emulation_window);
1448 status = gdk_pointer_grab (mv_resize->moveresize_emulation_window,
1450 GDK_BUTTON_RELEASE_MASK |
1451 GDK_POINTER_MOTION_MASK,
1456 if (status != GDK_GRAB_SUCCESS)
1458 /* If this fails, some other client has grabbed the window
1461 finish_drag (mv_resize);
1464 mv_resize->moveresize_process_time = 0;
1468 gdk_broadway_window_begin_resize_drag (GdkWindow *window,
1476 GdkBroadwayDisplay *broadway_display;
1477 MoveResizeData *mv_resize;
1479 if (GDK_WINDOW_DESTROYED (window) ||
1480 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1483 /* We need a connection to be able to get mouse events, if not, punt */
1484 broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (window));
1485 if (!broadway_display->output)
1488 mv_resize = get_move_resize_data (GDK_WINDOW_DISPLAY (window), TRUE);
1490 mv_resize->is_resize = TRUE;
1491 mv_resize->moveresize_button = button;
1492 mv_resize->resize_edge = edge;
1493 mv_resize->moveresize_x = root_x;
1494 mv_resize->moveresize_y = root_y;
1495 mv_resize->moveresize_window = g_object_ref (window);
1497 gdk_window_get_origin (mv_resize->moveresize_window,
1498 &mv_resize->moveresize_orig_x,
1499 &mv_resize->moveresize_orig_y);
1500 mv_resize->moveresize_orig_width = gdk_window_get_width (window);
1501 mv_resize->moveresize_orig_height = gdk_window_get_height (window);
1503 create_moveresize_window (mv_resize, timestamp);
1507 gdk_broadway_window_begin_move_drag (GdkWindow *window,
1514 if (GDK_WINDOW_DESTROYED (window) ||
1515 !WINDOW_IS_TOPLEVEL (window))
1521 gdk_broadway_window_enable_synchronized_configure (GdkWindow *window)
1523 if (!GDK_IS_WINDOW_IMPL_BROADWAY (window->impl))
1528 gdk_broadway_window_configure_finished (GdkWindow *window)
1530 if (!WINDOW_IS_TOPLEVEL (window))
1535 gdk_broadway_window_beep (GdkWindow *window)
1541 gdk_broadway_window_set_opacity (GdkWindow *window,
1544 g_return_if_fail (GDK_IS_WINDOW (window));
1546 if (GDK_WINDOW_DESTROYED (window) ||
1547 !WINDOW_IS_TOPLEVEL (window))
1552 else if (opacity > 1)
1557 gdk_broadway_window_set_composited (GdkWindow *window,
1558 gboolean composited)
1563 gdk_broadway_window_process_updates_recurse (GdkWindow *window,
1564 cairo_region_t *region)
1566 GdkWindowImplBroadway *impl;
1568 _gdk_window_process_updates_recurse (window, region);
1570 impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
1572 queue_dirty_flush (GDK_BROADWAY_DISPLAY (gdk_window_get_display (window)));
1576 _gdk_broadway_display_before_process_all_updates (GdkDisplay *display)
1581 _gdk_broadway_display_after_process_all_updates (GdkDisplay *display)
1586 _gdk_broadway_window_queue_antiexpose (GdkWindow *window,
1587 cairo_region_t *area)
1593 copy_region (cairo_surface_t *surface,
1594 cairo_region_t *area,
1600 cr = cairo_create (surface);
1602 gdk_cairo_region (cr, area);
1605 /* NB: This is a self-copy and Cairo doesn't support that yet.
1606 * So we do a litle trick.
1608 cairo_push_group (cr);
1610 cairo_set_source_surface (cr, surface, dx, dy);
1613 cairo_pop_group_to_source (cr);
1620 _gdk_broadway_window_translate (GdkWindow *window,
1621 cairo_region_t *area,
1625 GdkWindowImplBroadway *impl;
1626 GdkBroadwayDisplay *broadway_display;
1628 BroadwayRect *rects;
1629 cairo_rectangle_int_t rect;
1631 impl = GDK_WINDOW_IMPL_BROADWAY (window->impl);
1635 copy_region (impl->surface, area, dx, dy);
1636 broadway_display = GDK_BROADWAY_DISPLAY (gdk_window_get_display (window));
1637 if (GDK_WINDOW_IMPL_BROADWAY (impl)->last_synced &&
1638 broadway_display->output)
1640 copy_region (impl->last_surface, area, dx, dy);
1641 n_rects = cairo_region_num_rectangles (area);
1642 rects = g_new (BroadwayRect, n_rects);
1643 for (i = 0; i < n_rects; i++)
1645 cairo_region_get_rectangle (area, i, &rect);
1646 rects[i].x = rect.x;
1647 rects[i].y = rect.y;
1648 rects[i].width = rect.width;
1649 rects[i].height = rect.height;
1651 broadway_output_copy_rectangles (broadway_display->output,
1652 GDK_WINDOW_IMPL_BROADWAY (impl)->id,
1653 rects, n_rects, dx, dy);
1654 queue_dirty_flush (broadway_display);
1661 gdk_broadway_get_last_seen_time (GdkWindow *window)
1663 GdkDisplay *display;
1665 display = gdk_window_get_display (window);
1666 _gdk_broadway_display_consume_all_input (display);
1667 return (guint32) GDK_BROADWAY_DISPLAY (display)->last_seen_time;
1671 gdk_window_impl_broadway_class_init (GdkWindowImplBroadwayClass *klass)
1673 GObjectClass *object_class = G_OBJECT_CLASS (klass);
1674 GdkWindowImplClass *impl_class = GDK_WINDOW_IMPL_CLASS (klass);
1676 object_class->finalize = gdk_window_impl_broadway_finalize;
1678 impl_class->ref_cairo_surface = gdk_window_broadway_ref_cairo_surface;
1679 impl_class->show = gdk_window_broadway_show;
1680 impl_class->hide = gdk_window_broadway_hide;
1681 impl_class->withdraw = gdk_window_broadway_withdraw;
1682 impl_class->set_events = gdk_window_broadway_set_events;
1683 impl_class->get_events = gdk_window_broadway_get_events;
1684 impl_class->raise = gdk_window_broadway_raise;
1685 impl_class->lower = gdk_window_broadway_lower;
1686 impl_class->restack_under = gdk_window_broadway_restack_under;
1687 impl_class->restack_toplevel = gdk_window_broadway_restack_toplevel;
1688 impl_class->move_resize = gdk_window_broadway_move_resize;
1689 impl_class->set_background = gdk_window_broadway_set_background;
1690 impl_class->reparent = gdk_window_broadway_reparent;
1691 impl_class->set_device_cursor = gdk_window_broadway_set_device_cursor;
1692 impl_class->get_geometry = gdk_window_broadway_get_geometry;
1693 impl_class->get_root_coords = gdk_window_broadway_get_root_coords;
1694 impl_class->get_device_state = gdk_window_broadway_get_device_state;
1695 impl_class->shape_combine_region = gdk_window_broadway_shape_combine_region;
1696 impl_class->input_shape_combine_region = gdk_window_broadway_input_shape_combine_region;
1697 impl_class->set_static_gravities = gdk_window_broadway_set_static_gravities;
1698 impl_class->queue_antiexpose = _gdk_broadway_window_queue_antiexpose;
1699 impl_class->translate = _gdk_broadway_window_translate;
1700 impl_class->destroy = _gdk_broadway_window_destroy;
1701 impl_class->destroy_foreign = gdk_broadway_window_destroy_foreign;
1702 impl_class->resize_cairo_surface = gdk_window_broadway_resize_cairo_surface;
1703 impl_class->get_shape = gdk_broadway_window_get_shape;
1704 impl_class->get_input_shape = gdk_broadway_window_get_input_shape;
1705 impl_class->beep = gdk_broadway_window_beep;
1707 impl_class->focus = gdk_broadway_window_focus;
1708 impl_class->set_type_hint = gdk_broadway_window_set_type_hint;
1709 impl_class->get_type_hint = gdk_broadway_window_get_type_hint;
1710 impl_class->set_modal_hint = gdk_broadway_window_set_modal_hint;
1711 impl_class->set_skip_taskbar_hint = gdk_broadway_window_set_skip_taskbar_hint;
1712 impl_class->set_skip_pager_hint = gdk_broadway_window_set_skip_pager_hint;
1713 impl_class->set_urgency_hint = gdk_broadway_window_set_urgency_hint;
1714 impl_class->set_geometry_hints = gdk_broadway_window_set_geometry_hints;
1715 impl_class->set_title = gdk_broadway_window_set_title;
1716 impl_class->set_role = gdk_broadway_window_set_role;
1717 impl_class->set_startup_id = gdk_broadway_window_set_startup_id;
1718 impl_class->set_transient_for = gdk_broadway_window_set_transient_for;
1719 impl_class->get_root_origin = gdk_broadway_window_get_root_origin;
1720 impl_class->get_frame_extents = gdk_broadway_window_get_frame_extents;
1721 impl_class->set_override_redirect = gdk_broadway_window_set_override_redirect;
1722 impl_class->set_accept_focus = gdk_broadway_window_set_accept_focus;
1723 impl_class->set_focus_on_map = gdk_broadway_window_set_focus_on_map;
1724 impl_class->set_icon_list = gdk_broadway_window_set_icon_list;
1725 impl_class->set_icon_name = gdk_broadway_window_set_icon_name;
1726 impl_class->iconify = gdk_broadway_window_iconify;
1727 impl_class->deiconify = gdk_broadway_window_deiconify;
1728 impl_class->stick = gdk_broadway_window_stick;
1729 impl_class->unstick = gdk_broadway_window_unstick;
1730 impl_class->maximize = gdk_broadway_window_maximize;
1731 impl_class->unmaximize = gdk_broadway_window_unmaximize;
1732 impl_class->fullscreen = gdk_broadway_window_fullscreen;
1733 impl_class->unfullscreen = gdk_broadway_window_unfullscreen;
1734 impl_class->set_keep_above = gdk_broadway_window_set_keep_above;
1735 impl_class->set_keep_below = gdk_broadway_window_set_keep_below;
1736 impl_class->get_group = gdk_broadway_window_get_group;
1737 impl_class->set_group = gdk_broadway_window_set_group;
1738 impl_class->set_decorations = gdk_broadway_window_set_decorations;
1739 impl_class->get_decorations = gdk_broadway_window_get_decorations;
1740 impl_class->set_functions = gdk_broadway_window_set_functions;
1741 impl_class->set_functions = gdk_broadway_window_set_functions;
1742 impl_class->begin_resize_drag = gdk_broadway_window_begin_resize_drag;
1743 impl_class->begin_move_drag = gdk_broadway_window_begin_move_drag;
1744 impl_class->enable_synchronized_configure = gdk_broadway_window_enable_synchronized_configure;
1745 impl_class->configure_finished = gdk_broadway_window_configure_finished;
1746 impl_class->set_opacity = gdk_broadway_window_set_opacity;
1747 impl_class->set_composited = gdk_broadway_window_set_composited;
1748 impl_class->destroy_notify = gdk_broadway_window_destroy_notify;
1749 impl_class->register_dnd = _gdk_broadway_window_register_dnd;
1750 impl_class->drag_begin = _gdk_broadway_window_drag_begin;
1751 impl_class->process_updates_recurse = gdk_broadway_window_process_updates_recurse;
1752 impl_class->sync_rendering = _gdk_broadway_window_sync_rendering;
1753 impl_class->simulate_key = _gdk_broadway_window_simulate_key;
1754 impl_class->simulate_button = _gdk_broadway_window_simulate_button;
1755 impl_class->get_property = _gdk_broadway_window_get_property;
1756 impl_class->change_property = _gdk_broadway_window_change_property;
1757 impl_class->delete_property = _gdk_broadway_window_delete_property;
1758 impl_class->get_drag_protocol = _gdk_broadway_window_get_drag_protocol;