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"
38 #include <wayland-egl.h>
40 #define WINDOW_IS_TOPLEVEL_OR_FOREIGN(window) \
41 (GDK_WINDOW_TYPE (window) != GDK_WINDOW_CHILD && \
42 GDK_WINDOW_TYPE (window) != GDK_WINDOW_OFFSCREEN)
44 #define WINDOW_IS_TOPLEVEL(window) \
45 (GDK_WINDOW_TYPE (window) != GDK_WINDOW_CHILD && \
46 GDK_WINDOW_TYPE (window) != GDK_WINDOW_FOREIGN && \
47 GDK_WINDOW_TYPE (window) != GDK_WINDOW_OFFSCREEN)
49 /* Return whether time1 is considered later than time2 as far as xserver
50 * time is concerned. Accounts for wraparound.
52 #define XSERVER_TIME_IS_LATER(time1, time2) \
53 ( (( time1 > time2 ) && ( time1 - time2 < ((guint32)-1)/2 )) || \
54 (( time1 < time2 ) && ( time2 - time1 > ((guint32)-1)/2 )) \
57 typedef struct _GdkWaylandWindow GdkWaylandWindow;
58 typedef struct _GdkWaylandWindowClass GdkWaylandWindowClass;
60 struct _GdkWaylandWindow {
64 struct _GdkWaylandWindowClass {
65 GdkWindowClass parent_class;
68 G_DEFINE_TYPE (GdkWaylandWindow, _gdk_wayland_window, GDK_TYPE_WINDOW)
71 _gdk_wayland_window_class_init (GdkWaylandWindowClass *wayland_window_class)
76 _gdk_wayland_window_init (GdkWaylandWindow *wayland_window)
80 #define GDK_TYPE_WINDOW_IMPL_WAYLAND (_gdk_window_impl_wayland_get_type ())
81 #define GDK_WINDOW_IMPL_WAYLAND(object) (G_TYPE_CHECK_INSTANCE_CAST ((object), GDK_TYPE_WINDOW_IMPL_WAYLAND, GdkWindowImplWayland))
82 #define GDK_WINDOW_IMPL_WAYLAND_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GDK_TYPE_WINDOW_IMPL_WAYLAND, GdkWindowImplWaylandClass))
83 #define GDK_IS_WINDOW_IMPL_WAYLAND(object) (G_TYPE_CHECK_INSTANCE_TYPE ((object), GDK_TYPE_WINDOW_IMPL_WAYLAND))
84 #define GDK_IS_WINDOW_IMPL_WAYLAND_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GDK_TYPE_WINDOW_IMPL_WAYLAND))
85 #define GDK_WINDOW_IMPL_WAYLAND_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GDK_TYPE_WINDOW_IMPL_WAYLAND, GdkWindowImplWaylandClass))
87 typedef struct _GdkWindowImplWayland GdkWindowImplWayland;
88 typedef struct _GdkWindowImplWaylandClass GdkWindowImplWaylandClass;
90 struct _GdkWindowImplWayland
92 GdkWindowImpl parent_instance;
98 gint8 toplevel_window_type;
100 struct wl_surface *surface;
101 struct wl_shell_surface *shell_surface;
102 unsigned int mapped : 1;
103 GdkWindow *transient_for;
104 GdkWindowTypeHint hint;
106 /* The surface which is being "drawn to" to */
107 cairo_surface_t *cairo_surface;
109 /* The surface that was the last surface the Wayland buffer from which was attached
110 * to the Wayland surface. It will be the same as cairo_surface after a call
111 * to gdk_wayland_window_attach_image. But after a call to
112 * gdk_wayland_window_update_size and then
113 * gdk_wayland_window_ref_cairo_surface the above pointer will be different.
115 cairo_surface_t *server_surface;
117 uint32_t resize_edges;
121 gulong map_serial; /* Serial of last transition from unmapped */
123 cairo_surface_t *icon_pixmap;
124 cairo_surface_t *icon_mask;
126 /* Time of most recent user interaction. */
129 GdkGeometry geometry_hints;
130 GdkWindowHints geometry_mask;
132 struct wl_input_device *grab_input_device;
136 struct _GdkWindowImplWaylandClass
138 GdkWindowImplClass parent_class;
141 G_DEFINE_TYPE (GdkWindowImplWayland, _gdk_window_impl_wayland, GDK_TYPE_WINDOW_IMPL)
144 _gdk_window_impl_wayland_init (GdkWindowImplWayland *impl)
146 impl->toplevel_window_type = -1;
150 _gdk_wayland_window_add_focus (GdkWindow *window)
152 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
155 if (impl->focus_count == 1)
156 gdk_synthesize_window_state (window, 0, GDK_WINDOW_STATE_FOCUSED);
160 _gdk_wayland_window_remove_focus (GdkWindow *window)
162 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
165 if (impl->focus_count == 0)
166 gdk_synthesize_window_state (window, GDK_WINDOW_STATE_FOCUSED, 0);
170 * gdk_wayland_window_update_size:
171 * @drawable: a #GdkDrawableImplWayland.
173 * Updates the state of the drawable (in particular the drawable's
174 * cairo surface) when its size has changed.
177 gdk_wayland_window_update_size (GdkWindow *window,
178 int32_t width, int32_t height, uint32_t edges)
180 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
182 cairo_region_t *region;
184 if (impl->cairo_surface)
186 cairo_surface_destroy (impl->cairo_surface);
187 impl->cairo_surface = NULL;
190 window->width = width;
191 window->height = height;
192 impl->resize_edges = edges;
196 area.width = window->width;
197 area.height = window->height;
199 region = cairo_region_create_rectangle (&area);
200 _gdk_window_invalidate_for_expose (window, region);
201 cairo_region_destroy (region);
205 _gdk_wayland_screen_create_root_window (GdkScreen *screen,
206 int width, int height)
209 GdkWindowImplWayland *impl;
211 window = _gdk_display_create_window (gdk_screen_get_display (screen));
212 window->impl = g_object_new (GDK_TYPE_WINDOW_IMPL_WAYLAND, NULL);
213 window->impl_window = window;
214 window->visual = gdk_screen_get_system_visual (screen);
216 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
218 impl->wrapper = GDK_WINDOW (window);
220 window->window_type = GDK_WINDOW_ROOT;
227 window->width = width;
228 window->height = height;
229 window->viewable = TRUE;
231 /* see init_randr_support() in gdkscreen-wayland.c */
232 window->event_mask = GDK_STRUCTURE_MASK;
238 get_default_title (void)
242 title = g_get_application_name ();
244 title = g_get_prgname ();
252 _gdk_wayland_display_create_window_impl (GdkDisplay *display,
254 GdkWindow *real_parent,
256 GdkEventMask event_mask,
257 GdkWindowAttr *attributes,
258 gint attributes_mask)
260 GdkWindowImplWayland *impl;
263 impl = g_object_new (GDK_TYPE_WINDOW_IMPL_WAYLAND, NULL);
264 window->impl = GDK_WINDOW_IMPL (impl);
265 impl->wrapper = GDK_WINDOW (window);
267 if (window->width > 65535 ||
268 window->height > 65535)
270 g_warning ("Native Windows wider or taller than 65535 pixels are not supported");
272 if (window->width > 65535)
273 window->width = 65535;
274 if (window->height > 65535)
275 window->height = 65535;
278 g_object_ref (window);
280 switch (GDK_WINDOW_TYPE (window))
282 case GDK_WINDOW_TOPLEVEL:
283 case GDK_WINDOW_TEMP:
284 if (attributes_mask & GDK_WA_TITLE)
285 title = attributes->title;
287 title = get_default_title ();
289 gdk_window_set_title (window, title);
292 case GDK_WINDOW_CHILD:
297 if (attributes_mask & GDK_WA_TYPE_HINT)
298 gdk_window_set_type_hint (window, attributes->type_hint);
301 static const cairo_user_data_key_t gdk_wayland_cairo_key;
303 typedef struct _GdkWaylandCairoSurfaceData {
304 #ifdef GDK_WAYLAND_USE_EGL
307 struct wl_egl_pixmap *pixmap;
312 struct wl_buffer *buffer;
313 GdkDisplayWayland *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);
375 #ifdef GDK_WAYLAND_USE_EGL
377 gdk_wayland_cairo_surface_destroy (void *p)
379 GdkWaylandCairoSurfaceData *data = p;
381 data->display->destroy_image (data->display->egl_display, data->image);
382 cairo_device_acquire(data->display->cairo_device);
383 glDeleteTextures(1, &data->texture);
384 cairo_device_release(data->display->cairo_device);
386 wl_buffer_destroy(data->buffer);
390 static cairo_surface_t *
391 gdk_wayland_create_cairo_surface (GdkDisplayWayland *display,
392 int width, int height)
394 GdkWaylandCairoSurfaceData *data;
395 cairo_surface_t *surface;
397 data = g_new (GdkWaylandCairoSurfaceData, 1);
398 data->display = display;
401 data->height = height;
402 data->pixmap = wl_egl_pixmap_create(width, height, 0);
404 display->create_image(display->egl_display, NULL, EGL_NATIVE_PIXMAP_KHR,
405 (EGLClientBuffer) data->pixmap, NULL);
407 cairo_device_acquire(display->cairo_device);
408 glGenTextures(1, &data->texture);
409 glBindTexture(GL_TEXTURE_2D, data->texture);
410 display->image_target_texture_2d(GL_TEXTURE_2D, data->image);
411 cairo_device_release(display->cairo_device);
413 surface = cairo_gl_surface_create_for_texture(display->cairo_device,
414 CAIRO_CONTENT_COLOR_ALPHA,
415 data->texture, width, height);
417 cairo_surface_set_user_data (surface, &gdk_wayland_cairo_key,
418 data, gdk_wayland_cairo_surface_destroy);
420 if (cairo_surface_status (surface) != CAIRO_STATUS_SUCCESS)
421 fprintf (stderr, "create gl surface failed\n");
425 wl_egl_pixmap_create_buffer(data->pixmap);
430 static struct wl_buffer *
431 create_shm_buffer (struct wl_shm *shm,
438 char filename[] = "/tmp/wayland-shm-XXXXXX";
439 struct wl_buffer *buffer;
440 int fd, size, stride;
443 fd = mkstemp(filename);
445 fprintf(stderr, "open %s failed: %m\n", filename);
449 size = stride * height;
450 if (ftruncate(fd, size) < 0) {
451 fprintf(stderr, "ftruncate failed: %m\n");
456 data = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
459 if (data == MAP_FAILED) {
460 fprintf(stderr, "mmap failed: %m\n");
465 buffer = wl_shm_create_buffer(shm, fd,
477 gdk_wayland_cairo_surface_destroy (void *p)
479 GdkWaylandCairoSurfaceData *data = p;
482 wl_buffer_destroy(data->buffer);
484 munmap(data->buf, data->buf_length);
488 static cairo_surface_t *
489 gdk_wayland_create_cairo_surface (GdkDisplayWayland *display,
490 int width, int height)
492 GdkWaylandCairoSurfaceData *data;
493 cairo_surface_t *surface;
495 data = g_new (GdkWaylandCairoSurfaceData, 1);
496 data->display = display;
499 data->height = height;
501 data->buffer = create_shm_buffer (display->shm,
504 WL_SHM_FORMAT_XRGB8888,
508 surface = cairo_image_surface_create_for_data (data->buf,
514 cairo_surface_set_user_data (surface, &gdk_wayland_cairo_key,
515 data, gdk_wayland_cairo_surface_destroy);
517 if (cairo_surface_status (surface) != CAIRO_STATUS_SUCCESS)
518 fprintf (stderr, "create image surface failed\n");
524 /* On this first call this creates a double reference - the first reference
525 * is held by the GdkWindowImplWayland struct - since unlike other backends
526 * the Cairo surface is not just a cheap wrapper around some other backing.
527 * It is the buffer itself.
529 static cairo_surface_t *
530 gdk_wayland_window_ref_cairo_surface (GdkWindow *window)
532 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
533 GdkDisplayWayland *display_wayland =
534 GDK_DISPLAY_WAYLAND (gdk_window_get_display (impl->wrapper));
536 if (GDK_WINDOW_DESTROYED (impl->wrapper))
539 if (!impl->cairo_surface)
541 impl->cairo_surface =
542 gdk_wayland_create_cairo_surface (display_wayland,
543 impl->wrapper->width,
544 impl->wrapper->height);
547 cairo_surface_reference (impl->cairo_surface);
549 return impl->cairo_surface;
554 gdk_window_impl_wayland_finalize (GObject *object)
556 GdkWindowImplWayland *impl;
558 g_return_if_fail (GDK_IS_WINDOW_IMPL_WAYLAND (object));
560 impl = GDK_WINDOW_IMPL_WAYLAND (object);
563 g_object_unref (impl->cursor);
564 if (impl->server_surface)
565 cairo_surface_destroy (impl->server_surface);
567 G_OBJECT_CLASS (_gdk_window_impl_wayland_parent_class)->finalize (object);
571 gdk_wayland_window_configure (GdkWindow *window,
572 int width, int height, int edges)
574 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
578 display = gdk_window_get_display (window);
580 /* TODO: Only generate a configure event if width or height have actually
583 event = gdk_event_new (GDK_CONFIGURE);
584 event->configure.window = window;
585 event->configure.send_event = FALSE;
586 event->configure.width = width;
587 event->configure.height = height;
589 _gdk_window_update_size (window);
590 gdk_wayland_window_update_size (window, width, height, edges);
592 g_object_ref(window);
594 _gdk_wayland_display_deliver_event (display, event);
598 gdk_wayland_window_set_user_time (GdkWindow *window, guint32 user_time)
603 gdk_wayland_window_map (GdkWindow *window)
605 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
606 GdkWindowImplWayland *parent;
610 if (impl->transient_for)
612 parent = GDK_WINDOW_IMPL_WAYLAND (impl->transient_for->impl);
614 if (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)
618 /* Use the device that was used for the grab as the device for
619 * the popup window setup - so this relies on GTK+ taking the
620 * grab before showing the popup window.
622 wl_shell_surface_set_popup (impl->shell_surface,
623 parent->grab_input_device, parent->grab_time,
624 parent->shell_surface,
625 window->x, window->y, 0);
627 wl_shell_surface_set_transient (impl->shell_surface, parent->shell_surface,
628 window->x, window->y, 0);
633 wl_shell_surface_set_toplevel (impl->shell_surface);
640 shell_surface_handle_configure(void *data,
641 struct wl_shell_surface *shell_surface,
647 GdkWindow *window = GDK_WINDOW (data);
648 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
650 gdk_window_constrain_size (&impl->geometry_hints,
657 gdk_wayland_window_configure (window, width, height, edges);
661 shell_surface_popup_done (void *data,
662 struct wl_shell_surface *shell_surface)
664 GdkWindow *window = GDK_WINDOW (data);
666 /* When the popup is complete hide the window - this really relies on the
667 * fix in https://bugzilla.gnome.org/show_bug.cgi?id=670881 to work
670 gdk_window_hide (window);
673 static const struct wl_shell_surface_listener shell_surface_listener = {
674 shell_surface_handle_configure,
675 shell_surface_popup_done
679 gdk_wayland_window_show (GdkWindow *window, gboolean already_mapped)
682 GdkDisplayWayland *display_wayland;
683 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
686 display = gdk_window_get_display (window);
687 display_wayland = GDK_DISPLAY_WAYLAND (display);
689 if (impl->user_time != 0 &&
690 display_wayland->user_time != 0 &&
691 XSERVER_TIME_IS_LATER (display_wayland->user_time, impl->user_time))
692 gdk_wayland_window_set_user_time (window, impl->user_time);
694 impl->surface = wl_compositor_create_surface(display_wayland->compositor);
695 wl_surface_set_user_data(impl->surface, window);
697 impl->shell_surface = wl_shell_get_shell_surface (display_wayland->shell,
699 wl_shell_surface_add_listener(impl->shell_surface,
700 &shell_surface_listener, window);
702 gdk_window_set_type_hint (window, impl->hint);
704 _gdk_make_event (window, GDK_MAP, NULL, FALSE);
705 event = _gdk_make_event (window, GDK_VISIBILITY_NOTIFY, NULL, FALSE);
706 event->visibility.state = GDK_VISIBILITY_UNOBSCURED;
708 if (impl->cairo_surface)
709 gdk_wayland_window_attach_image (window);
713 gdk_wayland_window_hide (GdkWindow *window)
715 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
719 wl_surface_destroy(impl->surface);
720 impl->surface = NULL;
721 cairo_surface_destroy(impl->server_surface);
722 impl->server_surface = NULL;
723 impl->mapped = FALSE;
726 _gdk_window_clear_update_area (window);
730 gdk_window_wayland_withdraw (GdkWindow *window)
732 GdkWindowImplWayland *impl;
734 if (!window->destroyed)
736 if (GDK_WINDOW_IS_MAPPED (window))
737 gdk_synthesize_window_state (window, 0, GDK_WINDOW_STATE_WITHDRAWN);
739 g_assert (!GDK_WINDOW_IS_MAPPED (window));
741 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
744 wl_surface_destroy(impl->surface);
745 impl->surface = NULL;
746 cairo_surface_destroy(impl->server_surface);
747 impl->server_surface = NULL;
748 impl->mapped = FALSE;
754 gdk_window_wayland_set_events (GdkWindow *window,
755 GdkEventMask event_mask)
757 GDK_WINDOW (window)->event_mask = event_mask;
761 gdk_window_wayland_get_events (GdkWindow *window)
763 if (GDK_WINDOW_DESTROYED (window))
766 return GDK_WINDOW (window)->event_mask;
770 gdk_window_wayland_raise (GdkWindow *window)
772 /* FIXME: wl_shell_raise() */
776 gdk_window_wayland_lower (GdkWindow *window)
778 /* FIXME: wl_shell_lower() */
782 gdk_window_wayland_restack_under (GdkWindow *window,
783 GList *native_siblings)
788 gdk_window_wayland_restack_toplevel (GdkWindow *window,
795 gdk_window_wayland_move_resize (GdkWindow *window,
805 /* If this function is called with width and height = -1 then that means
806 * just move the window - don't update its size
808 if (width > 0 && height > 0)
809 gdk_wayland_window_configure (window, width, height, 0);
813 gdk_window_wayland_set_background (GdkWindow *window,
814 cairo_pattern_t *pattern)
819 gdk_window_wayland_reparent (GdkWindow *window,
820 GdkWindow *new_parent,
828 gdk_window_wayland_set_device_cursor (GdkWindow *window,
832 g_return_if_fail (GDK_IS_WINDOW (window));
833 g_return_if_fail (GDK_IS_DEVICE (device));
835 if (!GDK_WINDOW_DESTROYED (window))
836 GDK_DEVICE_GET_CLASS (device)->set_window_cursor (device, window, cursor);
840 gdk_window_wayland_get_geometry (GdkWindow *window,
846 if (!GDK_WINDOW_DESTROYED (window))
853 *width = window->width;
855 *height = window->height;
860 gdk_window_wayland_get_root_coords (GdkWindow *window,
866 /* We can't do this. */
876 gdk_window_wayland_get_device_state (GdkWindow *window,
880 GdkModifierType *mask)
884 g_return_val_if_fail (window == NULL || GDK_IS_WINDOW (window), FALSE);
888 if (!GDK_WINDOW_DESTROYED (window))
892 GDK_DEVICE_GET_CLASS (device)->query_state (device, window,
896 return_val = (child != NULL);
903 gdk_window_wayland_shape_combine_region (GdkWindow *window,
904 const cairo_region_t *shape_region,
911 gdk_window_wayland_input_shape_combine_region (GdkWindow *window,
912 const cairo_region_t *shape_region,
919 gdk_window_wayland_set_static_gravities (GdkWindow *window,
926 gdk_wayland_window_queue_antiexpose (GdkWindow *window,
927 cairo_region_t *area)
933 gdk_wayland_window_translate (GdkWindow *window,
934 cairo_region_t *area,
938 _gdk_window_invalidate_for_expose (window, area);
942 gdk_wayland_window_destroy (GdkWindow *window,
944 gboolean foreign_destroy)
946 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
948 g_return_if_fail (GDK_IS_WINDOW (window));
950 if (impl->cairo_surface)
952 cairo_surface_finish (impl->cairo_surface);
953 cairo_surface_set_user_data (impl->cairo_surface, &gdk_wayland_cairo_key,
957 if (!recursing && !foreign_destroy)
959 if (GDK_WINDOW_IMPL_WAYLAND (window->impl)->surface)
960 wl_surface_destroy(GDK_WINDOW_IMPL_WAYLAND (window->impl)->surface);
961 wl_shell_surface_destroy(GDK_WINDOW_IMPL_WAYLAND (window->impl)->shell_surface);
966 gdk_window_wayland_destroy_foreign (GdkWindow *window)
970 static cairo_surface_t *
971 gdk_window_wayland_resize_cairo_surface (GdkWindow *window,
972 cairo_surface_t *surface,
979 static cairo_region_t *
980 gdk_wayland_window_get_shape (GdkWindow *window)
985 static cairo_region_t *
986 gdk_wayland_window_get_input_shape (GdkWindow *window)
992 gdk_wayland_window_focus (GdkWindow *window,
995 /* FIXME: wl_shell_focus() */
999 gdk_wayland_window_set_type_hint (GdkWindow *window,
1000 GdkWindowTypeHint hint)
1002 GdkWindowImplWayland *impl;
1004 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1006 if (GDK_WINDOW_DESTROYED (window))
1013 case GDK_WINDOW_TYPE_HINT_MENU:
1014 case GDK_WINDOW_TYPE_HINT_TOOLBAR:
1015 case GDK_WINDOW_TYPE_HINT_UTILITY:
1016 case GDK_WINDOW_TYPE_HINT_DOCK:
1017 case GDK_WINDOW_TYPE_HINT_DESKTOP:
1018 case GDK_WINDOW_TYPE_HINT_DROPDOWN_MENU:
1019 case GDK_WINDOW_TYPE_HINT_POPUP_MENU:
1020 case GDK_WINDOW_TYPE_HINT_TOOLTIP:
1021 case GDK_WINDOW_TYPE_HINT_NOTIFICATION:
1022 case GDK_WINDOW_TYPE_HINT_COMBO:
1023 case GDK_WINDOW_TYPE_HINT_DND:
1026 g_warning ("Unknown hint %d passed to gdk_window_set_type_hint", hint);
1028 case GDK_WINDOW_TYPE_HINT_DIALOG:
1029 case GDK_WINDOW_TYPE_HINT_NORMAL:
1030 case GDK_WINDOW_TYPE_HINT_SPLASHSCREEN:
1031 if (impl->shell_surface)
1032 wl_shell_surface_set_toplevel (impl->shell_surface);
1037 static GdkWindowTypeHint
1038 gdk_wayland_window_get_type_hint (GdkWindow *window)
1040 return GDK_WINDOW_TYPE_HINT_NORMAL;
1044 gdk_wayland_window_set_modal_hint (GdkWindow *window,
1050 gdk_wayland_window_set_skip_taskbar_hint (GdkWindow *window,
1051 gboolean skips_taskbar)
1056 gdk_wayland_window_set_skip_pager_hint (GdkWindow *window,
1057 gboolean skips_pager)
1062 gdk_wayland_window_set_urgency_hint (GdkWindow *window,
1068 gdk_wayland_window_set_geometry_hints (GdkWindow *window,
1069 const GdkGeometry *geometry,
1070 GdkWindowHints geom_mask)
1072 GdkWindowImplWayland *impl;
1074 if (GDK_WINDOW_DESTROYED (window) ||
1075 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1078 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1080 impl->geometry_hints = *geometry;
1081 impl->geometry_mask = geom_mask;
1086 * GDK_HINT_USER_SIZE
1089 * GDK_HINT_BASE_SIZE
1090 * GDK_HINT_RESIZE_INC
1092 * GDK_HINT_WIN_GRAVITY
1097 gdk_wayland_window_set_title (GdkWindow *window,
1100 g_return_if_fail (title != NULL);
1102 if (GDK_WINDOW_DESTROYED (window))
1107 gdk_wayland_window_set_role (GdkWindow *window,
1113 gdk_wayland_window_set_startup_id (GdkWindow *window,
1114 const gchar *startup_id)
1119 gdk_wayland_window_set_transient_for (GdkWindow *window,
1122 GdkWindowImplWayland *impl;
1124 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1125 impl->transient_for = parent;
1129 gdk_wayland_window_get_root_origin (GdkWindow *window,
1141 gdk_wayland_window_get_frame_extents (GdkWindow *window,
1144 rect->x = window->x;
1145 rect->y = window->y;
1146 rect->width = window->width;
1147 rect->height = window->height;
1151 gdk_wayland_window_set_override_redirect (GdkWindow *window,
1152 gboolean override_redirect)
1157 gdk_wayland_window_set_accept_focus (GdkWindow *window,
1158 gboolean accept_focus)
1163 gdk_wayland_window_set_focus_on_map (GdkWindow *window,
1164 gboolean focus_on_map)
1166 focus_on_map = focus_on_map != FALSE;
1168 if (window->focus_on_map != focus_on_map)
1170 window->focus_on_map = focus_on_map;
1172 if ((!GDK_WINDOW_DESTROYED (window)) &&
1173 (!window->focus_on_map) &&
1174 WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1175 gdk_wayland_window_set_user_time (window, 0);
1180 gdk_wayland_window_set_icon_list (GdkWindow *window,
1186 gdk_wayland_window_set_icon_name (GdkWindow *window,
1189 if (GDK_WINDOW_DESTROYED (window))
1194 gdk_wayland_window_iconify (GdkWindow *window)
1199 gdk_wayland_window_deiconify (GdkWindow *window)
1201 if (GDK_WINDOW_DESTROYED (window) ||
1202 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1205 if (GDK_WINDOW_IS_MAPPED (window))
1207 gdk_window_show (window);
1211 /* Flip our client side flag, the real work happens on map. */
1212 gdk_synthesize_window_state (window, GDK_WINDOW_STATE_ICONIFIED, 0);
1217 gdk_wayland_window_stick (GdkWindow *window)
1219 if (GDK_WINDOW_DESTROYED (window))
1224 gdk_wayland_window_unstick (GdkWindow *window)
1226 if (GDK_WINDOW_DESTROYED (window))
1231 gdk_wayland_window_maximize (GdkWindow *window)
1233 if (GDK_WINDOW_DESTROYED (window))
1238 gdk_wayland_window_unmaximize (GdkWindow *window)
1240 if (GDK_WINDOW_DESTROYED (window))
1245 gdk_wayland_window_fullscreen (GdkWindow *window)
1247 if (GDK_WINDOW_DESTROYED (window))
1252 gdk_wayland_window_unfullscreen (GdkWindow *window)
1254 if (GDK_WINDOW_DESTROYED (window))
1259 gdk_wayland_window_set_keep_above (GdkWindow *window,
1262 g_return_if_fail (GDK_IS_WINDOW (window));
1264 if (GDK_WINDOW_DESTROYED (window))
1269 gdk_wayland_window_set_keep_below (GdkWindow *window, gboolean setting)
1271 g_return_if_fail (GDK_IS_WINDOW (window));
1273 if (GDK_WINDOW_DESTROYED (window))
1278 gdk_wayland_window_get_group (GdkWindow *window)
1280 if (GDK_WINDOW_DESTROYED (window) ||
1281 !WINDOW_IS_TOPLEVEL (window))
1288 gdk_wayland_window_set_group (GdkWindow *window,
1291 g_return_if_fail (GDK_IS_WINDOW (window));
1292 g_return_if_fail (GDK_WINDOW_TYPE (window) != GDK_WINDOW_CHILD);
1293 g_return_if_fail (leader == NULL || GDK_IS_WINDOW (leader));
1297 gdk_wayland_window_set_decorations (GdkWindow *window,
1298 GdkWMDecoration decorations)
1303 gdk_wayland_window_get_decorations (GdkWindow *window,
1304 GdkWMDecoration *decorations)
1310 gdk_wayland_window_set_functions (GdkWindow *window,
1311 GdkWMFunction functions)
1316 gdk_wayland_window_begin_resize_drag (GdkWindow *window,
1324 GdkWindowImplWayland *impl;
1327 if (GDK_WINDOW_DESTROYED (window) ||
1328 !WINDOW_IS_TOPLEVEL_OR_FOREIGN (window))
1333 case GDK_WINDOW_EDGE_NORTH_WEST:
1334 grab_type = WL_SHELL_SURFACE_RESIZE_TOP_LEFT;
1337 case GDK_WINDOW_EDGE_NORTH:
1338 grab_type = WL_SHELL_SURFACE_RESIZE_TOP;
1341 case GDK_WINDOW_EDGE_NORTH_EAST:
1342 grab_type = WL_SHELL_SURFACE_RESIZE_RIGHT;
1345 case GDK_WINDOW_EDGE_WEST:
1346 grab_type = WL_SHELL_SURFACE_RESIZE_LEFT;
1349 case GDK_WINDOW_EDGE_EAST:
1350 grab_type = WL_SHELL_SURFACE_RESIZE_RIGHT;
1353 case GDK_WINDOW_EDGE_SOUTH_WEST:
1354 grab_type = WL_SHELL_SURFACE_RESIZE_BOTTOM_LEFT;
1357 case GDK_WINDOW_EDGE_SOUTH:
1358 grab_type = WL_SHELL_SURFACE_RESIZE_BOTTOM;
1361 case GDK_WINDOW_EDGE_SOUTH_EAST:
1362 grab_type = WL_SHELL_SURFACE_RESIZE_BOTTOM_RIGHT;
1366 g_warning ("gdk_window_begin_resize_drag: bad resize edge %d!",
1371 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1373 wl_shell_surface_resize (impl->shell_surface,
1374 _gdk_wayland_device_get_device (device),
1375 timestamp, grab_type);
1377 /* This is needed since Wayland will absorb all the pointer events after the
1378 * above function - FIXME: Is this always safe..?
1380 gdk_device_ungrab (device, timestamp);
1384 gdk_wayland_window_begin_move_drag (GdkWindow *window,
1391 GdkWindowImplWayland *impl;
1393 if (GDK_WINDOW_DESTROYED (window) ||
1394 !WINDOW_IS_TOPLEVEL (window))
1397 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1399 wl_shell_surface_move (impl->shell_surface,
1400 _gdk_wayland_device_get_device (device), timestamp);
1402 /* This is needed since Wayland will absorb all the pointer events after the
1403 * above function - FIXME: Is this always safe..?
1405 gdk_device_ungrab (device, timestamp);
1409 gdk_wayland_window_enable_synchronized_configure (GdkWindow *window)
1414 gdk_wayland_window_configure_finished (GdkWindow *window)
1416 if (!WINDOW_IS_TOPLEVEL (window))
1419 if (!GDK_IS_WINDOW_IMPL_WAYLAND (window->impl))
1424 gdk_wayland_window_set_opacity (GdkWindow *window,
1430 gdk_wayland_window_set_composited (GdkWindow *window,
1431 gboolean composited)
1436 gdk_wayland_window_destroy_notify (GdkWindow *window)
1438 if (!GDK_WINDOW_DESTROYED (window))
1440 if (GDK_WINDOW_TYPE(window) != GDK_WINDOW_FOREIGN)
1441 g_warning ("GdkWindow %p unexpectedly destroyed", window);
1443 _gdk_window_destroy (window, TRUE);
1446 g_object_unref (window);
1450 gdk_wayland_window_process_updates_recurse (GdkWindow *window,
1451 cairo_region_t *region)
1453 GdkWindowImplWayland *impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1454 #ifndef GDK_WAYLAND_USE_EGL
1455 GdkWaylandCairoSurfaceData *data = NULL;
1457 cairo_rectangle_int_t rect;
1460 gdk_wayland_window_map (window);
1462 if (impl->cairo_surface)
1463 gdk_wayland_window_attach_image (window);
1465 #ifndef GDK_WAYLAND_USE_EGL
1466 if (impl->server_surface)
1467 data = cairo_surface_get_user_data (impl->server_surface,
1468 &gdk_wayland_cairo_key);
1471 n = cairo_region_num_rectangles(region);
1472 for (i = 0; i < n; i++)
1474 cairo_region_get_rectangle (region, i, &rect);
1475 #ifndef GDK_WAYLAND_USE_EGL
1476 if (data && data->buffer)
1477 wl_buffer_damage (data->buffer,
1478 rect.x, rect.y, rect.width, rect.height);
1480 wl_surface_damage (impl->surface,
1481 rect.x, rect.y, rect.width, rect.height);
1484 _gdk_window_process_updates_recurse (window, region);
1488 gdk_wayland_window_sync_rendering (GdkWindow *window)
1493 gdk_wayland_window_simulate_key (GdkWindow *window,
1497 GdkModifierType modifiers,
1498 GdkEventType key_pressrelease)
1504 gdk_wayland_window_simulate_button (GdkWindow *window,
1507 guint button, /*1..3*/
1508 GdkModifierType modifiers,
1509 GdkEventType button_pressrelease)
1515 gdk_wayland_window_get_property (GdkWindow *window,
1521 GdkAtom *actual_property_type,
1522 gint *actual_format_type,
1523 gint *actual_length,
1530 gdk_wayland_window_change_property (GdkWindow *window,
1541 gdk_wayland_window_delete_property (GdkWindow *window,
1547 _gdk_window_impl_wayland_class_init (GdkWindowImplWaylandClass *klass)
1549 GObjectClass *object_class = G_OBJECT_CLASS (klass);
1550 GdkWindowImplClass *impl_class = GDK_WINDOW_IMPL_CLASS (klass);
1552 object_class->finalize = gdk_window_impl_wayland_finalize;
1554 impl_class->ref_cairo_surface = gdk_wayland_window_ref_cairo_surface;
1555 impl_class->show = gdk_wayland_window_show;
1556 impl_class->hide = gdk_wayland_window_hide;
1557 impl_class->withdraw = gdk_window_wayland_withdraw;
1558 impl_class->set_events = gdk_window_wayland_set_events;
1559 impl_class->get_events = gdk_window_wayland_get_events;
1560 impl_class->raise = gdk_window_wayland_raise;
1561 impl_class->lower = gdk_window_wayland_lower;
1562 impl_class->restack_under = gdk_window_wayland_restack_under;
1563 impl_class->restack_toplevel = gdk_window_wayland_restack_toplevel;
1564 impl_class->move_resize = gdk_window_wayland_move_resize;
1565 impl_class->set_background = gdk_window_wayland_set_background;
1566 impl_class->reparent = gdk_window_wayland_reparent;
1567 impl_class->set_device_cursor = gdk_window_wayland_set_device_cursor;
1568 impl_class->get_geometry = gdk_window_wayland_get_geometry;
1569 impl_class->get_root_coords = gdk_window_wayland_get_root_coords;
1570 impl_class->get_device_state = gdk_window_wayland_get_device_state;
1571 impl_class->shape_combine_region = gdk_window_wayland_shape_combine_region;
1572 impl_class->input_shape_combine_region = gdk_window_wayland_input_shape_combine_region;
1573 impl_class->set_static_gravities = gdk_window_wayland_set_static_gravities;
1574 impl_class->queue_antiexpose = gdk_wayland_window_queue_antiexpose;
1575 impl_class->translate = gdk_wayland_window_translate;
1576 impl_class->destroy = gdk_wayland_window_destroy;
1577 impl_class->destroy_foreign = gdk_window_wayland_destroy_foreign;
1578 impl_class->resize_cairo_surface = gdk_window_wayland_resize_cairo_surface;
1579 impl_class->get_shape = gdk_wayland_window_get_shape;
1580 impl_class->get_input_shape = gdk_wayland_window_get_input_shape;
1581 /* impl_class->beep */
1583 impl_class->focus = gdk_wayland_window_focus;
1584 impl_class->set_type_hint = gdk_wayland_window_set_type_hint;
1585 impl_class->get_type_hint = gdk_wayland_window_get_type_hint;
1586 impl_class->set_modal_hint = gdk_wayland_window_set_modal_hint;
1587 impl_class->set_skip_taskbar_hint = gdk_wayland_window_set_skip_taskbar_hint;
1588 impl_class->set_skip_pager_hint = gdk_wayland_window_set_skip_pager_hint;
1589 impl_class->set_urgency_hint = gdk_wayland_window_set_urgency_hint;
1590 impl_class->set_geometry_hints = gdk_wayland_window_set_geometry_hints;
1591 impl_class->set_title = gdk_wayland_window_set_title;
1592 impl_class->set_role = gdk_wayland_window_set_role;
1593 impl_class->set_startup_id = gdk_wayland_window_set_startup_id;
1594 impl_class->set_transient_for = gdk_wayland_window_set_transient_for;
1595 impl_class->get_root_origin = gdk_wayland_window_get_root_origin;
1596 impl_class->get_frame_extents = gdk_wayland_window_get_frame_extents;
1597 impl_class->set_override_redirect = gdk_wayland_window_set_override_redirect;
1598 impl_class->set_accept_focus = gdk_wayland_window_set_accept_focus;
1599 impl_class->set_focus_on_map = gdk_wayland_window_set_focus_on_map;
1600 impl_class->set_icon_list = gdk_wayland_window_set_icon_list;
1601 impl_class->set_icon_name = gdk_wayland_window_set_icon_name;
1602 impl_class->iconify = gdk_wayland_window_iconify;
1603 impl_class->deiconify = gdk_wayland_window_deiconify;
1604 impl_class->stick = gdk_wayland_window_stick;
1605 impl_class->unstick = gdk_wayland_window_unstick;
1606 impl_class->maximize = gdk_wayland_window_maximize;
1607 impl_class->unmaximize = gdk_wayland_window_unmaximize;
1608 impl_class->fullscreen = gdk_wayland_window_fullscreen;
1609 impl_class->unfullscreen = gdk_wayland_window_unfullscreen;
1610 impl_class->set_keep_above = gdk_wayland_window_set_keep_above;
1611 impl_class->set_keep_below = gdk_wayland_window_set_keep_below;
1612 impl_class->get_group = gdk_wayland_window_get_group;
1613 impl_class->set_group = gdk_wayland_window_set_group;
1614 impl_class->set_decorations = gdk_wayland_window_set_decorations;
1615 impl_class->get_decorations = gdk_wayland_window_get_decorations;
1616 impl_class->set_functions = gdk_wayland_window_set_functions;
1617 impl_class->begin_resize_drag = gdk_wayland_window_begin_resize_drag;
1618 impl_class->begin_move_drag = gdk_wayland_window_begin_move_drag;
1619 impl_class->enable_synchronized_configure = gdk_wayland_window_enable_synchronized_configure;
1620 impl_class->configure_finished = gdk_wayland_window_configure_finished;
1621 impl_class->set_opacity = gdk_wayland_window_set_opacity;
1622 impl_class->set_composited = gdk_wayland_window_set_composited;
1623 impl_class->destroy_notify = gdk_wayland_window_destroy_notify;
1624 impl_class->get_drag_protocol = _gdk_wayland_window_get_drag_protocol;
1625 impl_class->register_dnd = _gdk_wayland_window_register_dnd;
1626 impl_class->drag_begin = _gdk_wayland_window_drag_begin;
1627 impl_class->process_updates_recurse = gdk_wayland_window_process_updates_recurse;
1628 impl_class->sync_rendering = gdk_wayland_window_sync_rendering;
1629 impl_class->simulate_key = gdk_wayland_window_simulate_key;
1630 impl_class->simulate_button = gdk_wayland_window_simulate_button;
1631 impl_class->get_property = gdk_wayland_window_get_property;
1632 impl_class->change_property = gdk_wayland_window_change_property;
1633 impl_class->delete_property = gdk_wayland_window_delete_property;
1638 _gdk_wayland_window_set_device_grabbed (GdkWindow *window,
1639 struct wl_input_device *input_device,
1642 GdkWindowImplWayland *impl;
1644 g_return_if_fail (window != NULL);
1646 impl = GDK_WINDOW_IMPL_WAYLAND (window->impl);
1648 impl->grab_input_device = input_device;
1649 impl->grab_time = time_;