2 * Copyright © 2010 Intel Corporation
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public License
6 * as published by the Free Software Foundation; either version 2 of
7 * the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful, but
10 * WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library. If not, see <http://www.gnu.org/licenses/>.
20 #include <netinet/in.h>
24 #include "gdkwayland.h"
26 #include "gdkwindow.h"
27 #include "gdkwindowimpl.h"
28 #include "gdkdisplay-wayland.h"
29 #include "gdkprivate-wayland.h"
30 #include "gdkinternals.h"
31 #include "gdkdeviceprivate.h"
39 #include <wayland-egl.h>
41 #define WINDOW_IS_TOPLEVEL_OR_FOREIGN(window) \
42 (GDK_WINDOW_TYPE (window) != GDK_WINDOW_CHILD && \
43 GDK_WINDOW_TYPE (window) != GDK_WINDOW_OFFSCREEN)
45 #define WINDOW_IS_TOPLEVEL(window) \
46 (GDK_WINDOW_TYPE (window) != GDK_WINDOW_CHILD && \
47 GDK_WINDOW_TYPE (window) != GDK_WINDOW_FOREIGN && \
48 GDK_WINDOW_TYPE (window) != GDK_WINDOW_OFFSCREEN)
50 /* Return whether time1 is considered later than time2 as far as xserver
51 * time is concerned. Accounts for wraparound.
53 #define XSERVER_TIME_IS_LATER(time1, time2) \
54 ( (( time1 > time2 ) && ( time1 - time2 < ((guint32)-1)/2 )) || \
55 (( time1 < time2 ) && ( time2 - time1 > ((guint32)-1)/2 )) \
58 typedef struct _GdkWaylandWindow GdkWaylandWindow;
59 typedef struct _GdkWaylandWindowClass GdkWaylandWindowClass;
61 struct _GdkWaylandWindow {
65 struct _GdkWaylandWindowClass {
66 GdkWindowClass parent_class;
69 G_DEFINE_TYPE (GdkWaylandWindow, gdk_wayland_window, GDK_TYPE_WINDOW)
72 gdk_wayland_window_class_init (GdkWaylandWindowClass *wayland_window_class)
77 gdk_wayland_window_init (GdkWaylandWindow *wayland_window)
81 #define GDK_TYPE_WINDOW_IMPL_WAYLAND (_gdk_window_impl_wayland_get_type ())
82 #define GDK_WINDOW_IMPL_WAYLAND(object) (G_TYPE_CHECK_INSTANCE_CAST ((object), GDK_TYPE_WINDOW_IMPL_WAYLAND, GdkWindowImplWayland))
83 #define GDK_WINDOW_IMPL_WAYLAND_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GDK_TYPE_WINDOW_IMPL_WAYLAND, GdkWindowImplWaylandClass))
84 #define GDK_IS_WINDOW_IMPL_WAYLAND(object) (G_TYPE_CHECK_INSTANCE_TYPE ((object), GDK_TYPE_WINDOW_IMPL_WAYLAND))
85 #define GDK_IS_WINDOW_IMPL_WAYLAND_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GDK_TYPE_WINDOW_IMPL_WAYLAND))
86 #define GDK_WINDOW_IMPL_WAYLAND_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GDK_TYPE_WINDOW_IMPL_WAYLAND, GdkWindowImplWaylandClass))
88 typedef struct _GdkWindowImplWayland GdkWindowImplWayland;
89 typedef struct _GdkWindowImplWaylandClass GdkWindowImplWaylandClass;
91 struct _GdkWindowImplWayland
93 GdkWindowImpl parent_instance;
99 gint8 toplevel_window_type;
101 struct wl_surface *surface;
102 struct wl_shell_surface *shell_surface;
103 unsigned int mapped : 1;
104 GdkWindow *transient_for;
105 GdkWindowTypeHint hint;
107 /* The surface which is being "drawn to" to */
108 cairo_surface_t *cairo_surface;
110 /* The surface that was the last surface the Wayland buffer from which was attached
111 * to the Wayland surface. It will be the same as cairo_surface after a call
112 * to gdk_wayland_window_attach_image. But after a call to
113 * gdk_wayland_window_update_size and then
114 * gdk_wayland_window_ref_cairo_surface the above pointer will be different.
116 cairo_surface_t *server_surface;
118 uint32_t resize_edges;
122 gulong map_serial; /* Serial of last transition from unmapped */
124 cairo_surface_t *icon_pixmap;
125 cairo_surface_t *icon_mask;
127 /* Time of most recent user interaction. */
130 GdkGeometry geometry_hints;
131 GdkWindowHints geometry_mask;
133 GdkDevice *grab_device;
134 struct wl_seat *grab_input_seat;
138 int saved_width, saved_height; /* before going fullscreen */
141 struct _GdkWindowImplWaylandClass
143 GdkWindowImplClass parent_class;
146 G_DEFINE_TYPE (GdkWindowImplWayland, _gdk_window_impl_wayland, GDK_TYPE_WINDOW_IMPL)
149 _gdk_window_impl_wayland_init (GdkWindowImplWayland *impl)
151 impl->toplevel_window_type = -1;
155 _gdk_wayland_window_add_focus (GdkWindow *window)
157 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
160 if (impl->focus_count == 1)
161 gdk_synthesize_window_state (window, 0, GDK_WINDOW_STATE_FOCUSED);
165 _gdk_wayland_window_remove_focus (GdkWindow *window)
167 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
170 if (impl->focus_count == 0)
171 gdk_synthesize_window_state (window, GDK_WINDOW_STATE_FOCUSED, 0);
175 * gdk_wayland_window_update_size:
176 * @drawable: a #GdkDrawableImplWayland.
178 * Updates the state of the drawable (in particular the drawable's
179 * cairo surface) when its size has changed.
182 gdk_wayland_window_update_size (GdkWindow *window,
183 int32_t width, int32_t height, uint32_t edges)
185 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
187 cairo_region_t *region;
189 if (impl->cairo_surface)
191 cairo_surface_destroy (impl->cairo_surface);
192 impl->cairo_surface = NULL;
195 window->width = width;
196 window->height = height;
197 impl->resize_edges = edges;
201 area.width = window->width;
202 area.height = window->height;
204 region = cairo_region_create_rectangle (&area);
205 _gdk_window_invalidate_for_expose (window, region);
206 cairo_region_destroy (region);
210 _gdk_wayland_screen_create_root_window (GdkScreen *screen,
211 int width, int height)
214 GdkWindowImplWayland *impl;
216 window = _gdk_display_create_window (gdk_screen_get_display (screen));
217 window->impl = g_object_new (GDK_TYPE_WINDOW_IMPL_WAYLAND, NULL);
218 window->impl_window = window;
219 window->visual = gdk_screen_get_system_visual (screen);
221 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
223 impl->wrapper = GDK_WINDOW (window);
225 window->window_type = GDK_WINDOW_ROOT;
232 window->width = width;
233 window->height = height;
234 window->viewable = TRUE;
236 /* see init_randr_support() in gdkscreen-wayland.c */
237 window->event_mask = GDK_STRUCTURE_MASK;
243 get_default_title (void)
247 title = g_get_application_name ();
249 title = g_get_prgname ();
257 _gdk_wayland_display_create_window_impl (GdkDisplay *display,
259 GdkWindow *real_parent,
261 GdkEventMask event_mask,
262 GdkWindowAttr *attributes,
263 gint attributes_mask)
265 GdkWindowImplWayland *impl;
268 impl = g_object_new (GDK_TYPE_WINDOW_IMPL_WAYLAND, NULL);
269 window->impl = GDK_WINDOW_IMPL (impl);
270 impl->wrapper = GDK_WINDOW (window);
272 if (window->width > 65535 ||
273 window->height > 65535)
275 g_warning ("Native Windows wider or taller than 65535 pixels are not supported");
277 if (window->width > 65535)
278 window->width = 65535;
279 if (window->height > 65535)
280 window->height = 65535;
283 g_object_ref (window);
285 switch (GDK_WINDOW_TYPE (window))
287 case GDK_WINDOW_TOPLEVEL:
288 case GDK_WINDOW_TEMP:
289 if (attributes_mask & GDK_WA_TITLE)
290 title = attributes->title;
292 title = get_default_title ();
294 gdk_window_set_title (window, title);
297 case GDK_WINDOW_CHILD:
302 if (attributes_mask & GDK_WA_TYPE_HINT)
303 gdk_window_set_type_hint (window, attributes->type_hint);
306 static const cairo_user_data_key_t gdk_wayland_cairo_key;
308 typedef struct _GdkWaylandCairoSurfaceData {
311 struct wl_shm_pool *pool;
312 struct wl_buffer *buffer;
313 GdkWaylandDisplay *display;
314 int32_t width, height;
315 } GdkWaylandCairoSurfaceData;
318 gdk_wayland_window_attach_image (GdkWindow *window)
320 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
321 GdkWaylandCairoSurfaceData *data;
322 int32_t server_width, server_height, dx, dy;
324 if (GDK_WINDOW_DESTROYED (window))
327 /* The "drawn to" Cairo surface is the same as the Cairo surface from which
328 * we are driving the buffer for the Wayland surface. Therefore we don't
329 * need to do anything here
331 if (impl->server_surface == impl->cairo_surface)
334 /* The wayland surface is attached to a buffer that is from the old "drawn
335 * to" surface. Unref the surface and restore the state.
337 if (impl->server_surface)
339 data = cairo_surface_get_user_data (impl->server_surface,
340 &gdk_wayland_cairo_key);
342 /* Save the old dimensions used for the surface */
343 server_width = data->width;
344 server_height = data->height;
346 cairo_surface_destroy (impl->server_surface);
354 /* Save the current "drawn to" surface for future calls into here */
355 impl->server_surface = cairo_surface_reference (impl->cairo_surface);
357 /* Get a Wayland buffer from this new surface */
358 data = cairo_surface_get_user_data (impl->cairo_surface,
359 &gdk_wayland_cairo_key);
361 if (impl->resize_edges & WL_SHELL_SURFACE_RESIZE_LEFT)
362 dx = server_width - data->width;
366 if (impl->resize_edges & WL_SHELL_SURFACE_RESIZE_TOP)
367 dy = server_height - data->height;
371 /* Attach this new buffer to the surface */
372 wl_surface_attach (impl->surface, data->buffer, dx, dy);
376 gdk_wayland_cairo_surface_destroy (void *p)
378 GdkWaylandCairoSurfaceData *data = p;
381 wl_buffer_destroy (data->buffer);
384 wl_shm_pool_destroy (data->pool);
386 munmap (data->buf, data->buf_length);
391 static struct wl_shm_pool *
392 _create_shm_pool (struct wl_shm *shm,
398 char filename[] = "/tmp/wayland-shm-XXXXXX";
399 struct wl_shm_pool *pool;
400 int fd, size, stride;
403 fd = mkstemp (filename);
405 g_critical (G_STRLOC ": Unable to create temporary file (%s): %s",
406 filename, g_strerror (errno));
410 size = stride * height;
411 if (ftruncate (fd, size) < 0) {
412 g_critical (G_STRLOC ": Truncating temporary file failed: %s",
418 data = mmap (NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
421 if (data == MAP_FAILED) {
422 g_critical (G_STRLOC ": mmap'ping temporary file failed: %s",
428 pool = wl_shm_create_pool(shm, fd, size);
438 static cairo_surface_t *
439 gdk_wayland_create_cairo_surface (GdkWaylandDisplay *display,
440 int width, int height)
442 GdkWaylandCairoSurfaceData *data;
443 cairo_surface_t *surface = NULL;
444 cairo_status_t status;
447 data = g_new (GdkWaylandCairoSurfaceData, 1);
448 data->display = display;
451 data->height = height;
455 data->pool = _create_shm_pool (display->shm,
460 data->buffer = wl_shm_pool_create_buffer (data->pool, 0,
462 stride, WL_SHM_FORMAT_ARGB8888);
464 surface = cairo_image_surface_create_for_data (data->buf,
470 cairo_surface_set_user_data (surface, &gdk_wayland_cairo_key,
471 data, gdk_wayland_cairo_surface_destroy);
473 status = cairo_surface_status (surface);
474 if (status != CAIRO_STATUS_SUCCESS)
476 g_critical (G_STRLOC ": Unable to create Cairo image surface: %s",
477 cairo_status_to_string (status));
483 /* On this first call this creates a double reference - the first reference
484 * is held by the GdkWindowImplWayland struct - since unlike other backends
485 * the Cairo surface is not just a cheap wrapper around some other backing.
486 * It is the buffer itself.
488 static cairo_surface_t *
489 gdk_wayland_window_ref_cairo_surface (GdkWindow *window)
491 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
492 GdkWaylandDisplay *display_wayland =
493 GDK_WAYLAND_DISPLAY (gdk_window_get_display (impl->wrapper));
495 if (GDK_WINDOW_DESTROYED (impl->wrapper))
498 if (!impl->cairo_surface)
500 impl->cairo_surface =
501 gdk_wayland_create_cairo_surface (display_wayland,
502 impl->wrapper->width,
503 impl->wrapper->height);
506 cairo_surface_reference (impl->cairo_surface);
508 return impl->cairo_surface;
513 gdk_window_impl_wayland_finalize (GObject *object)
515 GdkWindowImplWayland *impl;
517 g_return_if_fail (GDK_IS_WINDOW_IMPL_WAYLAND (object));
519 impl = GDK_WINDOW_IMPL_WAYLAND (object);
522 g_object_unref (impl->cursor);
523 if (impl->server_surface)
524 cairo_surface_destroy (impl->server_surface);
526 G_OBJECT_CLASS (_gdk_window_impl_wayland_parent_class)->finalize (object);
530 gdk_wayland_window_configure (GdkWindow *window,
531 int width, int height, int edges)
536 display = gdk_window_get_display (window);
538 /* TODO: Only generate a configure event if width or height have actually
541 event = gdk_event_new (GDK_CONFIGURE);
542 event->configure.window = window;
543 event->configure.send_event = FALSE;
544 event->configure.width = width;
545 event->configure.height = height;
547 _gdk_window_update_size (window);
548 gdk_wayland_window_update_size (window, width, height, edges);
550 g_object_ref(window);
552 _gdk_wayland_display_deliver_event (display, event);
556 gdk_wayland_window_set_user_time (GdkWindow *window, guint32 user_time)
561 gdk_wayland_window_map (GdkWindow *window)
563 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
564 GdkWindowImplWayland *parent;
565 GdkWaylandDisplay *wayland_display =
566 GDK_WAYLAND_DISPLAY (gdk_window_get_display (window));
567 GdkWindow *transient_for;
571 /* Popup menus can appear without a transient parent, which means they
572 * cannot be positioned properly on Wayland. This attempts to guess the
573 * surface they should be positioned with by finding the surface beneath
574 * the device that created the grab for the popup window */
576 if (!impl->transient_for && impl->hint == GDK_WINDOW_TYPE_HINT_POPUP_MENU)
578 transient_for = gdk_device_get_window_at_position (impl->grab_device, NULL, NULL);
579 transient_for = gdk_window_get_toplevel (transient_for);
581 /* start the popup at the position of the device that holds the grab */
582 gdk_window_get_device_position (transient_for,
584 &window->x, &window->y, NULL);
587 transient_for = impl->transient_for;
591 struct wl_seat *grab_input_seat = NULL;
592 GdkWindowImplWayland *tmp_impl;
594 parent = GDK_WINDOW_IMPL_WAYLAND (transient_for->impl);
596 /* Use the device that was used for the grab as the device for
597 * the popup window setup - so this relies on GTK+ taking the
598 * grab before showing the popup window.
600 grab_input_seat = impl->grab_input_seat;
603 while (!grab_input_seat)
605 grab_input_seat = tmp_impl->grab_input_seat;
607 if (tmp_impl->transient_for)
608 tmp_impl = GDK_WINDOW_IMPL_WAYLAND (tmp_impl->transient_for->impl);
613 if (grab_input_seat &&
614 (impl->hint == GDK_WINDOW_TYPE_HINT_POPUP_MENU ||
615 impl->hint == GDK_WINDOW_TYPE_HINT_DROPDOWN_MENU ||
616 impl->hint == GDK_WINDOW_TYPE_HINT_COMBO))
619 wl_shell_surface_set_popup (impl->shell_surface,
621 _gdk_wayland_display_get_serial (wayland_display),
623 window->x, window->y, 0);
629 if (impl->hint == GDK_WINDOW_TYPE_HINT_TOOLTIP)
630 flags = WL_SHELL_SURFACE_TRANSIENT_INACTIVE;
632 wl_shell_surface_set_transient (impl->shell_surface, parent->surface,
633 window->x, window->y, flags);
638 wl_shell_surface_set_toplevel (impl->shell_surface);
645 shell_surface_handle_configure(void *data,
646 struct wl_shell_surface *shell_surface,
651 GdkWindow *window = GDK_WINDOW (data);
652 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
654 gdk_window_constrain_size (&impl->geometry_hints,
661 gdk_wayland_window_configure (window, width, height, edges);
665 shell_surface_popup_done (void *data,
666 struct wl_shell_surface *shell_surface)
668 GdkWindow *window = GDK_WINDOW (data);
670 /* When the popup is complete hide the window - this really relies on the
671 * fix in https://bugzilla.gnome.org/show_bug.cgi?id=670881 to work
674 gdk_window_hide (window);
678 shell_surface_ping (void *data,
679 struct wl_shell_surface *shell_surface,
683 GdkWindow *window = GDK_WINDOW (data);
684 GdkWaylandDisplay *wayland_display =
685 GDK_WAYLAND_DISPLAY (gdk_window_get_display (window));
687 _gdk_wayland_display_update_serial (wayland_display, serial);
689 wl_shell_surface_pong(shell_surface, serial);
692 static const struct wl_shell_surface_listener shell_surface_listener = {
694 shell_surface_handle_configure,
695 shell_surface_popup_done
699 gdk_wayland_window_show (GdkWindow *window, gboolean already_mapped)
702 GdkWaylandDisplay *display_wayland;
703 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
706 display = gdk_window_get_display (window);
707 display_wayland = GDK_WAYLAND_DISPLAY (display);
709 if (impl->user_time != 0 &&
710 display_wayland->user_time != 0 &&
711 XSERVER_TIME_IS_LATER (display_wayland->user_time, impl->user_time))
712 gdk_wayland_window_set_user_time (window, impl->user_time);
714 impl->surface = wl_compositor_create_surface(display_wayland->compositor);
715 wl_surface_set_user_data(impl->surface, window);
717 impl->shell_surface = wl_shell_get_shell_surface (display_wayland->shell,
719 wl_shell_surface_add_listener(impl->shell_surface,
720 &shell_surface_listener, window);
722 gdk_window_set_type_hint (window, impl->hint);
724 _gdk_make_event (window, GDK_MAP, NULL, FALSE);
725 event = _gdk_make_event (window, GDK_VISIBILITY_NOTIFY, NULL, FALSE);
726 event->visibility.state = GDK_VISIBILITY_UNOBSCURED;
728 if (impl->cairo_surface)
729 gdk_wayland_window_attach_image (window);
733 gdk_wayland_window_hide (GdkWindow *window)
735 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
739 if (impl->shell_surface)
740 wl_shell_surface_destroy(impl->shell_surface);
742 wl_surface_destroy(impl->surface);
743 impl->shell_surface = NULL;
744 impl->surface = NULL;
745 cairo_surface_destroy(impl->server_surface);
746 impl->server_surface = NULL;
747 impl->mapped = FALSE;
750 _gdk_window_clear_update_area (window);
754 gdk_window_wayland_withdraw (GdkWindow *window)
756 GdkWindowImplWayland *impl;
758 if (!window->destroyed)
760 if (GDK_WINDOW_IS_MAPPED (window))
761 gdk_synthesize_window_state (window, 0, GDK_WINDOW_STATE_WITHDRAWN);
763 g_assert (!GDK_WINDOW_IS_MAPPED (window));
765 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
768 if (impl->shell_surface)
769 wl_shell_surface_destroy(impl->shell_surface);
771 wl_surface_destroy(impl->surface);
772 impl->shell_surface = NULL;
773 impl->surface = NULL;
774 cairo_surface_destroy(impl->server_surface);
775 impl->server_surface = NULL;
776 impl->mapped = FALSE;
782 gdk_window_wayland_set_events (GdkWindow *window,
783 GdkEventMask event_mask)
785 GDK_WINDOW (window)->event_mask = event_mask;
789 gdk_window_wayland_get_events (GdkWindow *window)
791 if (GDK_WINDOW_DESTROYED (window))
794 return GDK_WINDOW (window)->event_mask;
798 gdk_window_wayland_raise (GdkWindow *window)
800 /* FIXME: wl_shell_raise() */
804 gdk_window_wayland_lower (GdkWindow *window)
806 /* FIXME: wl_shell_lower() */
810 gdk_window_wayland_restack_under (GdkWindow *window,
811 GList *native_siblings)
816 gdk_window_wayland_restack_toplevel (GdkWindow *window,
823 gdk_window_wayland_move_resize (GdkWindow *window,
836 /* If this function is called with width and height = -1 then that means
837 * just move the window - don't update its size
839 if (width > 0 && height > 0)
840 gdk_wayland_window_configure (window, width, height, 0);
844 gdk_window_wayland_set_background (GdkWindow *window,
845 cairo_pattern_t *pattern)
850 gdk_window_wayland_reparent (GdkWindow *window,
851 GdkWindow *new_parent,
859 gdk_window_wayland_set_device_cursor (GdkWindow *window,
863 g_return_if_fail (GDK_IS_WINDOW (window));
864 g_return_if_fail (GDK_IS_DEVICE (device));
866 if (!GDK_WINDOW_DESTROYED (window))
867 GDK_DEVICE_GET_CLASS (device)->set_window_cursor (device, window, cursor);
871 gdk_window_wayland_get_geometry (GdkWindow *window,
877 if (!GDK_WINDOW_DESTROYED (window))
884 *width = window->width;
886 *height = window->height;
891 _gdk_wayland_window_offset (GdkWindow *window,
895 GdkWindowImplWayland *impl, *parent_impl;
896 GdkWindow *parent_window;
897 gint x_offset = 0, y_offset = 0;
899 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
901 parent_window = impl->transient_for;
902 while (parent_window)
904 parent_impl = GDK_WINDOW_IMPL_WAYLAND (parent_window->impl);
906 x_offset += window->x;
907 y_offset += window->y;
909 parent_window = parent_impl->transient_for;
917 gdk_window_wayland_get_root_coords (GdkWindow *window,
923 gint x_offset, y_offset;
925 _gdk_wayland_window_offset (window, &x_offset, &y_offset);
927 *root_x = x_offset + x;
928 *root_y = y_offset + y;
934 gdk_window_wayland_get_device_state (GdkWindow *window,
938 GdkModifierType *mask)
942 g_return_val_if_fail (window == NULL || GDK_IS_WINDOW (window), FALSE);
946 if (!GDK_WINDOW_DESTROYED (window))
950 GDK_DEVICE_GET_CLASS (device)->query_state (device, window,
954 return_val = (child != NULL);
961 gdk_window_wayland_shape_combine_region (GdkWindow *window,
962 const cairo_region_t *shape_region,
969 gdk_window_wayland_input_shape_combine_region (GdkWindow *window,
970 const cairo_region_t *shape_region,
977 gdk_window_wayland_set_static_gravities (GdkWindow *window,
984 gdk_wayland_window_queue_antiexpose (GdkWindow *window,
985 cairo_region_t *area)
991 gdk_wayland_window_translate (GdkWindow *window,
992 cairo_region_t *area,
996 _gdk_window_invalidate_for_expose (window, area);
1000 gdk_wayland_window_destroy (GdkWindow *window,
1002 gboolean foreign_destroy)
1004 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1006 g_return_if_fail (GDK_IS_WINDOW (window));
1008 if (impl->cairo_surface)
1010 cairo_surface_finish (impl->cairo_surface);
1011 cairo_surface_set_user_data (impl->cairo_surface, &gdk_wayland_cairo_key,
1015 if (!recursing && !foreign_destroy)
1017 if (impl->shell_surface)
1018 wl_shell_surface_destroy(impl->shell_surface);
1020 wl_surface_destroy(impl->surface);
1021 impl->shell_surface = NULL;
1022 impl->surface = NULL;
1027 gdk_window_wayland_destroy_foreign (GdkWindow *window)
1031 static cairo_surface_t *
1032 gdk_window_wayland_resize_cairo_surface (GdkWindow *window,
1033 cairo_surface_t *surface,
1040 static cairo_region_t *
1041 gdk_wayland_window_get_shape (GdkWindow *window)
1046 static cairo_region_t *
1047 gdk_wayland_window_get_input_shape (GdkWindow *window)
1053 gdk_wayland_window_focus (GdkWindow *window,
1056 /* FIXME: wl_shell_focus() */
1060 gdk_wayland_window_set_type_hint (GdkWindow *window,
1061 GdkWindowTypeHint hint)
1063 GdkWindowImplWayland *impl;
1065 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1067 if (GDK_WINDOW_DESTROYED (window))
1074 case GDK_WINDOW_TYPE_HINT_MENU:
1075 case GDK_WINDOW_TYPE_HINT_TOOLBAR:
1076 case GDK_WINDOW_TYPE_HINT_UTILITY:
1077 case GDK_WINDOW_TYPE_HINT_DOCK:
1078 case GDK_WINDOW_TYPE_HINT_DESKTOP:
1079 case GDK_WINDOW_TYPE_HINT_DROPDOWN_MENU:
1080 case GDK_WINDOW_TYPE_HINT_POPUP_MENU:
1081 case GDK_WINDOW_TYPE_HINT_TOOLTIP:
1082 case GDK_WINDOW_TYPE_HINT_NOTIFICATION:
1083 case GDK_WINDOW_TYPE_HINT_COMBO:
1084 case GDK_WINDOW_TYPE_HINT_DND:
1087 g_warning ("Unknown hint %d passed to gdk_window_set_type_hint", hint);
1089 case GDK_WINDOW_TYPE_HINT_DIALOG:
1090 case GDK_WINDOW_TYPE_HINT_NORMAL:
1091 case GDK_WINDOW_TYPE_HINT_SPLASHSCREEN:
1092 if (impl->shell_surface)
1093 wl_shell_surface_set_toplevel (impl->shell_surface);
1098 static GdkWindowTypeHint
1099 gdk_wayland_window_get_type_hint (GdkWindow *window)
1101 return GDK_WINDOW_TYPE_HINT_NORMAL;
1105 gdk_wayland_window_set_modal_hint (GdkWindow *window,
1111 gdk_wayland_window_set_skip_taskbar_hint (GdkWindow *window,
1112 gboolean skips_taskbar)
1117 gdk_wayland_window_set_skip_pager_hint (GdkWindow *window,
1118 gboolean skips_pager)
1123 gdk_wayland_window_set_urgency_hint (GdkWindow *window,
1129 gdk_wayland_window_set_geometry_hints (GdkWindow *window,
1130 const GdkGeometry *geometry,
1131 GdkWindowHints geom_mask)
1133 GdkWindowImplWayland *impl;
1135 if (GDK_WINDOW_DESTROYED (window) ||
1136 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1139 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1141 impl->geometry_hints = *geometry;
1142 impl->geometry_mask = geom_mask;
1147 * GDK_HINT_USER_SIZE
1150 * GDK_HINT_BASE_SIZE
1151 * GDK_HINT_RESIZE_INC
1153 * GDK_HINT_WIN_GRAVITY
1158 gdk_wayland_window_set_title (GdkWindow *window,
1161 g_return_if_fail (title != NULL);
1163 if (GDK_WINDOW_DESTROYED (window))
1168 gdk_wayland_window_set_role (GdkWindow *window,
1174 gdk_wayland_window_set_startup_id (GdkWindow *window,
1175 const gchar *startup_id)
1180 gdk_wayland_window_set_transient_for (GdkWindow *window,
1183 GdkWindowImplWayland *impl;
1185 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1186 impl->transient_for = parent;
1190 gdk_wayland_window_get_root_origin (GdkWindow *window,
1202 gdk_wayland_window_get_frame_extents (GdkWindow *window,
1205 rect->x = window->x;
1206 rect->y = window->y;
1207 rect->width = window->width;
1208 rect->height = window->height;
1212 gdk_wayland_window_set_override_redirect (GdkWindow *window,
1213 gboolean override_redirect)
1218 gdk_wayland_window_set_accept_focus (GdkWindow *window,
1219 gboolean accept_focus)
1224 gdk_wayland_window_set_focus_on_map (GdkWindow *window,
1225 gboolean focus_on_map)
1227 focus_on_map = focus_on_map != FALSE;
1229 if (window->focus_on_map != focus_on_map)
1231 window->focus_on_map = focus_on_map;
1233 if ((!GDK_WINDOW_DESTROYED (window)) &&
1234 (!window->focus_on_map) &&
1235 WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1236 gdk_wayland_window_set_user_time (window, 0);
1241 gdk_wayland_window_set_icon_list (GdkWindow *window,
1247 gdk_wayland_window_set_icon_name (GdkWindow *window,
1250 if (GDK_WINDOW_DESTROYED (window))
1255 gdk_wayland_window_iconify (GdkWindow *window)
1260 gdk_wayland_window_deiconify (GdkWindow *window)
1262 if (GDK_WINDOW_DESTROYED (window) ||
1263 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1266 if (GDK_WINDOW_IS_MAPPED (window))
1268 gdk_window_show (window);
1272 /* Flip our client side flag, the real work happens on map. */
1273 gdk_synthesize_window_state (window, GDK_WINDOW_STATE_ICONIFIED, 0);
1278 gdk_wayland_window_stick (GdkWindow *window)
1280 if (GDK_WINDOW_DESTROYED (window))
1285 gdk_wayland_window_unstick (GdkWindow *window)
1287 if (GDK_WINDOW_DESTROYED (window))
1292 gdk_wayland_window_maximize (GdkWindow *window)
1294 if (GDK_WINDOW_DESTROYED (window))
1299 gdk_wayland_window_unmaximize (GdkWindow *window)
1301 if (GDK_WINDOW_DESTROYED (window))
1306 gdk_wayland_window_fullscreen (GdkWindow *window)
1308 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1310 if (GDK_WINDOW_DESTROYED (window))
1313 if (impl->fullscreen)
1316 impl->saved_width = gdk_window_get_width (window);
1317 impl->saved_height = gdk_window_get_height (window);
1318 wl_shell_surface_set_fullscreen (impl->shell_surface,
1319 WL_SHELL_SURFACE_FULLSCREEN_METHOD_DEFAULT,
1322 impl->fullscreen = TRUE;
1326 gdk_wayland_window_unfullscreen (GdkWindow *window)
1328 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1330 if (GDK_WINDOW_DESTROYED (window))
1333 if (!impl->fullscreen)
1336 wl_shell_surface_set_toplevel (impl->shell_surface);
1337 gdk_wayland_window_configure (window, impl->saved_width, impl->saved_height,
1339 impl->fullscreen = FALSE;
1343 gdk_wayland_window_set_keep_above (GdkWindow *window,
1346 g_return_if_fail (GDK_IS_WINDOW (window));
1348 if (GDK_WINDOW_DESTROYED (window))
1353 gdk_wayland_window_set_keep_below (GdkWindow *window, gboolean setting)
1355 g_return_if_fail (GDK_IS_WINDOW (window));
1357 if (GDK_WINDOW_DESTROYED (window))
1362 gdk_wayland_window_get_group (GdkWindow *window)
1364 if (GDK_WINDOW_DESTROYED (window) ||
1365 !WINDOW_IS_TOPLEVEL (window))
1372 gdk_wayland_window_set_group (GdkWindow *window,
1375 g_return_if_fail (GDK_IS_WINDOW (window));
1376 g_return_if_fail (GDK_WINDOW_TYPE (window) != GDK_WINDOW_CHILD);
1377 g_return_if_fail (leader == NULL || GDK_IS_WINDOW (leader));
1381 gdk_wayland_window_set_decorations (GdkWindow *window,
1382 GdkWMDecoration decorations)
1387 gdk_wayland_window_get_decorations (GdkWindow *window,
1388 GdkWMDecoration *decorations)
1394 gdk_wayland_window_set_functions (GdkWindow *window,
1395 GdkWMFunction functions)
1400 gdk_wayland_window_begin_resize_drag (GdkWindow *window,
1408 GdkWindowImplWayland *impl;
1409 GdkWaylandDisplay *wayland_display =
1410 GDK_WAYLAND_DISPLAY (gdk_window_get_display (window));
1414 if (GDK_WINDOW_DESTROYED (window) ||
1415 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1420 case GDK_WINDOW_EDGE_NORTH_WEST:
1421 grab_type = WL_SHELL_SURFACE_RESIZE_TOP_LEFT;
1424 case GDK_WINDOW_EDGE_NORTH:
1425 grab_type = WL_SHELL_SURFACE_RESIZE_TOP;
1428 case GDK_WINDOW_EDGE_NORTH_EAST:
1429 grab_type = WL_SHELL_SURFACE_RESIZE_RIGHT;
1432 case GDK_WINDOW_EDGE_WEST:
1433 grab_type = WL_SHELL_SURFACE_RESIZE_LEFT;
1436 case GDK_WINDOW_EDGE_EAST:
1437 grab_type = WL_SHELL_SURFACE_RESIZE_RIGHT;
1440 case GDK_WINDOW_EDGE_SOUTH_WEST:
1441 grab_type = WL_SHELL_SURFACE_RESIZE_BOTTOM_LEFT;
1444 case GDK_WINDOW_EDGE_SOUTH:
1445 grab_type = WL_SHELL_SURFACE_RESIZE_BOTTOM;
1448 case GDK_WINDOW_EDGE_SOUTH_EAST:
1449 grab_type = WL_SHELL_SURFACE_RESIZE_BOTTOM_RIGHT;
1453 g_warning ("gdk_window_begin_resize_drag: bad resize edge %d!",
1458 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1460 wl_shell_surface_resize (impl->shell_surface,
1461 gdk_wayland_device_get_wl_seat (device),
1462 _gdk_wayland_display_get_serial (wayland_display),
1465 /* This is needed since Wayland will absorb all the pointer events after the
1466 * above function - FIXME: Is this always safe..?
1468 gdk_device_ungrab (device, timestamp);
1472 gdk_wayland_window_begin_move_drag (GdkWindow *window,
1479 GdkWindowImplWayland *impl;
1480 GdkWaylandDisplay *wayland_display =
1481 GDK_WAYLAND_DISPLAY (gdk_window_get_display (window));
1483 if (GDK_WINDOW_DESTROYED (window) ||
1484 !WINDOW_IS_TOPLEVEL (window))
1487 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1489 wl_shell_surface_move (impl->shell_surface,
1490 gdk_wayland_device_get_wl_seat (device),
1491 _gdk_wayland_display_get_serial (wayland_display));
1493 /* This is needed since Wayland will absorb all the pointer events after the
1494 * above function - FIXME: Is this always safe..?
1496 gdk_device_ungrab (device, timestamp);
1500 gdk_wayland_window_set_opacity (GdkWindow *window,
1506 gdk_wayland_window_set_composited (GdkWindow *window,
1507 gboolean composited)
1512 gdk_wayland_window_destroy_notify (GdkWindow *window)
1514 if (!GDK_WINDOW_DESTROYED (window))
1516 if (GDK_WINDOW_TYPE(window) != GDK_WINDOW_FOREIGN)
1517 g_warning ("GdkWindow %p unexpectedly destroyed", window);
1519 _gdk_window_destroy (window, TRUE);
1522 g_object_unref (window);
1526 gdk_wayland_window_process_updates_recurse (GdkWindow *window,
1527 cairo_region_t *region)
1529 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1530 cairo_rectangle_int_t rect;
1533 gdk_wayland_window_map (window);
1535 if (impl->cairo_surface)
1536 gdk_wayland_window_attach_image (window);
1538 n = cairo_region_num_rectangles(region);
1539 for (i = 0; i < n; i++)
1541 cairo_region_get_rectangle (region, i, &rect);
1542 wl_surface_damage (impl->surface,
1543 rect.x, rect.y, rect.width, rect.height);
1544 wl_surface_commit(impl->surface);
1547 _gdk_window_process_updates_recurse (window, region);
1551 gdk_wayland_window_sync_rendering (GdkWindow *window)
1556 gdk_wayland_window_simulate_key (GdkWindow *window,
1560 GdkModifierType modifiers,
1561 GdkEventType key_pressrelease)
1567 gdk_wayland_window_simulate_button (GdkWindow *window,
1570 guint button, /*1..3*/
1571 GdkModifierType modifiers,
1572 GdkEventType button_pressrelease)
1578 gdk_wayland_window_get_property (GdkWindow *window,
1584 GdkAtom *actual_property_type,
1585 gint *actual_format_type,
1586 gint *actual_length,
1593 gdk_wayland_window_change_property (GdkWindow *window,
1604 gdk_wayland_window_delete_property (GdkWindow *window,
1610 _gdk_window_impl_wayland_class_init (GdkWindowImplWaylandClass *klass)
1612 GObjectClass *object_class = G_OBJECT_CLASS (klass);
1613 GdkWindowImplClass *impl_class = GDK_WINDOW_IMPL_CLASS (klass);
1615 object_class->finalize = gdk_window_impl_wayland_finalize;
1617 impl_class->ref_cairo_surface = gdk_wayland_window_ref_cairo_surface;
1618 impl_class->show = gdk_wayland_window_show;
1619 impl_class->hide = gdk_wayland_window_hide;
1620 impl_class->withdraw = gdk_window_wayland_withdraw;
1621 impl_class->set_events = gdk_window_wayland_set_events;
1622 impl_class->get_events = gdk_window_wayland_get_events;
1623 impl_class->raise = gdk_window_wayland_raise;
1624 impl_class->lower = gdk_window_wayland_lower;
1625 impl_class->restack_under = gdk_window_wayland_restack_under;
1626 impl_class->restack_toplevel = gdk_window_wayland_restack_toplevel;
1627 impl_class->move_resize = gdk_window_wayland_move_resize;
1628 impl_class->set_background = gdk_window_wayland_set_background;
1629 impl_class->reparent = gdk_window_wayland_reparent;
1630 impl_class->set_device_cursor = gdk_window_wayland_set_device_cursor;
1631 impl_class->get_geometry = gdk_window_wayland_get_geometry;
1632 impl_class->get_root_coords = gdk_window_wayland_get_root_coords;
1633 impl_class->get_device_state = gdk_window_wayland_get_device_state;
1634 impl_class->shape_combine_region = gdk_window_wayland_shape_combine_region;
1635 impl_class->input_shape_combine_region = gdk_window_wayland_input_shape_combine_region;
1636 impl_class->set_static_gravities = gdk_window_wayland_set_static_gravities;
1637 impl_class->queue_antiexpose = gdk_wayland_window_queue_antiexpose;
1638 impl_class->translate = gdk_wayland_window_translate;
1639 impl_class->destroy = gdk_wayland_window_destroy;
1640 impl_class->destroy_foreign = gdk_window_wayland_destroy_foreign;
1641 impl_class->resize_cairo_surface = gdk_window_wayland_resize_cairo_surface;
1642 impl_class->get_shape = gdk_wayland_window_get_shape;
1643 impl_class->get_input_shape = gdk_wayland_window_get_input_shape;
1644 /* impl_class->beep */
1646 impl_class->focus = gdk_wayland_window_focus;
1647 impl_class->set_type_hint = gdk_wayland_window_set_type_hint;
1648 impl_class->get_type_hint = gdk_wayland_window_get_type_hint;
1649 impl_class->set_modal_hint = gdk_wayland_window_set_modal_hint;
1650 impl_class->set_skip_taskbar_hint = gdk_wayland_window_set_skip_taskbar_hint;
1651 impl_class->set_skip_pager_hint = gdk_wayland_window_set_skip_pager_hint;
1652 impl_class->set_urgency_hint = gdk_wayland_window_set_urgency_hint;
1653 impl_class->set_geometry_hints = gdk_wayland_window_set_geometry_hints;
1654 impl_class->set_title = gdk_wayland_window_set_title;
1655 impl_class->set_role = gdk_wayland_window_set_role;
1656 impl_class->set_startup_id = gdk_wayland_window_set_startup_id;
1657 impl_class->set_transient_for = gdk_wayland_window_set_transient_for;
1658 impl_class->get_root_origin = gdk_wayland_window_get_root_origin;
1659 impl_class->get_frame_extents = gdk_wayland_window_get_frame_extents;
1660 impl_class->set_override_redirect = gdk_wayland_window_set_override_redirect;
1661 impl_class->set_accept_focus = gdk_wayland_window_set_accept_focus;
1662 impl_class->set_focus_on_map = gdk_wayland_window_set_focus_on_map;
1663 impl_class->set_icon_list = gdk_wayland_window_set_icon_list;
1664 impl_class->set_icon_name = gdk_wayland_window_set_icon_name;
1665 impl_class->iconify = gdk_wayland_window_iconify;
1666 impl_class->deiconify = gdk_wayland_window_deiconify;
1667 impl_class->stick = gdk_wayland_window_stick;
1668 impl_class->unstick = gdk_wayland_window_unstick;
1669 impl_class->maximize = gdk_wayland_window_maximize;
1670 impl_class->unmaximize = gdk_wayland_window_unmaximize;
1671 impl_class->fullscreen = gdk_wayland_window_fullscreen;
1672 impl_class->unfullscreen = gdk_wayland_window_unfullscreen;
1673 impl_class->set_keep_above = gdk_wayland_window_set_keep_above;
1674 impl_class->set_keep_below = gdk_wayland_window_set_keep_below;
1675 impl_class->get_group = gdk_wayland_window_get_group;
1676 impl_class->set_group = gdk_wayland_window_set_group;
1677 impl_class->set_decorations = gdk_wayland_window_set_decorations;
1678 impl_class->get_decorations = gdk_wayland_window_get_decorations;
1679 impl_class->set_functions = gdk_wayland_window_set_functions;
1680 impl_class->begin_resize_drag = gdk_wayland_window_begin_resize_drag;
1681 impl_class->begin_move_drag = gdk_wayland_window_begin_move_drag;
1682 impl_class->set_opacity = gdk_wayland_window_set_opacity;
1683 impl_class->set_composited = gdk_wayland_window_set_composited;
1684 impl_class->destroy_notify = gdk_wayland_window_destroy_notify;
1685 impl_class->get_drag_protocol = _gdk_wayland_window_get_drag_protocol;
1686 impl_class->register_dnd = _gdk_wayland_window_register_dnd;
1687 impl_class->drag_begin = _gdk_wayland_window_drag_begin;
1688 impl_class->process_updates_recurse = gdk_wayland_window_process_updates_recurse;
1689 impl_class->sync_rendering = gdk_wayland_window_sync_rendering;
1690 impl_class->simulate_key = gdk_wayland_window_simulate_key;
1691 impl_class->simulate_button = gdk_wayland_window_simulate_button;
1692 impl_class->get_property = gdk_wayland_window_get_property;
1693 impl_class->change_property = gdk_wayland_window_change_property;
1694 impl_class->delete_property = gdk_wayland_window_delete_property;
1699 _gdk_wayland_window_set_device_grabbed (GdkWindow *window,
1701 struct wl_seat *seat,
1704 GdkWindowImplWayland *impl;
1706 g_return_if_fail (window != NULL);
1708 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1710 impl->grab_device = device;
1711 impl->grab_input_seat = seat;
1712 impl->grab_time = time_;
1716 * gdk_wayland_window_get_wl_surface
1717 * @window: (type GdkWaylandWindow): a #GdkWindow
1719 * Returns the Wayland surface of a #GdkWindow
1721 * Returns: (transfer none): a Wayland wl_surface
1726 gdk_wayland_window_get_wl_surface (GdkWindow *window)
1728 GdkWindowImplWayland *impl;
1730 g_return_val_if_fail (GDK_IS_WAYLAND_WINDOW (window), NULL);
1732 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1734 return impl->surface;
1738 * gdk_wayland_window_get_wl_shell_surface
1739 * @window: (type GdkWaylandWindow): a #GdkWindow
1741 * Returns the Wayland shell surface of a #GdkWindow
1743 * Returns: (transfer none): a Wayland wl_shell_surface
1747 struct wl_shell_surface *
1748 gdk_wayland_window_get_wl_shell_surface (GdkWindow *window)
1750 GdkWindowImplWayland *impl;
1752 g_return_val_if_fail (GDK_IS_WAYLAND_WINDOW (window), NULL);
1754 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1756 return impl->shell_surface;