mirror of
https://github.com/hyprwm/wlroots-hyprland.git
synced 2025-01-03 15:19:47 +01:00
backend/wayland: add touch support to the wayland backend
Goal currently is to get support working for a single output, thus there is only one touch device created. Multi-output support is left for later.
This commit is contained in:
parent
e410ff8dd4
commit
b98522b38f
2 changed files with 135 additions and 0 deletions
|
@ -276,6 +276,95 @@ static struct wl_keyboard_listener keyboard_listener = {
|
||||||
.repeat_info = keyboard_handle_repeat_info
|
.repeat_info = keyboard_handle_repeat_info
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void touch_coordinates_to_absolute(struct wlr_wl_input_device *device,
|
||||||
|
wl_fixed_t x, wl_fixed_t y, double *sx, double *sy) {
|
||||||
|
// TODO: each output needs its own touch
|
||||||
|
struct wlr_wl_output *output, *tmp;
|
||||||
|
wl_list_for_each_safe(output, tmp, &device->backend->outputs, link) {
|
||||||
|
*sx = wl_fixed_to_double(x) / output->wlr_output.width;
|
||||||
|
*sy = wl_fixed_to_double(y) / output->wlr_output.height;
|
||||||
|
return; // Choose the first output in the list
|
||||||
|
}
|
||||||
|
|
||||||
|
*sx = *sy = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void touch_handle_down(void *data, struct wl_touch *wl_touch,
|
||||||
|
uint32_t serial, uint32_t time, struct wl_surface *surface,
|
||||||
|
int32_t id, wl_fixed_t x, wl_fixed_t y) {
|
||||||
|
struct wlr_wl_input_device *device = data;
|
||||||
|
assert(device && device->wlr_input_device.touch);
|
||||||
|
|
||||||
|
double sx, sy;
|
||||||
|
touch_coordinates_to_absolute(device, x, y, &sx, &sy);
|
||||||
|
struct wlr_event_touch_down event = {
|
||||||
|
.device = &device->wlr_input_device,
|
||||||
|
.time_msec = time,
|
||||||
|
.touch_id = id,
|
||||||
|
.x = sx,
|
||||||
|
.y = sy
|
||||||
|
};
|
||||||
|
wlr_signal_emit_safe(&device->wlr_input_device.touch->events.down, &event);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void touch_handle_up(void *data, struct wl_touch *wl_touch,
|
||||||
|
uint32_t serial, uint32_t time, int32_t id) {
|
||||||
|
struct wlr_wl_input_device *device = data;
|
||||||
|
assert(device && device->wlr_input_device.touch);
|
||||||
|
|
||||||
|
struct wlr_event_touch_up event = {
|
||||||
|
.device = &device->wlr_input_device,
|
||||||
|
.time_msec = time,
|
||||||
|
.touch_id = id,
|
||||||
|
};
|
||||||
|
wlr_signal_emit_safe(&device->wlr_input_device.touch->events.up, &event);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void touch_handle_motion(void *data, struct wl_touch *wl_touch,
|
||||||
|
uint32_t time, int32_t id, wl_fixed_t x, wl_fixed_t y) {
|
||||||
|
struct wlr_wl_input_device *device = data;
|
||||||
|
assert(device && device->wlr_input_device.touch);
|
||||||
|
|
||||||
|
double sx, sy;
|
||||||
|
touch_coordinates_to_absolute(device, x, y, &sx, &sy);
|
||||||
|
struct wlr_event_touch_motion event = {
|
||||||
|
.device = &device->wlr_input_device,
|
||||||
|
.time_msec = time,
|
||||||
|
.touch_id = id,
|
||||||
|
.x = sx,
|
||||||
|
.y = sy
|
||||||
|
};
|
||||||
|
wlr_signal_emit_safe(&device->wlr_input_device.touch->events.motion, &event);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void touch_handle_frame(void *data, struct wl_touch *wl_touch) {
|
||||||
|
// no-op
|
||||||
|
}
|
||||||
|
|
||||||
|
static void touch_handle_cancel(void *data, struct wl_touch *wl_touch) {
|
||||||
|
// no-op
|
||||||
|
}
|
||||||
|
|
||||||
|
static void touch_handle_shape(void *data, struct wl_touch *wl_touch,
|
||||||
|
int32_t id, wl_fixed_t major, wl_fixed_t minor) {
|
||||||
|
// no-op
|
||||||
|
}
|
||||||
|
|
||||||
|
static void touch_handle_orientation(void *data, struct wl_touch *wl_touch,
|
||||||
|
int32_t id, wl_fixed_t orientation) {
|
||||||
|
// no-op
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct wl_touch_listener touch_listener = {
|
||||||
|
.down = touch_handle_down,
|
||||||
|
.up = touch_handle_up,
|
||||||
|
.motion = touch_handle_motion,
|
||||||
|
.frame = touch_handle_frame,
|
||||||
|
.cancel = touch_handle_cancel,
|
||||||
|
.shape = touch_handle_shape,
|
||||||
|
.orientation = touch_handle_orientation,
|
||||||
|
};
|
||||||
|
|
||||||
static struct wlr_wl_input_device *get_wl_input_device_from_input_device(
|
static struct wlr_wl_input_device *get_wl_input_device_from_input_device(
|
||||||
struct wlr_input_device *wlr_dev) {
|
struct wlr_input_device *wlr_dev) {
|
||||||
assert(wlr_input_device_is_wl(wlr_dev));
|
assert(wlr_input_device_is_wl(wlr_dev));
|
||||||
|
@ -563,6 +652,28 @@ void create_wl_keyboard(struct wl_keyboard *wl_keyboard, struct wlr_wl_backend *
|
||||||
wlr_signal_emit_safe(&wl->backend.events.new_input, wlr_dev);
|
wlr_signal_emit_safe(&wl->backend.events.new_input, wlr_dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void create_wl_touch(struct wl_touch *wl_touch, struct wlr_wl_backend *wl) {
|
||||||
|
struct wlr_wl_input_device *dev =
|
||||||
|
create_wl_input_device(wl, WLR_INPUT_DEVICE_TOUCH);
|
||||||
|
if (!dev) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct wlr_input_device *wlr_dev = &dev->wlr_input_device;
|
||||||
|
|
||||||
|
wlr_dev->touch = calloc(1, sizeof(*wlr_dev->touch));
|
||||||
|
if (!wlr_dev->touch) {
|
||||||
|
wlr_log_errno(WLR_ERROR, "Allocation failed");
|
||||||
|
free(dev);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
wlr_touch_init(wlr_dev->touch, NULL);
|
||||||
|
|
||||||
|
wl_touch_add_listener(wl_touch, &touch_listener, dev);
|
||||||
|
wlr_signal_emit_safe(&wl->backend.events.new_input, wlr_dev);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static void seat_handle_capabilities(void *data, struct wl_seat *wl_seat,
|
static void seat_handle_capabilities(void *data, struct wl_seat *wl_seat,
|
||||||
enum wl_seat_capability caps) {
|
enum wl_seat_capability caps) {
|
||||||
struct wlr_wl_backend *backend = data;
|
struct wlr_wl_backend *backend = data;
|
||||||
|
@ -614,6 +725,28 @@ static void seat_handle_capabilities(void *data, struct wl_seat *wl_seat,
|
||||||
}
|
}
|
||||||
assert(backend->keyboard == NULL); // free'ed by input_device_destroy
|
assert(backend->keyboard == NULL); // free'ed by input_device_destroy
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ((caps & WL_SEAT_CAPABILITY_TOUCH) && backend->touch == NULL) {
|
||||||
|
wlr_log(WLR_DEBUG, "seat %p offered touch", (void *)wl_seat);
|
||||||
|
|
||||||
|
backend->touch = wl_seat_get_touch(wl_seat);
|
||||||
|
if (backend->started) {
|
||||||
|
create_wl_touch(backend->touch, backend);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!(caps & WL_SEAT_CAPABILITY_TOUCH) && backend->touch != NULL) {
|
||||||
|
wlr_log(WLR_DEBUG, "seat %p dropped touch", (void *)wl_seat);
|
||||||
|
|
||||||
|
struct wlr_input_device *device, *tmp;
|
||||||
|
wl_list_for_each_safe(device, tmp, &backend->devices, link) {
|
||||||
|
if (device->type == WLR_INPUT_DEVICE_TOUCH) {
|
||||||
|
wlr_input_device_destroy(device);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
wl_touch_release(backend->touch);
|
||||||
|
backend->touch = NULL;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void seat_handle_name(void *data, struct wl_seat *wl_seat,
|
static void seat_handle_name(void *data, struct wl_seat *wl_seat,
|
||||||
|
|
|
@ -39,6 +39,7 @@ struct wlr_wl_backend {
|
||||||
struct zwp_linux_dmabuf_v1 *zwp_linux_dmabuf_v1;
|
struct zwp_linux_dmabuf_v1 *zwp_linux_dmabuf_v1;
|
||||||
struct zwp_relative_pointer_manager_v1 *zwp_relative_pointer_manager_v1;
|
struct zwp_relative_pointer_manager_v1 *zwp_relative_pointer_manager_v1;
|
||||||
struct wl_seat *seat;
|
struct wl_seat *seat;
|
||||||
|
struct wl_touch *touch;
|
||||||
struct wl_pointer *pointer;
|
struct wl_pointer *pointer;
|
||||||
struct wl_keyboard *keyboard;
|
struct wl_keyboard *keyboard;
|
||||||
struct wlr_wl_pointer *current_pointer;
|
struct wlr_wl_pointer *current_pointer;
|
||||||
|
@ -112,6 +113,7 @@ void update_wl_output_cursor(struct wlr_wl_output *output);
|
||||||
struct wlr_wl_pointer *pointer_get_wl(struct wlr_pointer *wlr_pointer);
|
struct wlr_wl_pointer *pointer_get_wl(struct wlr_pointer *wlr_pointer);
|
||||||
void create_wl_pointer(struct wl_pointer *wl_pointer, struct wlr_wl_output *output);
|
void create_wl_pointer(struct wl_pointer *wl_pointer, struct wlr_wl_output *output);
|
||||||
void create_wl_keyboard(struct wl_keyboard *wl_keyboard, struct wlr_wl_backend *wl);
|
void create_wl_keyboard(struct wl_keyboard *wl_keyboard, struct wlr_wl_backend *wl);
|
||||||
|
void create_wl_touch(struct wl_touch *wl_touch, struct wlr_wl_backend *wl);
|
||||||
struct wlr_wl_input_device *create_wl_input_device(
|
struct wlr_wl_input_device *create_wl_input_device(
|
||||||
struct wlr_wl_backend *backend, enum wlr_input_device_type type);
|
struct wlr_wl_backend *backend, enum wlr_input_device_type type);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue