summaryrefslogtreecommitdiff
path: root/src/fe-common/core
diff options
context:
space:
mode:
authorLemonBoy <thatlemon@gmail.com>2017-02-10 13:59:50 +0100
committerLemonBoy <thatlemon@gmail.com>2017-02-18 15:57:55 +0100
commit9cae98e6421cb454ecc9b3b48d06cce5a042bb6a (patch)
tree26bd2059b7dc5073180f197741dae22ac804e566 /src/fe-common/core
parent653c7fb05a67841acd4fcc12ce1fee3be9955453 (diff)
downloadirssi-9cae98e6421cb454ecc9b3b48d06cce5a042bb6a.zip
Execute what's left in the input queue when the timeout expires.
Similar to how vim behaves.
Diffstat (limited to 'src/fe-common/core')
-rw-r--r--src/fe-common/core/keyboard.c46
1 files changed, 45 insertions, 1 deletions
diff --git a/src/fe-common/core/keyboard.c b/src/fe-common/core/keyboard.c
index fed7f9cb..6f7907eb 100644
--- a/src/fe-common/core/keyboard.c
+++ b/src/fe-common/core/keyboard.c
@@ -35,6 +35,7 @@
GSList *keyinfos;
static GHashTable *keys, *default_keys;
+static int key_timeout;
/* A cache of some sort for key presses that generate a single char only.
If the key isn't used, used_keys[key] is zero. */
@@ -48,7 +49,8 @@ static int key_config_frozen;
struct _KEYBOARD_REC {
char *key_state; /* the ongoing key combo */
- void *gui_data; /* GUI specific data sent in "key pressed" signal */
+ guint timer_tag; /* used to check when a pending combo has expired */
+ void *gui_data; /* GUI specific data sent in "key pressed" signal */
};
/* Creates a new "keyboard" - this is used only for keeping track of
@@ -60,6 +62,7 @@ KEYBOARD_REC *keyboard_create(void *data)
rec = g_new0(KEYBOARD_REC, 1);
rec->gui_data = data;
+ rec->timer_tag = 0;
signal_emit("keyboard created", 1, rec);
return rec;
@@ -68,6 +71,11 @@ KEYBOARD_REC *keyboard_create(void *data)
/* Destroys a keyboard */
void keyboard_destroy(KEYBOARD_REC *keyboard)
{
+ if (keyboard->timer_tag > 0) {
+ g_source_remove(keyboard->timer_tag);
+ keyboard->timer_tag = 0;
+ }
+
signal_emit("keyboard destroyed", 1, keyboard);
g_free_not_null(keyboard->key_state);
@@ -592,6 +600,25 @@ static int key_states_search(const unsigned char *combo,
return 0;
}
+static gboolean key_timeout_expired(KEYBOARD_REC *keyboard)
+{
+ KEY_REC *rec;
+
+ keyboard->timer_tag = 0;
+
+ /* So, the timeout has expired with the input queue full, let's see if
+ * what we've got is bound to some action. */
+ rec = g_tree_lookup(key_states, keyboard->key_state);
+ /* Drain the queue anyway. */
+ g_free_and_null(keyboard->key_state);
+
+ if (rec != NULL) {
+ (void)key_emit_signal(keyboard, rec);
+ }
+
+ return FALSE;
+}
+
int key_pressed(KEYBOARD_REC *keyboard, const char *key)
{
KEY_REC *rec;
@@ -601,6 +628,11 @@ int key_pressed(KEYBOARD_REC *keyboard, const char *key)
g_return_val_if_fail(keyboard != NULL, FALSE);
g_return_val_if_fail(key != NULL && *key != '\0', FALSE);
+ if (keyboard->timer_tag > 0) {
+ g_source_remove(keyboard->timer_tag);
+ keyboard->timer_tag = 0;
+ }
+
if (keyboard->key_state == NULL && key[1] == '\0' &&
!used_keys[(int) (unsigned char) key[0]]) {
/* fast check - key not used */
@@ -625,6 +657,13 @@ int key_pressed(KEYBOARD_REC *keyboard, const char *key)
if (g_tree_lookup(key_states, combo) != rec) {
/* key combo continues.. */
keyboard->key_state = combo;
+ /* respect the timeout if specified by the user */
+ if (key_timeout > 0) {
+ keyboard->timer_tag =
+ g_timeout_add(key_timeout,
+ (GSourceFunc) key_timeout_expired,
+ keyboard);
+ }
return 0;
}
@@ -870,6 +909,9 @@ static void read_keyboard_config(void)
key_config_read(tmp->data);
key_configure_thaw();
+
+ /* any positive value other than 0 enables the timeout (in ms). */
+ key_timeout = settings_get_int("key_timeout");
}
void keyboard_init(void)
@@ -883,6 +925,8 @@ void keyboard_init(void)
key_config_frozen = 0;
memset(used_keys, 0, sizeof(used_keys));
+ settings_add_int("misc", "key_timeout", 0);
+
key_bind("command", "Run any command", NULL, NULL, (SIGNAL_FUNC) sig_command);
key_bind("key", "Specify name for key binding", NULL, NULL, (SIGNAL_FUNC) sig_key);
key_bind("multi", "Run multiple commands", NULL, NULL, (SIGNAL_FUNC) sig_multi);