2019-06-19 10:04:47 +02:00
|
|
|
#include "input.h"
|
|
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#include <signal.h>
|
|
|
|
|
#include <threads.h>
|
2019-07-09 10:00:54 +02:00
|
|
|
#include <locale.h>
|
2019-06-19 10:04:47 +02:00
|
|
|
#include <sys/mman.h>
|
2019-07-17 21:30:57 +02:00
|
|
|
#include <sys/time.h>
|
2019-08-05 19:33:01 +02:00
|
|
|
#include <sys/timerfd.h>
|
2019-06-19 10:04:47 +02:00
|
|
|
|
2019-07-10 14:17:44 +02:00
|
|
|
#include <linux/input-event-codes.h>
|
|
|
|
|
|
2019-06-19 10:04:47 +02:00
|
|
|
#include <xkbcommon/xkbcommon.h>
|
|
|
|
|
#include <xkbcommon/xkbcommon-keysyms.h>
|
2019-07-09 10:00:54 +02:00
|
|
|
#include <xkbcommon/xkbcommon-compose.h>
|
2019-06-19 10:04:47 +02:00
|
|
|
|
|
|
|
|
#define LOG_MODULE "input"
|
2019-07-03 20:21:03 +02:00
|
|
|
#define LOG_ENABLE_DBG 0
|
2019-06-19 10:04:47 +02:00
|
|
|
#include "log.h"
|
2019-07-09 16:26:36 +02:00
|
|
|
#include "commands.h"
|
2019-08-27 17:23:28 +02:00
|
|
|
#include "keymap.h"
|
|
|
|
|
#include "render.h"
|
|
|
|
|
#include "search.h"
|
2019-07-11 09:51:51 +02:00
|
|
|
#include "selection.h"
|
2019-08-27 17:23:28 +02:00
|
|
|
#include "terminal.h"
|
2019-07-15 15:42:21 +02:00
|
|
|
#include "vt.h"
|
2019-06-19 10:04:47 +02:00
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
keyboard_keymap(void *data, struct wl_keyboard *wl_keyboard,
|
|
|
|
|
uint32_t format, int32_t fd, uint32_t size)
|
|
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
struct wayland *wayl = data;
|
2019-06-19 10:04:47 +02:00
|
|
|
|
|
|
|
|
char *map_str = mmap(NULL, size, PROT_READ, MAP_PRIVATE, fd, 0);
|
|
|
|
|
|
|
|
|
|
/* TODO: free old context + keymap */
|
2019-10-27 18:43:07 +01:00
|
|
|
if (wayl->kbd.xkb_compose_state != NULL) {
|
|
|
|
|
xkb_compose_state_unref(wayl->kbd.xkb_compose_state);
|
|
|
|
|
wayl->kbd.xkb_compose_state = NULL;
|
2019-08-12 21:32:53 +02:00
|
|
|
}
|
2019-10-27 18:43:07 +01:00
|
|
|
if (wayl->kbd.xkb_compose_table != NULL) {
|
|
|
|
|
xkb_compose_table_unref(wayl->kbd.xkb_compose_table);
|
|
|
|
|
wayl->kbd.xkb_compose_table = NULL;
|
2019-08-12 21:32:53 +02:00
|
|
|
}
|
2019-10-27 18:43:07 +01:00
|
|
|
if (wayl->kbd.xkb_keymap != NULL) {
|
|
|
|
|
xkb_keymap_unref(wayl->kbd.xkb_keymap);
|
|
|
|
|
wayl->kbd.xkb_keymap = NULL;
|
2019-08-12 21:32:53 +02:00
|
|
|
}
|
2019-10-27 18:43:07 +01:00
|
|
|
if (wayl->kbd.xkb_state != NULL) {
|
|
|
|
|
xkb_state_unref(wayl->kbd.xkb_state);
|
|
|
|
|
wayl->kbd.xkb_state = NULL;
|
2019-08-12 21:32:53 +02:00
|
|
|
}
|
2019-10-27 18:43:07 +01:00
|
|
|
if (wayl->kbd.xkb != NULL) {
|
|
|
|
|
xkb_context_unref(wayl->kbd.xkb);
|
|
|
|
|
wayl->kbd.xkb = NULL;
|
2019-08-12 21:32:53 +02:00
|
|
|
}
|
2019-06-19 10:04:47 +02:00
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->kbd.xkb = xkb_context_new(XKB_CONTEXT_NO_FLAGS);
|
|
|
|
|
wayl->kbd.xkb_keymap = xkb_keymap_new_from_string(
|
|
|
|
|
wayl->kbd.xkb, map_str, XKB_KEYMAP_FORMAT_TEXT_V1,
|
2019-06-19 10:04:47 +02:00
|
|
|
XKB_KEYMAP_COMPILE_NO_FLAGS);
|
|
|
|
|
|
|
|
|
|
/* TODO: initialize in enter? */
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->kbd.xkb_state = xkb_state_new(wayl->kbd.xkb_keymap);
|
2019-06-19 10:04:47 +02:00
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->kbd.mod_shift = xkb_keymap_mod_get_index(wayl->kbd.xkb_keymap, "Shift");
|
|
|
|
|
wayl->kbd.mod_alt = xkb_keymap_mod_get_index(wayl->kbd.xkb_keymap, "Mod1") ;
|
|
|
|
|
wayl->kbd.mod_ctrl = xkb_keymap_mod_get_index(wayl->kbd.xkb_keymap, "Control");
|
|
|
|
|
wayl->kbd.mod_meta = xkb_keymap_mod_get_index(wayl->kbd.xkb_keymap, "Mod4");
|
2019-07-05 15:13:06 +02:00
|
|
|
|
2019-07-09 10:00:54 +02:00
|
|
|
/* Compose (dead keys) */
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->kbd.xkb_compose_table = xkb_compose_table_new_from_locale(
|
|
|
|
|
wayl->kbd.xkb, setlocale(LC_CTYPE, NULL), XKB_COMPOSE_COMPILE_NO_FLAGS);
|
|
|
|
|
wayl->kbd.xkb_compose_state = xkb_compose_state_new(
|
|
|
|
|
wayl->kbd.xkb_compose_table, XKB_COMPOSE_STATE_NO_FLAGS);
|
2019-07-09 10:00:54 +02:00
|
|
|
|
2019-06-19 10:04:47 +02:00
|
|
|
munmap(map_str, size);
|
|
|
|
|
close(fd);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
keyboard_enter(void *data, struct wl_keyboard *wl_keyboard, uint32_t serial,
|
|
|
|
|
struct wl_surface *surface, struct wl_array *keys)
|
|
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
struct wayland *wayl = data;
|
|
|
|
|
wayl->input_serial = serial;
|
|
|
|
|
wayl->focused = wayl_terminal_from_surface(wayl, surface);
|
|
|
|
|
assert(wayl->focused != NULL);
|
|
|
|
|
term_focus_in(wayl->focused);
|
2019-06-19 10:04:47 +02:00
|
|
|
}
|
|
|
|
|
|
2019-08-05 19:33:01 +02:00
|
|
|
static bool
|
2019-10-27 18:43:07 +01:00
|
|
|
start_repeater(struct wayland *wayl, uint32_t key)
|
2019-06-19 10:04:47 +02:00
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
if (wayl->kbd.repeat.dont_re_repeat)
|
2019-08-05 19:33:01 +02:00
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
struct itimerspec t = {
|
2019-10-27 18:43:07 +01:00
|
|
|
.it_value = {.tv_sec = 0, .tv_nsec = wayl->kbd.repeat.delay * 1000000},
|
|
|
|
|
.it_interval = {.tv_sec = 0, .tv_nsec = 1000000000 / wayl->kbd.repeat.rate},
|
2019-08-05 19:33:01 +02:00
|
|
|
};
|
2019-06-21 18:40:53 +02:00
|
|
|
|
2019-08-05 19:33:01 +02:00
|
|
|
if (t.it_value.tv_nsec >= 1000000000) {
|
|
|
|
|
t.it_value.tv_sec += t.it_value.tv_nsec / 1000000000;
|
|
|
|
|
t.it_value.tv_nsec %= 1000000000;
|
|
|
|
|
}
|
|
|
|
|
if (t.it_interval.tv_nsec >= 1000000000) {
|
|
|
|
|
t.it_interval.tv_sec += t.it_interval.tv_nsec / 1000000000;
|
|
|
|
|
t.it_interval.tv_nsec %= 1000000000;
|
|
|
|
|
}
|
2019-10-27 18:43:07 +01:00
|
|
|
if (timerfd_settime(wayl->kbd.repeat.fd, 0, &t, NULL) < 0) {
|
2019-08-05 19:33:01 +02:00
|
|
|
LOG_ERRNO("failed to arm keyboard repeat timer");
|
|
|
|
|
return false;
|
2019-06-21 18:40:53 +02:00
|
|
|
}
|
2019-08-02 18:19:07 +02:00
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->kbd.repeat.key = key;
|
2019-08-05 19:33:01 +02:00
|
|
|
return true;
|
2019-08-02 18:19:07 +02:00
|
|
|
}
|
|
|
|
|
|
2019-08-05 19:33:01 +02:00
|
|
|
static bool
|
2019-10-27 18:43:07 +01:00
|
|
|
stop_repeater(struct wayland *wayl, uint32_t key)
|
2019-08-02 18:19:07 +02:00
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
if (key != -1 && key != wayl->kbd.repeat.key)
|
2019-08-05 19:33:01 +02:00
|
|
|
return true;
|
|
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
if (timerfd_settime(wayl->kbd.repeat.fd, 0, &(struct itimerspec){{0}}, NULL) < 0) {
|
2019-08-05 19:33:01 +02:00
|
|
|
LOG_ERRNO("failed to disarm keyboard repeat timer");
|
|
|
|
|
return false;
|
2019-08-02 18:19:07 +02:00
|
|
|
}
|
2019-08-05 19:33:01 +02:00
|
|
|
|
|
|
|
|
return true;
|
2019-08-02 18:19:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2019-08-05 19:33:01 +02:00
|
|
|
keyboard_leave(void *data, struct wl_keyboard *wl_keyboard, uint32_t serial,
|
|
|
|
|
struct wl_surface *surface)
|
2019-08-02 18:19:07 +02:00
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
struct wayland *wayl = data;
|
2019-08-05 19:33:01 +02:00
|
|
|
|
2019-11-21 18:17:02 +01:00
|
|
|
assert(wayl->focused == NULL ||
|
|
|
|
|
wayl_terminal_from_surface(wayl, surface) == wayl->focused);
|
|
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
stop_repeater(wayl, -1);
|
2019-11-21 18:17:02 +01:00
|
|
|
if (wayl->focused != NULL)
|
|
|
|
|
term_focus_out(wayl->focused);
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->focused = NULL;
|
2019-06-19 10:04:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
keyboard_key(void *data, struct wl_keyboard *wl_keyboard, uint32_t serial,
|
|
|
|
|
uint32_t time, uint32_t key, uint32_t state)
|
|
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
struct wayland *wayl = data;
|
|
|
|
|
struct terminal *term = wayl->focused;
|
2019-11-22 21:49:50 +01:00
|
|
|
|
2019-11-22 21:56:13 +01:00
|
|
|
/* Workaround buggy Sway 1.2 */
|
|
|
|
|
if (term == NULL) {
|
|
|
|
|
static bool have_warned = false;
|
|
|
|
|
if (!have_warned) {
|
|
|
|
|
have_warned = true;
|
|
|
|
|
LOG_WARN("compositor sent keyboard_key event without first sending keyboard_enter");
|
|
|
|
|
}
|
|
|
|
|
stop_repeater(wayl, -1);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
assert(term != NULL);
|
2019-06-19 10:04:47 +02:00
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
const xkb_mod_mask_t ctrl = 1 << wayl->kbd.mod_ctrl;
|
|
|
|
|
const xkb_mod_mask_t alt = 1 << wayl->kbd.mod_alt;
|
|
|
|
|
const xkb_mod_mask_t shift = 1 << wayl->kbd.mod_shift;
|
|
|
|
|
const xkb_mod_mask_t meta = 1 << wayl->kbd.mod_meta;
|
2019-06-19 10:04:47 +02:00
|
|
|
|
|
|
|
|
if (state == XKB_KEY_UP) {
|
2019-10-27 18:43:07 +01:00
|
|
|
stop_repeater(wayl, key);
|
2019-06-19 10:04:47 +02:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
key += 8;
|
2019-10-27 18:43:07 +01:00
|
|
|
xkb_keysym_t sym = xkb_state_key_get_one_sym(wayl->kbd.xkb_state, key);
|
2019-06-19 10:04:47 +02:00
|
|
|
|
2019-08-19 21:18:43 +02:00
|
|
|
#if 0
|
2019-07-09 14:27:26 +02:00
|
|
|
char foo[100];
|
|
|
|
|
xkb_keysym_get_name(sym, foo, sizeof(foo));
|
2019-08-19 21:16:47 +02:00
|
|
|
LOG_INFO("%s", foo);
|
2019-07-09 14:27:26 +02:00
|
|
|
#endif
|
|
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
xkb_compose_state_feed(wayl->kbd.xkb_compose_state, sym);
|
2019-07-09 10:00:54 +02:00
|
|
|
enum xkb_compose_status compose_status = xkb_compose_state_get_status(
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->kbd.xkb_compose_state);
|
2019-07-09 10:00:54 +02:00
|
|
|
|
|
|
|
|
if (compose_status == XKB_COMPOSE_COMPOSING)
|
|
|
|
|
return;
|
|
|
|
|
|
2019-06-19 10:04:47 +02:00
|
|
|
xkb_mod_mask_t mods = xkb_state_serialize_mods(
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->kbd.xkb_state, XKB_STATE_MODS_DEPRESSED);
|
|
|
|
|
//xkb_mod_mask_t consumed = xkb_state_key_get_consumed_mods(wayl->kbd.xkb_state, key);
|
2019-06-22 20:31:53 +02:00
|
|
|
xkb_mod_mask_t consumed = 0x0;
|
2019-08-15 19:38:28 +02:00
|
|
|
xkb_mod_mask_t significant = ctrl | alt | shift | meta;
|
2019-06-19 10:04:47 +02:00
|
|
|
xkb_mod_mask_t effective_mods = mods & ~consumed & significant;
|
|
|
|
|
|
2019-08-27 17:23:28 +02:00
|
|
|
if (term->is_searching) {
|
2019-10-27 18:43:07 +01:00
|
|
|
start_repeater(wayl, key - 8);
|
2019-08-27 17:23:28 +02:00
|
|
|
search_input(term, key, sym, effective_mods);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2019-06-19 10:04:47 +02:00
|
|
|
#if 0
|
|
|
|
|
for (size_t i = 0; i < 32; i++) {
|
|
|
|
|
if (mods & (1 << i)) {
|
2019-10-27 18:43:07 +01:00
|
|
|
LOG_INFO("%s", xkb_keymap_mod_get_name(wayl->kbd.xkb_keymap, i));
|
2019-06-19 10:04:47 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
LOG_DBG("sym=%u, mod=0x%08x, consumed=0x%08x, significant=0x%08x, "
|
|
|
|
|
"effective=0x%08x",
|
|
|
|
|
sym, mods, consumed, significant, effective_mods);
|
|
|
|
|
|
2019-07-09 14:27:26 +02:00
|
|
|
bool found_map = false;
|
|
|
|
|
|
|
|
|
|
enum modifier keymap_mods = MOD_NONE;
|
2019-10-27 18:43:07 +01:00
|
|
|
keymap_mods |= wayl->kbd.shift ? MOD_SHIFT : MOD_NONE;
|
|
|
|
|
keymap_mods |= wayl->kbd.alt ? MOD_ALT : MOD_NONE;
|
|
|
|
|
keymap_mods |= wayl->kbd.ctrl ? MOD_CTRL : MOD_NONE;
|
|
|
|
|
keymap_mods |= wayl->kbd.meta ? MOD_META : MOD_NONE;
|
2019-07-09 14:27:26 +02:00
|
|
|
|
2019-07-09 16:26:36 +02:00
|
|
|
if (effective_mods == shift) {
|
|
|
|
|
if (sym == XKB_KEY_Page_Up) {
|
2019-07-10 09:15:37 +02:00
|
|
|
cmd_scrollback_up(term, term->rows);
|
2019-07-09 16:26:36 +02:00
|
|
|
found_map = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
else if (sym == XKB_KEY_Page_Down) {
|
2019-07-10 09:15:37 +02:00
|
|
|
cmd_scrollback_down(term, term->rows);
|
2019-07-09 16:26:36 +02:00
|
|
|
found_map = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-11 12:16:50 +02:00
|
|
|
else if (effective_mods == (shift | ctrl)) {
|
|
|
|
|
if (sym == XKB_KEY_C) {
|
|
|
|
|
selection_to_clipboard(term, serial);
|
|
|
|
|
found_map = true;
|
|
|
|
|
}
|
2019-07-11 12:33:31 +02:00
|
|
|
|
|
|
|
|
else if (sym == XKB_KEY_V) {
|
|
|
|
|
selection_from_clipboard(term, serial);
|
2019-11-20 10:41:40 +01:00
|
|
|
term_reset_view(term);
|
2019-07-11 12:33:31 +02:00
|
|
|
found_map = true;
|
|
|
|
|
}
|
2019-08-27 17:23:28 +02:00
|
|
|
|
|
|
|
|
else if (sym == XKB_KEY_R) {
|
|
|
|
|
search_begin(term);
|
|
|
|
|
found_map = true;
|
|
|
|
|
}
|
2019-07-11 12:16:50 +02:00
|
|
|
}
|
|
|
|
|
|
2019-07-09 15:05:07 +02:00
|
|
|
for (size_t i = 0; i < sizeof(key_map) / sizeof(key_map[0]) && !found_map; i++) {
|
2019-07-09 14:27:26 +02:00
|
|
|
const struct key_map *k = &key_map[i];
|
|
|
|
|
if (k->sym != sym)
|
|
|
|
|
continue;
|
|
|
|
|
|
2019-07-09 15:05:07 +02:00
|
|
|
for (size_t j = 0; j < k->count; j++) {
|
|
|
|
|
const struct key_data *info = &k->data[j];
|
|
|
|
|
if (info->modifiers != MOD_ANY && info->modifiers != keymap_mods)
|
|
|
|
|
continue;
|
2019-07-09 14:27:26 +02:00
|
|
|
|
2019-07-09 15:05:07 +02:00
|
|
|
if (info->cursor_keys_mode != CURSOR_KEYS_DONTCARE &&
|
|
|
|
|
info->cursor_keys_mode != term->cursor_keys_mode)
|
|
|
|
|
continue;
|
2019-07-09 14:27:26 +02:00
|
|
|
|
2019-07-09 15:05:07 +02:00
|
|
|
if (info->keypad_keys_mode != KEYPAD_DONTCARE &&
|
|
|
|
|
info->keypad_keys_mode != term->keypad_keys_mode)
|
|
|
|
|
continue;
|
2019-07-09 14:27:26 +02:00
|
|
|
|
2019-11-03 00:27:39 +01:00
|
|
|
term_to_slave(term, info->seq, strlen(info->seq));
|
2019-07-09 15:05:07 +02:00
|
|
|
found_map = true;
|
2019-07-09 16:26:36 +02:00
|
|
|
|
2019-08-28 17:27:17 +02:00
|
|
|
term_reset_view(term);
|
2019-07-11 09:51:51 +02:00
|
|
|
selection_cancel(term);
|
2019-07-09 15:05:07 +02:00
|
|
|
break;
|
|
|
|
|
}
|
2019-07-09 14:27:26 +02:00
|
|
|
}
|
2019-06-22 20:40:28 +02:00
|
|
|
|
2019-07-09 14:27:26 +02:00
|
|
|
if (!found_map) {
|
2019-07-30 20:27:19 +02:00
|
|
|
uint8_t buf[64] = {0};
|
2019-07-09 10:00:54 +02:00
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
|
|
if (compose_status == XKB_COMPOSE_COMPOSED) {
|
|
|
|
|
count = xkb_compose_state_get_utf8(
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->kbd.xkb_compose_state, (char *)buf, sizeof(buf));
|
|
|
|
|
xkb_compose_state_reset(wayl->kbd.xkb_compose_state);
|
2019-07-09 10:00:54 +02:00
|
|
|
} else {
|
|
|
|
|
count = xkb_state_key_get_utf8(
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->kbd.xkb_state, key, (char *)buf, sizeof(buf));
|
2019-07-09 10:00:54 +02:00
|
|
|
}
|
2019-06-19 10:04:47 +02:00
|
|
|
|
2019-06-22 20:31:53 +02:00
|
|
|
if (count > 0) {
|
2019-06-19 10:04:47 +02:00
|
|
|
|
2019-07-26 18:49:09 +02:00
|
|
|
#define is_control_key(x) ((x) >= 0x40 && (x) <= 0x7f)
|
|
|
|
|
#define IS_CTRL(x) ((x) < 0x20 || ((x) >= 0x7f && (x) <= 0x9f))
|
|
|
|
|
|
|
|
|
|
if ((keymap_mods & MOD_CTRL) &&
|
|
|
|
|
!is_control_key(sym) &&
|
|
|
|
|
(count == 1 && !IS_CTRL(buf[0])) &&
|
|
|
|
|
sym < 256)
|
|
|
|
|
{
|
2019-08-15 19:38:50 +02:00
|
|
|
static const int mod_param_map[32] = {
|
2019-07-26 18:49:09 +02:00
|
|
|
[MOD_SHIFT] = 2,
|
|
|
|
|
[MOD_ALT] = 3,
|
|
|
|
|
[MOD_SHIFT | MOD_ALT] = 4,
|
|
|
|
|
[MOD_CTRL] = 5,
|
|
|
|
|
[MOD_SHIFT | MOD_CTRL] = 6,
|
|
|
|
|
[MOD_ALT | MOD_CTRL] = 7,
|
|
|
|
|
[MOD_SHIFT | MOD_ALT | MOD_CTRL] = 8,
|
2019-08-15 19:38:50 +02:00
|
|
|
[MOD_META] = 9,
|
|
|
|
|
[MOD_META | MOD_SHIFT] = 10,
|
|
|
|
|
[MOD_META | MOD_ALT] = 11,
|
|
|
|
|
[MOD_META | MOD_SHIFT | MOD_ALT] = 12,
|
|
|
|
|
[MOD_META | MOD_CTRL] = 13,
|
|
|
|
|
[MOD_META | MOD_SHIFT | MOD_CTRL] = 14,
|
|
|
|
|
[MOD_META | MOD_ALT | MOD_CTRL] = 15,
|
|
|
|
|
[MOD_META | MOD_SHIFT | MOD_ALT | MOD_CTRL] = 16,
|
2019-07-26 18:49:09 +02:00
|
|
|
};
|
2019-08-15 19:38:50 +02:00
|
|
|
|
|
|
|
|
assert(keymap_mods < sizeof(mod_param_map) / sizeof(mod_param_map[0]));
|
2019-07-26 18:49:09 +02:00
|
|
|
int modify_param = mod_param_map[keymap_mods];
|
|
|
|
|
assert(modify_param != 0);
|
|
|
|
|
|
|
|
|
|
char reply[1024];
|
|
|
|
|
snprintf(reply, sizeof(reply), "\x1b[27;%d;%d~", modify_param, sym);
|
2019-11-03 00:27:39 +01:00
|
|
|
term_to_slave(term, reply, strlen(reply));
|
2019-07-26 18:49:09 +02:00
|
|
|
}
|
2019-07-09 16:26:36 +02:00
|
|
|
|
2019-07-26 18:49:09 +02:00
|
|
|
else {
|
|
|
|
|
if (effective_mods & alt)
|
2019-11-03 00:27:39 +01:00
|
|
|
term_to_slave(term, "\x1b", 1);
|
2019-07-26 18:49:09 +02:00
|
|
|
|
2019-11-03 00:27:39 +01:00
|
|
|
term_to_slave(term, buf, count);
|
2019-07-09 16:26:36 +02:00
|
|
|
}
|
|
|
|
|
|
2019-11-20 10:41:40 +01:00
|
|
|
term_reset_view(term);
|
2019-07-11 09:51:51 +02:00
|
|
|
selection_cancel(term);
|
2019-06-22 20:31:53 +02:00
|
|
|
}
|
2019-06-19 10:04:47 +02:00
|
|
|
}
|
|
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
start_repeater(wayl, key - 8);
|
2019-06-19 10:04:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
keyboard_modifiers(void *data, struct wl_keyboard *wl_keyboard, uint32_t serial,
|
|
|
|
|
uint32_t mods_depressed, uint32_t mods_latched,
|
|
|
|
|
uint32_t mods_locked, uint32_t group)
|
|
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
struct wayland *wayl = data;
|
2019-06-19 10:04:47 +02:00
|
|
|
|
|
|
|
|
LOG_DBG("modifiers: depressed=0x%x, latched=0x%x, locked=0x%x, group=%u",
|
|
|
|
|
mods_depressed, mods_latched, mods_locked, group);
|
|
|
|
|
|
|
|
|
|
xkb_state_update_mask(
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->kbd.xkb_state, mods_depressed, mods_latched, mods_locked, 0, 0, group);
|
2019-07-05 15:13:06 +02:00
|
|
|
|
|
|
|
|
/* Update state of modifiers we're interrested in for e.g mouse events */
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->kbd.shift = xkb_state_mod_index_is_active(
|
|
|
|
|
wayl->kbd.xkb_state, wayl->kbd.mod_shift, XKB_STATE_MODS_DEPRESSED);
|
|
|
|
|
wayl->kbd.alt = xkb_state_mod_index_is_active(
|
|
|
|
|
wayl->kbd.xkb_state, wayl->kbd.mod_alt, XKB_STATE_MODS_DEPRESSED);
|
|
|
|
|
wayl->kbd.ctrl = xkb_state_mod_index_is_active(
|
|
|
|
|
wayl->kbd.xkb_state, wayl->kbd.mod_ctrl, XKB_STATE_MODS_DEPRESSED);
|
|
|
|
|
wayl->kbd.meta = xkb_state_mod_index_is_active(
|
|
|
|
|
wayl->kbd.xkb_state, wayl->kbd.mod_meta, XKB_STATE_MODS_DEPRESSED);
|
2019-06-19 10:04:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
keyboard_repeat_info(void *data, struct wl_keyboard *wl_keyboard,
|
|
|
|
|
int32_t rate, int32_t delay)
|
|
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
struct wayland *wayl = data;
|
2019-06-19 10:04:47 +02:00
|
|
|
LOG_DBG("keyboard repeat: rate=%d, delay=%d", rate, delay);
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->kbd.repeat.rate = rate;
|
|
|
|
|
wayl->kbd.repeat.delay = delay;
|
2019-06-19 10:04:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const struct wl_keyboard_listener keyboard_listener = {
|
|
|
|
|
.keymap = &keyboard_keymap,
|
|
|
|
|
.enter = &keyboard_enter,
|
|
|
|
|
.leave = &keyboard_leave,
|
|
|
|
|
.key = &keyboard_key,
|
|
|
|
|
.modifiers = &keyboard_modifiers,
|
|
|
|
|
.repeat_info = &keyboard_repeat_info,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void
|
2019-10-27 18:44:58 +01:00
|
|
|
input_repeat(struct wayland *wayl, uint32_t key)
|
2019-06-19 10:04:47 +02:00
|
|
|
{
|
2019-11-22 21:49:50 +01:00
|
|
|
keyboard_key(wayl, NULL, wayl->input_serial, 0, key, XKB_KEY_DOWN);
|
2019-06-19 10:04:47 +02:00
|
|
|
}
|
2019-07-05 10:44:57 +02:00
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
wl_pointer_enter(void *data, struct wl_pointer *wl_pointer,
|
|
|
|
|
uint32_t serial, struct wl_surface *surface,
|
|
|
|
|
wl_fixed_t surface_x, wl_fixed_t surface_y)
|
|
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
struct wayland *wayl = data;
|
|
|
|
|
struct terminal *term = wayl_terminal_from_surface(wayl, surface);
|
|
|
|
|
|
|
|
|
|
wayl->moused = term;
|
2019-07-05 15:13:06 +02:00
|
|
|
|
2019-09-26 18:41:26 +02:00
|
|
|
int x = wl_fixed_to_int(surface_x) * term->scale;
|
|
|
|
|
int y = wl_fixed_to_int(surface_y) * term->scale;
|
2019-07-05 15:13:06 +02:00
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->mouse.col = x / term->cell_width;
|
|
|
|
|
wayl->mouse.row = y / term->cell_height;
|
2019-07-05 10:44:57 +02:00
|
|
|
|
2019-10-27 19:36:45 +01:00
|
|
|
wayl_update_cursor_surface(wayl, term);
|
2019-07-05 10:44:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
wl_pointer_leave(void *data, struct wl_pointer *wl_pointer,
|
|
|
|
|
uint32_t serial, struct wl_surface *surface)
|
|
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
struct wayland *wayl = data;
|
|
|
|
|
wayl->moused = NULL;
|
2019-07-05 10:44:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
wl_pointer_motion(void *data, struct wl_pointer *wl_pointer,
|
|
|
|
|
uint32_t time, wl_fixed_t surface_x, wl_fixed_t surface_y)
|
|
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
struct wayland *wayl = data;
|
|
|
|
|
struct terminal *term = wayl->moused;
|
2019-11-22 21:49:50 +01:00
|
|
|
|
2019-11-22 21:56:13 +01:00
|
|
|
/* Workaround buggy Sway 1.2 */
|
|
|
|
|
if (term == NULL) {
|
|
|
|
|
static bool have_warned = false;
|
|
|
|
|
if (!have_warned) {
|
|
|
|
|
have_warned = true;
|
|
|
|
|
LOG_WARN("compositor sent pointer_motion event without first sending pointer_enter");
|
|
|
|
|
}
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
assert(term != NULL);
|
2019-07-05 15:13:06 +02:00
|
|
|
|
2019-09-26 18:39:49 +02:00
|
|
|
int x = wl_fixed_to_int(surface_x) * term->scale;
|
|
|
|
|
int y = wl_fixed_to_int(surface_y) * term->scale;
|
2019-07-05 15:13:06 +02:00
|
|
|
|
2019-08-30 21:30:27 +02:00
|
|
|
int col = (x - term->x_margin) / term->cell_width;
|
|
|
|
|
int row = (y - term->y_margin) / term->cell_height;
|
2019-07-05 15:13:06 +02:00
|
|
|
|
2019-07-11 11:09:34 +02:00
|
|
|
if (col < 0 || row < 0 || col >= term->cols || row >= term->rows)
|
|
|
|
|
return;
|
|
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
bool update_selection = wayl->mouse.button == BTN_LEFT;
|
2019-07-11 09:51:51 +02:00
|
|
|
bool update_selection_early = term->selection.end.row == -1;
|
|
|
|
|
|
|
|
|
|
if (update_selection && update_selection_early)
|
|
|
|
|
selection_update(term, col, row);
|
|
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
if (col == wayl->mouse.col && row == wayl->mouse.row)
|
2019-07-05 15:13:06 +02:00
|
|
|
return;
|
|
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->mouse.col = col;
|
|
|
|
|
wayl->mouse.row = row;
|
2019-07-05 15:13:06 +02:00
|
|
|
|
2019-07-11 09:51:51 +02:00
|
|
|
if (update_selection && !update_selection_early)
|
|
|
|
|
selection_update(term, col, row);
|
2019-07-10 20:57:09 +02:00
|
|
|
|
2019-07-05 15:13:06 +02:00
|
|
|
term_mouse_motion(
|
2019-10-27 18:43:07 +01:00
|
|
|
term, wayl->mouse.button, wayl->mouse.row, wayl->mouse.col,
|
|
|
|
|
wayl->kbd.shift, wayl->kbd.alt, wayl->kbd.ctrl);
|
2019-07-05 10:44:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
wl_pointer_button(void *data, struct wl_pointer *wl_pointer,
|
|
|
|
|
uint32_t serial, uint32_t time, uint32_t button, uint32_t state)
|
|
|
|
|
{
|
2019-07-05 14:24:51 +02:00
|
|
|
LOG_DBG("BUTTON: button=%x, state=%u", button, state);
|
|
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
struct wayland *wayl = data;
|
|
|
|
|
struct terminal *term = wayl->moused;
|
|
|
|
|
assert(term != NULL);
|
2019-07-05 14:24:51 +02:00
|
|
|
|
2019-08-27 19:44:19 +02:00
|
|
|
search_cancel(term);
|
|
|
|
|
|
2019-07-05 14:24:51 +02:00
|
|
|
switch (state) {
|
2019-07-17 21:30:57 +02:00
|
|
|
case WL_POINTER_BUTTON_STATE_PRESSED: {
|
2019-08-06 19:32:06 +02:00
|
|
|
/* Time since last click */
|
|
|
|
|
struct timeval now, since_last;
|
2019-07-17 21:30:57 +02:00
|
|
|
gettimeofday(&now, NULL);
|
2019-10-27 18:43:07 +01:00
|
|
|
timersub(&now, &wayl->mouse.last_time, &since_last);
|
2019-08-06 19:32:06 +02:00
|
|
|
|
|
|
|
|
/* Double- or triple click? */
|
2019-10-27 18:43:07 +01:00
|
|
|
if (button == wayl->mouse.last_button &&
|
2019-08-06 19:32:06 +02:00
|
|
|
since_last.tv_sec == 0 &&
|
|
|
|
|
since_last.tv_usec <= 300 * 1000)
|
2019-07-17 21:30:57 +02:00
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->mouse.count++;
|
2019-08-06 19:32:06 +02:00
|
|
|
} else
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->mouse.count = 1;
|
2019-07-17 21:30:57 +02:00
|
|
|
|
|
|
|
|
if (button == BTN_LEFT) {
|
2019-10-27 18:43:07 +01:00
|
|
|
switch (wayl->mouse.count) {
|
2019-08-06 19:32:06 +02:00
|
|
|
case 1:
|
2019-10-27 18:43:07 +01:00
|
|
|
selection_start(term, wayl->mouse.col, wayl->mouse.row);
|
2019-08-06 19:32:06 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 2:
|
2019-10-27 18:43:07 +01:00
|
|
|
selection_mark_word(term, wayl->mouse.col, wayl->mouse.row,
|
|
|
|
|
wayl->kbd.ctrl, serial);
|
2019-08-06 19:32:06 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 3:
|
2019-10-27 18:43:07 +01:00
|
|
|
selection_mark_row(term, wayl->mouse.row, serial);
|
2019-08-06 19:32:06 +02:00
|
|
|
break;
|
|
|
|
|
}
|
2019-07-17 21:30:57 +02:00
|
|
|
} else {
|
2019-10-27 18:43:07 +01:00
|
|
|
if (wayl->mouse.count == 1 && button == BTN_MIDDLE && selection_enabled(term))
|
2019-07-11 16:42:59 +02:00
|
|
|
selection_from_primary(term);
|
2019-07-11 09:51:51 +02:00
|
|
|
selection_cancel(term);
|
2019-07-11 16:42:59 +02:00
|
|
|
}
|
2019-07-10 20:57:09 +02:00
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->mouse.button = button; /* For motion events */
|
|
|
|
|
wayl->mouse.last_button = button;
|
|
|
|
|
wayl->mouse.last_time = now;
|
|
|
|
|
term_mouse_down(term, button, wayl->mouse.row, wayl->mouse.col,
|
|
|
|
|
wayl->kbd.shift, wayl->kbd.alt, wayl->kbd.ctrl);
|
2019-07-05 14:24:51 +02:00
|
|
|
break;
|
2019-07-17 21:30:57 +02:00
|
|
|
}
|
2019-07-05 14:24:51 +02:00
|
|
|
|
|
|
|
|
case WL_POINTER_BUTTON_STATE_RELEASED:
|
2019-07-11 18:17:40 +02:00
|
|
|
if (button != BTN_LEFT || term->selection.end.col == -1)
|
2019-07-11 09:51:51 +02:00
|
|
|
selection_cancel(term);
|
2019-07-11 11:10:53 +02:00
|
|
|
else
|
2019-07-11 17:34:52 +02:00
|
|
|
selection_finalize(term, serial);
|
2019-07-10 20:57:09 +02:00
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->mouse.button = 0; /* For motion events */
|
|
|
|
|
term_mouse_up(term, button, wayl->mouse.row, wayl->mouse.col,
|
|
|
|
|
wayl->kbd.shift, wayl->kbd.alt, wayl->kbd.ctrl);
|
2019-07-05 14:24:51 +02:00
|
|
|
break;
|
|
|
|
|
}
|
2019-07-05 10:44:57 +02:00
|
|
|
}
|
|
|
|
|
|
2019-07-26 18:47:56 +02:00
|
|
|
static void
|
2019-10-27 18:43:07 +01:00
|
|
|
mouse_scroll(struct wayland *wayl, int amount)
|
2019-07-26 18:47:56 +02:00
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
struct terminal *term = wayl->moused;
|
|
|
|
|
assert(term != NULL);
|
|
|
|
|
|
2019-07-26 18:47:56 +02:00
|
|
|
int button = amount < 0 ? BTN_BACK : BTN_FORWARD;
|
|
|
|
|
|
|
|
|
|
void (*scrollback)(struct terminal *term, int rows)
|
|
|
|
|
= amount < 0 ? &cmd_scrollback_up : &cmd_scrollback_down;
|
|
|
|
|
|
|
|
|
|
amount = abs(amount);
|
|
|
|
|
|
2019-08-19 21:16:47 +02:00
|
|
|
if ((button == BTN_BACK || button == BTN_FORWARD) &&
|
|
|
|
|
term->grid == &term->alt && term->alt_scrolling)
|
|
|
|
|
{
|
|
|
|
|
/*
|
|
|
|
|
* alternateScroll/faux scrolling - translate mouse
|
|
|
|
|
* "back"/"forward" to up/down keys
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
xkb_keycode_t key = xkb_keymap_key_by_name(
|
2019-10-27 18:43:07 +01:00
|
|
|
wayl->kbd.xkb_keymap, button == BTN_BACK ? "UP" : "DOWN");
|
2019-08-19 21:16:47 +02:00
|
|
|
|
|
|
|
|
for (int i = 0; i < amount; i++)
|
2019-10-27 18:43:07 +01:00
|
|
|
keyboard_key(term, NULL, wayl->input_serial, 0, key - 8, XKB_KEY_DOWN);
|
|
|
|
|
keyboard_key(term, NULL, wayl->input_serial, 0, key - 8, XKB_KEY_UP);
|
2019-08-19 21:16:47 +02:00
|
|
|
} else {
|
|
|
|
|
for (int i = 0; i < amount; i++)
|
2019-10-27 18:43:07 +01:00
|
|
|
term_mouse_down(term, button, wayl->mouse.row, wayl->mouse.col,
|
|
|
|
|
wayl->kbd.shift, wayl->kbd.alt, wayl->kbd.ctrl);
|
2019-08-19 21:16:47 +02:00
|
|
|
|
|
|
|
|
scrollback(term, amount);
|
|
|
|
|
}
|
2019-07-26 18:47:56 +02:00
|
|
|
}
|
|
|
|
|
|
2019-07-05 10:44:57 +02:00
|
|
|
static void
|
|
|
|
|
wl_pointer_axis(void *data, struct wl_pointer *wl_pointer,
|
|
|
|
|
uint32_t time, uint32_t axis, wl_fixed_t value)
|
|
|
|
|
{
|
2019-07-26 18:47:56 +02:00
|
|
|
if (axis != WL_POINTER_AXIS_VERTICAL_SCROLL)
|
|
|
|
|
return;
|
|
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
struct wayland *wayl = data;
|
|
|
|
|
|
|
|
|
|
if (wayl->mouse.have_discrete)
|
2019-07-26 18:47:56 +02:00
|
|
|
return;
|
2019-07-10 09:15:37 +02:00
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
mouse_scroll(wayl, wl_fixed_to_int(value));
|
2019-07-26 18:47:56 +02:00
|
|
|
}
|
2019-07-10 09:15:37 +02:00
|
|
|
|
2019-07-26 18:47:56 +02:00
|
|
|
static void
|
|
|
|
|
wl_pointer_axis_discrete(void *data, struct wl_pointer *wl_pointer,
|
|
|
|
|
uint32_t axis, int32_t discrete)
|
|
|
|
|
{
|
2019-07-10 09:15:37 +02:00
|
|
|
if (axis != WL_POINTER_AXIS_VERTICAL_SCROLL)
|
|
|
|
|
return;
|
|
|
|
|
|
2019-10-27 18:43:07 +01:00
|
|
|
struct wayland *wayl = data;
|
|
|
|
|
wayl->mouse.have_discrete = true;
|
|
|
|
|
mouse_scroll(wayl, discrete);
|
2019-07-05 10:44:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
wl_pointer_frame(void *data, struct wl_pointer *wl_pointer)
|
|
|
|
|
{
|
2019-10-27 18:43:07 +01:00
|
|
|
struct wayland *wayl = data;
|
|
|
|
|
wayl->mouse.have_discrete = false;
|
2019-07-05 10:44:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
wl_pointer_axis_source(void *data, struct wl_pointer *wl_pointer,
|
|
|
|
|
uint32_t axis_source)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
wl_pointer_axis_stop(void *data, struct wl_pointer *wl_pointer,
|
|
|
|
|
uint32_t time, uint32_t axis)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const struct wl_pointer_listener pointer_listener = {
|
|
|
|
|
.enter = wl_pointer_enter,
|
|
|
|
|
.leave = wl_pointer_leave,
|
|
|
|
|
.motion = wl_pointer_motion,
|
|
|
|
|
.button = wl_pointer_button,
|
|
|
|
|
.axis = wl_pointer_axis,
|
|
|
|
|
.frame = wl_pointer_frame,
|
|
|
|
|
.axis_source = wl_pointer_axis_source,
|
|
|
|
|
.axis_stop = wl_pointer_axis_stop,
|
|
|
|
|
.axis_discrete = wl_pointer_axis_discrete,
|
|
|
|
|
};
|