|  |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright © 2010-2012 Intel Corporation
 | 
					
						
							|  |  |  |  * Copyright © 2011-2012 Collabora, Ltd.
 | 
					
						
							|  |  |  |  * Copyright © 2013 Raspberry Pi Foundation
 | 
					
						
							|  |  |  |  * Copyright © 2016 Quentin "Sardem FF7" Glidic
 | 
					
						
							|  |  |  |  *
 | 
					
						
							|  |  |  |  * Permission is hereby granted, free of charge, to any person obtaining a
 | 
					
						
							|  |  |  |  * copy of this software and associated documentation files (the "Software"),
 | 
					
						
							|  |  |  |  * to deal in the Software without restriction, including without limitation
 | 
					
						
							|  |  |  |  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 | 
					
						
							|  |  |  |  * and/or sell copies of the Software, and to permit persons to whom the
 | 
					
						
							|  |  |  |  * Software is furnished to do so, subject to the following conditions:
 | 
					
						
							|  |  |  |  *
 | 
					
						
							|  |  |  |  * The above copyright notice and this permission notice (including the next
 | 
					
						
							|  |  |  |  * paragraph) shall be included in all copies or substantial portions of the
 | 
					
						
							|  |  |  |  * Software.
 | 
					
						
							|  |  |  |  *
 | 
					
						
							|  |  |  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | 
					
						
							|  |  |  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | 
					
						
							|  |  |  |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
 | 
					
						
							|  |  |  |  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | 
					
						
							|  |  |  |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 | 
					
						
							|  |  |  |  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 | 
					
						
							|  |  |  |  * DEALINGS IN THE SOFTWARE.
 | 
					
						
							|  |  |  |  */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "config.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <assert.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <wayland-server.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "compositor.h"
 | 
					
						
							|  |  |  | #include "zalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "libweston-desktop.h"
 | 
					
						
							|  |  |  | #include "internal.h"
 | 
					
						
							|  |  |  | #include "shared/timespec-util.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct weston_desktop_seat {
 | 
					
						
							|  |  |  | 	struct wl_listener seat_destroy_listener;
 | 
					
						
							|  |  |  | 	struct weston_seat *seat;
 | 
					
						
							|  |  |  | 	struct {
 | 
					
						
							|  |  |  | 		struct weston_keyboard_grab keyboard;
 | 
					
						
							|  |  |  | 		struct weston_pointer_grab pointer;
 | 
					
						
							|  |  |  | 		struct weston_touch_grab touch;
 | 
					
						
							|  |  |  | 		bool initial_up;
 | 
					
						
							|  |  |  | 		struct wl_client *client;
 | 
					
						
							|  |  |  | 		struct wl_list surfaces;
 | 
					
						
							|  |  |  | 	} popup_grab;
 | 
					
						
							|  |  |  | };
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void weston_desktop_seat_popup_grab_end(struct weston_desktop_seat *seat);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_keyboard_key(struct weston_keyboard_grab *grab,
 | 
					
						
							|  |  |  | 					    const struct timespec *time,
 | 
					
						
							|  |  |  | 					    uint32_t key,
 | 
					
						
							|  |  |  | 					    enum wl_keyboard_key_state state)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	weston_keyboard_send_key(grab->keyboard, time, key, state);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_keyboard_modifiers(struct weston_keyboard_grab *grab,
 | 
					
						
							|  |  |  | 						  uint32_t serial,
 | 
					
						
							|  |  |  | 						  uint32_t mods_depressed,
 | 
					
						
							|  |  |  | 						  uint32_t mods_latched,
 | 
					
						
							|  |  |  | 						  uint32_t mods_locked,
 | 
					
						
							|  |  |  | 						  uint32_t group)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	weston_keyboard_send_modifiers(grab->keyboard, serial, mods_depressed,
 | 
					
						
							|  |  |  | 				       mods_latched, mods_locked, group);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_keyboard_cancel(struct weston_keyboard_grab *grab)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	struct weston_desktop_seat *seat =
 | 
					
						
							|  |  |  | 		wl_container_of(grab, seat, popup_grab.keyboard);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	weston_desktop_seat_popup_grab_end(seat);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct weston_keyboard_grab_interface weston_desktop_seat_keyboard_popup_grab_interface = {
 | 
					
						
							|  |  |  |    .key = weston_desktop_seat_popup_grab_keyboard_key,
 | 
					
						
							|  |  |  |    .modifiers = weston_desktop_seat_popup_grab_keyboard_modifiers,
 | 
					
						
							|  |  |  |    .cancel = weston_desktop_seat_popup_grab_keyboard_cancel,
 | 
					
						
							|  |  |  | };
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_pointer_focus(struct weston_pointer_grab *grab)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	struct weston_desktop_seat *seat =
 | 
					
						
							|  |  |  | 		wl_container_of(grab, seat, popup_grab.pointer);
 | 
					
						
							|  |  |  | 	struct weston_pointer *pointer = grab->pointer;
 | 
					
						
							|  |  |  | 	struct weston_view *view;
 | 
					
						
							|  |  |  | 	wl_fixed_t sx, sy;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	view = weston_compositor_pick_view(pointer->seat->compositor,
 | 
					
						
							|  |  |  | 					   pointer->x, pointer->y, &sx, &sy);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (view != NULL &&
 | 
					
						
							|  |  |  | 	    view->surface->resource != NULL &&
 | 
					
						
							|  |  |  | 	    wl_resource_get_client(view->surface->resource) == seat->popup_grab.client)
 | 
					
						
							|  |  |  | 		weston_pointer_set_focus(pointer, view, sx, sy);
 | 
					
						
							|  |  |  | 	else
 | 
					
						
							|  |  |  | 		weston_pointer_clear_focus(pointer);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_pointer_motion(struct weston_pointer_grab *grab,
 | 
					
						
							|  |  |  | 					      const struct timespec *time,
 | 
					
						
							|  |  |  | 					      struct weston_pointer_motion_event *event)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	weston_pointer_send_motion(grab->pointer, time, event);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_pointer_button(struct weston_pointer_grab *grab,
 | 
					
						
							|  |  |  | 					      const struct timespec *time,
 | 
					
						
							|  |  |  | 					      uint32_t button,
 | 
					
						
							|  |  |  | 					      enum wl_pointer_button_state state)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	struct weston_desktop_seat *seat =
 | 
					
						
							|  |  |  | 		wl_container_of(grab, seat, popup_grab.pointer);
 | 
					
						
							|  |  |  | 	struct weston_pointer *pointer = grab->pointer;
 | 
					
						
							|  |  |  | 	bool initial_up = seat->popup_grab.initial_up;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (state == WL_POINTER_BUTTON_STATE_RELEASED)
 | 
					
						
							|  |  |  | 		seat->popup_grab.initial_up = true;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (weston_pointer_has_focus_resource(pointer))
 | 
					
						
							|  |  |  | 		weston_pointer_send_button(pointer, time, button, state);
 | 
					
						
							|  |  |  | 	else if (state == WL_POINTER_BUTTON_STATE_RELEASED &&
 | 
					
						
							|  |  |  | 		 (initial_up ||
 | 
					
						
							|  |  |  | 		  (timespec_sub_to_msec(time, &grab->pointer->grab_time) > 500)))
 | 
					
						
							|  |  |  | 		weston_desktop_seat_popup_grab_end(seat);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_pointer_axis(struct weston_pointer_grab *grab,
 | 
					
						
							|  |  |  | 					    const struct timespec *time,
 | 
					
						
							|  |  |  | 					    struct weston_pointer_axis_event *event)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	weston_pointer_send_axis(grab->pointer, time, event);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_pointer_axis_source(struct weston_pointer_grab *grab,
 | 
					
						
							|  |  |  | 						   uint32_t source)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	weston_pointer_send_axis_source(grab->pointer, source);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_pointer_frame(struct weston_pointer_grab *grab)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	weston_pointer_send_frame(grab->pointer);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_pointer_cancel(struct weston_pointer_grab *grab)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	struct weston_desktop_seat *seat =
 | 
					
						
							|  |  |  | 		wl_container_of(grab, seat, popup_grab.pointer);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	weston_desktop_seat_popup_grab_end(seat);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct weston_pointer_grab_interface weston_desktop_seat_pointer_popup_grab_interface = {
 | 
					
						
							|  |  |  |    .focus = weston_desktop_seat_popup_grab_pointer_focus,
 | 
					
						
							|  |  |  |    .motion = weston_desktop_seat_popup_grab_pointer_motion,
 | 
					
						
							|  |  |  |    .button = weston_desktop_seat_popup_grab_pointer_button,
 | 
					
						
							|  |  |  |    .axis = weston_desktop_seat_popup_grab_pointer_axis,
 | 
					
						
							|  |  |  |    .axis_source = weston_desktop_seat_popup_grab_pointer_axis_source,
 | 
					
						
							|  |  |  |    .frame = weston_desktop_seat_popup_grab_pointer_frame,
 | 
					
						
							|  |  |  |    .cancel = weston_desktop_seat_popup_grab_pointer_cancel,
 | 
					
						
							|  |  |  | };
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_touch_down(struct weston_touch_grab *grab,
 | 
					
						
							|  |  |  | 					  const struct timespec *time,
 | 
					
						
							|  |  |  | 					  int touch_id,
 | 
					
						
							|  |  |  | 					  wl_fixed_t sx, wl_fixed_t sy)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	weston_touch_send_down(grab->touch, time, touch_id, sx, sy);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_touch_up(struct weston_touch_grab *grab,
 | 
					
						
							|  |  |  | 					const struct timespec *time,
 | 
					
						
							|  |  |  | 					int touch_id)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	weston_touch_send_up(grab->touch, time, touch_id);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_touch_motion(struct weston_touch_grab *grab,
 | 
					
						
							|  |  |  | 					    const struct timespec *time,
 | 
					
						
							|  |  |  | 					    int touch_id,
 | 
					
						
							|  |  |  | 					    wl_fixed_t sx, wl_fixed_t sy)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	weston_touch_send_motion(grab->touch, time, touch_id, sx, sy);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_touch_frame(struct weston_touch_grab *grab)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	weston_touch_send_frame(grab->touch);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_touch_cancel(struct weston_touch_grab *grab)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	struct weston_desktop_seat *seat =
 | 
					
						
							|  |  |  | 		wl_container_of(grab, seat, popup_grab.touch);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	weston_desktop_seat_popup_grab_end(seat);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct weston_touch_grab_interface weston_desktop_seat_touch_popup_grab_interface = {
 | 
					
						
							|  |  |  |    .down = weston_desktop_seat_popup_grab_touch_down,
 | 
					
						
							|  |  |  |    .up = weston_desktop_seat_popup_grab_touch_up,
 | 
					
						
							|  |  |  |    .motion = weston_desktop_seat_popup_grab_touch_motion,
 | 
					
						
							|  |  |  |    .frame = weston_desktop_seat_popup_grab_touch_frame,
 | 
					
						
							|  |  |  |    .cancel = weston_desktop_seat_popup_grab_touch_cancel,
 | 
					
						
							|  |  |  | };
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_destroy(struct wl_listener *listener, void *data)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	struct weston_desktop_seat *seat =
 | 
					
						
							|  |  |  | 		wl_container_of(listener, seat, seat_destroy_listener);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	free(seat);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct weston_desktop_seat *
 | 
					
						
							|  |  |  | weston_desktop_seat_from_seat(struct weston_seat *wseat)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	struct wl_listener *listener;
 | 
					
						
							|  |  |  | 	struct weston_desktop_seat *seat;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (wseat == NULL)
 | 
					
						
							|  |  |  | 		return NULL;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	listener = wl_signal_get(&wseat->destroy_signal,
 | 
					
						
							|  |  |  | 				 weston_desktop_seat_destroy);
 | 
					
						
							|  |  |  | 	if (listener != NULL)
 | 
					
						
							|  |  |  | 		return wl_container_of(listener, seat, seat_destroy_listener);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	seat = zalloc(sizeof(struct weston_desktop_seat));
 | 
					
						
							|  |  |  | 	if (seat == NULL)
 | 
					
						
							|  |  |  | 		return NULL;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	seat->seat = wseat;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	seat->seat_destroy_listener.notify = weston_desktop_seat_destroy;
 | 
					
						
							|  |  |  | 	wl_signal_add(&wseat->destroy_signal, &seat->seat_destroy_listener);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	seat->popup_grab.keyboard.interface =
 | 
					
						
							|  |  |  | 		&weston_desktop_seat_keyboard_popup_grab_interface;
 | 
					
						
							|  |  |  | 	seat->popup_grab.pointer.interface =
 | 
					
						
							|  |  |  | 		&weston_desktop_seat_pointer_popup_grab_interface;
 | 
					
						
							|  |  |  | 	seat->popup_grab.touch.interface =
 | 
					
						
							|  |  |  | 		&weston_desktop_seat_touch_popup_grab_interface;
 | 
					
						
							|  |  |  | 	wl_list_init(&seat->popup_grab.surfaces);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return seat;
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct weston_desktop_surface *
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_get_topmost_surface(struct weston_desktop_seat *seat)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	if (seat == NULL || wl_list_empty(&seat->popup_grab.surfaces))
 | 
					
						
							|  |  |  | 		return NULL;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct wl_list *grab_link = seat->popup_grab.surfaces.next;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return weston_desktop_surface_from_grab_link(grab_link);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_start(struct weston_desktop_seat *seat,
 | 
					
						
							|  |  |  | 				     struct wl_client *client, uint32_t serial)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	assert(seat == NULL || seat->popup_grab.client == NULL ||
 | 
					
						
							|  |  |  | 	       seat->popup_grab.client == client);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct weston_seat *wseat = seat != NULL ? seat->seat : NULL;
 | 
					
						
							|  |  |  | 	/* weston_seat_get_* functions can properly handle a NULL wseat */
 | 
					
						
							|  |  |  | 	struct weston_keyboard *keyboard = weston_seat_get_keyboard(wseat);
 | 
					
						
							|  |  |  | 	struct weston_pointer *pointer = weston_seat_get_pointer(wseat);
 | 
					
						
							|  |  |  | 	struct weston_touch *touch = weston_seat_get_touch(wseat);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((keyboard == NULL || keyboard->grab_serial != serial) &&
 | 
					
						
							|  |  |  | 	    (pointer == NULL || pointer->grab_serial != serial) &&
 | 
					
						
							|  |  |  | 	    (touch == NULL || touch->grab_serial != serial)) {
 | 
					
						
							|  |  |  | 		return false;
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (keyboard != NULL &&
 | 
					
						
							|  |  |  | 	    keyboard->grab->interface != &weston_desktop_seat_keyboard_popup_grab_interface)
 | 
					
						
							|  |  |  | 		weston_keyboard_start_grab(keyboard, &seat->popup_grab.keyboard);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (pointer != NULL &&
 | 
					
						
							|  |  |  | 	    pointer->grab->interface != &weston_desktop_seat_pointer_popup_grab_interface)
 | 
					
						
							|  |  |  | 		weston_pointer_start_grab(pointer, &seat->popup_grab.pointer);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (touch != NULL &&
 | 
					
						
							|  |  |  | 	    touch->grab->interface != &weston_desktop_seat_touch_popup_grab_interface)
 | 
					
						
							|  |  |  | 		weston_touch_start_grab(touch, &seat->popup_grab.touch);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	seat->popup_grab.initial_up =
 | 
					
						
							|  |  |  | 		(pointer == NULL || pointer->button_count == 0);
 | 
					
						
							|  |  |  | 	seat->popup_grab.client = client;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return true;
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_end(struct weston_desktop_seat *seat)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	struct weston_keyboard *keyboard = weston_seat_get_keyboard(seat->seat);
 | 
					
						
							|  |  |  | 	struct weston_pointer *pointer = weston_seat_get_pointer(seat->seat);
 | 
					
						
							|  |  |  | 	struct weston_touch *touch = weston_seat_get_touch(seat->seat);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (!wl_list_empty(&seat->popup_grab.surfaces)) {
 | 
					
						
							|  |  |  | 		struct wl_list *link = seat->popup_grab.surfaces.prev;
 | 
					
						
							|  |  |  | 		struct weston_desktop_surface *surface =
 | 
					
						
							|  |  |  | 			weston_desktop_surface_from_grab_link(link);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		wl_list_remove(link);
 | 
					
						
							|  |  |  | 		wl_list_init(link);
 | 
					
						
							|  |  |  | 		weston_desktop_surface_popup_dismiss(surface);
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (keyboard != NULL &&
 | 
					
						
							|  |  |  | 	    keyboard->grab->interface == &weston_desktop_seat_keyboard_popup_grab_interface)
 | 
					
						
							|  |  |  | 		weston_keyboard_end_grab(keyboard);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (pointer != NULL &&
 | 
					
						
							|  |  |  | 	    pointer->grab->interface == &weston_desktop_seat_pointer_popup_grab_interface)
 | 
					
						
							|  |  |  | 		weston_pointer_end_grab(pointer);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (touch != NULL &&
 | 
					
						
							|  |  |  | 	    touch->grab->interface == &weston_desktop_seat_touch_popup_grab_interface)
 | 
					
						
							|  |  |  | 		weston_touch_end_grab(touch);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	seat->popup_grab.client = NULL;
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_add_surface(struct weston_desktop_seat *seat,
 | 
					
						
							|  |  |  | 					   struct wl_list *link)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	assert(seat->popup_grab.client != NULL);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	wl_list_insert(&seat->popup_grab.surfaces, link);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void
 | 
					
						
							|  |  |  | weston_desktop_seat_popup_grab_remove_surface(struct weston_desktop_seat *seat,
 | 
					
						
							|  |  |  | 					      struct wl_list *link)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	assert(seat->popup_grab.client != NULL);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	wl_list_remove(link);
 | 
					
						
							|  |  |  | 	wl_list_init(link);
 | 
					
						
							|  |  |  | 	if (wl_list_empty(&seat->popup_grab.surfaces))
 | 
					
						
							|  |  |  | 		weston_desktop_seat_popup_grab_end(seat);
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | WL_EXPORT void
 | 
					
						
							|  |  |  | weston_seat_break_desktop_grabs(struct weston_seat *wseat)
 | 
					
						
							|  |  |  | {
 | 
					
						
							|  |  |  | 	struct weston_desktop_seat *seat = weston_desktop_seat_from_seat(wseat);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	weston_desktop_seat_popup_grab_end(seat);
 | 
					
						
							|  |  |  | }
 |