#include #include #include #include #include #include #include "backend/libinput/backend.h" #include "common/log.h" #include "types.h" static struct wlr_input_device *get_appropriate_device( enum wlr_input_device_type desired_type, struct libinput_device *device) { list_t *devices = libinput_device_get_user_data(device); if (!devices) { return NULL; } for (size_t i = 0; i < devices->length; ++i) { struct wlr_input_device *dev = devices->items[i]; if (dev->type == desired_type) { return dev; } } return NULL; } static void wlr_libinput_keyboard_destroy(struct wlr_keyboard_state *state) { free(state); } static struct wlr_keyboard_impl keyboard_impl = { .destroy = wlr_libinput_keyboard_destroy }; static struct wlr_keyboard *wlr_libinput_keyboard_create( struct libinput_device *device) { assert(device); struct wlr_keyboard_state *kbstate = calloc(1, sizeof(struct wlr_keyboard_state)); kbstate->handle = device; libinput_device_ref(device); return wlr_keyboard_create(&keyboard_impl, kbstate); } static void handle_keyboard_key(struct libinput_event *event, struct libinput_device *device) { struct wlr_input_device *dev = get_appropriate_device(WLR_INPUT_DEVICE_KEYBOARD, device); if (!dev) { wlr_log(L_DEBUG, "Got a keyboard event for a device with no keyboards?"); return; } struct libinput_event_keyboard *kbevent = libinput_event_get_keyboard_event(event); struct wlr_keyboard_key *wlr_event = calloc(1, sizeof(struct wlr_keyboard_key)); wlr_event->time_sec = libinput_event_keyboard_get_time(kbevent); wlr_event->time_usec = libinput_event_keyboard_get_time_usec(kbevent); wlr_event->keycode = libinput_event_keyboard_get_key(kbevent); enum libinput_key_state state = libinput_event_keyboard_get_key_state(kbevent); switch (state) { case LIBINPUT_KEY_STATE_RELEASED: wlr_event->state = WLR_KEY_RELEASED; break; case LIBINPUT_KEY_STATE_PRESSED: wlr_event->state = WLR_KEY_PRESSED; break; } wl_signal_emit(&dev->keyboard->events.key, wlr_event); } static void handle_device_added(struct wlr_backend_state *state, struct libinput_device *device) { assert(state && device); /* * Note: the wlr API exposes only devices with a single capability, because * that meshes better with how Wayland does things and is a bit simpler. * However, libinput devices often have multiple capabilities - in such * cases we have to create several devices. */ int vendor = libinput_device_get_id_vendor(device); int product = libinput_device_get_id_product(device); const char *name = libinput_device_get_name(device); list_t *devices = list_create(); wlr_log(L_DEBUG, "Added %s [%d:%d]", name, vendor, product); if (libinput_device_has_capability(device, LIBINPUT_DEVICE_CAP_KEYBOARD)) { struct wlr_input_device *wlr_device = wlr_input_device_create( WLR_INPUT_DEVICE_KEYBOARD, name, vendor, product); wlr_device->keyboard = wlr_libinput_keyboard_create(device); wl_signal_emit(&state->backend->events.input_add, wlr_device); list_add(devices, wlr_device); } if (libinput_device_has_capability(device, LIBINPUT_DEVICE_CAP_POINTER)) { // TODO } if (libinput_device_has_capability(device, LIBINPUT_DEVICE_CAP_TOUCH)) { // TODO } if (libinput_device_has_capability(device, LIBINPUT_DEVICE_CAP_TABLET_TOOL)) { // TODO } if (libinput_device_has_capability(device, LIBINPUT_DEVICE_CAP_TABLET_PAD)) { // TODO } if (libinput_device_has_capability(device, LIBINPUT_DEVICE_CAP_GESTURE)) { // TODO } if (libinput_device_has_capability(device, LIBINPUT_DEVICE_CAP_SWITCH)) { // TODO } if (devices->length > 0) { libinput_device_set_user_data(device, devices); } else { list_free(devices); } } static void handle_device_removed(struct wlr_backend_state *state, struct libinput_device *device) { wlr_log(L_DEBUG, "libinput device removed"); // TODO } void wlr_libinput_event(struct wlr_backend_state *state, struct libinput_event *event) { assert(state && event); struct libinput *context = libinput_event_get_context(event); struct libinput_device *device = libinput_event_get_device(event); enum libinput_event_type event_type = libinput_event_get_type(event); (void)context; switch (event_type) { case LIBINPUT_EVENT_DEVICE_ADDED: handle_device_added(state, device); break; case LIBINPUT_EVENT_DEVICE_REMOVED: handle_device_removed(state, device); break; case LIBINPUT_EVENT_KEYBOARD_KEY: handle_keyboard_key(event, device); break; default: wlr_log(L_DEBUG, "Unknown libinput event %d", event_type); break; } }