/* Copyright (c), Niclas Meyer * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this file, * You can obtain one at https://mozilla.org/MPL/2.0/. */ #include "input/cursor.h" #include #include #include #include #include #include #include #include #include #include "desktop/desktop.h" #include "desktop/layer_shell.h" #include "desktop/output.h" #include "desktop/view.h" #include "input/seat.h" #include "server.h" static void process_cursor_motion(struct kiwmi_server *server, uint32_t time) { struct kiwmi_input *input = &server->input; struct kiwmi_cursor *cursor = input->cursor; struct wlr_seat *seat = input->seat->seat; switch (cursor->cursor_mode) { case KIWMI_CURSOR_MOVE: { struct kiwmi_view *view = cursor->grabbed.view; view_set_pos( view, cursor->cursor->x - cursor->grabbed.orig_x, cursor->cursor->y - cursor->grabbed.orig_y); return; } case KIWMI_CURSOR_RESIZE: { struct kiwmi_view *view = cursor->grabbed.view; int dx = cursor->cursor->x - cursor->grabbed.orig_x; int dy = cursor->cursor->y - cursor->grabbed.orig_y; struct wlr_box new_geom = { .x = cursor->grabbed.orig_geom.x, .y = cursor->grabbed.orig_geom.y, .width = cursor->grabbed.orig_geom.width, .height = cursor->grabbed.orig_geom.height, }; if (cursor->grabbed.resize_edges & WLR_EDGE_TOP) { new_geom.y += dy; new_geom.height -= dy; if (new_geom.height < 1) { new_geom.y += new_geom.height; } } if (cursor->grabbed.resize_edges & WLR_EDGE_BOTTOM) { new_geom.height += dy; } if (cursor->grabbed.resize_edges & WLR_EDGE_LEFT) { new_geom.x += dx; new_geom.width -= dx; if (new_geom.width < 1) { new_geom.x += new_geom.width; } } if (cursor->grabbed.resize_edges & WLR_EDGE_RIGHT) { new_geom.width += dx; } view_set_pos(view, new_geom.x, new_geom.y); view_set_size(view, new_geom.width, new_geom.height); return; } default: // EMPTY break; } struct wlr_surface *old_focus = seat->pointer_state.focused_surface; struct wlr_surface *new_focus; double sx, sy; cursor_refresh_focus(cursor, &new_focus, &sx, &sy); if (new_focus && new_focus == old_focus) { wlr_seat_pointer_notify_enter(seat, new_focus, sx, sy); wlr_seat_pointer_notify_motion(seat, time, sx, sy); } } static void cursor_motion_notify(struct wl_listener *listener, void *data) { struct kiwmi_cursor *cursor = wl_container_of(listener, cursor, cursor_motion); struct kiwmi_server *server = cursor->server; struct wlr_event_pointer_motion *event = data; struct kiwmi_cursor_motion_event new_event = { .oldx = cursor->cursor->x, .oldy = cursor->cursor->y, }; wlr_cursor_move( cursor->cursor, event->device, event->delta_x, event->delta_y); new_event.newx = cursor->cursor->x; new_event.newy = cursor->cursor->y; wl_signal_emit(&cursor->events.motion, &new_event); process_cursor_motion(server, event->time_msec); } static void cursor_motion_absolute_notify(struct wl_listener *listener, void *data) { struct kiwmi_cursor *cursor = wl_container_of(listener, cursor, cursor_motion_absolute); struct kiwmi_server *server = cursor->server; struct wlr_event_pointer_motion_absolute *event = data; struct kiwmi_cursor_motion_event new_event = { .oldx = cursor->cursor->x, .oldy = cursor->cursor->y, }; wlr_cursor_warp_absolute(cursor->cursor, event->device, event->x, event->y); new_event.newx = cursor->cursor->x; new_event.newy = cursor->cursor->y; wl_signal_emit(&cursor->events.motion, &new_event); process_cursor_motion(server, event->time_msec); } static void cursor_button_notify(struct wl_listener *listener, void *data) { struct kiwmi_cursor *cursor = wl_container_of(listener, cursor, cursor_button); struct kiwmi_server *server = cursor->server; struct kiwmi_input *input = &server->input; struct wlr_event_pointer_button *event = data; struct kiwmi_cursor_button_event new_event = { .wlr_event = event, .handled = false, }; if (event->state == WLR_BUTTON_PRESSED) { wl_signal_emit(&cursor->events.button_down, &new_event); } else { wl_signal_emit(&cursor->events.button_up, &new_event); } if (!new_event.handled) { wlr_seat_pointer_notify_button( input->seat->seat, event->time_msec, event->button, event->state); } } static void cursor_axis_notify(struct wl_listener *listener, void *data) { struct kiwmi_cursor *cursor = wl_container_of(listener, cursor, cursor_axis); struct kiwmi_server *server = cursor->server; struct kiwmi_input *input = &server->input; struct wlr_event_pointer_axis *event = data; struct kiwmi_cursor_scroll_event new_event = { .device_name = event->device->name, .is_vertical = event->orientation == WLR_AXIS_ORIENTATION_VERTICAL, .length = event->delta, .handled = false, }; wl_signal_emit(&cursor->events.scroll, &new_event); if (!new_event.handled) { wlr_seat_pointer_notify_axis( input->seat->seat, event->time_msec, event->orientation, event->delta, event->delta_discrete, event->source); } } static void cursor_frame_notify(struct wl_listener *listener, void *UNUSED(data)) { struct kiwmi_cursor *cursor = wl_container_of(listener, cursor, cursor_frame); struct kiwmi_server *server = cursor->server; struct kiwmi_input *input = &server->input; wlr_seat_pointer_notify_frame(input->seat->seat); } struct kiwmi_cursor * cursor_create( struct kiwmi_server *server, struct wlr_output_layout *output_layout) { wlr_log(WLR_DEBUG, "Creating cursor"); struct kiwmi_cursor *cursor = malloc(sizeof(*cursor)); if (!cursor) { wlr_log(WLR_ERROR, "Failed to allocate kiwmi_cursor"); return NULL; } cursor->server = server; cursor->cursor_mode = KIWMI_CURSOR_PASSTHROUGH; cursor->cursor = wlr_cursor_create(); if (!cursor->cursor) { wlr_log(WLR_ERROR, "Failed to create cursor"); free(cursor); return NULL; } wlr_cursor_attach_output_layout(cursor->cursor, output_layout); cursor->xcursor_manager = wlr_xcursor_manager_create(NULL, 24); cursor->cursor_motion.notify = cursor_motion_notify; wl_signal_add(&cursor->cursor->events.motion, &cursor->cursor_motion); cursor->cursor_motion_absolute.notify = cursor_motion_absolute_notify; wl_signal_add( &cursor->cursor->events.motion_absolute, &cursor->cursor_motion_absolute); cursor->cursor_button.notify = cursor_button_notify; wl_signal_add(&cursor->cursor->events.button, &cursor->cursor_button); cursor->cursor_axis.notify = cursor_axis_notify; wl_signal_add(&cursor->cursor->events.axis, &cursor->cursor_axis); cursor->cursor_frame.notify = cursor_frame_notify; wl_signal_add(&cursor->cursor->events.frame, &cursor->cursor_frame); wl_signal_init(&cursor->events.button_down); wl_signal_init(&cursor->events.button_up); wl_signal_init(&cursor->events.destroy); wl_signal_init(&cursor->events.motion); wl_signal_init(&cursor->events.scroll); return cursor; } void cursor_destroy(struct kiwmi_cursor *cursor) { wl_signal_emit(&cursor->events.destroy, cursor); wlr_cursor_destroy(cursor->cursor); wlr_xcursor_manager_destroy(cursor->xcursor_manager); // The wlr_cursor is already destroyed, don't unregister listeners free(cursor); } void cursor_refresh_focus( struct kiwmi_cursor *cursor, struct wlr_surface **new_surface, double *cursor_sx, double *cursor_sy) { struct kiwmi_desktop *desktop = &cursor->server->desktop; struct wlr_seat *seat = cursor->server->input.seat->seat; double ox = cursor->cursor->x; double oy = cursor->cursor->y; struct wlr_output *wlr_output = wlr_output_layout_output_at( desktop->output_layout, cursor->cursor->x, cursor->cursor->y); wlr_output_layout_output_coords( desktop->output_layout, wlr_output, &ox, &oy); struct kiwmi_output *output = wlr_output->data; struct wlr_surface *surface = NULL; double sx; double sy; struct kiwmi_layer *layer = layer_at( &output->layers[ZWLR_LAYER_SHELL_V1_LAYER_OVERLAY], &surface, ox, oy, &sx, &sy); struct kiwmi_view *view; if (!layer) { layer = layer_at( &output->layers[ZWLR_LAYER_SHELL_V1_LAYER_TOP], &surface, ox, oy, &sx, &sy); } if (!layer) { view = view_at( desktop, cursor->cursor->x, cursor->cursor->y, &surface, &sx, &sy); } if (!layer) { layer = layer_at( &output->layers[ZWLR_LAYER_SHELL_V1_LAYER_BOTTOM], &surface, ox, oy, &sx, &sy); } if (!layer) { layer = layer_at( &output->layers[ZWLR_LAYER_SHELL_V1_LAYER_BACKGROUND], &surface, ox, oy, &sx, &sy); } if (!layer && !view) { wlr_xcursor_manager_set_cursor_image( cursor->xcursor_manager, "left_ptr", cursor->cursor); } if (surface && surface != seat->pointer_state.focused_surface) { wlr_seat_pointer_notify_enter(seat, surface, sx, sy); } else if (!surface) { wlr_seat_pointer_clear_focus(seat); } if (new_surface) { *new_surface = surface; } if (cursor_sx) { *cursor_sx = surface ? sx : 0; } if (cursor_sy) { *cursor_sy = surface ? sy : 0; } }