2023-09-03 18:56:20 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2025-07-28 01:02:01 -04:00
|
|
|
#include "input/gestures.h"
|
2025-09-06 20:06:32 -04:00
|
|
|
#include <wlr/types/wlr_cursor.h>
|
2023-09-03 18:56:20 +02:00
|
|
|
#include <wlr/types/wlr_pointer_gestures_v1.h>
|
2025-05-18 04:41:57 +02:00
|
|
|
#include "action.h"
|
2024-11-10 19:05:55 +01:00
|
|
|
#include "common/macros.h"
|
2025-05-18 04:41:57 +02:00
|
|
|
#include "config/mousebind.h"
|
|
|
|
|
#include "config/rcxml.h"
|
|
|
|
|
#include "input/gestures.h"
|
2023-09-03 18:56:20 +02:00
|
|
|
#include "labwc.h"
|
2024-11-10 19:07:27 +01:00
|
|
|
#include "idle.h"
|
2023-09-03 18:56:20 +02:00
|
|
|
|
2025-05-18 04:41:57 +02:00
|
|
|
static struct {
|
|
|
|
|
double dx;
|
|
|
|
|
double dy;
|
|
|
|
|
uint32_t fingers;
|
|
|
|
|
} swipe_state;
|
|
|
|
|
|
2023-09-03 18:56:20 +02:00
|
|
|
static void
|
2024-11-10 19:05:55 +01:00
|
|
|
handle_pinch_begin(struct wl_listener *listener, void *data)
|
2023-09-03 18:56:20 +02:00
|
|
|
{
|
|
|
|
|
struct seat *seat = wl_container_of(listener, seat, pinch_begin);
|
|
|
|
|
struct wlr_pointer_pinch_begin_event *event = data;
|
2024-11-10 19:07:27 +01:00
|
|
|
|
|
|
|
|
idle_manager_notify_activity(seat->seat);
|
2024-11-10 19:07:56 +01:00
|
|
|
cursor_set_visible(seat, /* visible */ true);
|
2024-11-10 19:07:27 +01:00
|
|
|
|
2023-09-03 18:56:20 +02:00
|
|
|
wlr_pointer_gestures_v1_send_pinch_begin(seat->pointer_gestures,
|
|
|
|
|
seat->seat, event->time_msec, event->fingers);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2024-11-10 19:05:55 +01:00
|
|
|
handle_pinch_update(struct wl_listener *listener, void *data)
|
2023-09-03 18:56:20 +02:00
|
|
|
{
|
|
|
|
|
struct seat *seat = wl_container_of(listener, seat, pinch_update);
|
|
|
|
|
struct wlr_pointer_pinch_update_event *event = data;
|
2024-11-10 19:07:27 +01:00
|
|
|
|
|
|
|
|
idle_manager_notify_activity(seat->seat);
|
2024-11-10 19:07:56 +01:00
|
|
|
cursor_set_visible(seat, /* visible */ true);
|
2024-11-10 19:07:27 +01:00
|
|
|
|
2023-09-03 18:56:20 +02:00
|
|
|
wlr_pointer_gestures_v1_send_pinch_update(seat->pointer_gestures,
|
|
|
|
|
seat->seat, event->time_msec, event->dx, event->dy,
|
|
|
|
|
event->scale, event->rotation);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2024-11-10 19:05:55 +01:00
|
|
|
handle_pinch_end(struct wl_listener *listener, void *data)
|
2023-09-03 18:56:20 +02:00
|
|
|
{
|
|
|
|
|
struct seat *seat = wl_container_of(listener, seat, pinch_end);
|
|
|
|
|
struct wlr_pointer_pinch_end_event *event = data;
|
2024-11-10 19:07:27 +01:00
|
|
|
|
|
|
|
|
idle_manager_notify_activity(seat->seat);
|
2024-11-10 19:07:56 +01:00
|
|
|
cursor_set_visible(seat, /* visible */ true);
|
2024-11-10 19:07:27 +01:00
|
|
|
|
2023-09-03 18:56:20 +02:00
|
|
|
wlr_pointer_gestures_v1_send_pinch_end(seat->pointer_gestures,
|
|
|
|
|
seat->seat, event->time_msec, event->cancelled);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2024-11-10 19:05:55 +01:00
|
|
|
handle_swipe_begin(struct wl_listener *listener, void *data)
|
2023-09-03 18:56:20 +02:00
|
|
|
{
|
|
|
|
|
struct seat *seat = wl_container_of(listener, seat, swipe_begin);
|
|
|
|
|
struct wlr_pointer_swipe_begin_event *event = data;
|
2024-11-10 19:07:27 +01:00
|
|
|
|
|
|
|
|
idle_manager_notify_activity(seat->seat);
|
2024-11-10 19:07:56 +01:00
|
|
|
cursor_set_visible(seat, /* visible */ true);
|
2024-11-10 19:07:27 +01:00
|
|
|
|
2023-09-03 18:56:20 +02:00
|
|
|
wlr_pointer_gestures_v1_send_swipe_begin(seat->pointer_gestures,
|
|
|
|
|
seat->seat, event->time_msec, event->fingers);
|
2025-05-18 04:41:57 +02:00
|
|
|
|
|
|
|
|
swipe_state.dx = 0;
|
|
|
|
|
swipe_state.dy = 0;
|
|
|
|
|
swipe_state.fingers = event->fingers;
|
2023-09-03 18:56:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2024-11-10 19:05:55 +01:00
|
|
|
handle_swipe_update(struct wl_listener *listener, void *data)
|
2023-09-03 18:56:20 +02:00
|
|
|
{
|
|
|
|
|
struct seat *seat = wl_container_of(listener, seat, swipe_update);
|
|
|
|
|
struct wlr_pointer_swipe_update_event *event = data;
|
2024-11-10 19:07:27 +01:00
|
|
|
|
|
|
|
|
idle_manager_notify_activity(seat->seat);
|
2024-11-10 19:07:56 +01:00
|
|
|
cursor_set_visible(seat, /* visible */ true);
|
2024-11-10 19:07:27 +01:00
|
|
|
|
2023-09-03 18:56:20 +02:00
|
|
|
wlr_pointer_gestures_v1_send_swipe_update(seat->pointer_gestures,
|
|
|
|
|
seat->seat, event->time_msec, event->dx, event->dy);
|
2025-05-18 04:41:57 +02:00
|
|
|
|
|
|
|
|
if (swipe_state.fingers == event->fingers) {
|
|
|
|
|
swipe_state.dx += event->dx;
|
|
|
|
|
swipe_state.dy += event->dy;
|
|
|
|
|
} else {
|
|
|
|
|
swipe_state.dx = 0;
|
|
|
|
|
swipe_state.dy = 0;
|
|
|
|
|
}
|
2023-09-03 18:56:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2024-11-10 19:05:55 +01:00
|
|
|
handle_swipe_end(struct wl_listener *listener, void *data)
|
2023-09-03 18:56:20 +02:00
|
|
|
{
|
|
|
|
|
struct seat *seat = wl_container_of(listener, seat, swipe_end);
|
|
|
|
|
struct wlr_pointer_swipe_end_event *event = data;
|
2024-11-10 19:07:27 +01:00
|
|
|
|
|
|
|
|
idle_manager_notify_activity(seat->seat);
|
2024-11-10 19:07:56 +01:00
|
|
|
cursor_set_visible(seat, /* visible */ true);
|
2024-11-10 19:07:27 +01:00
|
|
|
|
2023-09-03 18:56:20 +02:00
|
|
|
wlr_pointer_gestures_v1_send_swipe_end(seat->pointer_gestures,
|
|
|
|
|
seat->seat, event->time_msec, event->cancelled);
|
2025-05-18 04:41:57 +02:00
|
|
|
|
|
|
|
|
// TODO: check for !event->cancelled ?
|
|
|
|
|
if (swipe_state.dx || swipe_state.dy) {
|
|
|
|
|
enum direction direction;
|
|
|
|
|
if (swipe_state.dx * swipe_state.dx > swipe_state.dy * swipe_state.dy) {
|
|
|
|
|
direction = swipe_state.dx > 0
|
|
|
|
|
? LAB_DIRECTION_RIGHT
|
|
|
|
|
: LAB_DIRECTION_LEFT;
|
|
|
|
|
} else {
|
|
|
|
|
direction = swipe_state.dy > 0
|
|
|
|
|
? LAB_DIRECTION_DOWN
|
|
|
|
|
: LAB_DIRECTION_UP;
|
|
|
|
|
}
|
|
|
|
|
struct mousebind *mousebind;
|
|
|
|
|
wl_list_for_each(mousebind, &rc.mousebinds, link) {
|
|
|
|
|
if (mousebind->mouse_event != MOUSE_ACTION_SWIPE) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if (mousebind->fingers != swipe_state.fingers) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if (mousebind->direction != direction) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
actions_run(/*view*/ NULL, seat->server,
|
|
|
|
|
&mousebind->actions, /*cursor_ctx*/ NULL);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
swipe_state.dx = 0;
|
|
|
|
|
swipe_state.dy = 0;
|
|
|
|
|
swipe_state.fingers = 0;
|
2023-09-03 18:56:20 +02:00
|
|
|
}
|
|
|
|
|
|
2024-11-10 19:22:22 +01:00
|
|
|
static void
|
|
|
|
|
handle_hold_begin(struct wl_listener *listener, void *data)
|
|
|
|
|
{
|
|
|
|
|
struct seat *seat = wl_container_of(listener, seat, hold_begin);
|
|
|
|
|
struct wlr_pointer_hold_begin_event *event = data;
|
|
|
|
|
|
|
|
|
|
idle_manager_notify_activity(seat->seat);
|
|
|
|
|
cursor_set_visible(seat, /* visible */ true);
|
|
|
|
|
|
|
|
|
|
wlr_pointer_gestures_v1_send_hold_begin(seat->pointer_gestures,
|
|
|
|
|
seat->seat, event->time_msec, event->fingers);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
handle_hold_end(struct wl_listener *listener, void *data)
|
|
|
|
|
{
|
|
|
|
|
struct seat *seat = wl_container_of(listener, seat, hold_end);
|
|
|
|
|
struct wlr_pointer_hold_end_event *event = data;
|
|
|
|
|
|
|
|
|
|
idle_manager_notify_activity(seat->seat);
|
|
|
|
|
cursor_set_visible(seat, /* visible */ true);
|
|
|
|
|
|
|
|
|
|
wlr_pointer_gestures_v1_send_hold_end(seat->pointer_gestures,
|
|
|
|
|
seat->seat, event->time_msec, event->cancelled);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-03 18:56:20 +02:00
|
|
|
void
|
|
|
|
|
gestures_init(struct seat *seat)
|
|
|
|
|
{
|
|
|
|
|
seat->pointer_gestures = wlr_pointer_gestures_v1_create(seat->server->wl_display);
|
|
|
|
|
|
2024-11-10 19:05:55 +01:00
|
|
|
CONNECT_SIGNAL(seat->cursor, seat, pinch_begin);
|
|
|
|
|
CONNECT_SIGNAL(seat->cursor, seat, pinch_update);
|
|
|
|
|
CONNECT_SIGNAL(seat->cursor, seat, pinch_end);
|
|
|
|
|
CONNECT_SIGNAL(seat->cursor, seat, swipe_begin);
|
|
|
|
|
CONNECT_SIGNAL(seat->cursor, seat, swipe_update);
|
|
|
|
|
CONNECT_SIGNAL(seat->cursor, seat, swipe_end);
|
2024-11-10 19:22:22 +01:00
|
|
|
CONNECT_SIGNAL(seat->cursor, seat, hold_begin);
|
|
|
|
|
CONNECT_SIGNAL(seat->cursor, seat, hold_end);
|
2023-09-03 18:56:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
gestures_finish(struct seat *seat)
|
|
|
|
|
{
|
|
|
|
|
wl_list_remove(&seat->pinch_begin.link);
|
|
|
|
|
wl_list_remove(&seat->pinch_update.link);
|
|
|
|
|
wl_list_remove(&seat->pinch_end.link);
|
|
|
|
|
wl_list_remove(&seat->swipe_begin.link);
|
|
|
|
|
wl_list_remove(&seat->swipe_update.link);
|
|
|
|
|
wl_list_remove(&seat->swipe_end.link);
|
2024-11-10 19:22:22 +01:00
|
|
|
wl_list_remove(&seat->hold_begin.link);
|
|
|
|
|
wl_list_remove(&seat->hold_end.link);
|
2023-09-03 18:56:20 +02:00
|
|
|
}
|