| /* |
| * Copyright © 2010-2012 Intel Corporation |
| * Copyright © 2011-2012 Collabora, Ltd. |
| * Copyright © 2013 Raspberry Pi Foundation |
| * |
| * 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 <stdlib.h> |
| #include <stdint.h> |
| #include <stdio.h> |
| #include <string.h> |
| #include <unistd.h> |
| #include <linux/input.h> |
| #include <assert.h> |
| #include <signal.h> |
| #include <math.h> |
| #include <sys/types.h> |
| |
| #include "shell.h" |
| #include "compositor/weston.h" |
| #include "weston-desktop-shell-server-protocol.h" |
| #include "shared/config-parser.h" |
| #include "shared/helpers.h" |
| #include "shared/timespec-util.h" |
| #include "libweston-desktop/libweston-desktop.h" |
| |
| #define DEFAULT_NUM_WORKSPACES 1 |
| #define DEFAULT_WORKSPACE_CHANGE_ANIMATION_LENGTH 200 |
| |
| #ifndef static_assert |
| #define static_assert(cond, msg) |
| #endif |
| |
| struct focus_state { |
| struct desktop_shell *shell; |
| struct weston_seat *seat; |
| struct workspace *ws; |
| struct weston_surface *keyboard_focus; |
| struct wl_list link; |
| struct wl_listener seat_destroy_listener; |
| struct wl_listener surface_destroy_listener; |
| }; |
| |
| /* |
| * Surface stacking and ordering. |
| * |
| * This is handled using several linked lists of surfaces, organised into |
| * ‘layers’. The layers are ordered, and each of the surfaces in one layer are |
| * above all of the surfaces in the layer below. The set of layers is static and |
| * in the following order (top-most first): |
| * • Lock layer (only ever displayed on its own) |
| * • Cursor layer |
| * • Input panel layer |
| * • Fullscreen layer |
| * • Panel layer |
| * • Workspace layers |
| * • Background layer |
| * |
| * The list of layers may be manipulated to remove whole layers of surfaces from |
| * display. For example, when locking the screen, all layers except the lock |
| * layer are removed. |
| * |
| * A surface’s layer is modified on configuring the surface, in |
| * set_surface_type() (which is only called when the surface’s type change is |
| * _committed_). If a surface’s type changes (e.g. when making a window |
| * fullscreen) its layer changes too. |
| * |
| * In order to allow popup and transient surfaces to be correctly stacked above |
| * their parent surfaces, each surface tracks both its parent surface, and a |
| * linked list of its children. When a surface’s layer is updated, so are the |
| * layers of its children. Note that child surfaces are *not* the same as |
| * subsurfaces — child/parent surfaces are purely for maintaining stacking |
| * order. |
| * |
| * The children_link list of siblings of a surface (i.e. those surfaces which |
| * have the same parent) only contains weston_surfaces which have a |
| * shell_surface. Stacking is not implemented for non-shell_surface |
| * weston_surfaces. This means that the following implication does *not* hold: |
| * (shsurf->parent != NULL) ⇒ !wl_list_is_empty(shsurf->children_link) |
| */ |
| |
| struct shell_surface { |
| struct wl_signal destroy_signal; |
| |
| struct weston_desktop_surface *desktop_surface; |
| struct weston_view *view; |
| int32_t last_width, last_height; |
| |
| struct desktop_shell *shell; |
| |
| int32_t saved_x, saved_y; |
| bool saved_position_valid; |
| bool saved_rotation_valid; |
| int unresponsive, grabbed; |
| uint32_t resize_edges; |
| |
| struct { |
| struct weston_transform transform; |
| struct weston_matrix rotation; |
| } rotation; |
| |
| struct { |
| struct weston_transform transform; /* matrix from x, y */ |
| struct weston_view *black_view; |
| } fullscreen; |
| |
| struct weston_transform workspace_transform; |
| |
| struct weston_output *fullscreen_output; |
| struct weston_output *output; |
| |
| struct surface_state { |
| bool fullscreen; |
| bool maximized; |
| bool lowered; |
| } state; |
| |
| struct { |
| bool is_set; |
| int32_t x; |
| int32_t y; |
| } xwayland; |
| |
| int focus_count; |
| |
| bool destroying; |
| }; |
| |
| struct shell_grab { |
| struct weston_pointer_grab grab; |
| struct shell_surface *shsurf; |
| struct wl_listener shsurf_destroy_listener; |
| }; |
| |
| struct shell_touch_grab { |
| struct weston_touch_grab grab; |
| struct shell_surface *shsurf; |
| struct wl_listener shsurf_destroy_listener; |
| struct weston_touch *touch; |
| }; |
| |
| struct weston_move_grab { |
| struct shell_grab base; |
| wl_fixed_t dx, dy; |
| bool client_initiated; |
| }; |
| |
| struct weston_touch_move_grab { |
| struct shell_touch_grab base; |
| int active; |
| wl_fixed_t dx, dy; |
| }; |
| |
| struct rotate_grab { |
| struct shell_grab base; |
| struct weston_matrix rotation; |
| struct { |
| float x; |
| float y; |
| } center; |
| }; |
| |
| struct shell_seat { |
| struct weston_seat *seat; |
| struct wl_listener seat_destroy_listener; |
| struct weston_surface *focused_surface; |
| |
| struct wl_listener caps_changed_listener; |
| struct wl_listener pointer_focus_listener; |
| struct wl_listener keyboard_focus_listener; |
| }; |
| |
| |
| static struct desktop_shell * |
| shell_surface_get_shell(struct shell_surface *shsurf); |
| |
| static void |
| set_busy_cursor(struct shell_surface *shsurf, struct weston_pointer *pointer); |
| |
| static void |
| surface_rotate(struct shell_surface *surface, struct weston_pointer *pointer); |
| |
| static void |
| shell_fade_startup(struct desktop_shell *shell); |
| |
| static void |
| shell_fade(struct desktop_shell *shell, enum fade_type type); |
| |
| static struct shell_seat * |
| get_shell_seat(struct weston_seat *seat); |
| |
| static void |
| get_output_panel_size(struct desktop_shell *shell, |
| struct weston_output *output, |
| int *width, int *height); |
| |
| static void |
| shell_surface_update_child_surface_layers(struct shell_surface *shsurf); |
| |
| static int |
| shell_surface_get_label(struct weston_surface *surface, char *buf, size_t len) |
| { |
| const char *t, *c; |
| struct weston_desktop_surface *desktop_surface = |
| weston_surface_get_desktop_surface(surface); |
| |
| t = weston_desktop_surface_get_title(desktop_surface); |
| c = weston_desktop_surface_get_app_id(desktop_surface); |
| |
| return snprintf(buf, len, "%s window%s%s%s%s%s", |
| "top-level", |
| t ? " '" : "", t ?: "", t ? "'" : "", |
| c ? " of " : "", c ?: ""); |
| } |
| |
| static void |
| destroy_shell_grab_shsurf(struct wl_listener *listener, void *data) |
| { |
| struct shell_grab *grab; |
| |
| grab = container_of(listener, struct shell_grab, |
| shsurf_destroy_listener); |
| |
| grab->shsurf = NULL; |
| } |
| |
| struct weston_view * |
| get_default_view(struct weston_surface *surface) |
| { |
| struct shell_surface *shsurf; |
| struct weston_view *view; |
| |
| if (!surface || wl_list_empty(&surface->views)) |
| return NULL; |
| |
| shsurf = get_shell_surface(surface); |
| if (shsurf) |
| return shsurf->view; |
| |
| wl_list_for_each(view, &surface->views, surface_link) |
| if (weston_view_is_mapped(view)) |
| return view; |
| |
| return container_of(surface->views.next, struct weston_view, surface_link); |
| } |
| |
| static void |
| shell_grab_start(struct shell_grab *grab, |
| const struct weston_pointer_grab_interface *interface, |
| struct shell_surface *shsurf, |
| struct weston_pointer *pointer, |
| enum weston_desktop_shell_cursor cursor) |
| { |
| struct desktop_shell *shell = shsurf->shell; |
| |
| weston_seat_break_desktop_grabs(pointer->seat); |
| |
| grab->grab.interface = interface; |
| grab->shsurf = shsurf; |
| grab->shsurf_destroy_listener.notify = destroy_shell_grab_shsurf; |
| wl_signal_add(&shsurf->destroy_signal, |
| &grab->shsurf_destroy_listener); |
| |
| shsurf->grabbed = 1; |
| weston_pointer_start_grab(pointer, &grab->grab); |
| if (shell->child.desktop_shell) { |
| weston_desktop_shell_send_grab_cursor(shell->child.desktop_shell, |
| cursor); |
| weston_pointer_set_focus(pointer, |
| get_default_view(shell->grab_surface), |
| wl_fixed_from_int(0), |
| wl_fixed_from_int(0)); |
| } |
| } |
| |
| static void |
| get_panel_size(struct desktop_shell *shell, |
| struct weston_view *view, |
| int *width, |
| int *height) |
| { |
| float x1, y1; |
| float x2, y2; |
| weston_view_to_global_float(view, 0, 0, &x1, &y1); |
| weston_view_to_global_float(view, |
| view->surface->width, |
| view->surface->height, |
| &x2, &y2); |
| *width = (int)(x2 - x1); |
| *height = (int)(y2 - y1); |
| } |
| |
| static void |
| get_output_panel_size(struct desktop_shell *shell, |
| struct weston_output *output, |
| int *width, |
| int *height) |
| { |
| struct weston_view *view; |
| |
| *width = 0; |
| *height = 0; |
| |
| if (!output) |
| return; |
| |
| wl_list_for_each(view, &shell->panel_layer.view_list.link, layer_link.link) { |
| if (view->surface->output == output) { |
| get_panel_size(shell, view, width, height); |
| return; |
| } |
| } |
| |
| /* the correct view wasn't found */ |
| } |
| |
| static void |
| get_output_work_area(struct desktop_shell *shell, |
| struct weston_output *output, |
| pixman_rectangle32_t *area) |
| { |
| int32_t panel_width = 0, panel_height = 0; |
| |
| area->x = output->x; |
| area->y = output->y; |
| |
| get_output_panel_size(shell, output, &panel_width, &panel_height); |
| switch (shell->panel_position) { |
| case WESTON_DESKTOP_SHELL_PANEL_POSITION_TOP: |
| default: |
| area->y += panel_height; |
| /* fallthrough */ |
| case WESTON_DESKTOP_SHELL_PANEL_POSITION_BOTTOM: |
| area->width = output->width; |
| area->height = output->height - panel_height; |
| break; |
| case WESTON_DESKTOP_SHELL_PANEL_POSITION_LEFT: |
| area->x += panel_width; |
| /* fallthrough */ |
| case WESTON_DESKTOP_SHELL_PANEL_POSITION_RIGHT: |
| area->width = output->width - panel_width; |
| area->height = output->height; |
| break; |
| } |
| } |
| |
| static void |
| shell_grab_end(struct shell_grab *grab) |
| { |
| if (grab->shsurf) { |
| wl_list_remove(&grab->shsurf_destroy_listener.link); |
| grab->shsurf->grabbed = 0; |
| |
| if (grab->shsurf->resize_edges) { |
| grab->shsurf->resize_edges = 0; |
| } |
| } |
| |
| weston_pointer_end_grab(grab->grab.pointer); |
| } |
| |
| static void |
| shell_touch_grab_start(struct shell_touch_grab *grab, |
| const struct weston_touch_grab_interface *interface, |
| struct shell_surface *shsurf, |
| struct weston_touch *touch) |
| { |
| struct desktop_shell *shell = shsurf->shell; |
| |
| weston_seat_break_desktop_grabs(touch->seat); |
| |
| grab->grab.interface = interface; |
| grab->shsurf = shsurf; |
| grab->shsurf_destroy_listener.notify = destroy_shell_grab_shsurf; |
| wl_signal_add(&shsurf->destroy_signal, |
| &grab->shsurf_destroy_listener); |
| |
| grab->touch = touch; |
| shsurf->grabbed = 1; |
| |
| weston_touch_start_grab(touch, &grab->grab); |
| if (shell->child.desktop_shell) |
| weston_touch_set_focus(touch, |
| get_default_view(shell->grab_surface)); |
| } |
| |
| static void |
| shell_touch_grab_end(struct shell_touch_grab *grab) |
| { |
| if (grab->shsurf) { |
| wl_list_remove(&grab->shsurf_destroy_listener.link); |
| grab->shsurf->grabbed = 0; |
| } |
| |
| weston_touch_end_grab(grab->touch); |
| } |
| |
| static void |
| center_on_output(struct weston_view *view, |
| struct weston_output *output); |
| |
| static enum weston_keyboard_modifier |
| get_modifier(char *modifier) |
| { |
| if (!modifier) |
| return MODIFIER_SUPER; |
| |
| if (!strcmp("ctrl", modifier)) |
| return MODIFIER_CTRL; |
| else if (!strcmp("alt", modifier)) |
| return MODIFIER_ALT; |
| else if (!strcmp("super", modifier)) |
| return MODIFIER_SUPER; |
| else if (!strcmp("none", modifier)) |
| return 0; |
| else |
| return MODIFIER_SUPER; |
| } |
| |
| static enum animation_type |
| get_animation_type(char *animation) |
| { |
| if (!animation) |
| return ANIMATION_NONE; |
| |
| if (!strcmp("zoom", animation)) |
| return ANIMATION_ZOOM; |
| else if (!strcmp("fade", animation)) |
| return ANIMATION_FADE; |
| else if (!strcmp("dim-layer", animation)) |
| return ANIMATION_DIM_LAYER; |
| else |
| return ANIMATION_NONE; |
| } |
| |
| static void |
| shell_configuration(struct desktop_shell *shell) |
| { |
| struct weston_config_section *section; |
| char *s, *client; |
| int ret; |
| int allow_zap; |
| |
| section = weston_config_get_section(wet_get_config(shell->compositor), |
| "shell", NULL, NULL); |
| ret = asprintf(&client, "%s/%s", weston_config_get_libexec_dir(), |
| WESTON_SHELL_CLIENT); |
| if (ret < 0) |
| client = NULL; |
| weston_config_section_get_string(section, |
| "client", &s, client); |
| free(client); |
| shell->client = s; |
| |
| weston_config_section_get_bool(section, |
| "allow-zap", &allow_zap, true); |
| shell->allow_zap = allow_zap; |
| |
| weston_config_section_get_string(section, |
| "binding-modifier", &s, "super"); |
| shell->binding_modifier = get_modifier(s); |
| free(s); |
| |
| weston_config_section_get_string(section, |
| "exposay-modifier", &s, "none"); |
| shell->exposay_modifier = get_modifier(s); |
| free(s); |
| |
| weston_config_section_get_string(section, "animation", &s, "none"); |
| shell->win_animation_type = get_animation_type(s); |
| free(s); |
| weston_config_section_get_string(section, "close-animation", &s, "fade"); |
| shell->win_close_animation_type = get_animation_type(s); |
| free(s); |
| weston_config_section_get_string(section, |
| "startup-animation", &s, "fade"); |
| shell->startup_animation_type = get_animation_type(s); |
| free(s); |
| if (shell->startup_animation_type == ANIMATION_ZOOM) |
| shell->startup_animation_type = ANIMATION_NONE; |
| weston_config_section_get_string(section, "focus-animation", &s, "none"); |
| shell->focus_animation_type = get_animation_type(s); |
| free(s); |
| weston_config_section_get_uint(section, "num-workspaces", |
| &shell->workspaces.num, |
| DEFAULT_NUM_WORKSPACES); |
| } |
| |
| struct weston_output * |
| get_default_output(struct weston_compositor *compositor) |
| { |
| if (wl_list_empty(&compositor->output_list)) |
| return NULL; |
| |
| return container_of(compositor->output_list.next, |
| struct weston_output, link); |
| } |
| |
| static int |
| focus_surface_get_label(struct weston_surface *surface, char *buf, size_t len) |
| { |
| return snprintf(buf, len, "focus highlight effect for output %s", |
| surface->output->name); |
| } |
| |
| /* no-op func for checking focus surface */ |
| static void |
| focus_surface_committed(struct weston_surface *es, int32_t sx, int32_t sy) |
| { |
| } |
| |
| static struct focus_surface * |
| get_focus_surface(struct weston_surface *surface) |
| { |
| if (surface->committed == focus_surface_committed) |
| return surface->committed_private; |
| else |
| return NULL; |
| } |
| |
| static bool |
| is_focus_surface (struct weston_surface *es) |
| { |
| return (es->committed == focus_surface_committed); |
| } |
| |
| static bool |
| is_focus_view (struct weston_view *view) |
| { |
| return is_focus_surface (view->surface); |
| } |
| |
| static struct focus_surface * |
| create_focus_surface(struct weston_compositor *ec, |
| struct weston_output *output) |
| { |
| struct focus_surface *fsurf = NULL; |
| struct weston_surface *surface = NULL; |
| |
| fsurf = malloc(sizeof *fsurf); |
| if (!fsurf) |
| return NULL; |
| |
| fsurf->surface = weston_surface_create(ec); |
| surface = fsurf->surface; |
| if (surface == NULL) { |
| free(fsurf); |
| return NULL; |
| } |
| |
| surface->committed = focus_surface_committed; |
| surface->output = output; |
| surface->is_mapped = true; |
| surface->committed_private = fsurf; |
| weston_surface_set_label_func(surface, focus_surface_get_label); |
| |
| fsurf->view = weston_view_create(surface); |
| if (fsurf->view == NULL) { |
| weston_surface_destroy(surface); |
| free(fsurf); |
| return NULL; |
| } |
| weston_view_set_output(fsurf->view, output); |
| fsurf->view->is_mapped = true; |
| |
| weston_surface_set_size(surface, output->width, output->height); |
| weston_view_set_position(fsurf->view, output->x, output->y); |
| weston_surface_set_color(surface, 0.0, 0.0, 0.0, 1.0); |
| pixman_region32_fini(&surface->opaque); |
| pixman_region32_init_rect(&surface->opaque, output->x, output->y, |
| output->width, output->height); |
| pixman_region32_fini(&surface->input); |
| pixman_region32_init(&surface->input); |
| |
| wl_list_init(&fsurf->workspace_transform.link); |
| |
| return fsurf; |
| } |
| |
| static void |
| focus_surface_destroy(struct focus_surface *fsurf) |
| { |
| weston_surface_destroy(fsurf->surface); |
| free(fsurf); |
| } |
| |
| static void |
| focus_animation_done(struct weston_view_animation *animation, void *data) |
| { |
| struct workspace *ws = data; |
| |
| ws->focus_animation = NULL; |
| } |
| |
| static void |
| focus_state_destroy(struct focus_state *state) |
| { |
| wl_list_remove(&state->seat_destroy_listener.link); |
| wl_list_remove(&state->surface_destroy_listener.link); |
| free(state); |
| } |
| |
| static void |
| focus_state_seat_destroy(struct wl_listener *listener, void *data) |
| { |
| struct focus_state *state = container_of(listener, |
| struct focus_state, |
| seat_destroy_listener); |
| |
| wl_list_remove(&state->link); |
| focus_state_destroy(state); |
| } |
| |
| static void |
| focus_state_surface_destroy(struct wl_listener *listener, void *data) |
| { |
| struct focus_state *state = container_of(listener, |
| struct focus_state, |
| surface_destroy_listener); |
| struct weston_surface *main_surface; |
| struct weston_view *next; |
| struct weston_view *view; |
| |
| main_surface = weston_surface_get_main_surface(state->keyboard_focus); |
| |
| next = NULL; |
| wl_list_for_each(view, |
| &state->ws->layer.view_list.link, layer_link.link) { |
| if (view->surface == main_surface) |
| continue; |
| if (is_focus_view(view)) |
| continue; |
| if (!get_shell_surface(view->surface)) |
| continue; |
| |
| next = view; |
| break; |
| } |
| |
| /* if the focus was a sub-surface, activate its main surface */ |
| if (main_surface != state->keyboard_focus) |
| next = get_default_view(main_surface); |
| |
| if (next) { |
| state->keyboard_focus = NULL; |
| activate(state->shell, next, state->seat, |
| WESTON_ACTIVATE_FLAG_CONFIGURE); |
| } else { |
| if (state->shell->focus_animation_type == ANIMATION_DIM_LAYER) { |
| if (state->ws->focus_animation) |
| weston_view_animation_destroy(state->ws->focus_animation); |
| |
| state->ws->focus_animation = weston_fade_run( |
| state->ws->fsurf_front->view, |
| state->ws->fsurf_front->view->alpha, 0.0, 300, |
| focus_animation_done, state->ws); |
| } |
| |
| wl_list_remove(&state->link); |
| focus_state_destroy(state); |
| } |
| } |
| |
| static struct focus_state * |
| focus_state_create(struct desktop_shell *shell, struct weston_seat *seat, |
| struct workspace *ws) |
| { |
| struct focus_state *state; |
| |
| state = malloc(sizeof *state); |
| if (state == NULL) |
| return NULL; |
| |
| state->shell = shell; |
| state->keyboard_focus = NULL; |
| state->ws = ws; |
| state->seat = seat; |
| wl_list_insert(&ws->focus_list, &state->link); |
| |
| state->seat_destroy_listener.notify = focus_state_seat_destroy; |
| state->surface_destroy_listener.notify = focus_state_surface_destroy; |
| wl_signal_add(&seat->destroy_signal, |
| &state->seat_destroy_listener); |
| wl_list_init(&state->surface_destroy_listener.link); |
| |
| return state; |
| } |
| |
| static struct focus_state * |
| ensure_focus_state(struct desktop_shell *shell, struct weston_seat *seat) |
| { |
| struct workspace *ws = get_current_workspace(shell); |
| struct focus_state *state; |
| |
| wl_list_for_each(state, &ws->focus_list, link) |
| if (state->seat == seat) |
| break; |
| |
| if (&state->link == &ws->focus_list) |
| state = focus_state_create(shell, seat, ws); |
| |
| return state; |
| } |
| |
| static void |
| focus_state_set_focus(struct focus_state *state, |
| struct weston_surface *surface) |
| { |
| if (state->keyboard_focus) { |
| wl_list_remove(&state->surface_destroy_listener.link); |
| wl_list_init(&state->surface_destroy_listener.link); |
| } |
| |
| state->keyboard_focus = surface; |
| if (surface) |
| wl_signal_add(&surface->destroy_signal, |
| &state->surface_destroy_listener); |
| } |
| |
| static void |
| restore_focus_state(struct desktop_shell *shell, struct workspace *ws) |
| { |
| struct focus_state *state, *next; |
| struct weston_surface *surface; |
| struct wl_list pending_seat_list; |
| struct weston_seat *seat, *next_seat; |
| |
| /* Temporarily steal the list of seats so that we can keep |
| * track of the seats we've already processed */ |
| wl_list_init(&pending_seat_list); |
| wl_list_insert_list(&pending_seat_list, &shell->compositor->seat_list); |
| wl_list_init(&shell->compositor->seat_list); |
| |
| wl_list_for_each_safe(state, next, &ws->focus_list, link) { |
| struct weston_keyboard *keyboard = |
| weston_seat_get_keyboard(state->seat); |
| |
| wl_list_remove(&state->seat->link); |
| wl_list_insert(&shell->compositor->seat_list, |
| &state->seat->link); |
| |
| if (!keyboard) |
| continue; |
| |
| surface = state->keyboard_focus; |
| |
| weston_keyboard_set_focus(keyboard, surface); |
| } |
| |
| /* For any remaining seats that we don't have a focus state |
| * for we'll reset the keyboard focus to NULL */ |
| wl_list_for_each_safe(seat, next_seat, &pending_seat_list, link) { |
| struct weston_keyboard *keyboard = |
| weston_seat_get_keyboard(seat); |
| |
| wl_list_insert(&shell->compositor->seat_list, &seat->link); |
| |
| if (!keyboard) |
| continue; |
| |
| weston_keyboard_set_focus(keyboard, NULL); |
| } |
| } |
| |
| static void |
| replace_focus_state(struct desktop_shell *shell, struct workspace *ws, |
| struct weston_seat *seat) |
| { |
| struct weston_keyboard *keyboard = weston_seat_get_keyboard(seat); |
| struct focus_state *state; |
| |
| wl_list_for_each(state, &ws->focus_list, link) { |
| if (state->seat == seat) { |
| focus_state_set_focus(state, keyboard->focus); |
| return; |
| } |
| } |
| } |
| |
| static void |
| drop_focus_state(struct desktop_shell *shell, struct workspace *ws, |
| struct weston_surface *surface) |
| { |
| struct focus_state *state; |
| |
| wl_list_for_each(state, &ws->focus_list, link) |
| if (state->keyboard_focus == surface) |
| focus_state_set_focus(state, NULL); |
| } |
| |
| static void |
| animate_focus_change(struct desktop_shell *shell, struct workspace *ws, |
| struct weston_view *from, struct weston_view *to) |
| { |
| struct weston_output *output; |
| bool focus_surface_created = false; |
| |
| /* FIXME: Only support dim animation using two layers */ |
| if (from == to || shell->focus_animation_type != ANIMATION_DIM_LAYER) |
| return; |
| |
| output = get_default_output(shell->compositor); |
| if (ws->fsurf_front == NULL && (from || to)) { |
| ws->fsurf_front = create_focus_surface(shell->compositor, output); |
| if (ws->fsurf_front == NULL) |
| return; |
| ws->fsurf_front->view->alpha = 0.0; |
| |
| ws->fsurf_back = create_focus_surface(shell->compositor, output); |
| if (ws->fsurf_back == NULL) { |
| focus_surface_destroy(ws->fsurf_front); |
| return; |
| } |
| ws->fsurf_back->view->alpha = 0.0; |
| |
| focus_surface_created = true; |
| } else { |
| weston_layer_entry_remove(&ws->fsurf_front->view->layer_link); |
| weston_layer_entry_remove(&ws->fsurf_back->view->layer_link); |
| } |
| |
| if (ws->focus_animation) { |
| weston_view_animation_destroy(ws->focus_animation); |
| ws->focus_animation = NULL; |
| } |
| |
| if (to) |
| weston_layer_entry_insert(&to->layer_link, |
| &ws->fsurf_front->view->layer_link); |
| else if (from) |
| weston_layer_entry_insert(&ws->layer.view_list, |
| &ws->fsurf_front->view->layer_link); |
| |
| if (focus_surface_created) { |
| ws->focus_animation = weston_fade_run( |
| ws->fsurf_front->view, |
| ws->fsurf_front->view->alpha, 0.4, 300, |
| focus_animation_done, ws); |
| } else if (from) { |
| weston_layer_entry_insert(&from->layer_link, |
| &ws->fsurf_back->view->layer_link); |
| ws->focus_animation = weston_stable_fade_run( |
| ws->fsurf_front->view, 0.0, |
| ws->fsurf_back->view, 0.4, |
| focus_animation_done, ws); |
| } else if (to) { |
| weston_layer_entry_insert(&ws->layer.view_list, |
| &ws->fsurf_back->view->layer_link); |
| ws->focus_animation = weston_stable_fade_run( |
| ws->fsurf_front->view, 0.0, |
| ws->fsurf_back->view, 0.4, |
| focus_animation_done, ws); |
| } |
| } |
| |
| static void |
| workspace_destroy(struct workspace *ws) |
| { |
| struct focus_state *state, *next; |
| |
| wl_list_for_each_safe(state, next, &ws->focus_list, link) |
| focus_state_destroy(state); |
| |
| if (ws->fsurf_front) |
| focus_surface_destroy(ws->fsurf_front); |
| if (ws->fsurf_back) |
| focus_surface_destroy(ws->fsurf_back); |
| |
| free(ws); |
| } |
| |
| static void |
| seat_destroyed(struct wl_listener *listener, void *data) |
| { |
| struct weston_seat *seat = data; |
| struct focus_state *state, *next; |
| struct workspace *ws = container_of(listener, |
| struct workspace, |
| seat_destroyed_listener); |
| |
| wl_list_for_each_safe(state, next, &ws->focus_list, link) |
| if (state->seat == seat) |
| wl_list_remove(&state->link); |
| } |
| |
| static struct workspace * |
| workspace_create(struct desktop_shell *shell) |
| { |
| struct workspace *ws = malloc(sizeof *ws); |
| if (ws == NULL) |
| return NULL; |
| |
| weston_layer_init(&ws->layer, shell->compositor); |
| |
| wl_list_init(&ws->focus_list); |
| wl_list_init(&ws->seat_destroyed_listener.link); |
| ws->seat_destroyed_listener.notify = seat_destroyed; |
| ws->fsurf_front = NULL; |
| ws->fsurf_back = NULL; |
| ws->focus_animation = NULL; |
| |
| return ws; |
| } |
| |
| static int |
| workspace_is_empty(struct workspace *ws) |
| { |
| return wl_list_empty(&ws->layer.view_list.link); |
| } |
| |
| static struct workspace * |
| get_workspace(struct desktop_shell *shell, unsigned int index) |
| { |
| struct workspace **pws = shell->workspaces.array.data; |
| assert(index < shell->workspaces.num); |
| pws += index; |
| return *pws; |
| } |
| |
| struct workspace * |
| get_current_workspace(struct desktop_shell *shell) |
| { |
| return get_workspace(shell, shell->workspaces.current); |
| } |
| |
| static void |
| activate_workspace(struct desktop_shell *shell, unsigned int index) |
| { |
| struct workspace *ws; |
| |
| ws = get_workspace(shell, index); |
| weston_layer_set_position(&ws->layer, WESTON_LAYER_POSITION_NORMAL); |
| |
| shell->workspaces.current = index; |
| } |
| |
| static unsigned int |
| get_output_height(struct weston_output *output) |
| { |
| return abs(output->region.extents.y1 - output->region.extents.y2); |
| } |
| |
| static void |
| view_translate(struct workspace *ws, struct weston_view *view, double d) |
| { |
| struct weston_transform *transform; |
| |
| if (is_focus_view(view)) { |
| struct focus_surface *fsurf = get_focus_surface(view->surface); |
| transform = &fsurf->workspace_transform; |
| } else { |
| struct shell_surface *shsurf = get_shell_surface(view->surface); |
| transform = &shsurf->workspace_transform; |
| } |
| |
| if (wl_list_empty(&transform->link)) |
| wl_list_insert(view->geometry.transformation_list.prev, |
| &transform->link); |
| |
| weston_matrix_init(&transform->matrix); |
| weston_matrix_translate(&transform->matrix, |
| 0.0, d, 0.0); |
| weston_view_geometry_dirty(view); |
| } |
| |
| static void |
| workspace_translate_out(struct workspace *ws, double fraction) |
| { |
| struct weston_view *view; |
| unsigned int height; |
| double d; |
| |
| wl_list_for_each(view, &ws->layer.view_list.link, layer_link.link) { |
| height = get_output_height(view->surface->output); |
| d = height * fraction; |
| |
| view_translate(ws, view, d); |
| } |
| } |
| |
| static void |
| workspace_translate_in(struct workspace *ws, double fraction) |
| { |
| struct weston_view *view; |
| unsigned int height; |
| double d; |
| |
| wl_list_for_each(view, &ws->layer.view_list.link, layer_link.link) { |
| height = get_output_height(view->surface->output); |
| |
| if (fraction > 0) |
| d = -(height - height * fraction); |
| else |
| d = height + height * fraction; |
| |
| view_translate(ws, view, d); |
| } |
| } |
| |
| static void |
| reverse_workspace_change_animation(struct desktop_shell *shell, |
| unsigned int index, |
| struct workspace *from, |
| struct workspace *to) |
| { |
| shell->workspaces.current = index; |
| |
| shell->workspaces.anim_to = to; |
| shell->workspaces.anim_from = from; |
| shell->workspaces.anim_dir = -1 * shell->workspaces.anim_dir; |
| shell->workspaces.anim_timestamp = (struct timespec) { 0 }; |
| |
| weston_layer_set_position(&to->layer, WESTON_LAYER_POSITION_NORMAL); |
| weston_layer_set_position(&from->layer, WESTON_LAYER_POSITION_NORMAL - 1); |
| |
| weston_compositor_schedule_repaint(shell->compositor); |
| } |
| |
| static void |
| workspace_deactivate_transforms(struct workspace *ws) |
| { |
| struct weston_view *view; |
| struct weston_transform *transform; |
| |
| wl_list_for_each(view, &ws->layer.view_list.link, layer_link.link) { |
| if (is_focus_view(view)) { |
| struct focus_surface *fsurf = get_focus_surface(view->surface); |
| transform = &fsurf->workspace_transform; |
| } else { |
| struct shell_surface *shsurf = get_shell_surface(view->surface); |
| transform = &shsurf->workspace_transform; |
| } |
| |
| if (!wl_list_empty(&transform->link)) { |
| wl_list_remove(&transform->link); |
| wl_list_init(&transform->link); |
| } |
| weston_view_geometry_dirty(view); |
| } |
| } |
| |
| static void |
| finish_workspace_change_animation(struct desktop_shell *shell, |
| struct workspace *from, |
| struct workspace *to) |
| { |
| struct weston_view *view; |
| |
| weston_compositor_schedule_repaint(shell->compositor); |
| |
| /* Views that extend past the bottom of the output are still |
| * visible after the workspace animation ends but before its layer |
| * is hidden. In that case, we need to damage below those views so |
| * that the screen is properly repainted. */ |
| wl_list_for_each(view, &from->layer.view_list.link, layer_link.link) |
| weston_view_damage_below(view); |
| |
| wl_list_remove(&shell->workspaces.animation.link); |
| workspace_deactivate_transforms(from); |
| workspace_deactivate_transforms(to); |
| shell->workspaces.anim_to = NULL; |
| |
| weston_layer_unset_position(&shell->workspaces.anim_from->layer); |
| } |
| |
| static void |
| animate_workspace_change_frame(struct weston_animation *animation, |
| struct weston_output *output, |
| const struct timespec *time) |
| { |
| struct desktop_shell *shell = |
| container_of(animation, struct desktop_shell, |
| workspaces.animation); |
| struct workspace *from = shell->workspaces.anim_from; |
| struct workspace *to = shell->workspaces.anim_to; |
| int64_t t; |
| double x, y; |
| |
| if (workspace_is_empty(from) && workspace_is_empty(to)) { |
| finish_workspace_change_animation(shell, from, to); |
| return; |
| } |
| |
| if (timespec_is_zero(&shell->workspaces.anim_timestamp)) { |
| if (shell->workspaces.anim_current == 0.0) |
| shell->workspaces.anim_timestamp = *time; |
| else |
| timespec_add_msec(&shell->workspaces.anim_timestamp, |
| time, |
| /* Invers of movement function 'y' below. */ |
| -(asin(1.0 - shell->workspaces.anim_current) * |
| DEFAULT_WORKSPACE_CHANGE_ANIMATION_LENGTH * |
| M_2_PI)); |
| } |
| |
| t = timespec_sub_to_msec(time, &shell->workspaces.anim_timestamp); |
| |
| /* |
| * x = [0, π/2] |
| * y(x) = sin(x) |
| */ |
| x = t * (1.0/DEFAULT_WORKSPACE_CHANGE_ANIMATION_LENGTH) * M_PI_2; |
| y = sin(x); |
| |
| if (t < DEFAULT_WORKSPACE_CHANGE_ANIMATION_LENGTH) { |
| weston_compositor_schedule_repaint(shell->compositor); |
| |
| workspace_translate_out(from, shell->workspaces.anim_dir * y); |
| workspace_translate_in(to, shell->workspaces.anim_dir * y); |
| shell->workspaces.anim_current = y; |
| |
| weston_compositor_schedule_repaint(shell->compositor); |
| } |
| else |
| finish_workspace_change_animation(shell, from, to); |
| } |
| |
| static void |
| animate_workspace_change(struct desktop_shell *shell, |
| unsigned int index, |
| struct workspace *from, |
| struct workspace *to) |
| { |
| struct weston_output *output; |
| |
| int dir; |
| |
| if (index > shell->workspaces.current) |
| dir = -1; |
| else |
| dir = 1; |
| |
| shell->workspaces.current = index; |
| |
| shell->workspaces.anim_dir = dir; |
| shell->workspaces.anim_from = from; |
| shell->workspaces.anim_to = to; |
| shell->workspaces.anim_current = 0.0; |
| shell->workspaces.anim_timestamp = (struct timespec) { 0 }; |
| |
| output = container_of(shell->compositor->output_list.next, |
| struct weston_output, link); |
| wl_list_insert(&output->animation_list, |
| &shell->workspaces.animation.link); |
| |
| weston_layer_set_position(&to->layer, WESTON_LAYER_POSITION_NORMAL); |
| weston_layer_set_position(&from->layer, WESTON_LAYER_POSITION_NORMAL - 1); |
| |
| workspace_translate_in(to, 0); |
| |
| restore_focus_state(shell, to); |
| |
| weston_compositor_schedule_repaint(shell->compositor); |
| } |
| |
| static void |
| update_workspace(struct desktop_shell *shell, unsigned int index, |
| struct workspace *from, struct workspace *to) |
| { |
| shell->workspaces.current = index; |
| weston_layer_set_position(&to->layer, WESTON_LAYER_POSITION_NORMAL); |
| weston_layer_unset_position(&from->layer); |
| } |
| |
| static void |
| change_workspace(struct desktop_shell *shell, unsigned int index) |
| { |
| struct workspace *from; |
| struct workspace *to; |
| struct focus_state *state; |
| |
| if (index == shell->workspaces.current) |
| return; |
| |
| /* Don't change workspace when there is any fullscreen surfaces. */ |
| if (!wl_list_empty(&shell->fullscreen_layer.view_list.link)) |
| return; |
| |
| from = get_current_workspace(shell); |
| to = get_workspace(shell, index); |
| |
| if (shell->workspaces.anim_from == to && |
| shell->workspaces.anim_to == from) { |
| restore_focus_state(shell, to); |
| reverse_workspace_change_animation(shell, index, from, to); |
| return; |
| } |
| |
| if (shell->workspaces.anim_to != NULL) |
| finish_workspace_change_animation(shell, |
| shell->workspaces.anim_from, |
| shell->workspaces.anim_to); |
| |
| restore_focus_state(shell, to); |
| |
| if (shell->focus_animation_type != ANIMATION_NONE) { |
| wl_list_for_each(state, &from->focus_list, link) |
| if (state->keyboard_focus) |
| animate_focus_change(shell, from, |
| get_default_view(state->keyboard_focus), NULL); |
| |
| wl_list_for_each(state, &to->focus_list, link) |
| if (state->keyboard_focus) |
| animate_focus_change(shell, to, |
| NULL, get_default_view(state->keyboard_focus)); |
| } |
| |
| if (workspace_is_empty(to) && workspace_is_empty(from)) |
| update_workspace(shell, index, from, to); |
| else |
| animate_workspace_change(shell, index, from, to); |
| } |
| |
| static bool |
| workspace_has_only(struct workspace *ws, struct weston_surface *surface) |
| { |
| struct wl_list *list = &ws->layer.view_list.link; |
| struct wl_list *e; |
| |
| if (wl_list_empty(list)) |
| return false; |
| |
| e = list->next; |
| |
| if (e->next != list) |
| return false; |
| |
| return container_of(e, struct weston_view, layer_link.link)->surface == surface; |
| } |
| |
| static void |
| surface_keyboard_focus_lost(struct weston_surface *surface) |
| { |
| struct weston_compositor *compositor = surface->compositor; |
| struct weston_seat *seat; |
| struct weston_surface *focus; |
| |
| wl_list_for_each(seat, &compositor->seat_list, link) { |
| struct weston_keyboard *keyboard = |
| weston_seat_get_keyboard(seat); |
| |
| if (!keyboard) |
| continue; |
| |
| focus = weston_surface_get_main_surface(keyboard->focus); |
| if (focus == surface) |
| weston_keyboard_set_focus(keyboard, NULL); |
| } |
| } |
| |
| static void |
| take_surface_to_workspace_by_seat(struct desktop_shell *shell, |
| struct weston_seat *seat, |
| unsigned int index) |
| { |
| struct weston_keyboard *keyboard = weston_seat_get_keyboard(seat); |
| struct weston_surface *surface; |
| struct weston_view *view; |
| struct shell_surface *shsurf; |
| struct workspace *from; |
| struct workspace *to; |
| struct focus_state *state; |
| |
| surface = weston_surface_get_main_surface(keyboard->focus); |
| view = get_default_view(surface); |
| if (view == NULL || |
| index == shell->workspaces.current || |
| is_focus_view(view)) |
| return; |
| |
| from = get_current_workspace(shell); |
| to = get_workspace(shell, index); |
| |
| weston_layer_entry_remove(&view->layer_link); |
| weston_layer_entry_insert(&to->layer.view_list, &view->layer_link); |
| |
| shsurf = get_shell_surface(surface); |
| if (shsurf != NULL) |
| shell_surface_update_child_surface_layers(shsurf); |
| |
| replace_focus_state(shell, to, seat); |
| drop_focus_state(shell, from, surface); |
| |
| if (shell->workspaces.anim_from == to && |
| shell->workspaces.anim_to == from) { |
| reverse_workspace_change_animation(shell, index, from, to); |
| |
| return; |
| } |
| |
| if (shell->workspaces.anim_to != NULL) |
| finish_workspace_change_animation(shell, |
| shell->workspaces.anim_from, |
| shell->workspaces.anim_to); |
| |
| if (workspace_is_empty(from) && |
| workspace_has_only(to, surface)) |
| update_workspace(shell, index, from, to); |
| else { |
| if (shsurf != NULL && |
| wl_list_empty(&shsurf->workspace_transform.link)) |
| wl_list_insert(&shell->workspaces.anim_sticky_list, |
| &shsurf->workspace_transform.link); |
| |
| animate_workspace_change(shell, index, from, to); |
| } |
| |
| state = ensure_focus_state(shell, seat); |
| if (state != NULL) |
| focus_state_set_focus(state, surface); |
| } |
| |
| static void |
| touch_move_grab_down(struct weston_touch_grab *grab, |
| const struct timespec *time, |
| int touch_id, wl_fixed_t x, wl_fixed_t y) |
| { |
| } |
| |
| static void |
| touch_move_grab_up(struct weston_touch_grab *grab, const struct timespec *time, |
| int touch_id) |
| { |
| struct weston_touch_move_grab *move = |
| (struct weston_touch_move_grab *) container_of( |
| grab, struct shell_touch_grab, grab); |
| |
| if (touch_id == 0) |
| move->active = 0; |
| |
| if (grab->touch->num_tp == 0) { |
| shell_touch_grab_end(&move->base); |
| free(move); |
| } |
| } |
| |
| static void |
| touch_move_grab_motion(struct weston_touch_grab *grab, |
| const struct timespec *time, int touch_id, |
| wl_fixed_t x, wl_fixed_t y) |
| { |
| struct weston_touch_move_grab *move = (struct weston_touch_move_grab *) grab; |
| struct shell_surface *shsurf = move->base.shsurf; |
| struct weston_surface *es; |
| int dx = wl_fixed_to_int(grab->touch->grab_x + move->dx); |
| int dy = wl_fixed_to_int(grab->touch->grab_y + move->dy); |
| |
| if (!shsurf || !move->active) |
| return; |
| |
| es = weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| |
| weston_view_set_position(shsurf->view, dx, dy); |
| |
| weston_compositor_schedule_repaint(es->compositor); |
| } |
| |
| static void |
| touch_move_grab_frame(struct weston_touch_grab *grab) |
| { |
| } |
| |
| static void |
| touch_move_grab_cancel(struct weston_touch_grab *grab) |
| { |
| struct weston_touch_move_grab *move = |
| (struct weston_touch_move_grab *) container_of( |
| grab, struct shell_touch_grab, grab); |
| |
| shell_touch_grab_end(&move->base); |
| free(move); |
| } |
| |
| static const struct weston_touch_grab_interface touch_move_grab_interface = { |
| touch_move_grab_down, |
| touch_move_grab_up, |
| touch_move_grab_motion, |
| touch_move_grab_frame, |
| touch_move_grab_cancel, |
| }; |
| |
| static int |
| surface_touch_move(struct shell_surface *shsurf, struct weston_touch *touch) |
| { |
| struct weston_touch_move_grab *move; |
| |
| if (!shsurf) |
| return -1; |
| |
| if (weston_desktop_surface_get_fullscreen(shsurf->desktop_surface) || |
| weston_desktop_surface_get_maximized(shsurf->desktop_surface)) |
| return 0; |
| |
| move = malloc(sizeof *move); |
| if (!move) |
| return -1; |
| |
| move->active = 1; |
| move->dx = wl_fixed_from_double(shsurf->view->geometry.x) - |
| touch->grab_x; |
| move->dy = wl_fixed_from_double(shsurf->view->geometry.y) - |
| touch->grab_y; |
| |
| shell_touch_grab_start(&move->base, &touch_move_grab_interface, shsurf, |
| touch); |
| |
| return 0; |
| } |
| |
| static void |
| noop_grab_focus(struct weston_pointer_grab *grab) |
| { |
| } |
| |
| static void |
| noop_grab_axis(struct weston_pointer_grab *grab, |
| const struct timespec *time, |
| struct weston_pointer_axis_event *event) |
| { |
| } |
| |
| static void |
| noop_grab_axis_source(struct weston_pointer_grab *grab, |
| uint32_t source) |
| { |
| } |
| |
| static void |
| noop_grab_frame(struct weston_pointer_grab *grab) |
| { |
| } |
| |
| static void |
| constrain_position(struct weston_move_grab *move, int *cx, int *cy) |
| { |
| struct shell_surface *shsurf = move->base.shsurf; |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| struct weston_pointer *pointer = move->base.grab.pointer; |
| int x, y, bottom; |
| const int safety = 50; |
| pixman_rectangle32_t area; |
| struct weston_geometry geometry; |
| |
| x = wl_fixed_to_int(pointer->x + move->dx); |
| y = wl_fixed_to_int(pointer->y + move->dy); |
| |
| if (shsurf->shell->panel_position == |
| WESTON_DESKTOP_SHELL_PANEL_POSITION_TOP) { |
| get_output_work_area(shsurf->shell, surface->output, &area); |
| geometry = |
| weston_desktop_surface_get_geometry(shsurf->desktop_surface); |
| |
| bottom = y + geometry.height + geometry.y; |
| if (bottom - safety < area.y) |
| y = area.y + safety - geometry.height |
| - geometry.y; |
| |
| if (move->client_initiated && |
| y + geometry.y < area.y) |
| y = area.y - geometry.y; |
| } |
| |
| *cx = x; |
| *cy = y; |
| } |
| |
| static void |
| move_grab_motion(struct weston_pointer_grab *grab, |
| const struct timespec *time, |
| struct weston_pointer_motion_event *event) |
| { |
| struct weston_move_grab *move = (struct weston_move_grab *) grab; |
| struct weston_pointer *pointer = grab->pointer; |
| struct shell_surface *shsurf = move->base.shsurf; |
| struct weston_surface *surface; |
| int cx, cy; |
| |
| weston_pointer_move(pointer, event); |
| if (!shsurf) |
| return; |
| |
| surface = weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| |
| constrain_position(move, &cx, &cy); |
| |
| weston_view_set_position(shsurf->view, cx, cy); |
| |
| weston_compositor_schedule_repaint(surface->compositor); |
| } |
| |
| static void |
| move_grab_button(struct weston_pointer_grab *grab, |
| const struct timespec *time, uint32_t button, uint32_t state_w) |
| { |
| struct shell_grab *shell_grab = container_of(grab, struct shell_grab, |
| grab); |
| struct weston_pointer *pointer = grab->pointer; |
| enum wl_pointer_button_state state = state_w; |
| |
| if (pointer->button_count == 0 && |
| state == WL_POINTER_BUTTON_STATE_RELEASED) { |
| shell_grab_end(shell_grab); |
| free(grab); |
| } |
| } |
| |
| static void |
| move_grab_cancel(struct weston_pointer_grab *grab) |
| { |
| struct shell_grab *shell_grab = |
| container_of(grab, struct shell_grab, grab); |
| |
| shell_grab_end(shell_grab); |
| free(grab); |
| } |
| |
| static const struct weston_pointer_grab_interface move_grab_interface = { |
| noop_grab_focus, |
| move_grab_motion, |
| move_grab_button, |
| noop_grab_axis, |
| noop_grab_axis_source, |
| noop_grab_frame, |
| move_grab_cancel, |
| }; |
| |
| static int |
| surface_move(struct shell_surface *shsurf, struct weston_pointer *pointer, |
| bool client_initiated) |
| { |
| struct weston_move_grab *move; |
| |
| if (!shsurf) |
| return -1; |
| |
| if (shsurf->grabbed || |
| weston_desktop_surface_get_fullscreen(shsurf->desktop_surface) || |
| weston_desktop_surface_get_maximized(shsurf->desktop_surface)) |
| return 0; |
| |
| move = malloc(sizeof *move); |
| if (!move) |
| return -1; |
| |
| move->dx = wl_fixed_from_double(shsurf->view->geometry.x) - |
| pointer->grab_x; |
| move->dy = wl_fixed_from_double(shsurf->view->geometry.y) - |
| pointer->grab_y; |
| move->client_initiated = client_initiated; |
| |
| shell_grab_start(&move->base, &move_grab_interface, shsurf, |
| pointer, WESTON_DESKTOP_SHELL_CURSOR_MOVE); |
| |
| return 0; |
| } |
| |
| struct weston_resize_grab { |
| struct shell_grab base; |
| uint32_t edges; |
| int32_t width, height; |
| }; |
| |
| static void |
| resize_grab_motion(struct weston_pointer_grab *grab, |
| const struct timespec *time, |
| struct weston_pointer_motion_event *event) |
| { |
| struct weston_resize_grab *resize = (struct weston_resize_grab *) grab; |
| struct weston_pointer *pointer = grab->pointer; |
| struct shell_surface *shsurf = resize->base.shsurf; |
| int32_t width, height; |
| struct weston_size min_size, max_size; |
| wl_fixed_t from_x, from_y; |
| wl_fixed_t to_x, to_y; |
| |
| weston_pointer_move(pointer, event); |
| |
| if (!shsurf) |
| return; |
| |
| weston_view_from_global_fixed(shsurf->view, |
| pointer->grab_x, pointer->grab_y, |
| &from_x, &from_y); |
| weston_view_from_global_fixed(shsurf->view, |
| pointer->x, pointer->y, &to_x, &to_y); |
| |
| width = resize->width; |
| if (resize->edges & WL_SHELL_SURFACE_RESIZE_LEFT) { |
| width += wl_fixed_to_int(from_x - to_x); |
| } else if (resize->edges & WL_SHELL_SURFACE_RESIZE_RIGHT) { |
| width += wl_fixed_to_int(to_x - from_x); |
| } |
| |
| height = resize->height; |
| if (resize->edges & WL_SHELL_SURFACE_RESIZE_TOP) { |
| height += wl_fixed_to_int(from_y - to_y); |
| } else if (resize->edges & WL_SHELL_SURFACE_RESIZE_BOTTOM) { |
| height += wl_fixed_to_int(to_y - from_y); |
| } |
| |
| max_size = weston_desktop_surface_get_max_size(shsurf->desktop_surface); |
| min_size = weston_desktop_surface_get_min_size(shsurf->desktop_surface); |
| |
| min_size.width = MAX(1, min_size.width); |
| min_size.height = MAX(1, min_size.height); |
| |
| if (width < min_size.width) |
| width = min_size.width; |
| else if (max_size.width > 0 && width > max_size.width) |
| width = max_size.width; |
| if (height < min_size.height) |
| height = min_size.height; |
| else if (max_size.width > 0 && width > max_size.width) |
| width = max_size.width; |
| weston_desktop_surface_set_size(shsurf->desktop_surface, width, height); |
| } |
| |
| static void |
| resize_grab_button(struct weston_pointer_grab *grab, |
| const struct timespec *time, |
| uint32_t button, uint32_t state_w) |
| { |
| struct weston_resize_grab *resize = (struct weston_resize_grab *) grab; |
| struct weston_pointer *pointer = grab->pointer; |
| enum wl_pointer_button_state state = state_w; |
| struct weston_desktop_surface *desktop_surface = |
| resize->base.shsurf->desktop_surface; |
| |
| if (pointer->button_count == 0 && |
| state == WL_POINTER_BUTTON_STATE_RELEASED) { |
| weston_desktop_surface_set_resizing(desktop_surface, false); |
| shell_grab_end(&resize->base); |
| free(grab); |
| } |
| } |
| |
| static void |
| resize_grab_cancel(struct weston_pointer_grab *grab) |
| { |
| struct weston_resize_grab *resize = (struct weston_resize_grab *) grab; |
| struct weston_desktop_surface *desktop_surface = |
| resize->base.shsurf->desktop_surface; |
| |
| weston_desktop_surface_set_resizing(desktop_surface, false); |
| shell_grab_end(&resize->base); |
| free(grab); |
| } |
| |
| static const struct weston_pointer_grab_interface resize_grab_interface = { |
| noop_grab_focus, |
| resize_grab_motion, |
| resize_grab_button, |
| noop_grab_axis, |
| noop_grab_axis_source, |
| noop_grab_frame, |
| resize_grab_cancel, |
| }; |
| |
| /* |
| * Returns the bounding box of a surface and all its sub-surfaces, |
| * in surface-local coordinates. */ |
| static void |
| surface_subsurfaces_boundingbox(struct weston_surface *surface, int32_t *x, |
| int32_t *y, int32_t *w, int32_t *h) { |
| pixman_region32_t region; |
| pixman_box32_t *box; |
| struct weston_subsurface *subsurface; |
| |
| pixman_region32_init_rect(®ion, 0, 0, |
| surface->width, |
| surface->height); |
| |
| wl_list_for_each(subsurface, &surface->subsurface_list, parent_link) { |
| pixman_region32_union_rect(®ion, ®ion, |
| subsurface->position.x, |
| subsurface->position.y, |
| subsurface->surface->width, |
| subsurface->surface->height); |
| } |
| |
| box = pixman_region32_extents(®ion); |
| if (x) |
| *x = box->x1; |
| if (y) |
| *y = box->y1; |
| if (w) |
| *w = box->x2 - box->x1; |
| if (h) |
| *h = box->y2 - box->y1; |
| |
| pixman_region32_fini(®ion); |
| } |
| |
| static int |
| surface_resize(struct shell_surface *shsurf, |
| struct weston_pointer *pointer, uint32_t edges) |
| { |
| struct weston_resize_grab *resize; |
| const unsigned resize_topbottom = |
| WL_SHELL_SURFACE_RESIZE_TOP | WL_SHELL_SURFACE_RESIZE_BOTTOM; |
| const unsigned resize_leftright = |
| WL_SHELL_SURFACE_RESIZE_LEFT | WL_SHELL_SURFACE_RESIZE_RIGHT; |
| const unsigned resize_any = resize_topbottom | resize_leftright; |
| struct weston_geometry geometry; |
| |
| if (shsurf->grabbed || |
| weston_desktop_surface_get_fullscreen(shsurf->desktop_surface) || |
| weston_desktop_surface_get_maximized(shsurf->desktop_surface)) |
| return 0; |
| |
| /* Check for invalid edge combinations. */ |
| if (edges == WL_SHELL_SURFACE_RESIZE_NONE || edges > resize_any || |
| (edges & resize_topbottom) == resize_topbottom || |
| (edges & resize_leftright) == resize_leftright) |
| return 0; |
| |
| resize = malloc(sizeof *resize); |
| if (!resize) |
| return -1; |
| |
| resize->edges = edges; |
| |
| geometry = weston_desktop_surface_get_geometry(shsurf->desktop_surface); |
| resize->width = geometry.width; |
| resize->height = geometry.height; |
| |
| shsurf->resize_edges = edges; |
| weston_desktop_surface_set_resizing(shsurf->desktop_surface, true); |
| shell_grab_start(&resize->base, &resize_grab_interface, shsurf, |
| pointer, edges); |
| |
| return 0; |
| } |
| |
| static void |
| busy_cursor_grab_focus(struct weston_pointer_grab *base) |
| { |
| struct shell_grab *grab = (struct shell_grab *) base; |
| struct weston_pointer *pointer = base->pointer; |
| struct weston_desktop_surface *desktop_surface; |
| struct weston_view *view; |
| wl_fixed_t sx, sy; |
| |
| view = weston_compositor_pick_view(pointer->seat->compositor, |
| pointer->x, pointer->y, |
| &sx, &sy); |
| desktop_surface = weston_surface_get_desktop_surface(view->surface); |
| |
| if (!grab->shsurf || grab->shsurf->desktop_surface != desktop_surface) { |
| shell_grab_end(grab); |
| free(grab); |
| } |
| } |
| |
| static void |
| busy_cursor_grab_motion(struct weston_pointer_grab *grab, |
| const struct timespec *time, |
| struct weston_pointer_motion_event *event) |
| { |
| weston_pointer_move(grab->pointer, event); |
| } |
| |
| static void |
| busy_cursor_grab_button(struct weston_pointer_grab *base, |
| const struct timespec *time, |
| uint32_t button, uint32_t state) |
| { |
| struct shell_grab *grab = (struct shell_grab *) base; |
| struct shell_surface *shsurf = grab->shsurf; |
| struct weston_pointer *pointer = grab->grab.pointer; |
| struct weston_seat *seat = pointer->seat; |
| |
| if (shsurf && button == BTN_LEFT && state) { |
| activate(shsurf->shell, shsurf->view, seat, |
| WESTON_ACTIVATE_FLAG_CONFIGURE); |
| surface_move(shsurf, pointer, false); |
| } else if (shsurf && button == BTN_RIGHT && state) { |
| activate(shsurf->shell, shsurf->view, seat, |
| WESTON_ACTIVATE_FLAG_CONFIGURE); |
| surface_rotate(shsurf, pointer); |
| } |
| } |
| |
| static void |
| busy_cursor_grab_cancel(struct weston_pointer_grab *base) |
| { |
| struct shell_grab *grab = (struct shell_grab *) base; |
| |
| shell_grab_end(grab); |
| free(grab); |
| } |
| |
| static const struct weston_pointer_grab_interface busy_cursor_grab_interface = { |
| busy_cursor_grab_focus, |
| busy_cursor_grab_motion, |
| busy_cursor_grab_button, |
| noop_grab_axis, |
| noop_grab_axis_source, |
| noop_grab_frame, |
| busy_cursor_grab_cancel, |
| }; |
| |
| static void |
| handle_pointer_focus(struct wl_listener *listener, void *data) |
| { |
| struct weston_pointer *pointer = data; |
| struct weston_view *view = pointer->focus; |
| struct shell_surface *shsurf; |
| struct weston_desktop_client *client; |
| |
| if (!view) |
| return; |
| |
| shsurf = get_shell_surface(view->surface); |
| if (!shsurf) |
| return; |
| |
| client = weston_desktop_surface_get_client(shsurf->desktop_surface); |
| |
| if (shsurf->unresponsive) |
| set_busy_cursor(shsurf, pointer); |
| else |
| weston_desktop_client_ping(client); |
| } |
| |
| static void |
| shell_surface_lose_keyboard_focus(struct shell_surface *shsurf) |
| { |
| if (--shsurf->focus_count == 0) |
| weston_desktop_surface_set_activated(shsurf->desktop_surface, false); |
| } |
| |
| static void |
| shell_surface_gain_keyboard_focus(struct shell_surface *shsurf) |
| { |
| if (shsurf->focus_count++ == 0) |
| weston_desktop_surface_set_activated(shsurf->desktop_surface, true); |
| } |
| |
| static void |
| handle_keyboard_focus(struct wl_listener *listener, void *data) |
| { |
| struct weston_keyboard *keyboard = data; |
| struct shell_seat *seat = get_shell_seat(keyboard->seat); |
| |
| if (seat->focused_surface) { |
| struct shell_surface *shsurf = get_shell_surface(seat->focused_surface); |
| if (shsurf) |
| shell_surface_lose_keyboard_focus(shsurf); |
| } |
| |
| seat->focused_surface = weston_surface_get_main_surface(keyboard->focus); |
| |
| if (seat->focused_surface) { |
| struct shell_surface *shsurf = get_shell_surface(seat->focused_surface); |
| if (shsurf) |
| shell_surface_gain_keyboard_focus(shsurf); |
| } |
| } |
| |
| /* The surface will be inserted into the list immediately after the link |
| * returned by this function (i.e. will be stacked immediately above the |
| * returned link). */ |
| static struct weston_layer_entry * |
| shell_surface_calculate_layer_link (struct shell_surface *shsurf) |
| { |
| struct workspace *ws; |
| |
| if (weston_desktop_surface_get_fullscreen(shsurf->desktop_surface) && |
| !shsurf->state.lowered) { |
| return &shsurf->shell->fullscreen_layer.view_list; |
| } |
| |
| /* Move the surface to a normal workspace layer so that surfaces |
| * which were previously fullscreen or transient are no longer |
| * rendered on top. */ |
| ws = get_current_workspace(shsurf->shell); |
| return &ws->layer.view_list; |
| } |
| |
| static void |
| shell_surface_update_child_surface_layers (struct shell_surface *shsurf) |
| { |
| weston_desktop_surface_propagate_layer(shsurf->desktop_surface); |
| } |
| |
| /* Update the surface’s layer. Mark both the old and new views as having dirty |
| * geometry to ensure the changes are redrawn. |
| * |
| * If any child surfaces exist and are mapped, ensure they’re in the same layer |
| * as this surface. */ |
| static void |
| shell_surface_update_layer(struct shell_surface *shsurf) |
| { |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| struct weston_layer_entry *new_layer_link; |
| |
| new_layer_link = shell_surface_calculate_layer_link(shsurf); |
| |
| if (new_layer_link == NULL) |
| return; |
| if (new_layer_link == &shsurf->view->layer_link) |
| return; |
| |
| weston_view_geometry_dirty(shsurf->view); |
| weston_layer_entry_remove(&shsurf->view->layer_link); |
| weston_layer_entry_insert(new_layer_link, &shsurf->view->layer_link); |
| weston_view_geometry_dirty(shsurf->view); |
| weston_surface_damage(surface); |
| |
| shell_surface_update_child_surface_layers(shsurf); |
| } |
| |
| static void |
| shell_surface_set_output(struct shell_surface *shsurf, |
| struct weston_output *output) |
| { |
| struct weston_surface *es = |
| weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| |
| /* get the default output, if the client set it as NULL |
| check whether the output is available */ |
| if (output) |
| shsurf->output = output; |
| else if (es->output) |
| shsurf->output = es->output; |
| else |
| shsurf->output = get_default_output(es->compositor); |
| } |
| |
| static void |
| weston_view_set_initial_position(struct weston_view *view, |
| struct desktop_shell *shell); |
| |
| static void |
| unset_fullscreen(struct shell_surface *shsurf) |
| { |
| /* Unset the fullscreen output, driver configuration and transforms. */ |
| wl_list_remove(&shsurf->fullscreen.transform.link); |
| wl_list_init(&shsurf->fullscreen.transform.link); |
| |
| if (shsurf->fullscreen.black_view) |
| weston_surface_destroy(shsurf->fullscreen.black_view->surface); |
| shsurf->fullscreen.black_view = NULL; |
| |
| if (shsurf->saved_position_valid) |
| weston_view_set_position(shsurf->view, |
| shsurf->saved_x, shsurf->saved_y); |
| else |
| weston_view_set_initial_position(shsurf->view, shsurf->shell); |
| shsurf->saved_position_valid = false; |
| |
| if (shsurf->saved_rotation_valid) { |
| wl_list_insert(&shsurf->view->geometry.transformation_list, |
| &shsurf->rotation.transform.link); |
| shsurf->saved_rotation_valid = false; |
| } |
| } |
| |
| static void |
| unset_maximized(struct shell_surface *shsurf) |
| { |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| |
| /* undo all maximized things here */ |
| shsurf->output = get_default_output(surface->compositor); |
| |
| if (shsurf->saved_position_valid) |
| weston_view_set_position(shsurf->view, |
| shsurf->saved_x, shsurf->saved_y); |
| else |
| weston_view_set_initial_position(shsurf->view, shsurf->shell); |
| shsurf->saved_position_valid = false; |
| |
| if (shsurf->saved_rotation_valid) { |
| wl_list_insert(&shsurf->view->geometry.transformation_list, |
| &shsurf->rotation.transform.link); |
| shsurf->saved_rotation_valid = false; |
| } |
| } |
| |
| static void |
| set_minimized(struct weston_surface *surface) |
| { |
| struct shell_surface *shsurf; |
| struct workspace *current_ws; |
| struct weston_view *view; |
| |
| view = get_default_view(surface); |
| if (!view) |
| return; |
| |
| assert(weston_surface_get_main_surface(view->surface) == view->surface); |
| |
| shsurf = get_shell_surface(surface); |
| current_ws = get_current_workspace(shsurf->shell); |
| |
| weston_layer_entry_remove(&view->layer_link); |
| weston_layer_entry_insert(&shsurf->shell->minimized_layer.view_list, &view->layer_link); |
| |
| drop_focus_state(shsurf->shell, current_ws, view->surface); |
| surface_keyboard_focus_lost(surface); |
| |
| shell_surface_update_child_surface_layers(shsurf); |
| weston_view_damage_below(view); |
| } |
| |
| |
| static struct desktop_shell * |
| shell_surface_get_shell(struct shell_surface *shsurf) |
| { |
| return shsurf->shell; |
| } |
| |
| static int |
| black_surface_get_label(struct weston_surface *surface, char *buf, size_t len) |
| { |
| struct weston_view *fs_view = surface->committed_private; |
| struct weston_surface *fs_surface = fs_view->surface; |
| int n; |
| int rem; |
| int ret; |
| |
| n = snprintf(buf, len, "black background surface for "); |
| if (n < 0) |
| return n; |
| |
| rem = (int)len - n; |
| if (rem < 0) |
| rem = 0; |
| |
| if (fs_surface->get_label) |
| ret = fs_surface->get_label(fs_surface, buf + n, rem); |
| else |
| ret = snprintf(buf + n, rem, "<unknown>"); |
| |
| if (ret < 0) |
| return n; |
| |
| return n + ret; |
| } |
| |
| static void |
| black_surface_committed(struct weston_surface *es, int32_t sx, int32_t sy); |
| |
| static struct weston_view * |
| create_black_surface(struct weston_compositor *ec, |
| struct weston_view *fs_view, |
| float x, float y, int w, int h) |
| { |
| struct weston_surface *surface = NULL; |
| struct weston_view *view; |
| |
| surface = weston_surface_create(ec); |
| if (surface == NULL) { |
| weston_log("no memory\n"); |
| return NULL; |
| } |
| view = weston_view_create(surface); |
| if (surface == NULL) { |
| weston_log("no memory\n"); |
| weston_surface_destroy(surface); |
| return NULL; |
| } |
| |
| surface->committed = black_surface_committed; |
| surface->committed_private = fs_view; |
| weston_surface_set_label_func(surface, black_surface_get_label); |
| weston_surface_set_color(surface, 0.0, 0.0, 0.0, 1); |
| pixman_region32_fini(&surface->opaque); |
| pixman_region32_init_rect(&surface->opaque, 0, 0, w, h); |
| pixman_region32_fini(&surface->input); |
| pixman_region32_init_rect(&surface->input, 0, 0, w, h); |
| |
| weston_surface_set_size(surface, w, h); |
| weston_view_set_position(view, x, y); |
| |
| return view; |
| } |
| |
| static void |
| shell_ensure_fullscreen_black_view(struct shell_surface *shsurf) |
| { |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| struct weston_output *output = shsurf->fullscreen_output; |
| |
| assert(weston_desktop_surface_get_fullscreen(shsurf->desktop_surface)); |
| |
| if (!shsurf->fullscreen.black_view) |
| shsurf->fullscreen.black_view = |
| create_black_surface(surface->compositor, |
| shsurf->view, |
| output->x, output->y, |
| output->width, |
| output->height); |
| |
| weston_view_geometry_dirty(shsurf->fullscreen.black_view); |
| weston_layer_entry_remove(&shsurf->fullscreen.black_view->layer_link); |
| weston_layer_entry_insert(&shsurf->view->layer_link, |
| &shsurf->fullscreen.black_view->layer_link); |
| weston_view_geometry_dirty(shsurf->fullscreen.black_view); |
| weston_surface_damage(surface); |
| |
| shsurf->fullscreen.black_view->is_mapped = true; |
| shsurf->state.lowered = false; |
| } |
| |
| /* Create black surface and append it to the associated fullscreen surface. |
| * Handle size dismatch and positioning according to the method. */ |
| static void |
| shell_configure_fullscreen(struct shell_surface *shsurf) |
| { |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| int32_t surf_x, surf_y, surf_width, surf_height; |
| |
| /* Reverse the effect of lower_fullscreen_layer() */ |
| weston_layer_entry_remove(&shsurf->view->layer_link); |
| weston_layer_entry_insert(&shsurf->shell->fullscreen_layer.view_list, |
| &shsurf->view->layer_link); |
| |
| shell_ensure_fullscreen_black_view(shsurf); |
| |
| surface_subsurfaces_boundingbox(surface, &surf_x, &surf_y, |
| &surf_width, &surf_height); |
| |
| if (surface->buffer_ref.buffer) |
| center_on_output(shsurf->view, shsurf->fullscreen_output); |
| } |
| |
| static void |
| shell_map_fullscreen(struct shell_surface *shsurf) |
| { |
| shell_configure_fullscreen(shsurf); |
| } |
| |
| static struct weston_output * |
| get_focused_output(struct weston_compositor *compositor) |
| { |
| struct weston_seat *seat; |
| struct weston_output *output = NULL; |
| |
| wl_list_for_each(seat, &compositor->seat_list, link) { |
| struct weston_touch *touch = weston_seat_get_touch(seat); |
| struct weston_pointer *pointer = weston_seat_get_pointer(seat); |
| struct weston_keyboard *keyboard = |
| weston_seat_get_keyboard(seat); |
| |
| /* Priority has touch focus, then pointer and |
| * then keyboard focus. We should probably have |
| * three for loops and check frist for touch, |
| * then for pointer, etc. but unless somebody has some |
| * objections, I think this is sufficient. */ |
| if (touch && touch->focus) |
| output = touch->focus->output; |
| else if (pointer && pointer->focus) |
| output = pointer->focus->output; |
| else if (keyboard && keyboard->focus) |
| output = keyboard->focus->output; |
| |
| if (output) |
| break; |
| } |
| |
| return output; |
| } |
| |
| static void |
| destroy_shell_seat(struct wl_listener *listener, void *data) |
| { |
| struct shell_seat *shseat = |
| container_of(listener, |
| struct shell_seat, seat_destroy_listener); |
| |
| wl_list_remove(&shseat->seat_destroy_listener.link); |
| free(shseat); |
| } |
| |
| static void |
| shell_seat_caps_changed(struct wl_listener *listener, void *data) |
| { |
| struct weston_keyboard *keyboard; |
| struct weston_pointer *pointer; |
| struct shell_seat *seat; |
| |
| seat = container_of(listener, struct shell_seat, caps_changed_listener); |
| keyboard = weston_seat_get_keyboard(seat->seat); |
| pointer = weston_seat_get_pointer(seat->seat); |
| |
| if (keyboard && |
| wl_list_empty(&seat->keyboard_focus_listener.link)) { |
| wl_signal_add(&keyboard->focus_signal, |
| &seat->keyboard_focus_listener); |
| } else if (!keyboard) { |
| wl_list_remove(&seat->keyboard_focus_listener.link); |
| wl_list_init(&seat->keyboard_focus_listener.link); |
| } |
| |
| if (pointer && |
| wl_list_empty(&seat->pointer_focus_listener.link)) { |
| wl_signal_add(&pointer->focus_signal, |
| &seat->pointer_focus_listener); |
| } else if (!pointer) { |
| wl_list_remove(&seat->pointer_focus_listener.link); |
| wl_list_init(&seat->pointer_focus_listener.link); |
| } |
| } |
| |
| static struct shell_seat * |
| create_shell_seat(struct weston_seat *seat) |
| { |
| struct shell_seat *shseat; |
| |
| shseat = calloc(1, sizeof *shseat); |
| if (!shseat) { |
| weston_log("no memory to allocate shell seat\n"); |
| return NULL; |
| } |
| |
| shseat->seat = seat; |
| |
| shseat->seat_destroy_listener.notify = destroy_shell_seat; |
| wl_signal_add(&seat->destroy_signal, |
| &shseat->seat_destroy_listener); |
| |
| shseat->keyboard_focus_listener.notify = handle_keyboard_focus; |
| wl_list_init(&shseat->keyboard_focus_listener.link); |
| |
| shseat->pointer_focus_listener.notify = handle_pointer_focus; |
| wl_list_init(&shseat->pointer_focus_listener.link); |
| |
| shseat->caps_changed_listener.notify = shell_seat_caps_changed; |
| wl_signal_add(&seat->updated_caps_signal, |
| &shseat->caps_changed_listener); |
| shell_seat_caps_changed(&shseat->caps_changed_listener, NULL); |
| |
| return shseat; |
| } |
| |
| static struct shell_seat * |
| get_shell_seat(struct weston_seat *seat) |
| { |
| struct wl_listener *listener; |
| |
| listener = wl_signal_get(&seat->destroy_signal, destroy_shell_seat); |
| assert(listener != NULL); |
| |
| return container_of(listener, |
| struct shell_seat, seat_destroy_listener); |
| } |
| |
| static void |
| fade_out_done_idle_cb(void *data) |
| { |
| struct shell_surface *shsurf = data; |
| |
| weston_surface_destroy(shsurf->view->surface); |
| free(shsurf); |
| } |
| |
| static void |
| fade_out_done(struct weston_view_animation *animation, void *data) |
| { |
| struct shell_surface *shsurf = data; |
| struct wl_event_loop *loop; |
| |
| loop = wl_display_get_event_loop(shsurf->shell->compositor->wl_display); |
| |
| if (weston_view_is_mapped(shsurf->view)) { |
| shsurf->view->is_mapped = false; |
| wl_event_loop_add_idle(loop, fade_out_done_idle_cb, shsurf); |
| } |
| } |
| |
| struct shell_surface * |
| get_shell_surface(struct weston_surface *surface) |
| { |
| if (weston_surface_is_desktop_surface(surface)) { |
| struct weston_desktop_surface *desktop_surface = |
| weston_surface_get_desktop_surface(surface); |
| return weston_desktop_surface_get_user_data(desktop_surface); |
| } |
| return NULL; |
| } |
| |
| /* |
| * libweston-desktop |
| */ |
| |
| static void |
| desktop_surface_added(struct weston_desktop_surface *desktop_surface, |
| void *shell) |
| { |
| struct weston_desktop_client *client = |
| weston_desktop_surface_get_client(desktop_surface); |
| struct wl_client *wl_client = |
| weston_desktop_client_get_client(client); |
| struct weston_view *view; |
| struct shell_surface *shsurf; |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(desktop_surface); |
| |
| view = weston_desktop_surface_create_view(desktop_surface); |
| if (!view) |
| return; |
| |
| shsurf = calloc(1, sizeof *shsurf); |
| if (!shsurf) { |
| if (wl_client) |
| wl_client_post_no_memory(wl_client); |
| else |
| weston_log("no memory to allocate shell surface\n"); |
| return; |
| } |
| |
| weston_surface_set_label_func(surface, shell_surface_get_label); |
| |
| shsurf->shell = (struct desktop_shell *) shell; |
| shsurf->unresponsive = 0; |
| shsurf->saved_position_valid = false; |
| shsurf->saved_rotation_valid = false; |
| shsurf->desktop_surface = desktop_surface; |
| shsurf->view = view; |
| shsurf->fullscreen.black_view = NULL; |
| wl_list_init(&shsurf->fullscreen.transform.link); |
| |
| shsurf->output = get_default_output(shsurf->shell->compositor); |
| |
| wl_signal_init(&shsurf->destroy_signal); |
| |
| /* empty when not in use */ |
| wl_list_init(&shsurf->rotation.transform.link); |
| weston_matrix_init(&shsurf->rotation.rotation); |
| |
| wl_list_init(&shsurf->workspace_transform.link); |
| |
| weston_desktop_surface_set_user_data(desktop_surface, shsurf); |
| weston_desktop_surface_set_activated(desktop_surface, |
| shsurf->focus_count > 0); |
| } |
| |
| static void |
| desktop_surface_removed(struct weston_desktop_surface *desktop_surface, |
| void *shell) |
| { |
| struct shell_surface *shsurf = |
| weston_desktop_surface_get_user_data(desktop_surface); |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(desktop_surface); |
| |
| if (!shsurf) |
| return; |
| |
| wl_signal_emit(&shsurf->destroy_signal, shsurf); |
| |
| if (shsurf->fullscreen.black_view) |
| weston_surface_destroy(shsurf->fullscreen.black_view->surface); |
| |
| weston_surface_set_label_func(surface, NULL); |
| weston_desktop_surface_set_user_data(shsurf->desktop_surface, NULL); |
| shsurf->desktop_surface = NULL; |
| |
| weston_desktop_surface_unlink_view(shsurf->view); |
| if (weston_surface_is_mapped(surface) && |
| shsurf->shell->win_close_animation_type == ANIMATION_FADE) { |
| pixman_region32_fini(&surface->pending.input); |
| pixman_region32_init(&surface->pending.input); |
| pixman_region32_fini(&surface->input); |
| pixman_region32_init(&surface->input); |
| weston_fade_run(shsurf->view, 1.0, 0.0, 300.0, |
| fade_out_done, shsurf); |
| } else { |
| weston_view_destroy(shsurf->view); |
| free(shsurf); |
| } |
| } |
| |
| static void |
| set_maximized_position(struct desktop_shell *shell, |
| struct shell_surface *shsurf) |
| { |
| pixman_rectangle32_t area; |
| struct weston_geometry geometry; |
| |
| get_output_work_area(shell, shsurf->output, &area); |
| geometry = weston_desktop_surface_get_geometry(shsurf->desktop_surface); |
| |
| weston_view_set_position(shsurf->view, |
| area.x - geometry.x, |
| area.y - geometry.y); |
| } |
| |
| static void |
| set_position_from_xwayland(struct shell_surface *shsurf) |
| { |
| struct weston_geometry geometry; |
| float x; |
| float y; |
| |
| assert(shsurf->xwayland.is_set); |
| |
| geometry = weston_desktop_surface_get_geometry(shsurf->desktop_surface); |
| x = shsurf->xwayland.x - geometry.x; |
| y = shsurf->xwayland.y - geometry.y; |
| |
| weston_view_set_position(shsurf->view, x, y); |
| |
| #ifdef WM_DEBUG |
| weston_log("%s: XWM %d, %d; geometry %d, %d; view %f, %f\n", |
| __func__, shsurf->xwayland.x, shsurf->xwayland.y, |
| geometry.x, geometry.y, x, y); |
| #endif |
| } |
| |
| static void |
| map(struct desktop_shell *shell, struct shell_surface *shsurf, |
| int32_t sx, int32_t sy) |
| { |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| struct weston_compositor *compositor = shell->compositor; |
| struct weston_seat *seat; |
| |
| /* initial positioning, see also configure() */ |
| if (shsurf->state.fullscreen) { |
| center_on_output(shsurf->view, shsurf->fullscreen_output); |
| shell_map_fullscreen(shsurf); |
| } else if (shsurf->state.maximized) { |
| set_maximized_position(shell, shsurf); |
| } else if (shsurf->xwayland.is_set) { |
| set_position_from_xwayland(shsurf); |
| } else { |
| weston_view_set_initial_position(shsurf->view, shell); |
| } |
| |
| /* Surface stacking order, see also activate(). */ |
| shell_surface_update_layer(shsurf); |
| |
| weston_view_update_transform(shsurf->view); |
| shsurf->view->is_mapped = true; |
| if (shsurf->state.maximized) { |
| surface->output = shsurf->output; |
| weston_view_set_output(shsurf->view, shsurf->output); |
| } |
| |
| if (!shell->locked) { |
| wl_list_for_each(seat, &compositor->seat_list, link) |
| activate(shell, shsurf->view, seat, |
| WESTON_ACTIVATE_FLAG_CONFIGURE); |
| } |
| |
| if (!shsurf->state.fullscreen && !shsurf->state.maximized) { |
| switch (shell->win_animation_type) { |
| case ANIMATION_FADE: |
| weston_fade_run(shsurf->view, 0.0, 1.0, 300.0, NULL, NULL); |
| break; |
| case ANIMATION_ZOOM: |
| weston_zoom_run(shsurf->view, 0.5, 1.0, NULL, NULL); |
| break; |
| case ANIMATION_NONE: |
| default: |
| break; |
| } |
| } |
| } |
| |
| static void |
| desktop_surface_committed(struct weston_desktop_surface *desktop_surface, |
| int32_t sx, int32_t sy, void *data) |
| { |
| struct shell_surface *shsurf = |
| weston_desktop_surface_get_user_data(desktop_surface); |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(desktop_surface); |
| struct weston_view *view = shsurf->view; |
| struct desktop_shell *shell = data; |
| bool was_fullscreen; |
| bool was_maximized; |
| |
| if (surface->width == 0) |
| return; |
| |
| was_fullscreen = shsurf->state.fullscreen; |
| was_maximized = shsurf->state.maximized; |
| |
| shsurf->state.fullscreen = |
| weston_desktop_surface_get_fullscreen(desktop_surface); |
| shsurf->state.maximized = |
| weston_desktop_surface_get_maximized(desktop_surface); |
| |
| if (!weston_surface_is_mapped(surface)) { |
| map(shell, shsurf, sx, sy); |
| surface->is_mapped = true; |
| if (shsurf->shell->win_close_animation_type == ANIMATION_FADE) |
| ++surface->ref_count; |
| return; |
| } |
| |
| if (sx == 0 && sy == 0 && |
| shsurf->last_width == surface->width && |
| shsurf->last_height == surface->height && |
| was_fullscreen == shsurf->state.fullscreen && |
| was_maximized == shsurf->state.maximized) |
| return; |
| |
| if (was_fullscreen) |
| unset_fullscreen(shsurf); |
| if (was_maximized) |
| unset_maximized(shsurf); |
| |
| if ((shsurf->state.fullscreen || shsurf->state.maximized) && |
| !shsurf->saved_position_valid) { |
| shsurf->saved_x = shsurf->view->geometry.x; |
| shsurf->saved_y = shsurf->view->geometry.y; |
| shsurf->saved_position_valid = true; |
| |
| if (!wl_list_empty(&shsurf->rotation.transform.link)) { |
| wl_list_remove(&shsurf->rotation.transform.link); |
| wl_list_init(&shsurf->rotation.transform.link); |
| weston_view_geometry_dirty(shsurf->view); |
| shsurf->saved_rotation_valid = true; |
| } |
| } |
| |
| if (shsurf->state.fullscreen) { |
| shell_configure_fullscreen(shsurf); |
| } else if (shsurf->state.maximized) { |
| set_maximized_position(shell, shsurf); |
| surface->output = shsurf->output; |
| } else { |
| float from_x, from_y; |
| float to_x, to_y; |
| float x, y; |
| |
| if (shsurf->resize_edges) { |
| sx = 0; |
| sy = 0; |
| } |
| |
| if (shsurf->resize_edges & WL_SHELL_SURFACE_RESIZE_LEFT) |
| sx = shsurf->last_width - surface->width; |
| if (shsurf->resize_edges & WL_SHELL_SURFACE_RESIZE_TOP) |
| sy = shsurf->last_height - surface->height; |
| |
| weston_view_to_global_float(shsurf->view, 0, 0, &from_x, &from_y); |
| weston_view_to_global_float(shsurf->view, sx, sy, &to_x, &to_y); |
| x = shsurf->view->geometry.x + to_x - from_x; |
| y = shsurf->view->geometry.y + to_y - from_y; |
| |
| weston_view_set_position(shsurf->view, x, y); |
| } |
| |
| shsurf->last_width = surface->width; |
| shsurf->last_height = surface->height; |
| |
| /* XXX: would a fullscreen surface need the same handling? */ |
| if (surface->output) { |
| wl_list_for_each(view, &surface->views, surface_link) |
| weston_view_update_transform(view); |
| } |
| } |
| |
| static void |
| get_maximized_size(struct shell_surface *shsurf, int32_t *width, int32_t *height) |
| { |
| struct desktop_shell *shell; |
| pixman_rectangle32_t area; |
| |
| shell = shell_surface_get_shell(shsurf); |
| get_output_work_area(shell, shsurf->output, &area); |
| |
| *width = area.width; |
| *height = area.height; |
| } |
| |
| static void |
| set_fullscreen(struct shell_surface *shsurf, bool fullscreen, |
| struct weston_output *output) |
| { |
| struct weston_desktop_surface *desktop_surface = shsurf->desktop_surface; |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| int32_t width = 0, height = 0; |
| |
| if (fullscreen) { |
| /* handle clients launching in fullscreen */ |
| if (output == NULL && !weston_surface_is_mapped(surface)) { |
| /* Set the output to the one that has focus currently. */ |
| output = get_focused_output(surface->compositor); |
| } |
| |
| shell_surface_set_output(shsurf, output); |
| shsurf->fullscreen_output = shsurf->output; |
| |
| width = shsurf->output->width; |
| height = shsurf->output->height; |
| } else if (weston_desktop_surface_get_maximized(desktop_surface)) { |
| get_maximized_size(shsurf, &width, &height); |
| } |
| weston_desktop_surface_set_fullscreen(desktop_surface, fullscreen); |
| weston_desktop_surface_set_size(desktop_surface, width, height); |
| } |
| |
| static void |
| desktop_surface_move(struct weston_desktop_surface *desktop_surface, |
| struct weston_seat *seat, uint32_t serial, void *shell) |
| { |
| struct weston_pointer *pointer = weston_seat_get_pointer(seat); |
| struct weston_touch *touch = weston_seat_get_touch(seat); |
| struct shell_surface *shsurf = |
| weston_desktop_surface_get_user_data(desktop_surface); |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| struct wl_resource *resource = surface->resource; |
| struct weston_surface *focus; |
| |
| if (pointer && |
| pointer->focus && |
| pointer->button_count > 0 && |
| pointer->grab_serial == serial) { |
| focus = weston_surface_get_main_surface(pointer->focus->surface); |
| if ((focus == surface) && |
| (surface_move(shsurf, pointer, true) < 0)) |
| wl_resource_post_no_memory(resource); |
| } else if (touch && |
| touch->focus && |
| touch->grab_serial == serial) { |
| focus = weston_surface_get_main_surface(touch->focus->surface); |
| if ((focus == surface) && |
| (surface_touch_move(shsurf, touch) < 0)) |
| wl_resource_post_no_memory(resource); |
| } |
| } |
| |
| static void |
| desktop_surface_resize(struct weston_desktop_surface *desktop_surface, |
| struct weston_seat *seat, uint32_t serial, |
| enum weston_desktop_surface_edge edges, void *shell) |
| { |
| struct weston_pointer *pointer = weston_seat_get_pointer(seat); |
| struct shell_surface *shsurf = |
| weston_desktop_surface_get_user_data(desktop_surface); |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| struct wl_resource *resource = surface->resource; |
| struct weston_surface *focus; |
| |
| if (!pointer || |
| pointer->button_count == 0 || |
| pointer->grab_serial != serial || |
| pointer->focus == NULL) |
| return; |
| |
| focus = weston_surface_get_main_surface(pointer->focus->surface); |
| if (focus != surface) |
| return; |
| |
| if (surface_resize(shsurf, pointer, edges) < 0) |
| wl_resource_post_no_memory(resource); |
| } |
| |
| static void |
| desktop_surface_fullscreen_requested(struct weston_desktop_surface *desktop_surface, |
| bool fullscreen, |
| struct weston_output *output, void *shell) |
| { |
| struct shell_surface *shsurf = |
| weston_desktop_surface_get_user_data(desktop_surface); |
| |
| set_fullscreen(shsurf, fullscreen, output); |
| } |
| |
| static void |
| set_maximized(struct shell_surface *shsurf, bool maximized) |
| { |
| struct weston_desktop_surface *desktop_surface = shsurf->desktop_surface; |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| int32_t width = 0, height = 0; |
| |
| if (maximized) { |
| struct weston_output *output; |
| |
| if (!weston_surface_is_mapped(surface)) |
| output = get_focused_output(surface->compositor); |
| else |
| output = surface->output; |
| |
| shell_surface_set_output(shsurf, output); |
| |
| get_maximized_size(shsurf, &width, &height); |
| } |
| weston_desktop_surface_set_maximized(desktop_surface, maximized); |
| weston_desktop_surface_set_size(desktop_surface, width, height); |
| } |
| |
| static void |
| desktop_surface_maximized_requested(struct weston_desktop_surface *desktop_surface, |
| bool maximized, void *shell) |
| { |
| struct shell_surface *shsurf = |
| weston_desktop_surface_get_user_data(desktop_surface); |
| |
| set_maximized(shsurf, maximized); |
| } |
| |
| static void |
| desktop_surface_minimized_requested(struct weston_desktop_surface *desktop_surface, |
| void *shell) |
| { |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(desktop_surface); |
| |
| /* apply compositor's own minimization logic (hide) */ |
| set_minimized(surface); |
| } |
| |
| static void |
| set_busy_cursor(struct shell_surface *shsurf, struct weston_pointer *pointer) |
| { |
| struct shell_grab *grab; |
| |
| if (pointer->grab->interface == &busy_cursor_grab_interface) |
| return; |
| |
| grab = malloc(sizeof *grab); |
| if (!grab) |
| return; |
| |
| shell_grab_start(grab, &busy_cursor_grab_interface, shsurf, pointer, |
| WESTON_DESKTOP_SHELL_CURSOR_BUSY); |
| /* Mark the shsurf as ungrabbed so that button binding is able |
| * to move it. */ |
| shsurf->grabbed = 0; |
| } |
| |
| static void |
| end_busy_cursor(struct weston_compositor *compositor, |
| struct weston_desktop_client *desktop_client) |
| { |
| struct shell_surface *shsurf; |
| struct shell_grab *grab; |
| struct weston_seat *seat; |
| |
| wl_list_for_each(seat, &compositor->seat_list, link) { |
| struct weston_pointer *pointer = weston_seat_get_pointer(seat); |
| struct weston_desktop_client *grab_client; |
| |
| if (!pointer) |
| continue; |
| |
| if (pointer->grab->interface != &busy_cursor_grab_interface) |
| continue; |
| |
| grab = (struct shell_grab *) pointer->grab; |
| shsurf = grab->shsurf; |
| if (!shsurf) |
| continue; |
| |
| grab_client = |
| weston_desktop_surface_get_client(shsurf->desktop_surface); |
| if (grab_client == desktop_client) { |
| shell_grab_end(grab); |
| free(grab); |
| } |
| } |
| } |
| |
| static void |
| desktop_surface_set_unresponsive(struct weston_desktop_surface *desktop_surface, |
| void *user_data) |
| { |
| struct shell_surface *shsurf = |
| weston_desktop_surface_get_user_data(desktop_surface); |
| bool *unresponsive = user_data; |
| |
| shsurf->unresponsive = *unresponsive; |
| } |
| |
| static void |
| desktop_surface_ping_timeout(struct weston_desktop_client *desktop_client, |
| void *shell_) |
| { |
| struct desktop_shell *shell = shell_; |
| struct shell_surface *shsurf; |
| struct weston_seat *seat; |
| bool unresponsive = true; |
| |
| weston_desktop_client_for_each_surface(desktop_client, |
| desktop_surface_set_unresponsive, |
| &unresponsive); |
| |
| |
| wl_list_for_each(seat, &shell->compositor->seat_list, link) { |
| struct weston_pointer *pointer = weston_seat_get_pointer(seat); |
| struct weston_desktop_client *grab_client; |
| |
| if (!pointer || !pointer->focus) |
| continue; |
| |
| shsurf = get_shell_surface(pointer->focus->surface); |
| if (!shsurf) |
| continue; |
| |
| grab_client = |
| weston_desktop_surface_get_client(shsurf->desktop_surface); |
| if (grab_client == desktop_client) |
| set_busy_cursor(shsurf, pointer); |
| } |
| } |
| |
| static void |
| desktop_surface_pong(struct weston_desktop_client *desktop_client, |
| void *shell_) |
| { |
| struct desktop_shell *shell = shell_; |
| bool unresponsive = false; |
| |
| weston_desktop_client_for_each_surface(desktop_client, |
| desktop_surface_set_unresponsive, |
| &unresponsive); |
| end_busy_cursor(shell->compositor, desktop_client); |
| } |
| |
| static void |
| desktop_surface_set_xwayland_position(struct weston_desktop_surface *surface, |
| int32_t x, int32_t y, void *shell_) |
| { |
| struct shell_surface *shsurf = |
| weston_desktop_surface_get_user_data(surface); |
| |
| shsurf->xwayland.x = x; |
| shsurf->xwayland.y = y; |
| shsurf->xwayland.is_set = true; |
| } |
| |
| static const struct weston_desktop_api shell_desktop_api = { |
| .struct_size = sizeof(struct weston_desktop_api), |
| .surface_added = desktop_surface_added, |
| .surface_removed = desktop_surface_removed, |
| .committed = desktop_surface_committed, |
| .move = desktop_surface_move, |
| .resize = desktop_surface_resize, |
| .fullscreen_requested = desktop_surface_fullscreen_requested, |
| .maximized_requested = desktop_surface_maximized_requested, |
| .minimized_requested = desktop_surface_minimized_requested, |
| .ping_timeout = desktop_surface_ping_timeout, |
| .pong = desktop_surface_pong, |
| .set_xwayland_position = desktop_surface_set_xwayland_position, |
| }; |
| |
| /* ************************ * |
| * end of libweston-desktop * |
| * ************************ */ |
| static void |
| configure_static_view(struct weston_view *ev, struct weston_layer *layer, int x, int y) |
| { |
| struct weston_view *v, *next; |
| |
| if (!ev->output) |
| return; |
| |
| wl_list_for_each_safe(v, next, &layer->view_list.link, layer_link.link) { |
| if (v->output == ev->output && v != ev) { |
| weston_view_unmap(v); |
| v->surface->committed = NULL; |
| weston_surface_set_label_func(v->surface, NULL); |
| } |
| } |
| |
| weston_view_set_position(ev, ev->output->x + x, ev->output->y + y); |
| ev->surface->is_mapped = true; |
| ev->is_mapped = true; |
| |
| if (wl_list_empty(&ev->layer_link.link)) { |
| weston_layer_entry_insert(&layer->view_list, &ev->layer_link); |
| weston_compositor_schedule_repaint(ev->surface->compositor); |
| } |
| } |
| |
| |
| static struct shell_output * |
| find_shell_output_from_weston_output(struct desktop_shell *shell, |
| struct weston_output *output) |
| { |
| struct shell_output *shell_output; |
| |
| wl_list_for_each(shell_output, &shell->output_list, link) { |
| if (shell_output->output == output) |
| return shell_output; |
| } |
| |
| return NULL; |
| } |
| |
| static int |
| background_get_label(struct weston_surface *surface, char *buf, size_t len) |
| { |
| return snprintf(buf, len, "background for output %s", |
| surface->output->name); |
| } |
| |
| static void |
| background_committed(struct weston_surface *es, int32_t sx, int32_t sy) |
| { |
| struct desktop_shell *shell = es->committed_private; |
| struct weston_view *view; |
| |
| view = container_of(es->views.next, struct weston_view, surface_link); |
| |
| configure_static_view(view, &shell->background_layer, 0, 0); |
| } |
| |
| static void |
| handle_background_surface_destroy(struct wl_listener *listener, void *data) |
| { |
| struct shell_output *output = |
| container_of(listener, struct shell_output, background_surface_listener); |
| |
| weston_log("background surface gone\n"); |
| wl_list_remove(&output->background_surface_listener.link); |
| output->background_surface = NULL; |
| } |
| |
| static void |
| desktop_shell_set_background(struct wl_client *client, |
| struct wl_resource *resource, |
| struct wl_resource *output_resource, |
| struct wl_resource *surface_resource) |
| { |
| struct desktop_shell *shell = wl_resource_get_user_data(resource); |
| struct weston_surface *surface = |
| wl_resource_get_user_data(surface_resource); |
| struct shell_output *sh_output; |
| struct weston_view *view, *next; |
| |
| if (surface->committed) { |
| wl_resource_post_error(surface_resource, |
| WL_DISPLAY_ERROR_INVALID_OBJECT, |
| "surface role already assigned"); |
| return; |
| } |
| |
| wl_list_for_each_safe(view, next, &surface->views, surface_link) |
| weston_view_destroy(view); |
| view = weston_view_create(surface); |
| |
| surface->committed = background_committed; |
| surface->committed_private = shell; |
| weston_surface_set_label_func(surface, background_get_label); |
| surface->output = weston_head_from_resource(output_resource)->output; |
| weston_view_set_output(view, surface->output); |
| |
| sh_output = find_shell_output_from_weston_output(shell, surface->output); |
| if (sh_output->background_surface) { |
| /* The output already has a background, tell our helper |
| * there is no need for another one. */ |
| weston_desktop_shell_send_configure(resource, 0, |
| surface_resource, |
| 0, 0); |
| } else { |
| weston_desktop_shell_send_configure(resource, 0, |
| surface_resource, |
| surface->output->width, |
| surface->output->height); |
| |
| sh_output->background_surface = surface; |
| |
| sh_output->background_surface_listener.notify = |
| handle_background_surface_destroy; |
| wl_signal_add(&surface->destroy_signal, |
| &sh_output->background_surface_listener); |
| } |
| } |
| |
| static int |
| panel_get_label(struct weston_surface *surface, char *buf, size_t len) |
| { |
| return snprintf(buf, len, "panel for output %s", |
| surface->output->name); |
| } |
| |
| static void |
| panel_committed(struct weston_surface *es, int32_t sx, int32_t sy) |
| { |
| struct desktop_shell *shell = es->committed_private; |
| struct weston_view *view; |
| int width, height; |
| int x = 0, y = 0; |
| |
| view = container_of(es->views.next, struct weston_view, surface_link); |
| |
| get_panel_size(shell, view, &width, &height); |
| switch (shell->panel_position) { |
| case WESTON_DESKTOP_SHELL_PANEL_POSITION_TOP: |
| break; |
| case WESTON_DESKTOP_SHELL_PANEL_POSITION_BOTTOM: |
| y = view->output->height - height; |
| break; |
| case WESTON_DESKTOP_SHELL_PANEL_POSITION_LEFT: |
| break; |
| case WESTON_DESKTOP_SHELL_PANEL_POSITION_RIGHT: |
| x = view->output->width - width; |
| break; |
| } |
| |
| configure_static_view(view, &shell->panel_layer, x, y); |
| } |
| |
| static void |
| handle_panel_surface_destroy(struct wl_listener *listener, void *data) |
| { |
| struct shell_output *output = |
| container_of(listener, struct shell_output, panel_surface_listener); |
| |
| weston_log("panel surface gone\n"); |
| wl_list_remove(&output->panel_surface_listener.link); |
| output->panel_surface = NULL; |
| } |
| |
| |
| static void |
| desktop_shell_set_panel(struct wl_client *client, |
| struct wl_resource *resource, |
| struct wl_resource *output_resource, |
| struct wl_resource *surface_resource) |
| { |
| struct desktop_shell *shell = wl_resource_get_user_data(resource); |
| struct weston_surface *surface = |
| wl_resource_get_user_data(surface_resource); |
| struct weston_view *view, *next; |
| struct shell_output *sh_output; |
| |
| if (surface->committed) { |
| wl_resource_post_error(surface_resource, |
| WL_DISPLAY_ERROR_INVALID_OBJECT, |
| "surface role already assigned"); |
| return; |
| } |
| |
| wl_list_for_each_safe(view, next, &surface->views, surface_link) |
| weston_view_destroy(view); |
| view = weston_view_create(surface); |
| |
| surface->committed = panel_committed; |
| surface->committed_private = shell; |
| weston_surface_set_label_func(surface, panel_get_label); |
| surface->output = weston_head_from_resource(output_resource)->output; |
| weston_view_set_output(view, surface->output); |
| |
| sh_output = find_shell_output_from_weston_output(shell, surface->output); |
| if (sh_output->panel_surface) { |
| /* The output already has a panel, tell our helper |
| * there is no need for another one. */ |
| weston_desktop_shell_send_configure(resource, 0, |
| surface_resource, |
| 0, 0); |
| } else { |
| weston_desktop_shell_send_configure(resource, 0, |
| surface_resource, |
| surface->output->width, |
| surface->output->height); |
| |
| sh_output->panel_surface = surface; |
| |
| sh_output->panel_surface_listener.notify = handle_panel_surface_destroy; |
| wl_signal_add(&surface->destroy_signal, &sh_output->panel_surface_listener); |
| } |
| } |
| |
| static int |
| lock_surface_get_label(struct weston_surface *surface, char *buf, size_t len) |
| { |
| return snprintf(buf, len, "lock window"); |
| } |
| |
| static void |
| lock_surface_committed(struct weston_surface *surface, int32_t sx, int32_t sy) |
| { |
| struct desktop_shell *shell = surface->committed_private; |
| struct weston_view *view; |
| |
| view = container_of(surface->views.next, struct weston_view, surface_link); |
| |
| if (surface->width == 0) |
| return; |
| |
| center_on_output(view, get_default_output(shell->compositor)); |
| |
| if (!weston_surface_is_mapped(surface)) { |
| weston_layer_entry_insert(&shell->lock_layer.view_list, |
| &view->layer_link); |
| weston_view_update_transform(view); |
| surface->is_mapped = true; |
| view->is_mapped = true; |
| shell_fade(shell, FADE_IN); |
| } |
| } |
| |
| static void |
| handle_lock_surface_destroy(struct wl_listener *listener, void *data) |
| { |
| struct desktop_shell *shell = |
| container_of(listener, struct desktop_shell, lock_surface_listener); |
| |
| weston_log("lock surface gone\n"); |
| shell->lock_surface = NULL; |
| } |
| |
| static void |
| desktop_shell_set_lock_surface(struct wl_client *client, |
| struct wl_resource *resource, |
| struct wl_resource *surface_resource) |
| { |
| struct desktop_shell *shell = wl_resource_get_user_data(resource); |
| struct weston_surface *surface = |
| wl_resource_get_user_data(surface_resource); |
| |
| shell->prepare_event_sent = false; |
| |
| if (!shell->locked) |
| return; |
| |
| shell->lock_surface = surface; |
| |
| shell->lock_surface_listener.notify = handle_lock_surface_destroy; |
| wl_signal_add(&surface->destroy_signal, |
| &shell->lock_surface_listener); |
| |
| weston_view_create(surface); |
| surface->committed = lock_surface_committed; |
| surface->committed_private = shell; |
| weston_surface_set_label_func(surface, lock_surface_get_label); |
| } |
| |
| static void |
| resume_desktop(struct desktop_shell *shell) |
| { |
| struct workspace *ws = get_current_workspace(shell); |
| |
| weston_layer_unset_position(&shell->lock_layer); |
| |
| if (shell->showing_input_panels) |
| weston_layer_set_position(&shell->input_panel_layer, |
| WESTON_LAYER_POSITION_TOP_UI); |
| weston_layer_set_position(&shell->fullscreen_layer, |
| WESTON_LAYER_POSITION_FULLSCREEN); |
| weston_layer_set_position(&shell->panel_layer, |
| WESTON_LAYER_POSITION_UI); |
| weston_layer_set_position(&ws->layer, WESTON_LAYER_POSITION_NORMAL); |
| |
| restore_focus_state(shell, get_current_workspace(shell)); |
| |
| shell->locked = false; |
| shell_fade(shell, FADE_IN); |
| weston_compositor_damage_all(shell->compositor); |
| } |
| |
| static void |
| desktop_shell_unlock(struct wl_client *client, |
| struct wl_resource *resource) |
| { |
| struct desktop_shell *shell = wl_resource_get_user_data(resource); |
| |
| shell->prepare_event_sent = false; |
| |
| if (shell->locked) |
| resume_desktop(shell); |
| } |
| |
| static void |
| desktop_shell_set_grab_surface(struct wl_client *client, |
| struct wl_resource *resource, |
| struct wl_resource *surface_resource) |
| { |
| struct desktop_shell *shell = wl_resource_get_user_data(resource); |
| |
| shell->grab_surface = wl_resource_get_user_data(surface_resource); |
| weston_view_create(shell->grab_surface); |
| } |
| |
| static void |
| desktop_shell_desktop_ready(struct wl_client *client, |
| struct wl_resource *resource) |
| { |
| struct desktop_shell *shell = wl_resource_get_user_data(resource); |
| |
| shell_fade_startup(shell); |
| } |
| |
| static void |
| desktop_shell_set_panel_position(struct wl_client *client, |
| struct wl_resource *resource, |
| uint32_t position) |
| { |
| struct desktop_shell *shell = wl_resource_get_user_data(resource); |
| |
| if (position != WESTON_DESKTOP_SHELL_PANEL_POSITION_TOP && |
| position != WESTON_DESKTOP_SHELL_PANEL_POSITION_BOTTOM && |
| position != WESTON_DESKTOP_SHELL_PANEL_POSITION_LEFT && |
| position != WESTON_DESKTOP_SHELL_PANEL_POSITION_RIGHT) { |
| wl_resource_post_error(resource, |
| WESTON_DESKTOP_SHELL_ERROR_INVALID_ARGUMENT, |
| "bad position argument"); |
| return; |
| } |
| |
| shell->panel_position = position; |
| } |
| |
| static const struct weston_desktop_shell_interface desktop_shell_implementation = { |
| desktop_shell_set_background, |
| desktop_shell_set_panel, |
| desktop_shell_set_lock_surface, |
| desktop_shell_unlock, |
| desktop_shell_set_grab_surface, |
| desktop_shell_desktop_ready, |
| desktop_shell_set_panel_position |
| }; |
| |
| static void |
| move_binding(struct weston_pointer *pointer, const struct timespec *time, |
| uint32_t button, void *data) |
| { |
| struct weston_surface *focus; |
| struct weston_surface *surface; |
| struct shell_surface *shsurf; |
| |
| if (pointer->focus == NULL) |
| return; |
| |
| focus = pointer->focus->surface; |
| |
| surface = weston_surface_get_main_surface(focus); |
| if (surface == NULL) |
| return; |
| |
| shsurf = get_shell_surface(surface); |
| if (shsurf == NULL || |
| weston_desktop_surface_get_fullscreen(shsurf->desktop_surface) || |
| weston_desktop_surface_get_maximized(shsurf->desktop_surface)) |
| return; |
| |
| surface_move(shsurf, pointer, false); |
| } |
| |
| static void |
| maximize_binding(struct weston_keyboard *keyboard, const struct timespec *time, |
| uint32_t button, void *data) |
| { |
| struct weston_surface *focus = keyboard->focus; |
| struct weston_surface *surface; |
| struct shell_surface *shsurf; |
| |
| surface = weston_surface_get_main_surface(focus); |
| if (surface == NULL) |
| return; |
| |
| shsurf = get_shell_surface(surface); |
| if (shsurf == NULL) |
| return; |
| |
| set_maximized(shsurf, !weston_desktop_surface_get_maximized(shsurf->desktop_surface)); |
| } |
| |
| static void |
| fullscreen_binding(struct weston_keyboard *keyboard, |
| const struct timespec *time, uint32_t button, void *data) |
| { |
| struct weston_surface *focus = keyboard->focus; |
| struct weston_surface *surface; |
| struct shell_surface *shsurf; |
| bool fullscreen; |
| |
| surface = weston_surface_get_main_surface(focus); |
| if (surface == NULL) |
| return; |
| |
| shsurf = get_shell_surface(surface); |
| if (shsurf == NULL) |
| return; |
| |
| fullscreen = |
| weston_desktop_surface_get_fullscreen(shsurf->desktop_surface); |
| |
| set_fullscreen(shsurf, !fullscreen, NULL); |
| } |
| |
| static void |
| touch_move_binding(struct weston_touch *touch, const struct timespec *time, void *data) |
| { |
| struct weston_surface *focus; |
| struct weston_surface *surface; |
| struct shell_surface *shsurf; |
| |
| if (touch->focus == NULL) |
| return; |
| |
| focus = touch->focus->surface; |
| surface = weston_surface_get_main_surface(focus); |
| if (surface == NULL) |
| return; |
| |
| shsurf = get_shell_surface(surface); |
| if (shsurf == NULL || |
| weston_desktop_surface_get_fullscreen(shsurf->desktop_surface) || |
| weston_desktop_surface_get_maximized(shsurf->desktop_surface)) |
| return; |
| |
| surface_touch_move(shsurf, touch); |
| } |
| |
| static void |
| resize_binding(struct weston_pointer *pointer, const struct timespec *time, |
| uint32_t button, void *data) |
| { |
| struct weston_surface *focus; |
| struct weston_surface *surface; |
| uint32_t edges = 0; |
| int32_t x, y; |
| struct shell_surface *shsurf; |
| |
| if (pointer->focus == NULL) |
| return; |
| |
| focus = pointer->focus->surface; |
| |
| surface = weston_surface_get_main_surface(focus); |
| if (surface == NULL) |
| return; |
| |
| shsurf = get_shell_surface(surface); |
| if (shsurf == NULL || |
| weston_desktop_surface_get_fullscreen(shsurf->desktop_surface) || |
| weston_desktop_surface_get_maximized(shsurf->desktop_surface)) |
| return; |
| |
| weston_view_from_global(shsurf->view, |
| wl_fixed_to_int(pointer->grab_x), |
| wl_fixed_to_int(pointer->grab_y), |
| &x, &y); |
| |
| if (x < surface->width / 3) |
| edges |= WL_SHELL_SURFACE_RESIZE_LEFT; |
| else if (x < 2 * surface->width / 3) |
| edges |= 0; |
| else |
| edges |= WL_SHELL_SURFACE_RESIZE_RIGHT; |
| |
| if (y < surface->height / 3) |
| edges |= WL_SHELL_SURFACE_RESIZE_TOP; |
| else if (y < 2 * surface->height / 3) |
| edges |= 0; |
| else |
| edges |= WL_SHELL_SURFACE_RESIZE_BOTTOM; |
| |
| surface_resize(shsurf, pointer, edges); |
| } |
| |
| static void |
| surface_opacity_binding(struct weston_pointer *pointer, |
| const struct timespec *time, |
| struct weston_pointer_axis_event *event, |
| void *data) |
| { |
| float step = 0.005; |
| struct shell_surface *shsurf; |
| struct weston_surface *focus = pointer->focus->surface; |
| struct weston_surface *surface; |
| |
| /* XXX: broken for windows containing sub-surfaces */ |
| surface = weston_surface_get_main_surface(focus); |
| if (surface == NULL) |
| return; |
| |
| shsurf = get_shell_surface(surface); |
| if (!shsurf) |
| return; |
| |
| shsurf->view->alpha -= event->value * step; |
| |
| if (shsurf->view->alpha > 1.0) |
| shsurf->view->alpha = 1.0; |
| if (shsurf->view->alpha < step) |
| shsurf->view->alpha = step; |
| |
| weston_view_geometry_dirty(shsurf->view); |
| weston_surface_damage(surface); |
| } |
| |
| static void |
| do_zoom(struct weston_seat *seat, const struct timespec *time, uint32_t key, |
| uint32_t axis, double value) |
| { |
| struct weston_compositor *compositor = seat->compositor; |
| struct weston_pointer *pointer = weston_seat_get_pointer(seat); |
| struct weston_output *output; |
| float increment; |
| |
| if (!pointer) { |
| weston_log("Zoom hotkey pressed but seat '%s' contains no pointer.\n", seat->seat_name); |
| return; |
| } |
| |
| wl_list_for_each(output, &compositor->output_list, link) { |
| if (pixman_region32_contains_point(&output->region, |
| wl_fixed_to_double(pointer->x), |
| wl_fixed_to_double(pointer->y), |
| NULL)) { |
| if (key == KEY_PAGEUP) |
| increment = output->zoom.increment; |
| else if (key == KEY_PAGEDOWN) |
| increment = -output->zoom.increment; |
| else if (axis == WL_POINTER_AXIS_VERTICAL_SCROLL) |
| /* For every pixel zoom 20th of a step */ |
| increment = output->zoom.increment * |
| -value / 20.0; |
| else |
| increment = 0; |
| |
| output->zoom.level += increment; |
| |
| if (output->zoom.level < 0.0) |
| output->zoom.level = 0.0; |
| else if (output->zoom.level > output->zoom.max_level) |
| output->zoom.level = output->zoom.max_level; |
| |
| if (!output->zoom.active) { |
| if (output->zoom.level <= 0.0) |
| continue; |
| weston_output_activate_zoom(output, seat); |
| } |
| |
| output->zoom.spring_z.target = output->zoom.level; |
| |
| weston_output_update_zoom(output); |
| } |
| } |
| } |
| |
| static void |
| zoom_axis_binding(struct weston_pointer *pointer, const struct timespec *time, |
| struct weston_pointer_axis_event *event, |
| void *data) |
| { |
| do_zoom(pointer->seat, time, 0, event->axis, event->value); |
| } |
| |
| static void |
| zoom_key_binding(struct weston_keyboard *keyboard, const struct timespec *time, |
| uint32_t key, void *data) |
| { |
| do_zoom(keyboard->seat, time, key, 0, 0); |
| } |
| |
| static void |
| terminate_binding(struct weston_keyboard *keyboard, const struct timespec *time, |
| uint32_t key, void *data) |
| { |
| struct weston_compositor *compositor = data; |
| |
| wl_display_terminate(compositor->wl_display); |
| } |
| |
| static void |
| rotate_grab_motion(struct weston_pointer_grab *grab, |
| const struct timespec *time, |
| struct weston_pointer_motion_event *event) |
| { |
| struct rotate_grab *rotate = |
| container_of(grab, struct rotate_grab, base.grab); |
| struct weston_pointer *pointer = grab->pointer; |
| struct shell_surface *shsurf = rotate->base.shsurf; |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| float cx, cy, dx, dy, cposx, cposy, dposx, dposy, r; |
| |
| weston_pointer_move(pointer, event); |
| |
| if (!shsurf) |
| return; |
| |
| cx = 0.5f * surface->width; |
| cy = 0.5f * surface->height; |
| |
| dx = wl_fixed_to_double(pointer->x) - rotate->center.x; |
| dy = wl_fixed_to_double(pointer->y) - rotate->center.y; |
| r = sqrtf(dx * dx + dy * dy); |
| |
| wl_list_remove(&shsurf->rotation.transform.link); |
| weston_view_geometry_dirty(shsurf->view); |
| |
| if (r > 20.0f) { |
| struct weston_matrix *matrix = |
| &shsurf->rotation.transform.matrix; |
| |
| weston_matrix_init(&rotate->rotation); |
| weston_matrix_rotate_xy(&rotate->rotation, dx / r, dy / r); |
| |
| weston_matrix_init(matrix); |
| weston_matrix_translate(matrix, -cx, -cy, 0.0f); |
| weston_matrix_multiply(matrix, &shsurf->rotation.rotation); |
| weston_matrix_multiply(matrix, &rotate->rotation); |
| weston_matrix_translate(matrix, cx, cy, 0.0f); |
| |
| wl_list_insert( |
| &shsurf->view->geometry.transformation_list, |
| &shsurf->rotation.transform.link); |
| } else { |
| wl_list_init(&shsurf->rotation.transform.link); |
| weston_matrix_init(&shsurf->rotation.rotation); |
| weston_matrix_init(&rotate->rotation); |
| } |
| |
| /* We need to adjust the position of the surface |
| * in case it was resized in a rotated state before */ |
| cposx = shsurf->view->geometry.x + cx; |
| cposy = shsurf->view->geometry.y + cy; |
| dposx = rotate->center.x - cposx; |
| dposy = rotate->center.y - cposy; |
| if (dposx != 0.0f || dposy != 0.0f) { |
| weston_view_set_position(shsurf->view, |
| shsurf->view->geometry.x + dposx, |
| shsurf->view->geometry.y + dposy); |
| } |
| |
| /* Repaint implies weston_view_update_transform(), which |
| * lazily applies the damage due to rotation update. |
| */ |
| weston_compositor_schedule_repaint(surface->compositor); |
| } |
| |
| static void |
| rotate_grab_button(struct weston_pointer_grab *grab, |
| const struct timespec *time, |
| uint32_t button, uint32_t state_w) |
| { |
| struct rotate_grab *rotate = |
| container_of(grab, struct rotate_grab, base.grab); |
| struct weston_pointer *pointer = grab->pointer; |
| struct shell_surface *shsurf = rotate->base.shsurf; |
| enum wl_pointer_button_state state = state_w; |
| |
| if (pointer->button_count == 0 && |
| state == WL_POINTER_BUTTON_STATE_RELEASED) { |
| if (shsurf) |
| weston_matrix_multiply(&shsurf->rotation.rotation, |
| &rotate->rotation); |
| shell_grab_end(&rotate->base); |
| free(rotate); |
| } |
| } |
| |
| static void |
| rotate_grab_cancel(struct weston_pointer_grab *grab) |
| { |
| struct rotate_grab *rotate = |
| container_of(grab, struct rotate_grab, base.grab); |
| |
| shell_grab_end(&rotate->base); |
| free(rotate); |
| } |
| |
| static const struct weston_pointer_grab_interface rotate_grab_interface = { |
| noop_grab_focus, |
| rotate_grab_motion, |
| rotate_grab_button, |
| noop_grab_axis, |
| noop_grab_axis_source, |
| noop_grab_frame, |
| rotate_grab_cancel, |
| }; |
| |
| static void |
| surface_rotate(struct shell_surface *shsurf, struct weston_pointer *pointer) |
| { |
| struct weston_surface *surface = |
| weston_desktop_surface_get_surface(shsurf->desktop_surface); |
| struct rotate_grab *rotate; |
| float dx, dy; |
| float r; |
| |
| rotate = malloc(sizeof *rotate); |
| if (!rotate) |
| return; |
| |
| weston_view_to_global_float(shsurf->view, |
| surface->width * 0.5f, |
| surface->height * 0.5f, |
| &rotate->center.x, &rotate->center.y); |
| |
| dx = wl_fixed_to_double(pointer->x) - rotate->center.x; |
| dy = wl_fixed_to_double(pointer->y) - rotate->center.y; |
| r = sqrtf(dx * dx + dy * dy); |
| if (r > 20.0f) { |
| struct weston_matrix inverse; |
| |
| weston_matrix_init(&inverse); |
| weston_matrix_rotate_xy(&inverse, dx / r, -dy / r); |
| weston_matrix_multiply(&shsurf->rotation.rotation, &inverse); |
| |
| weston_matrix_init(&rotate->rotation); |
| weston_matrix_rotate_xy(&rotate->rotation, dx / r, dy / r); |
| } else { |
| weston_matrix_init(&shsurf->rotation.rotation); |
| weston_matrix_init(&rotate->rotation); |
| } |
| |
| shell_grab_start(&rotate->base, &rotate_grab_interface, shsurf, |
| pointer, WESTON_DESKTOP_SHELL_CURSOR_ARROW); |
| } |
| |
| static void |
| rotate_binding(struct weston_pointer *pointer, const struct timespec *time, |
| uint32_t button, void *data) |
| { |
| struct weston_surface *focus; |
| struct weston_surface *base_surface; |
| struct shell_surface *surface; |
| |
| if (pointer->focus == NULL) |
| return; |
| |
| focus = pointer->focus->surface; |
| |
| base_surface = weston_surface_get_main_surface(focus); |
| if (base_surface == NULL) |
| return; |
| |
| surface = get_shell_surface(base_surface); |
| if (surface == NULL || |
| weston_desktop_surface_get_fullscreen(surface->desktop_surface) || |
| weston_desktop_surface_get_maximized(surface->desktop_surface)) |
| return; |
| |
| surface_rotate(surface, pointer); |
| } |
| |
| /* Move all fullscreen layers down to the current workspace and hide their |
| * black views. The surfaces' state is set to both fullscreen and lowered, |
| * and this is reversed when such a surface is re-configured, see |
| * shell_configure_fullscreen() and shell_ensure_fullscreen_black_view(). |
| * |
| * lowering_output = NULL - Lower on all outputs, else only lower on the |
| * specified output. |
| * |
| * This should be used when implementing shell-wide overlays, such as |
| * the alt-tab switcher, which need to de-promote fullscreen layers. */ |
| void |
| lower_fullscreen_layer(struct desktop_shell *shell, |
| struct weston_output *lowering_output) |
| { |
| struct workspace *ws; |
| struct weston_view *view, *prev; |
| |
| ws = get_current_workspace(shell); |
| wl_list_for_each_reverse_safe(view, prev, |
| &shell->fullscreen_layer.view_list.link, |
| layer_link.link) { |
| struct shell_surface *shsurf = get_shell_surface(view->surface); |
| |
| if (!shsurf) |
| continue; |
| |
| /* Only lower surfaces which have lowering_output as their fullscreen |
| * output, unless a NULL output asks for lowering on all outputs. |
| */ |
| if (lowering_output && (shsurf->fullscreen_output != lowering_output)) |
| continue; |
| |
| /* We can have a non-fullscreen popup for a fullscreen surface |
| * in the fullscreen layer. */ |
| if (weston_desktop_surface_get_fullscreen(shsurf->desktop_surface)) { |
| /* Hide the black view */ |
| weston_layer_entry_remove(&shsurf->fullscreen.black_view->layer_link); |
| wl_list_init(&shsurf->fullscreen.black_view->layer_link.link); |
| weston_view_damage_below(shsurf->fullscreen.black_view); |
| |
| } |
| |
| /* Lower the view to the workspace layer */ |
| weston_layer_entry_remove(&view->layer_link); |
| weston_layer_entry_insert(&ws->layer.view_list, &view->layer_link); |
| weston_view_damage_below(view); |
| weston_surface_damage(view->surface); |
| |
| shsurf->state.lowered = true; |
| } |
| } |
| |
| void |
| activate(struct desktop_shell *shell, struct weston_view *view, |
| struct weston_seat *seat, uint32_t flags) |
| { |
| struct weston_surface *es = view->surface; |
| struct weston_surface *main_surface; |
| struct focus_state *state; |
| struct workspace *ws; |
| struct weston_surface *old_es; |
| struct shell_surface *shsurf; |
| |
| main_surface = weston_surface_get_main_surface(es); |
| shsurf = get_shell_surface(main_surface); |
| assert(shsurf); |
| |
| /* Only demote fullscreen surfaces on the output of activated shsurf. |
| * Leave fullscreen surfaces on unrelated outputs alone. */ |
| lower_fullscreen_layer(shell, shsurf->output); |
| |
| weston_view_activate(view, seat, flags); |
| |
| state = ensure_focus_state(shell, seat); |
| if (state == NULL) |
| return; |
| |
| old_es = state->keyboard_focus; |
| focus_state_set_focus(state, es); |
| |
| if (weston_desktop_surface_get_fullscreen(shsurf->desktop_surface) && |
| flags & WESTON_ACTIVATE_FLAG_CONFIGURE) |
| shell_configure_fullscreen(shsurf); |
| |
| /* Update the surface’s layer. This brings it to the top of the stacking |
| * order as appropriate. */ |
| shell_surface_update_layer(shsurf); |
| |
| if (shell->focus_animation_type != ANIMATION_NONE) { |
| ws = get_current_workspace(shell); |
| animate_focus_change(shell, ws, get_default_view(old_es), get_default_view(es)); |
| } |
| } |
| |
| /* no-op func for checking black surface */ |
| static void |
| black_surface_committed(struct weston_surface *es, int32_t sx, int32_t sy) |
| { |
| } |
| |
| static bool |
| is_black_surface_view(struct weston_view *view, struct weston_view **fs_view) |
| { |
| struct weston_surface *surface = view->surface; |
| |
| if (surface->committed == black_surface_committed) { |
| if (fs_view) |
| *fs_view = surface->committed_private; |
| return true; |
| } |
| return false; |
| } |
| |
| static void |
| activate_binding(struct weston_seat *seat, |
| struct desktop_shell *shell, |
| struct weston_view *focus_view, |
| uint32_t flags) |
| { |
| struct weston_view *main_view; |
| struct weston_surface *main_surface; |
| |
| if (!focus_view) |
| return; |
| |
| if (is_black_surface_view(focus_view, &main_view)) |
| focus_view = main_view; |
| |
| main_surface = weston_surface_get_main_surface(focus_view->surface); |
| if (!get_shell_surface(main_surface)) |
| return; |
| |
| activate(shell, focus_view, seat, flags); |
| } |
| |
| static void |
| click_to_activate_binding(struct weston_pointer *pointer, |
| const struct timespec *time, |
| uint32_t button, void *data) |
| { |
| if (pointer->grab != &pointer->default_grab) |
| return; |
| if (pointer->focus == NULL) |
| return; |
| |
| activate_binding(pointer->seat, data, pointer->focus, |
| WESTON_ACTIVATE_FLAG_CLICKED | |
| WESTON_ACTIVATE_FLAG_CONFIGURE); |
| } |
| |
| static void |
| touch_to_activate_binding(struct weston_touch *touch, |
| const struct timespec *time, |
| void *data) |
| { |
| if (touch->grab != &touch->default_grab) |
| return; |
| if (touch->focus == NULL) |
| return; |
| |
| activate_binding(touch->seat, data, touch->focus, |
| WESTON_ACTIVATE_FLAG_CONFIGURE); |
| } |
| |
| static void |
| unfocus_all_seats(struct desktop_shell *shell) |
| { |
| struct weston_seat *seat, *next; |
| |
| wl_list_for_each_safe(seat, next, &shell->compositor->seat_list, link) { |
| struct weston_keyboard *keyboard = |
| weston_seat_get_keyboard(seat); |
| |
| if (!keyboard) |
| continue; |
| |
| weston_keyboard_set_focus(keyboard, NULL); |
| } |
| } |
| |
| static void |
| lock(struct desktop_shell *shell) |
| { |
| struct workspace *ws = get_current_workspace(shell); |
| |
| if (shell->locked) { |
| weston_compositor_sleep(shell->compositor); |
| return; |
| } |
| |
| shell->locked = true; |
| |
| /* Hide all surfaces by removing the fullscreen, panel and |
| * toplevel layers. This way nothing else can show or receive |
| * input events while we are locked. */ |
| |
| weston_layer_unset_position(&shell->panel_layer); |
| weston_layer_unset_position(&shell->fullscreen_layer); |
| if (shell->showing_input_panels) |
| weston_layer_unset_position(&shell->input_panel_layer); |
| weston_layer_unset_position(&ws->layer); |
| |
| weston_layer_set_position(&shell->lock_layer, |
| WESTON_LAYER_POSITION_LOCK); |
| |
| weston_compositor_sleep(shell->compositor); |
| |
| /* Remove the keyboard focus on all seats. This will be |
| * restored to the workspace's saved state via |
| * restore_focus_state when the compositor is unlocked */ |
| unfocus_all_seats(shell); |
| |
| /* TODO: disable bindings that should not work while locked. */ |
| |
| /* All this must be undone in resume_desktop(). */ |
| } |
| |
| static void |
| unlock(struct desktop_shell *shell) |
| { |
| struct wl_resource *shell_resource; |
| |
| if (!shell->locked || shell->lock_surface) { |
| shell_fade(shell, FADE_IN); |
| return; |
| } |
| |
| /* If desktop-shell client has gone away, unlock immediately. */ |
| if (!shell->child.desktop_shell) { |
| resume_desktop(shell); |
| return; |
| } |
| |
| if (shell->prepare_event_sent) |
| return; |
| |
| shell_resource = shell->child.desktop_shell; |
| weston_desktop_shell_send_prepare_lock_surface(shell_resource); |
| shell->prepare_event_sent = true; |
| } |
| |
| static void |
| shell_fade_done_for_output(struct weston_view_animation *animation, void *data) |
| { |
| struct shell_output *shell_output = data; |
| struct desktop_shell *shell = shell_output->shell; |
| |
| shell_output->fade.animation = NULL; |
| switch (shell_output->fade.type) { |
| case FADE_IN: |
| weston_surface_destroy(shell_output->fade.view->surface); |
| shell_output->fade.view = NULL; |
| break; |
| case FADE_OUT: |
| lock(shell); |
| break; |
| default: |
| break; |
| } |
| } |
| |
| static struct weston_view * |
| shell_fade_create_surface_for_output(struct desktop_shell *shell, struct shell_output *shell_output) |
| { |
| struct weston_compositor *compositor = shell->compositor; |
| struct weston_surface *surface; |
| struct weston_view *view; |
| |
| surface = weston_surface_create(compositor); |
| if (!surface) |
| return NULL; |
| |
| view = weston_view_create(surface); |
| if (!view) { |
| weston_surface_destroy(surface); |
| return NULL; |
| } |
| |
| weston_surface_set_size(surface, shell_output->output->width, shell_output->output->height); |
| weston_view_set_position(view, shell_output->output->x, shell_output->output->y); |
| weston_surface_set_color(surface, 0.0, 0.0, 0.0, 1.0); |
| weston_layer_entry_insert(&compositor->fade_layer.view_list, |
| &view->layer_link); |
| pixman_region32_init(&surface->input); |
| surface->is_mapped = true; |
| view->is_mapped = true; |
| |
| return view; |
| } |
| |
| static void |
| shell_fade(struct desktop_shell *shell, enum fade_type type) |
| { |
| float tint; |
| struct shell_output *shell_output; |
| |
| switch (type) { |
| case FADE_IN: |
| tint = 0.0; |
| break; |
| case FADE_OUT: |
| tint = 1.0; |
| break; |
| default: |
| weston_log("shell: invalid fade type\n"); |
| return; |
| } |
| |
| /* Create a separate fade surface for each output */ |
| wl_list_for_each(shell_output, &shell->output_list, link) { |
| shell_output->fade.type = type; |
| |
| if (shell_output->fade.view == NULL) { |
| shell_output->fade.view = shell_fade_create_surface_for_output(shell, shell_output); |
| if (!shell_output->fade.view) |
| continue; |
| |
| shell_output->fade.view->alpha = 1.0 - tint; |
| weston_view_update_transform(shell_output->fade.view); |
| } |
| |
| if (shell_output->fade.view->output == NULL) { |
| /* If the black view gets a NULL output, we lost the |
| * last output and we'll just cancel the fade. This |
| * happens when you close the last window under the |
| * X11 or Wayland backends. */ |
| shell->locked = false; |
| weston_surface_destroy(shell_output->fade.view->surface); |
| shell_output->fade.view = NULL; |
| } else if (shell_output->fade.animation) { |
| weston_fade_update(shell_output->fade.animation, tint); |
| } else { |
| shell_output->fade.animation = |
| weston_fade_run(shell_output->fade.view, |
| 1.0 - tint, tint, 300.0, |
| shell_fade_done_for_output, shell_output); |
| } |
| } |
| } |
| |
| static void |
| do_shell_fade_startup(void *data) |
| { |
| struct desktop_shell *shell = data; |
| struct shell_output *shell_output; |
| |
| if (shell->startup_animation_type == ANIMATION_FADE) { |
| shell_fade(shell, FADE_IN); |
| } else { |
| weston_log("desktop shell: " |
| "unexpected fade-in animation type %d\n", |
| shell->startup_animation_type); |
| wl_list_for_each(shell_output, &shell->output_list, link) { |
| weston_surface_destroy(shell_output->fade.view->surface); |
| shell_output->fade.view = NULL; |
| } |
| } |
| } |
| |
| static void |
| shell_fade_startup(struct desktop_shell *shell) |
| { |
| struct wl_event_loop *loop; |
| struct shell_output *shell_output; |
| bool has_fade = false; |
| |
| wl_list_for_each(shell_output, &shell->output_list, link) { |
| if (!shell_output->fade.startup_timer) |
| continue; |
| |
| wl_event_source_remove(shell_output->fade.startup_timer); |
| shell_output->fade.startup_timer = NULL; |
| has_fade = true; |
| } |
| |
| if (has_fade) { |
| loop = wl_display_get_event_loop(shell->compositor->wl_display); |
| wl_event_loop_add_idle(loop, do_shell_fade_startup, shell); |
| } |
| } |
| |
| static int |
| fade_startup_timeout(void *data) |
| { |
| struct desktop_shell *shell = data; |
| |
| shell_fade_startup(shell); |
| return 0; |
| } |
| |
| static void |
| shell_fade_init(struct desktop_shell *shell) |
| { |
| /* Make compositor output all black, and wait for the desktop-shell |
| * client to signal it is ready, then fade in. The timer triggers a |
| * fade-in, in case the desktop-shell client takes too long. |
| */ |
| |
| struct wl_event_loop *loop; |
| struct shell_output *shell_output; |
| |
| if (shell->startup_animation_type == ANIMATION_NONE) |
| return; |
| |
| wl_list_for_each(shell_output, &shell->output_list, link) { |
| if (shell_output->fade.view != NULL) { |
| weston_log("%s: warning: fade surface already exists\n", |
| __func__); |
| continue; |
| } |
| |
| shell_output->fade.view = shell_fade_create_surface_for_output(shell, shell_output); |
| if (!shell_output->fade.view) |
| continue; |
| |
| weston_view_update_transform(shell_output->fade.view); |
| weston_surface_damage(shell_output->fade.view->surface); |
| |
| loop = wl_display_get_event_loop(shell->compositor->wl_display); |
| shell_output->fade.startup_timer = |
| wl_event_loop_add_timer(loop, fade_startup_timeout, shell); |
| wl_event_source_timer_update(shell_output->fade.startup_timer, 15000); |
| } |
| } |
| |
| static void |
| idle_handler(struct wl_listener *listener, void *data) |
| { |
| struct desktop_shell *shell = |
| container_of(listener, struct desktop_shell, idle_listener); |
| |
| struct weston_seat *seat; |
| |
| wl_list_for_each(seat, &shell->compositor->seat_list, link) |
| weston_seat_break_desktop_grabs(seat); |
| |
| shell_fade(shell, FADE_OUT); |
| /* lock() is called from shell_fade_done_for_output() */ |
| } |
| |
| static void |
| wake_handler(struct wl_listener *listener, void *data) |
| { |
| struct desktop_shell *shell = |
| container_of(listener, struct desktop_shell, wake_listener); |
| |
| unlock(shell); |
| } |
| |
| static void |
| transform_handler(struct wl_listener *listener, void *data) |
| { |
| struct weston_surface *surface = data; |
| struct shell_surface *shsurf = get_shell_surface(surface); |
| const struct weston_xwayland_surface_api *api; |
| int x, y; |
| |
| if (!shsurf) |
| return; |
| |
| api = shsurf->shell->xwayland_surface_api; |
| if (!api) { |
| api = weston_xwayland_surface_get_api(shsurf->shell->compositor); |
| shsurf->shell->xwayland_surface_api = api; |
| } |
| |
| if (!api || !api->is_xwayland_surface(surface)) |
| return; |
| |
| if (!weston_view_is_mapped(shsurf->view)) |
| return; |
| |
| x = shsurf->view->geometry.x; |
| y = shsurf->view->geometry.y; |
| |
| api->send_position(surface, x, y); |
| } |
| |
| static void |
| center_on_output(struct weston_view *view, struct weston_output *output) |
| { |
| int32_t surf_x, surf_y, width, height; |
| float x, y; |
| |
| surface_subsurfaces_boundingbox(view->surface, &surf_x, &surf_y, &width, &height); |
| |
| x = output->x + (output->width - width) / 2 - surf_x / 2; |
| y = output->y + (output->height - height) / 2 - surf_y / 2; |
| |
| weston_view_set_position(view, x, y); |
| } |
| |
| static void |
| weston_view_set_initial_position(struct weston_view *view, |
| struct desktop_shell *shell) |
| { |
| struct weston_compositor *compositor = shell->compositor; |
| int ix = 0, iy = 0; |
| int32_t range_x, range_y; |
| int32_t x, y; |
| struct weston_output *output, *target_output = NULL; |
| struct weston_seat *seat; |
| pixman_rectangle32_t area; |
| |
| /* As a heuristic place the new window on the same output as the |
| * pointer. Falling back to the output containing 0, 0. |
| * |
| * TODO: Do something clever for touch too? |
| */ |
| wl_list_for_each(seat, &compositor->seat_list, link) { |
| struct weston_pointer *pointer = weston_seat_get_pointer(seat); |
| |
| if (pointer) { |
| ix = wl_fixed_to_int(pointer->x); |
| iy = wl_fixed_to_int(pointer->y); |
| break; |
| } |
| } |
| |
| wl_list_for_each(output, &compositor->output_list, link) { |
| if (pixman_region32_contains_point(&output->region, ix, iy, NULL)) { |
| target_output = output; |
| break; |
| } |
| } |
| |
| if (!target_output) { |
| weston_view_set_position(view, 10 + random() % 400, |
| 10 + random() % 400); |
| return; |
| } |
| |
| /* Valid range within output where the surface will still be onscreen. |
| * If this is negative it means that the surface is bigger than |
| * output. |
| */ |
| get_output_work_area(shell, target_output, &area); |
| |
| x = area.x; |
| y = area.y; |
| range_x = area.width - view->surface->width; |
| range_y = area.height - view->surface->height; |
| |
| if (range_x > 0) |
| x += random() % range_x; |
| |
| if (range_y > 0) |
| y += random() % range_y; |
| |
| weston_view_set_position(view, x, y); |
| } |
| |
| static bool |
| check_desktop_shell_crash_too_early(struct desktop_shell *shell) |
| { |
| struct timespec now; |
| |
| if (clock_gettime(CLOCK_MONOTONIC, &now) < 0) |
| return false; |
| |
| /* |
| * If the shell helper client dies before the session has been |
| * up for roughly 30 seconds, better just make Weston shut down, |
| * because the user likely has no way to interact with the desktop |
| * anyway. |
| */ |
| if (now.tv_sec - shell->startup_time.tv_sec < 30) { |
| weston_log("Error: %s apparently cannot run at all.\n", |
| shell->client); |
| weston_log_continue(STAMP_SPACE "Quitting..."); |
| wl_display_terminate(shell->compositor->wl_display); |
| |
| return true; |
| } |
| |
| return false; |
| } |
| |
| static void launch_desktop_shell_process(void *data); |
| |
| static void |
| respawn_desktop_shell_process(struct desktop_shell *shell) |
| { |
| struct timespec time; |
| |
| /* if desktop-shell dies more than 5 times in 30 seconds, give up */ |
| weston_compositor_get_time(&time); |
| if (timespec_sub_to_msec(&time, &shell->child.deathstamp) > 30000) { |
| shell->child.deathstamp = time; |
| shell->child.deathcount = 0; |
| } |
| |
| shell->child.deathcount++; |
| if (shell->child.deathcount > 5) { |
| weston_log("%s disconnected, giving up.\n", shell->client); |
| return; |
| } |
| |
| weston_log("%s disconnected, respawning...\n", shell->client); |
| launch_desktop_shell_process(shell); |
| } |
| |
| static void |
| desktop_shell_client_destroy(struct wl_listener *listener, void *data) |
| { |
| struct desktop_shell *shell; |
| |
| shell = container_of(listener, struct desktop_shell, |
| child.client_destroy_listener); |
| |
| wl_list_remove(&shell->child.client_destroy_listener.link); |
| shell->child.client = NULL; |
| /* |
| * unbind_desktop_shell() will reset shell->child.desktop_shell |
| * before the respawned process has a chance to create a new |
| * desktop_shell object, because we are being called from the |
| * wl_client destructor which destroys all wl_resources before |
| * returning. |
| */ |
| |
| if (!check_desktop_shell_crash_too_early(shell)) |
| respawn_desktop_shell_process(shell); |
| |
| shell_fade_startup(shell); |
| } |
| |
| static void |
| launch_desktop_shell_process(void *data) |
| { |
| struct desktop_shell *shell = data; |
| |
| shell->child.client = weston_client_start(shell->compositor, |
| shell->client); |
| |
| if (!shell->child.client) { |
| weston_log("not able to start %s\n", shell->client); |
| return; |
| } |
| |
| shell->child.client_destroy_listener.notify = |
| desktop_shell_client_destroy; |
| wl_client_add_destroy_listener(shell->child.client, |
| &shell->child.client_destroy_listener); |
| } |
| |
| static void |
| unbind_desktop_shell(struct wl_resource *resource) |
| { |
| struct desktop_shell *shell = wl_resource_get_user_data(resource); |
| |
| if (shell->locked) |
| resume_desktop(shell); |
| |
| shell->child.desktop_shell = NULL; |
| shell->prepare_event_sent = false; |
| } |
| |
| static void |
| bind_desktop_shell(struct wl_client *client, |
| void *data, uint32_t version, uint32_t id) |
| { |
| struct desktop_shell *shell = data; |
| struct wl_resource *resource; |
| |
| resource = wl_resource_create(client, &weston_desktop_shell_interface, |
| 1, id); |
| |
| if (client == shell->child.client) { |
| wl_resource_set_implementation(resource, |
| &desktop_shell_implementation, |
| shell, unbind_desktop_shell); |
| shell->child.desktop_shell = resource; |
| return; |
| } |
| |
| wl_resource_post_error(resource, WL_DISPLAY_ERROR_INVALID_OBJECT, |
| "permission to bind desktop_shell denied"); |
| } |
| |
| struct switcher { |
| struct desktop_shell *shell; |
| struct weston_view *current; |
| struct wl_listener listener; |
| struct weston_keyboard_grab grab; |
| struct wl_array minimized_array; |
| }; |
| |
| static void |
| switcher_next(struct switcher *switcher) |
| { |
| struct weston_view *view; |
| struct weston_view *first = NULL, *prev = NULL, *next = NULL; |
| struct shell_surface *shsurf; |
| struct workspace *ws = get_current_workspace(switcher->shell); |
| |
| /* temporary re-display minimized surfaces */ |
| struct weston_view *tmp; |
| struct weston_view **minimized; |
| wl_list_for_each_safe(view, tmp, &switcher->shell->minimized_layer.view_list.link, layer_link.link) { |
| weston_layer_entry_remove(&view->layer_link); |
| weston_layer_entry_insert(&ws->layer.view_list, &view->layer_link); |
| minimized = wl_array_add(&switcher->minimized_array, sizeof *minimized); |
| *minimized = view; |
| } |
| |
| wl_list_for_each(view, &ws->layer.view_list.link, layer_link.link) { |
| shsurf = get_shell_surface(view->surface); |
| if (shsurf) { |
| if (first == NULL) |
| first = view; |
| if (prev == switcher->current) |
| next = view; |
| prev = view; |
| view->alpha = 0.25; |
| weston_view_geometry_dirty(view); |
| weston_surface_damage(view->surface); |
| } |
| |
| if (is_black_surface_view(view, NULL)) { |
| view->alpha = 0.25; |
| weston_view_geometry_dirty(view); |
| weston_surface_damage(view->surface); |
| } |
| } |
| |
| if (next == NULL) |
| next = first; |
| |
| if (next == NULL) |
| return; |
| |
| wl_list_remove(&switcher->listener.link); |
| wl_signal_add(&next->destroy_signal, &switcher->listener); |
| |
| switcher->current = next; |
| wl_list_for_each(view, &next->surface->views, surface_link) |
| view->alpha = 1.0; |
| |
| shsurf = get_shell_surface(switcher->current->surface); |
| if (shsurf && weston_desktop_surface_get_fullscreen(shsurf->desktop_surface)) |
| shsurf->fullscreen.black_view->alpha = 1.0; |
| } |
| |
| static void |
| switcher_handle_view_destroy(struct wl_listener *listener, void *data) |
| { |
| struct switcher *switcher = |
| container_of(listener, struct switcher, listener); |
| |
| switcher_next(switcher); |
| } |
| |
| static void |
| switcher_destroy(struct switcher *switcher) |
| { |
| struct weston_view *view; |
| struct weston_keyboard *keyboard = switcher->grab.keyboard; |
| struct workspace *ws = get_current_workspace(switcher->shell); |
| |
| wl_list_for_each(view, &ws->layer.view_list.link, layer_link.link) { |
| if (is_focus_view(view)) |
| continue; |
| |
| view->alpha = 1.0; |
| weston_surface_damage(view->surface); |
| } |
| |
| if (switcher->current) { |
| activate(switcher->shell, switcher->current, |
| keyboard->seat, |
| WESTON_ACTIVATE_FLAG_CONFIGURE); |
| } |
| |
| wl_list_remove(&switcher->listener.link); |
| weston_keyboard_end_grab(keyboard); |
| if (keyboard->input_method_resource) |
| keyboard->grab = &keyboard->input_method_grab; |
| |
| /* re-hide surfaces that were temporary shown during the switch */ |
| struct weston_view **minimized; |
| wl_array_for_each(minimized, &switcher->minimized_array) { |
| /* with the exception of the current selected */ |
| if ((*minimized)->surface != switcher->current->surface) { |
| weston_layer_entry_remove(&(*minimized)->layer_link); |
| weston_layer_entry_insert(&switcher->shell->minimized_layer.view_list, &(*minimized)->layer_link); |
| weston_view_damage_below(*minimized); |
| } |
| } |
| wl_array_release(&switcher->minimized_array); |
| |
| free(switcher); |
| } |
| |
| static void |
| switcher_key(struct weston_keyboard_grab *grab, |
| const struct timespec *time, uint32_t key, uint32_t state_w) |
| { |
| struct switcher *switcher = container_of(grab, struct switcher, grab); |
| enum wl_keyboard_key_state state = state_w; |
| |
| if (key == KEY_TAB && state == WL_KEYBOARD_KEY_STATE_PRESSED) |
| switcher_next(switcher); |
| } |
| |
| static void |
| switcher_modifier(struct weston_keyboard_grab *grab, uint32_t serial, |
| uint32_t mods_depressed, uint32_t mods_latched, |
| uint32_t mods_locked, uint32_t group) |
| { |
| struct switcher *switcher = container_of(grab, struct switcher, grab); |
| struct weston_seat *seat = grab->keyboard->seat; |
| |
| if ((seat->modifier_state & switcher->shell->binding_modifier) == 0) |
| switcher_destroy(switcher); |
| } |
| |
| static void |
| switcher_cancel(struct weston_keyboard_grab *grab) |
| { |
| struct switcher *switcher = container_of(grab, struct switcher, grab); |
| |
| switcher_destroy(switcher); |
| } |
| |
| static const struct weston_keyboard_grab_interface switcher_grab = { |
| switcher_key, |
| switcher_modifier, |
| switcher_cancel, |
| }; |
| |
| static void |
| switcher_binding(struct weston_keyboard *keyboard, const struct timespec *time, |
| uint32_t key, void *data) |
| { |
| struct desktop_shell *shell = data; |
| struct switcher *switcher; |
| |
| switcher = malloc(sizeof *switcher); |
| switcher->shell = shell; |
| switcher->current = NULL; |
| switcher->listener.notify = switcher_handle_view_destroy; |
| wl_list_init(&switcher->listener.link); |
| wl_array_init(&switcher->minimized_array); |
| |
| lower_fullscreen_layer(switcher->shell, NULL); |
| switcher->grab.interface = &switcher_grab; |
| weston_keyboard_start_grab(keyboard, &switcher->grab); |
| weston_keyboard_set_focus(keyboard, NULL); |
| switcher_next(switcher); |
| } |
| |
| static void |
| backlight_binding(struct weston_keyboard *keyboard, const struct timespec *time, |
| uint32_t key, void *data) |
| { |
| struct weston_compositor *compositor = data; |
| struct weston_output *output; |
| long backlight_new = 0; |
| |
| /* TODO: we're limiting to simple use cases, where we assume just |
| * control on the primary display. We'd have to extend later if we |
| * ever get support for setting backlights on random desktop LCD |
| * panels though */ |
| output = get_default_output(compositor); |
| if (!output) |
| return; |
| |
| if (!output->set_backlight) |
| return; |
| |
| if (key == KEY_F9 || key == KEY_BRIGHTNESSDOWN) |
| backlight_new = output->backlight_current - 25; |
| else if (key == KEY_F10 || key == KEY_BRIGHTNESSUP) |
| backlight_new = output->backlight_current + 25; |
| |
| if (backlight_new < 5) |
| backlight_new = 5; |
| if (backlight_new > 255) |
| backlight_new = 255; |
| |
| output->backlight_current = backlight_new; |
| output->set_backlight(output, output->backlight_current); |
| } |
| |
| static void |
| force_kill_binding(struct weston_keyboard *keyboard, |
| const struct timespec *time, uint32_t key, void *data) |
| { |
| struct weston_surface *focus_surface; |
| struct wl_client *client; |
| struct desktop_shell *shell = data; |
| struct weston_compositor *compositor = shell->compositor; |
| pid_t pid; |
| |
| focus_surface = keyboard->focus; |
| if (!focus_surface) |
| return; |
| |
| wl_signal_emit(&compositor->kill_signal, focus_surface); |
| |
| client = wl_resource_get_client(focus_surface->resource); |
| wl_client_get_credentials(client, &pid, NULL, NULL); |
| |
| /* Skip clients that we launched ourselves (the credentials of |
| * the socketpair is ours) */ |
| if (pid == getpid()) |
| return; |
| |
| kill(pid, SIGKILL); |
| } |
| |
| static void |
| workspace_up_binding(struct weston_keyboard *keyboard, |
| const struct timespec *time, uint32_t key, void *data) |
| { |
| struct desktop_shell *shell = data; |
| unsigned int new_index = shell->workspaces.current; |
| |
| if (shell->locked) |
| return; |
| if (new_index != 0) |
| new_index--; |
| |
| change_workspace(shell, new_index); |
| } |
| |
| static void |
| workspace_down_binding(struct weston_keyboard *keyboard, |
| const struct timespec *time, uint32_t key, void *data) |
| { |
| struct desktop_shell *shell = data; |
| unsigned int new_index = shell->workspaces.current; |
| |
| if (shell->locked) |
| return; |
| if (new_index < shell->workspaces.num - 1) |
| new_index++; |
| |
| change_workspace(shell, new_index); |
| } |
| |
| static void |
| workspace_f_binding(struct weston_keyboard *keyboard, |
| const struct timespec *time, uint32_t key, void *data) |
| { |
| struct desktop_shell *shell = data; |
| unsigned int new_index; |
| |
| if (shell->locked) |
| return; |
| new_index = key - KEY_F1; |
| if (new_index >= shell->workspaces.num) |
| new_index = shell->workspaces.num - 1; |
| |
| change_workspace(shell, new_index); |
| } |
| |
| static void |
| workspace_move_surface_up_binding(struct weston_keyboard *keyboard, |
| const struct timespec *time, uint32_t key, |
| void *data) |
| { |
| struct desktop_shell *shell = data; |
| unsigned int new_index = shell->workspaces.current; |
| |
| if (shell->locked) |
| return; |
| |
| if (new_index != 0) |
| new_index--; |
| |
| take_surface_to_workspace_by_seat(shell, keyboard->seat, new_index); |
| } |
| |
| static void |
| workspace_move_surface_down_binding(struct weston_keyboard *keyboard, |
| const struct timespec *time, uint32_t key, |
| void *data) |
| { |
| struct desktop_shell *shell = data; |
| unsigned int new_index = shell->workspaces.current; |
| |
| if (shell->locked) |
| return; |
| |
| if (new_index < shell->workspaces.num - 1) |
| new_index++; |
| |
| take_surface_to_workspace_by_seat(shell, keyboard->seat, new_index); |
| } |
| |
| static void |
| shell_reposition_view_on_output_destroy(struct weston_view *view) |
| { |
| struct weston_output *output, *first_output; |
| struct weston_compositor *ec = view->surface->compositor; |
| struct shell_surface *shsurf; |
| float x, y; |
| int visible; |
| |
| x = view->geometry.x; |
| y = view->geometry.y; |
| |
| /* At this point the destroyed output is not in the list anymore. |
| * If the view is still visible somewhere, we leave where it is, |
| * otherwise, move it to the first output. */ |
| visible = 0; |
| wl_list_for_each(output, &ec->output_list, link) { |
| if (pixman_region32_contains_point(&output->region, |
| x, y, NULL)) { |
| visible = 1; |
| break; |
| } |
| } |
| |
| if (!visible) { |
| first_output = container_of(ec->output_list.next, |
| struct weston_output, link); |
| |
| x = first_output->x + first_output->width / 4; |
| y = first_output->y + first_output->height / 4; |
| |
| weston_view_set_position(view, x, y); |
| } else { |
| weston_view_geometry_dirty(view); |
| } |
| |
| |
| shsurf = get_shell_surface(view->surface); |
| if (!shsurf) |
| return; |
| |
| shsurf->saved_position_valid = false; |
| set_maximized(shsurf, false); |
| set_fullscreen(shsurf, false, NULL); |
| } |
| |
| void |
| shell_for_each_layer(struct desktop_shell *shell, |
| shell_for_each_layer_func_t func, void *data) |
| { |
| struct workspace **ws; |
| |
| func(shell, &shell->fullscreen_layer, data); |
| func(shell, &shell->panel_layer, data); |
| func(shell, &shell->background_layer, data); |
| func(shell, &shell->lock_layer, data); |
| func(shell, &shell->input_panel_layer, data); |
| |
| wl_array_for_each(ws, &shell->workspaces.array) |
| func(shell, &(*ws)->layer, data); |
| } |
| |
| static void |
| shell_output_destroy_move_layer(struct desktop_shell *shell, |
| struct weston_layer *layer, |
| void *data) |
| { |
| struct weston_view *view; |
| |
| wl_list_for_each(view, &layer->view_list.link, layer_link.link) |
| shell_reposition_view_on_output_destroy(view); |
| } |
| |
| static void |
| handle_output_destroy(struct wl_listener *listener, void *data) |
| { |
| struct shell_output *output_listener = |
| container_of(listener, struct shell_output, destroy_listener); |
| struct desktop_shell *shell = output_listener->shell; |
| |
| shell_for_each_layer(shell, shell_output_destroy_move_layer, NULL); |
| |
| if (output_listener->panel_surface) |
| wl_list_remove(&output_listener->panel_surface_listener.link); |
| if (output_listener->background_surface) |
| wl_list_remove(&output_listener->background_surface_listener.link); |
| wl_list_remove(&output_listener->destroy_listener.link); |
| wl_list_remove(&output_listener->link); |
| free(output_listener); |
| } |
| |
| static void |
| shell_resize_surface_to_output(struct desktop_shell *shell, |
| struct weston_surface *surface, |
| const struct weston_output *output) |
| { |
| if (!surface) |
| return; |
| |
| weston_desktop_shell_send_configure(shell->child.desktop_shell, 0, |
| surface->resource, |
| output->width, |
| output->height); |
| } |
| |
| |
| static void |
| handle_output_resized(struct wl_listener *listener, void *data) |
| { |
| struct desktop_shell *shell = |
| container_of(listener, struct desktop_shell, resized_listener); |
| struct weston_output *output = (struct weston_output *)data; |
| struct shell_output *sh_output = find_shell_output_from_weston_output(shell, output); |
| |
| shell_resize_surface_to_output(shell, sh_output->background_surface, output); |
| shell_resize_surface_to_output(shell, sh_output->panel_surface, output); |
| } |
| |
| static void |
| create_shell_output(struct desktop_shell *shell, |
| struct weston_output *output) |
| { |
| struct shell_output *shell_output; |
| |
| shell_output = zalloc(sizeof *shell_output); |
| if (shell_output == NULL) |
| return; |
| |
| shell_output->output = output; |
| shell_output->shell = shell; |
| shell_output->destroy_listener.notify = handle_output_destroy; |
| wl_signal_add(&output->destroy_signal, |
| &shell_output->destroy_listener); |
| wl_list_insert(shell->output_list.prev, &shell_output->link); |
| } |
| |
| static void |
| handle_output_create(struct wl_listener *listener, void *data) |
| { |
| struct desktop_shell *shell = |
| container_of(listener, struct desktop_shell, output_create_listener); |
| struct weston_output *output = (struct weston_output *)data; |
| |
| create_shell_output(shell, output); |
| } |
| |
| static void |
| handle_output_move_layer(struct desktop_shell *shell, |
| struct weston_layer *layer, void *data) |
| { |
| struct weston_output *output = data; |
| struct weston_view *view; |
| float x, y; |
| |
| wl_list_for_each(view, &layer->view_list.link, layer_link.link) { |
| if (view->output != output) |
| continue; |
| |
| x = view->geometry.x + output->move_x; |
| y = view->geometry.y + output->move_y; |
| weston_view_set_position(view, x, y); |
| } |
| } |
| |
| static void |
| handle_output_move(struct wl_listener *listener, void *data) |
| { |
| struct desktop_shell *shell; |
| |
| shell = container_of(listener, struct desktop_shell, |
| output_move_listener); |
| |
| shell_for_each_layer(shell, handle_output_move_layer, data); |
| } |
| |
| static void |
| setup_output_destroy_handler(struct weston_compositor *ec, |
| struct desktop_shell *shell) |
| { |
| struct weston_output *output; |
| |
| wl_list_init(&shell->output_list); |
| wl_list_for_each(output, &ec->output_list, link) |
| create_shell_output(shell, output); |
| |
| shell->output_create_listener.notify = handle_output_create; |
| wl_signal_add(&ec->output_created_signal, |
| &shell->output_create_listener); |
| |
| shell->output_move_listener.notify = handle_output_move; |
| wl_signal_add(&ec->output_moved_signal, &shell->output_move_listener); |
| } |
| |
| static void |
| shell_destroy(struct wl_listener *listener, void *data) |
| { |
| struct desktop_shell *shell = |
| container_of(listener, struct desktop_shell, destroy_listener); |
| struct workspace **ws; |
| struct shell_output *shell_output, *tmp; |
| |
| /* Force state to unlocked so we don't try to fade */ |
| shell->locked = false; |
| |
| if (shell->child.client) { |
| /* disable respawn */ |
| wl_list_remove(&shell->child.client_destroy_listener.link); |
| wl_client_destroy(shell->child.client); |
| } |
| |
| wl_list_remove(&shell->idle_listener.link); |
| wl_list_remove(&shell->wake_listener.link); |
| wl_list_remove(&shell->transform_listener.link); |
| |
| text_backend_destroy(shell->text_backend); |
| input_panel_destroy(shell); |
| |
| wl_list_for_each_safe(shell_output, tmp, &shell->output_list, link) { |
| wl_list_remove(&shell_output->destroy_listener.link); |
| wl_list_remove(&shell_output->link); |
| free(shell_output); |
| } |
| |
| wl_list_remove(&shell->output_create_listener.link); |
| wl_list_remove(&shell->output_move_listener.link); |
| wl_list_remove(&shell->resized_listener.link); |
| |
| wl_array_for_each(ws, &shell->workspaces.array) |
| workspace_destroy(*ws); |
| wl_array_release(&shell->workspaces.array); |
| |
| free(shell->client); |
| free(shell); |
| } |
| |
| static void |
| shell_add_bindings(struct weston_compositor *ec, struct desktop_shell *shell) |
| { |
| uint32_t mod; |
| int i, num_workspace_bindings; |
| |
| if (shell->allow_zap) |
| weston_compositor_add_key_binding(ec, KEY_BACKSPACE, |
| MODIFIER_CTRL | MODIFIER_ALT, |
| terminate_binding, ec); |
| |
| /* fixed bindings */ |
| weston_compositor_add_button_binding(ec, BTN_LEFT, 0, |
| click_to_activate_binding, |
| shell); |
| weston_compositor_add_button_binding(ec, BTN_RIGHT, 0, |
| click_to_activate_binding, |
| shell); |
| weston_compositor_add_touch_binding(ec, 0, |
| touch_to_activate_binding, |
| shell); |
| weston_compositor_add_key_binding(ec, KEY_BRIGHTNESSDOWN, 0, |
| backlight_binding, ec); |
| weston_compositor_add_key_binding(ec, KEY_BRIGHTNESSUP, 0, |
| backlight_binding, ec); |
| |
| /* configurable bindings */ |
| if (shell->exposay_modifier) |
| weston_compositor_add_modifier_binding(ec, shell->exposay_modifier, |
| exposay_binding, shell); |
| |
| mod = shell->binding_modifier; |
| if (!mod) |
| return; |
| |
| /* This binding is not configurable, but is only enabled if there is a |
| * valid binding modifier. */ |
| weston_compositor_add_axis_binding(ec, WL_POINTER_AXIS_VERTICAL_SCROLL, |
| MODIFIER_SUPER | MODIFIER_ALT, |
| surface_opacity_binding, NULL); |
| |
| weston_compositor_add_axis_binding(ec, WL_POINTER_AXIS_VERTICAL_SCROLL, |
| mod, zoom_axis_binding, |
| NULL); |
| |
| weston_compositor_add_key_binding(ec, KEY_PAGEUP, mod, |
| zoom_key_binding, NULL); |
| weston_compositor_add_key_binding(ec, KEY_PAGEDOWN, mod, |
| zoom_key_binding, NULL); |
| weston_compositor_add_key_binding(ec, KEY_M, mod | MODIFIER_SHIFT, |
| maximize_binding, NULL); |
| weston_compositor_add_key_binding(ec, KEY_F, mod | MODIFIER_SHIFT, |
| fullscreen_binding, NULL); |
| weston_compositor_add_button_binding(ec, BTN_LEFT, mod, move_binding, |
| shell); |
| weston_compositor_add_touch_binding(ec, mod, touch_move_binding, shell); |
| weston_compositor_add_button_binding(ec, BTN_RIGHT, mod, |
| resize_binding, shell); |
| weston_compositor_add_button_binding(ec, BTN_LEFT, |
| mod | MODIFIER_SHIFT, |
| resize_binding, shell); |
| |
| if (ec->capabilities & WESTON_CAP_ROTATION_ANY) |
| weston_compositor_add_button_binding(ec, BTN_MIDDLE, mod, |
| rotate_binding, NULL); |
| |
| weston_compositor_add_key_binding(ec, KEY_TAB, mod, switcher_binding, |
| shell); |
| weston_compositor_add_key_binding(ec, KEY_F9, mod, backlight_binding, |
| ec); |
| weston_compositor_add_key_binding(ec, KEY_F10, mod, backlight_binding, |
| ec); |
| weston_compositor_add_key_binding(ec, KEY_K, mod, |
| force_kill_binding, shell); |
| weston_compositor_add_key_binding(ec, KEY_UP, mod, |
| workspace_up_binding, shell); |
| weston_compositor_add_key_binding(ec, KEY_DOWN, mod, |
| workspace_down_binding, shell); |
| weston_compositor_add_key_binding(ec, KEY_UP, mod | MODIFIER_SHIFT, |
| workspace_move_surface_up_binding, |
| shell); |
| weston_compositor_add_key_binding(ec, KEY_DOWN, mod | MODIFIER_SHIFT, |
| workspace_move_surface_down_binding, |
| shell); |
| |
| /* Add bindings for mod+F[1-6] for workspace 1 to 6. */ |
| if (shell->workspaces.num > 1) { |
| num_workspace_bindings = shell->workspaces.num; |
| if (num_workspace_bindings > 6) |
| num_workspace_bindings = 6; |
| for (i = 0; i < num_workspace_bindings; i++) |
| weston_compositor_add_key_binding(ec, KEY_F1 + i, mod, |
| workspace_f_binding, |
| shell); |
| } |
| |
| weston_install_debug_key_binding(ec, mod); |
| } |
| |
| static void |
| handle_seat_created(struct wl_listener *listener, void *data) |
| { |
| struct weston_seat *seat = data; |
| |
| create_shell_seat(seat); |
| } |
| |
| WL_EXPORT int |
| wet_shell_init(struct weston_compositor *ec, |
| int *argc, char *argv[]) |
| { |
| struct weston_seat *seat; |
| struct desktop_shell *shell; |
| struct workspace **pws; |
| unsigned int i; |
| struct wl_event_loop *loop; |
| |
| shell = zalloc(sizeof *shell); |
| if (shell == NULL) |
| return -1; |
| |
| shell->compositor = ec; |
| |
| shell->destroy_listener.notify = shell_destroy; |
| wl_signal_add(&ec->destroy_signal, &shell->destroy_listener); |
| shell->idle_listener.notify = idle_handler; |
| wl_signal_add(&ec->idle_signal, &shell->idle_listener); |
| shell->wake_listener.notify = wake_handler; |
| wl_signal_add(&ec->wake_signal, &shell->wake_listener); |
| shell->transform_listener.notify = transform_handler; |
| wl_signal_add(&ec->transform_signal, &shell->transform_listener); |
| |
| weston_layer_init(&shell->fullscreen_layer, ec); |
| weston_layer_init(&shell->panel_layer, ec); |
| weston_layer_init(&shell->background_layer, ec); |
| weston_layer_init(&shell->lock_layer, ec); |
| weston_layer_init(&shell->input_panel_layer, ec); |
| |
| weston_layer_set_position(&shell->fullscreen_layer, |
| WESTON_LAYER_POSITION_FULLSCREEN); |
| weston_layer_set_position(&shell->panel_layer, |
| WESTON_LAYER_POSITION_UI); |
| weston_layer_set_position(&shell->background_layer, |
| WESTON_LAYER_POSITION_BACKGROUND); |
| |
| wl_array_init(&shell->workspaces.array); |
| wl_list_init(&shell->workspaces.client_list); |
| |
| if (input_panel_setup(shell) < 0) |
| return -1; |
| |
| shell->text_backend = text_backend_init(ec); |
| if (!shell->text_backend) |
| return -1; |
| |
| shell_configuration(shell); |
| |
| shell->exposay.state_cur = EXPOSAY_LAYOUT_INACTIVE; |
| shell->exposay.state_target = EXPOSAY_TARGET_CANCEL; |
| |
| for (i = 0; i < shell->workspaces.num; i++) { |
| pws = wl_array_add(&shell->workspaces.array, sizeof *pws); |
| if (pws == NULL) |
| return -1; |
| |
| *pws = workspace_create(shell); |
| if (*pws == NULL) |
| return -1; |
| } |
| activate_workspace(shell, 0); |
| |
| weston_layer_init(&shell->minimized_layer, ec); |
| |
| wl_list_init(&shell->workspaces.anim_sticky_list); |
| wl_list_init(&shell->workspaces.animation.link); |
| shell->workspaces.animation.frame = animate_workspace_change_frame; |
| |
| shell->desktop = weston_desktop_create(ec, &shell_desktop_api, shell); |
| if (!shell->desktop) |
| return -1; |
| |
| if (wl_global_create(ec->wl_display, |
| &weston_desktop_shell_interface, 1, |
| shell, bind_desktop_shell) == NULL) |
| return -1; |
| |
| weston_compositor_get_time(&shell->child.deathstamp); |
| |
| shell->panel_position = WESTON_DESKTOP_SHELL_PANEL_POSITION_TOP; |
| |
| setup_output_destroy_handler(ec, shell); |
| |
| loop = wl_display_get_event_loop(ec->wl_display); |
| wl_event_loop_add_idle(loop, launch_desktop_shell_process, shell); |
| |
| wl_list_for_each(seat, &ec->seat_list, link) |
| handle_seat_created(NULL, seat); |
| shell->seat_create_listener.notify = handle_seat_created; |
| wl_signal_add(&ec->seat_created_signal, &shell->seat_create_listener); |
| |
| shell->resized_listener.notify = handle_output_resized; |
| wl_signal_add(&ec->output_resized_signal, &shell->resized_listener); |
| |
| screenshooter_create(ec); |
| |
| shell_add_bindings(ec, shell); |
| |
| shell_fade_init(shell); |
| |
| clock_gettime(CLOCK_MONOTONIC, &shell->startup_time); |
| |
| return 0; |
| } |