#define _POSIX_C_SOURCE 200809L #include #include #include #include #include #include #include static void resource_destroy(struct wl_client *client, struct wl_resource *resource) { wl_resource_destroy(resource); } static void wl_pointer_set_cursor(struct wl_client *client, struct wl_resource *resource, uint32_t serial, struct wl_resource *surface, int32_t hotspot_x, int32_t hotspot_y) { wlr_log(L_DEBUG, "TODO: wl_pointer_set_cursor"); } static const struct wl_pointer_interface wl_pointer_impl = { .set_cursor = wl_pointer_set_cursor, .release = resource_destroy }; static void wl_pointer_destroy(struct wl_resource *resource) { struct wlr_seat_handle *handle = wl_resource_get_user_data(resource); if (handle->pointer) { handle->pointer = NULL; } } static void wl_seat_get_pointer(struct wl_client *client, struct wl_resource *_handle, uint32_t id) { struct wlr_seat_handle *handle = wl_resource_get_user_data(_handle); if (!(handle->wlr_seat->capabilities & WL_SEAT_CAPABILITY_POINTER)) { return; } if (handle->pointer) { // TODO: this is probably a protocol violation but it simplifies our // code and it'd be stupid for clients to create several pointers for // the same seat wl_resource_destroy(handle->pointer); } handle->pointer = wl_resource_create(client, &wl_pointer_interface, wl_resource_get_version(_handle), id); wl_resource_set_implementation(handle->pointer, &wl_pointer_impl, handle, &wl_pointer_destroy); } static const struct wl_keyboard_interface wl_keyboard_impl = { .release = resource_destroy }; static void wl_keyboard_destroy(struct wl_resource *resource) { struct wlr_seat_handle *handle = wl_resource_get_user_data(resource); if (handle->keyboard) { handle->keyboard = NULL; } } static void wl_seat_get_keyboard(struct wl_client *client, struct wl_resource *_handle, uint32_t id) { struct wlr_seat_handle *handle = wl_resource_get_user_data(_handle); if (!(handle->wlr_seat->capabilities & WL_SEAT_CAPABILITY_KEYBOARD)) { return; } if (handle->keyboard) { // TODO: this is probably a protocol violation but it simplifies our // code and it'd be stupid for clients to create several keyboards for // the same seat wl_resource_destroy(handle->keyboard); } handle->keyboard = wl_resource_create(client, &wl_keyboard_interface, wl_resource_get_version(_handle), id); wl_resource_set_implementation(handle->keyboard, &wl_keyboard_impl, handle, &wl_keyboard_destroy); if (wl_resource_get_version(handle->keyboard) >= WL_KEYBOARD_REPEAT_INFO_SINCE_VERSION) { wl_keyboard_send_repeat_info(handle->keyboard, 25, 600); } if (handle->wlr_seat->keyboard_state.keymap_size) { // TODO: handle no keymap wl_keyboard_send_keymap(handle->keyboard, WL_KEYBOARD_KEYMAP_FORMAT_XKB_V1, handle->wlr_seat->keyboard_state.keymap_fd, handle->wlr_seat->keyboard_state.keymap_size); } wl_signal_emit(&handle->wlr_seat->events.keyboard_bound, handle); } static const struct wl_touch_interface wl_touch_impl = { .release = resource_destroy }; static void wl_touch_destroy(struct wl_resource *resource) { struct wlr_seat_handle *handle = wl_resource_get_user_data(resource); if (handle->touch) { handle->touch = NULL; } } static void wl_seat_get_touch(struct wl_client *client, struct wl_resource *_handle, uint32_t id) { struct wlr_seat_handle *handle = wl_resource_get_user_data(_handle); if (!(handle->wlr_seat->capabilities & WL_SEAT_CAPABILITY_TOUCH)) { return; } if (handle->touch) { // TODO: this is probably a protocol violation but it simplifies our // code and it'd be stupid for clients to create several pointers for // the same seat wl_resource_destroy(handle->touch); } handle->touch = wl_resource_create(client, &wl_touch_interface, wl_resource_get_version(_handle), id); wl_resource_set_implementation(handle->touch, &wl_touch_impl, handle, &wl_touch_destroy); } static void wlr_seat_handle_resource_destroy(struct wl_resource *resource) { struct wlr_seat_handle *handle = wl_resource_get_user_data(resource); if (handle == handle->wlr_seat->pointer_state.focused_handle) { handle->wlr_seat->pointer_state.focused_handle = NULL; } if (handle == handle->wlr_seat->keyboard_state.focused_handle) { handle->wlr_seat->keyboard_state.focused_handle = NULL; } if (handle->pointer) { wl_resource_destroy(handle->pointer); } if (handle->keyboard) { wl_resource_destroy(handle->keyboard); } if (handle->touch) { wl_resource_destroy(handle->touch); } if (handle->data_device) { wl_resource_destroy(handle->data_device); } wl_signal_emit(&handle->wlr_seat->events.client_unbound, handle); wl_list_remove(&handle->link); free(handle); } struct wl_seat_interface wl_seat_impl = { .get_pointer = wl_seat_get_pointer, .get_keyboard = wl_seat_get_keyboard, .get_touch = wl_seat_get_touch, .release = resource_destroy }; static void wl_seat_bind(struct wl_client *wl_client, void *_wlr_seat, uint32_t version, uint32_t id) { struct wlr_seat *wlr_seat = _wlr_seat; assert(wl_client && wlr_seat); if (version > 6) { wlr_log(L_ERROR, "Client requested unsupported wl_seat version, disconnecting"); wl_client_destroy(wl_client); return; } struct wlr_seat_handle *handle = calloc(1, sizeof(struct wlr_seat_handle)); handle->wl_resource = wl_resource_create( wl_client, &wl_seat_interface, version, id); handle->wlr_seat = wlr_seat; wl_resource_set_implementation(handle->wl_resource, &wl_seat_impl, handle, wlr_seat_handle_resource_destroy); wl_list_insert(&wlr_seat->handles, &handle->link); wl_seat_send_capabilities(handle->wl_resource, wlr_seat->capabilities); wl_signal_emit(&wlr_seat->events.client_bound, handle); } struct wlr_seat *wlr_seat_create(struct wl_display *display, const char *name) { struct wlr_seat *wlr_seat = calloc(1, sizeof(struct wlr_seat)); if (!wlr_seat) { return NULL; } wlr_seat->pointer_state.wlr_seat = wlr_seat; wl_list_init(&wlr_seat->pointer_state.focus_resource_destroy_listener.link); wl_list_init(&wlr_seat->pointer_state.focus_surface_destroy_listener.link); wlr_seat->keyboard_state.wlr_seat = wlr_seat; wl_list_init( &wlr_seat->keyboard_state.focus_resource_destroy_listener.link); wl_list_init( &wlr_seat->keyboard_state.focus_surface_destroy_listener.link); struct wl_global *wl_global = wl_global_create(display, &wl_seat_interface, 6, wlr_seat, wl_seat_bind); if (!wl_global) { free(wlr_seat); return NULL; } wlr_seat->wl_global = wl_global; wlr_seat->display = display; wlr_seat->name = strdup(name); wl_list_init(&wlr_seat->handles); wl_signal_init(&wlr_seat->events.client_bound); wl_signal_init(&wlr_seat->events.client_unbound); wl_signal_init(&wlr_seat->events.keyboard_bound); return wlr_seat; } void wlr_seat_destroy(struct wlr_seat *wlr_seat) { if (!wlr_seat) { return; } struct wlr_seat_handle *handle, *tmp; wl_list_for_each_safe(handle, tmp, &wlr_seat->handles, link) { // will destroy other resources as well wl_resource_destroy(handle->wl_resource); } wl_global_destroy(wlr_seat->wl_global); free(wlr_seat->data_device); free(wlr_seat->name); free(wlr_seat); } struct wlr_seat_handle *wlr_seat_handle_for_client(struct wlr_seat *wlr_seat, struct wl_client *client) { assert(wlr_seat); struct wlr_seat_handle *handle; wl_list_for_each(handle, &wlr_seat->handles, link) { if (wl_resource_get_client(handle->wl_resource) == client) { return handle; } } return NULL; } void wlr_seat_set_capabilities(struct wlr_seat *wlr_seat, uint32_t capabilities) { wlr_seat->capabilities = capabilities; struct wlr_seat_handle *handle; wl_list_for_each(handle, &wlr_seat->handles, link) { wl_seat_send_capabilities(handle->wl_resource, capabilities); } } void wlr_seat_set_name(struct wlr_seat *wlr_seat, const char *name) { free(wlr_seat->name); wlr_seat->name = strdup(name); struct wlr_seat_handle *handle; wl_list_for_each(handle, &wlr_seat->handles, link) { wl_seat_send_name(handle->wl_resource, name); } } bool wlr_seat_pointer_surface_has_focus(struct wlr_seat *wlr_seat, struct wlr_surface *surface) { return surface == wlr_seat->pointer_state.focused_surface; } static void handle_pointer_focus_surface_destroyed( struct wl_listener *listener, void *data) { struct wlr_seat_pointer_state *state = wl_container_of(listener, state, focus_surface_destroy_listener); state->focused_surface = NULL; wlr_seat_pointer_clear_focus(state->wlr_seat); } static void handle_pointer_focus_resource_destroyed( struct wl_listener *listener, void *data) { struct wlr_seat_pointer_state *state = wl_container_of(listener, state, focus_resource_destroy_listener); state->focused_surface = NULL; wlr_seat_pointer_clear_focus(state->wlr_seat); } static bool wlr_seat_pointer_has_focus_resource(struct wlr_seat *wlr_seat) { return wlr_seat->pointer_state.focused_handle && wlr_seat->pointer_state.focused_handle->pointer; } void wlr_seat_pointer_enter(struct wlr_seat *wlr_seat, struct wlr_surface *surface, double sx, double sy) { assert(wlr_seat); if (wlr_seat->pointer_state.focused_surface == surface) { // this surface already got an enter notify return; } struct wlr_seat_handle *handle = NULL; if (surface) { struct wl_client *client = wl_resource_get_client(surface->resource); handle = wlr_seat_handle_for_client(wlr_seat, client); } struct wlr_seat_handle *focused_handle = wlr_seat->pointer_state.focused_handle; struct wlr_surface *focused_surface = wlr_seat->pointer_state.focused_surface; // leave the previously entered surface if (focused_handle && focused_handle->pointer && focused_surface) { uint32_t serial = wl_display_next_serial(wlr_seat->display); wl_pointer_send_leave(focused_handle->pointer, serial, focused_surface->resource); wl_pointer_send_frame(focused_handle->pointer); } // enter the current surface if (handle && handle->pointer) { uint32_t serial = wl_display_next_serial(wlr_seat->display); wl_pointer_send_enter(handle->pointer, serial, surface->resource, wl_fixed_from_double(sx), wl_fixed_from_double(sy)); wl_pointer_send_frame(handle->pointer); } // reinitialize the focus destroy events wl_list_remove( &wlr_seat->pointer_state.focus_surface_destroy_listener.link); wl_list_init(&wlr_seat->pointer_state.focus_surface_destroy_listener.link); wl_list_remove( &wlr_seat->pointer_state.focus_resource_destroy_listener.link); wl_list_init(&wlr_seat->pointer_state.focus_resource_destroy_listener.link); if (surface) { wl_signal_add(&surface->signals.destroy, &wlr_seat->pointer_state.focus_surface_destroy_listener); wl_resource_add_destroy_listener(surface->resource, &wlr_seat->pointer_state.focus_resource_destroy_listener); wlr_seat->pointer_state.focus_resource_destroy_listener.notify = handle_pointer_focus_resource_destroyed; wlr_seat->pointer_state.focus_surface_destroy_listener.notify = handle_pointer_focus_surface_destroyed; } wlr_seat->pointer_state.focused_handle = handle; wlr_seat->pointer_state.focused_surface = surface; // TODO: send focus change event } void wlr_seat_pointer_clear_focus(struct wlr_seat *wlr_seat) { wlr_seat_pointer_enter(wlr_seat, NULL, 0, 0); } void wlr_seat_pointer_send_motion(struct wlr_seat *wlr_seat, uint32_t time, double sx, double sy) { if (!wlr_seat_pointer_has_focus_resource(wlr_seat)) { return; } wl_pointer_send_motion(wlr_seat->pointer_state.focused_handle->pointer, time, wl_fixed_from_double(sx), wl_fixed_from_double(sy)); wl_pointer_send_frame(wlr_seat->pointer_state.focused_handle->pointer); } uint32_t wlr_seat_pointer_send_button(struct wlr_seat *wlr_seat, uint32_t time, uint32_t button, uint32_t state) { if (!wlr_seat_pointer_has_focus_resource(wlr_seat)) { return 0; } uint32_t serial = wl_display_next_serial(wlr_seat->display); wl_pointer_send_button(wlr_seat->pointer_state.focused_handle->pointer, serial, time, button, state); wl_pointer_send_frame(wlr_seat->pointer_state.focused_handle->pointer); return serial; } void wlr_seat_pointer_send_axis(struct wlr_seat *wlr_seat, uint32_t time, enum wlr_axis_orientation orientation, double value) { if (!wlr_seat_pointer_has_focus_resource(wlr_seat)) { return; } struct wl_resource *pointer = wlr_seat->pointer_state.focused_handle->pointer; if (value) { wl_pointer_send_axis(pointer, time, orientation, wl_fixed_from_double(value)); } else { wl_pointer_send_axis_stop(pointer, time, orientation); } wl_pointer_send_frame(pointer); } static void handle_keyboard_focus_surface_destroyed( struct wl_listener *listener, void *data) { struct wlr_seat_keyboard_state *state = wl_container_of(listener, state, focus_surface_destroy_listener); state->focused_surface = NULL; wlr_seat_keyboard_clear_focus(state->wlr_seat); } static void handle_keyboard_focus_resource_destroyed( struct wl_listener *listener, void *data) { struct wlr_seat_keyboard_state *state = wl_container_of(listener, state, focus_resource_destroy_listener); state->focused_surface = NULL; wlr_seat_keyboard_clear_focus(state->wlr_seat); } void wlr_seat_keyboard_enter(struct wlr_seat *wlr_seat, struct wlr_surface *surface, struct wl_array keys) { if (wlr_seat->keyboard_state.focused_surface == surface) { // this surface already got an enter notify return; } struct wlr_seat_handle *handle = NULL; if (surface) { struct wl_client *client = wl_resource_get_client(surface->resource); handle = wlr_seat_handle_for_client(wlr_seat, client); } struct wlr_seat_handle *focused_handle = wlr_seat->keyboard_state.focused_handle; struct wlr_surface *focused_surface = wlr_seat->keyboard_state.focused_surface; // leave the previously entered surface if (focused_handle && focused_handle->keyboard && focused_surface) { uint32_t serial = wl_display_next_serial(wlr_seat->display); wl_keyboard_send_leave(focused_handle->keyboard, serial, focused_surface->resource); } // enter the current surface if (handle && handle->keyboard) { uint32_t serial = wl_display_next_serial(wlr_seat->display); wl_keyboard_send_enter(handle->keyboard, serial, surface->resource, &keys); // TODO: send modifiers } // reinitialize the focus destroy events wl_list_remove( &wlr_seat->keyboard_state.focus_surface_destroy_listener.link); wl_list_init(&wlr_seat->keyboard_state.focus_surface_destroy_listener.link); wl_list_remove( &wlr_seat->keyboard_state.focus_resource_destroy_listener.link); wl_list_init( &wlr_seat->keyboard_state.focus_resource_destroy_listener.link); if (surface) { wl_signal_add(&surface->signals.destroy, &wlr_seat->keyboard_state.focus_surface_destroy_listener); wl_resource_add_destroy_listener(surface->resource, &wlr_seat->keyboard_state.focus_resource_destroy_listener); wlr_seat->keyboard_state.focus_resource_destroy_listener.notify = handle_keyboard_focus_resource_destroyed; wlr_seat->keyboard_state.focus_surface_destroy_listener.notify = handle_keyboard_focus_surface_destroyed; } wlr_seat->keyboard_state.focused_handle = handle; wlr_seat->keyboard_state.focused_surface = surface; } void wlr_seat_keyboard_clear_focus(struct wlr_seat *wlr_seat) { struct wl_array keys; wl_array_init(&keys); wlr_seat_keyboard_enter(wlr_seat, NULL, keys); } static bool wlr_seat_keyboard_has_focus_resource(struct wlr_seat *wlr_seat) { return wlr_seat->keyboard_state.focused_handle && wlr_seat->keyboard_state.focused_handle->keyboard; } uint32_t wlr_seat_keyboard_send_key(struct wlr_seat *wlr_seat, uint32_t time, uint32_t key, uint32_t state) { if (!wlr_seat_keyboard_has_focus_resource(wlr_seat)) { return 0; } uint32_t serial = wl_display_next_serial(wlr_seat->display); wl_keyboard_send_key(wlr_seat->keyboard_state.focused_handle->keyboard, serial, time, key, state); return serial; } void wlr_seat_keyboard_send_modifiers(struct wlr_seat *wlr_seat, uint32_t mods_depressed, uint32_t mods_latched, uint32_t mods_locked, uint32_t group) { uint32_t serial = 0; struct wl_resource *keyboard; if (wlr_seat_keyboard_has_focus_resource(wlr_seat)) { serial = wl_display_next_serial(wlr_seat->display); keyboard = wlr_seat->keyboard_state.focused_handle->keyboard; wl_keyboard_send_modifiers(keyboard, serial, mods_depressed, mods_latched, mods_locked, group); } if (wlr_seat_pointer_has_focus_resource(wlr_seat) && wlr_seat->pointer_state.focused_handle->keyboard && wlr_seat->pointer_state.focused_handle != wlr_seat->keyboard_state.focused_handle) { if (serial == 0) { serial = wl_display_next_serial(wlr_seat->display); } keyboard = wlr_seat->pointer_state.focused_handle->keyboard; wl_keyboard_send_modifiers(keyboard, serial, mods_depressed, mods_latched, mods_locked, group); } } void wlr_seat_keyboard_set_keymap(struct wlr_seat *wlr_seat, int keymap_fd, size_t keymap_size) { // TODO: we probably should wait to send the keymap if keys are pressed struct wlr_seat_handle *handle; wl_list_for_each(handle, &wlr_seat->handles, link) { if (handle->keyboard) { wl_keyboard_send_keymap(handle->keyboard, WL_KEYBOARD_KEYMAP_FORMAT_XKB_V1, keymap_fd, keymap_size); } } wlr_seat->keyboard_state.keymap_fd = keymap_fd; wlr_seat->keyboard_state.keymap_size = keymap_size; }