shell: Revive super-tab application switcher
We can do this right, now that we have keyboard grabs.
This commit is contained in:
parent
afa264c6b6
commit
0704539ec4
@ -88,7 +88,6 @@ desktop_shell_la_LIBADD = $(COMPOSITOR_LIBS) \
|
||||
desktop_shell_la_CFLAGS = $(GCC_CFLAGS)
|
||||
desktop_shell_la_SOURCES = \
|
||||
shell.c \
|
||||
switcher.c \
|
||||
desktop-shell-protocol.c \
|
||||
desktop-shell-server-protocol.h
|
||||
endif
|
||||
|
@ -179,8 +179,6 @@ struct weston_compositor {
|
||||
struct timespec previous_swap;
|
||||
struct wl_array vertices, indices;
|
||||
|
||||
struct weston_surface *overlay;
|
||||
struct weston_switcher *switcher;
|
||||
uint32_t focus;
|
||||
|
||||
PFNGLEGLIMAGETARGETRENDERBUFFERSTORAGEOESPROC
|
||||
@ -430,9 +428,6 @@ weston_input_device_init(struct weston_input_device *device,
|
||||
void
|
||||
weston_input_device_release(struct weston_input_device *device);
|
||||
|
||||
void
|
||||
weston_switcher_init(struct weston_compositor *compositor);
|
||||
|
||||
enum {
|
||||
TTY_ENTER_VT,
|
||||
TTY_LEAVE_VT
|
||||
|
119
src/shell.c
119
src/shell.c
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright © 2010 Intel Corporation
|
||||
* Copyright © 2010-2012 Intel Corporation
|
||||
* Copyright © 2011-2012 Collabora, Ltd.
|
||||
*
|
||||
* Permission to use, copy, modify, distribute, and sell this software and
|
||||
@ -1653,6 +1653,120 @@ bind_screensaver(struct wl_client *client,
|
||||
wl_resource_destroy(resource, 0);
|
||||
}
|
||||
|
||||
struct switcher {
|
||||
struct weston_compositor *compositor;
|
||||
struct weston_surface *current;
|
||||
struct wl_listener listener;
|
||||
struct wl_keyboard_grab grab;
|
||||
};
|
||||
|
||||
static void
|
||||
switcher_next(struct switcher *switcher)
|
||||
{
|
||||
struct weston_compositor *compositor = switcher->compositor;
|
||||
struct weston_surface *surface;
|
||||
struct weston_surface *first = NULL, *prev = NULL, *next = NULL;
|
||||
|
||||
wl_list_for_each(surface, &compositor->surface_list, link) {
|
||||
/* Workaround for cursor surfaces. */
|
||||
if (surface->surface.resource.destroy_listener_list.next == NULL)
|
||||
continue;
|
||||
|
||||
switch (get_shell_surface_type(surface)) {
|
||||
case SHELL_SURFACE_TOPLEVEL:
|
||||
case SHELL_SURFACE_FULLSCREEN:
|
||||
case SHELL_SURFACE_MAXIMIZED:
|
||||
if (first == NULL)
|
||||
first = surface;
|
||||
if (prev == switcher->current)
|
||||
next = surface;
|
||||
prev = surface;
|
||||
surface->alpha = 64;
|
||||
weston_surface_damage(surface);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (next == NULL)
|
||||
next = first;
|
||||
|
||||
wl_list_remove(&switcher->listener.link);
|
||||
wl_list_insert(next->surface.resource.destroy_listener_list.prev,
|
||||
&switcher->listener.link);
|
||||
|
||||
switcher->current = next;
|
||||
next->alpha = 255;
|
||||
}
|
||||
|
||||
static void
|
||||
switcher_handle_surface_destroy(struct wl_listener *listener,
|
||||
struct wl_resource *resource, uint32_t time)
|
||||
{
|
||||
struct switcher *switcher =
|
||||
container_of(listener, struct switcher, listener);
|
||||
|
||||
switcher_next(switcher);
|
||||
}
|
||||
|
||||
static void
|
||||
switcher_destroy(struct switcher *switcher, uint32_t time)
|
||||
{
|
||||
struct weston_compositor *compositor = switcher->compositor;
|
||||
struct weston_surface *surface;
|
||||
struct weston_input_device *device =
|
||||
(struct weston_input_device *) switcher->grab.input_device;
|
||||
|
||||
wl_list_for_each(surface, &compositor->surface_list, link) {
|
||||
surface->alpha = 255;
|
||||
weston_surface_damage(surface);
|
||||
}
|
||||
|
||||
activate(compositor->shell, switcher->current, device, time);
|
||||
wl_list_remove(&switcher->listener.link);
|
||||
wl_input_device_end_keyboard_grab(&device->input_device, time);
|
||||
free(switcher);
|
||||
}
|
||||
|
||||
static void
|
||||
switcher_key(struct wl_keyboard_grab *grab,
|
||||
uint32_t time, uint32_t key, int32_t state)
|
||||
{
|
||||
struct switcher *switcher = container_of(grab, struct switcher, grab);
|
||||
struct weston_input_device *device =
|
||||
(struct weston_input_device *) grab->input_device;
|
||||
|
||||
if ((device->modifier_state & MODIFIER_SUPER) == 0) {
|
||||
switcher_destroy(switcher, time);
|
||||
} else if (key == KEY_TAB && state) {
|
||||
switcher_next(switcher);
|
||||
}
|
||||
};
|
||||
|
||||
static const struct wl_keyboard_grab_interface switcher_grab = {
|
||||
switcher_key
|
||||
};
|
||||
|
||||
static void
|
||||
switcher_binding(struct wl_input_device *device, uint32_t time,
|
||||
uint32_t key, uint32_t button,
|
||||
uint32_t state, void *data)
|
||||
{
|
||||
struct weston_compositor *compositor = data;
|
||||
struct switcher *switcher;
|
||||
|
||||
switcher = malloc(sizeof *switcher);
|
||||
switcher->compositor = compositor;
|
||||
switcher->current = NULL;
|
||||
switcher->listener.func = switcher_handle_surface_destroy;
|
||||
wl_list_init(&switcher->listener.link);
|
||||
|
||||
switcher->grab.interface = &switcher_grab;
|
||||
wl_input_device_start_keyboard_grab(device, &switcher->grab, time);
|
||||
switcher_next(switcher);
|
||||
}
|
||||
|
||||
static void
|
||||
shell_destroy(struct weston_shell *base)
|
||||
{
|
||||
@ -1722,6 +1836,9 @@ shell_init(struct weston_compositor *ec)
|
||||
MODIFIER_SUPER | MODIFIER_ALT,
|
||||
rotate_binding, NULL);
|
||||
|
||||
weston_compositor_add_binding(ec, KEY_TAB, 0, MODIFIER_SUPER,
|
||||
switcher_binding, ec);
|
||||
|
||||
ec->shell = &shell->shell;
|
||||
|
||||
return 0;
|
||||
|
131
src/switcher.c
131
src/switcher.c
@ -1,131 +0,0 @@
|
||||
/*
|
||||
* Copyright © 2011 Intel Corporation
|
||||
*
|
||||
* Permission to use, copy, modify, distribute, and sell this software and its
|
||||
* documentation for any purpose is hereby granted without fee, provided that
|
||||
* the above copyright notice appear in all copies and that both that copyright
|
||||
* notice and this permission notice appear in supporting documentation, and
|
||||
* that the name of the copyright holders not be used in advertising or
|
||||
* publicity pertaining to distribution of the software without specific,
|
||||
* written prior permission. The copyright holders make no representations
|
||||
* about the suitability of this software for any purpose. It is provided "as
|
||||
* is" without express or implied warranty.
|
||||
*
|
||||
* THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
|
||||
* INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
|
||||
* EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
|
||||
* CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
|
||||
* DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
||||
* TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
|
||||
* OF THIS SOFTWARE.
|
||||
*/
|
||||
|
||||
#include <stdlib.h>
|
||||
#include <stdio.h>
|
||||
#include <linux/input.h>
|
||||
|
||||
#include "compositor.h"
|
||||
|
||||
struct weston_switcher {
|
||||
struct weston_compositor *compositor;
|
||||
struct weston_surface *current;
|
||||
struct wl_listener listener;
|
||||
};
|
||||
|
||||
static void
|
||||
weston_switcher_next(struct weston_switcher *switcher)
|
||||
{
|
||||
struct wl_list *l;
|
||||
struct wl_surface *current;
|
||||
|
||||
weston_surface_damage(switcher->current);
|
||||
l = switcher->current->link.next;
|
||||
if (l == &switcher->compositor->surface_list)
|
||||
l = switcher->compositor->surface_list.next;
|
||||
switcher->current = container_of(l, struct weston_surface, link);
|
||||
wl_list_remove(&switcher->listener.link);
|
||||
current = &switcher->current->surface;
|
||||
wl_list_insert(current->resource.destroy_listener_list.prev,
|
||||
&switcher->listener.link);
|
||||
switcher->compositor->overlay = switcher->current;
|
||||
weston_surface_damage(switcher->current);
|
||||
}
|
||||
|
||||
static void
|
||||
switcher_handle_surface_destroy(struct wl_listener *listener,
|
||||
struct wl_resource *resource, uint32_t time)
|
||||
{
|
||||
struct weston_switcher *switcher =
|
||||
container_of(listener, struct weston_switcher, listener);
|
||||
|
||||
weston_switcher_next(switcher);
|
||||
}
|
||||
|
||||
static struct weston_switcher *
|
||||
weston_switcher_create(struct weston_compositor *compositor)
|
||||
{
|
||||
struct weston_switcher *switcher;
|
||||
|
||||
switcher = malloc(sizeof *switcher);
|
||||
switcher->compositor = compositor;
|
||||
switcher->current = container_of(compositor->surface_list.next,
|
||||
struct weston_surface, link);
|
||||
switcher->listener.func = switcher_handle_surface_destroy;
|
||||
wl_list_init(&switcher->listener.link);
|
||||
|
||||
return switcher;
|
||||
}
|
||||
|
||||
static void
|
||||
weston_switcher_destroy(struct weston_switcher *switcher)
|
||||
{
|
||||
wl_list_remove(&switcher->listener.link);
|
||||
free(switcher);
|
||||
}
|
||||
|
||||
static void
|
||||
switcher_next_binding(struct wl_input_device *device, uint32_t time,
|
||||
uint32_t key, uint32_t button,
|
||||
uint32_t state, void *data)
|
||||
{
|
||||
struct weston_compositor *compositor = data;
|
||||
|
||||
if (!state)
|
||||
return;
|
||||
if (wl_list_empty(&compositor->surface_list))
|
||||
return;
|
||||
if (compositor->switcher == NULL)
|
||||
compositor->switcher = weston_switcher_create(compositor);
|
||||
|
||||
weston_switcher_next(compositor->switcher);
|
||||
}
|
||||
|
||||
static void
|
||||
switcher_terminate_binding(struct wl_input_device *device,
|
||||
uint32_t time, uint32_t key, uint32_t button,
|
||||
uint32_t state, void *data)
|
||||
{
|
||||
struct weston_compositor *compositor = data;
|
||||
struct weston_input_device *wd = (struct weston_input_device *) device;
|
||||
|
||||
if (compositor->switcher && !state) {
|
||||
weston_surface_activate(compositor->switcher->current, wd, time);
|
||||
weston_switcher_destroy(compositor->switcher);
|
||||
compositor->switcher = NULL;
|
||||
compositor->overlay = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
weston_switcher_init(struct weston_compositor *compositor)
|
||||
{
|
||||
weston_compositor_add_binding(compositor,
|
||||
KEY_TAB, 0, MODIFIER_SUPER,
|
||||
switcher_next_binding, compositor);
|
||||
weston_compositor_add_binding(compositor,
|
||||
KEY_LEFTMETA, 0, MODIFIER_SUPER,
|
||||
switcher_terminate_binding, compositor);
|
||||
weston_compositor_add_binding(compositor,
|
||||
KEY_RIGHTMETA, 0, MODIFIER_SUPER,
|
||||
switcher_terminate_binding, compositor);
|
||||
}
|
Loading…
Reference in New Issue
Block a user