mirror of
https://github.com/RGBCube/serenity
synced 2025-05-31 02:18:11 +00:00
LibLine: Support multi-character key callbacks
This commit is contained in:
parent
f79e28bd65
commit
c057225a36
5 changed files with 259 additions and 89 deletions
|
@ -1,6 +1,7 @@
|
||||||
set(SOURCES
|
set(SOURCES
|
||||||
Editor.cpp
|
Editor.cpp
|
||||||
InternalFunctions.cpp
|
InternalFunctions.cpp
|
||||||
|
KeyCallbackMachine.cpp
|
||||||
SuggestionManager.cpp
|
SuggestionManager.cpp
|
||||||
XtermSuggestionDisplay.cpp
|
XtermSuggestionDisplay.cpp
|
||||||
)
|
)
|
||||||
|
|
|
@ -80,9 +80,9 @@ Configuration Configuration::from_config(const StringView& libname)
|
||||||
GenericLexer key_lexer(binding_key);
|
GenericLexer key_lexer(binding_key);
|
||||||
auto has_ctrl = false;
|
auto has_ctrl = false;
|
||||||
auto alt = false;
|
auto alt = false;
|
||||||
unsigned key = 0;
|
Vector<Key> keys;
|
||||||
|
|
||||||
while (!key && !key_lexer.is_eof()) {
|
while (!key_lexer.is_eof()) {
|
||||||
if (key_lexer.next_is("alt+")) {
|
if (key_lexer.next_is("alt+")) {
|
||||||
alt = key_lexer.consume_specific("alt+");
|
alt = key_lexer.consume_specific("alt+");
|
||||||
continue;
|
continue;
|
||||||
|
@ -100,21 +100,24 @@ Configuration Configuration::from_config(const StringView& libname)
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
// FIXME: Support utf?
|
// FIXME: Support utf?
|
||||||
key = key_lexer.consume();
|
unsigned key = key_lexer.consume();
|
||||||
}
|
if (has_ctrl)
|
||||||
|
key = ctrl(key);
|
||||||
|
|
||||||
if (has_ctrl)
|
keys.append(Key { key, alt ? Key::Alt : Key::None });
|
||||||
key = ctrl(key);
|
alt = false;
|
||||||
|
has_ctrl = false;
|
||||||
|
}
|
||||||
|
|
||||||
auto value = config_file->read_entry("keybinds", binding_key);
|
auto value = config_file->read_entry("keybinds", binding_key);
|
||||||
if (value.starts_with("internal:")) {
|
if (value.starts_with("internal:")) {
|
||||||
configuration.set(KeyBinding {
|
configuration.set(KeyBinding {
|
||||||
Key { key, alt ? Key::Alt : Key::None },
|
keys,
|
||||||
KeyBinding::Kind::InternalFunction,
|
KeyBinding::Kind::InternalFunction,
|
||||||
value.substring(9, value.length() - 9) });
|
value.substring(9, value.length() - 9) });
|
||||||
} else {
|
} else {
|
||||||
configuration.set(KeyBinding {
|
configuration.set(KeyBinding {
|
||||||
Key { key, alt ? Key::Alt : Key::None },
|
keys,
|
||||||
KeyBinding::Kind::Insertion,
|
KeyBinding::Kind::Insertion,
|
||||||
value });
|
value });
|
||||||
}
|
}
|
||||||
|
@ -146,16 +149,16 @@ void Editor::set_default_keybinds()
|
||||||
register_key_input_callback('\n', EDITOR_INTERNAL_FUNCTION(finish));
|
register_key_input_callback('\n', EDITOR_INTERNAL_FUNCTION(finish));
|
||||||
|
|
||||||
// ^[.: alt-.: insert last arg of previous command (similar to `!$`)
|
// ^[.: alt-.: insert last arg of previous command (similar to `!$`)
|
||||||
register_key_input_callback({ '.', Key::Alt }, EDITOR_INTERNAL_FUNCTION(insert_last_words));
|
register_key_input_callback(Key { '.', Key::Alt }, EDITOR_INTERNAL_FUNCTION(insert_last_words));
|
||||||
register_key_input_callback({ 'b', Key::Alt }, EDITOR_INTERNAL_FUNCTION(cursor_left_word));
|
register_key_input_callback(Key { 'b', Key::Alt }, EDITOR_INTERNAL_FUNCTION(cursor_left_word));
|
||||||
register_key_input_callback({ 'f', Key::Alt }, EDITOR_INTERNAL_FUNCTION(cursor_right_word));
|
register_key_input_callback(Key { 'f', Key::Alt }, EDITOR_INTERNAL_FUNCTION(cursor_right_word));
|
||||||
// ^[^H: alt-backspace: backward delete word
|
// ^[^H: alt-backspace: backward delete word
|
||||||
register_key_input_callback({ '\b', Key::Alt }, EDITOR_INTERNAL_FUNCTION(erase_alnum_word_backwards));
|
register_key_input_callback(Key { '\b', Key::Alt }, EDITOR_INTERNAL_FUNCTION(erase_alnum_word_backwards));
|
||||||
register_key_input_callback({ 'd', Key::Alt }, EDITOR_INTERNAL_FUNCTION(erase_alnum_word_forwards));
|
register_key_input_callback(Key { 'd', Key::Alt }, EDITOR_INTERNAL_FUNCTION(erase_alnum_word_forwards));
|
||||||
register_key_input_callback({ 'c', Key::Alt }, EDITOR_INTERNAL_FUNCTION(capitalize_word));
|
register_key_input_callback(Key { 'c', Key::Alt }, EDITOR_INTERNAL_FUNCTION(capitalize_word));
|
||||||
register_key_input_callback({ 'l', Key::Alt }, EDITOR_INTERNAL_FUNCTION(lowercase_word));
|
register_key_input_callback(Key { 'l', Key::Alt }, EDITOR_INTERNAL_FUNCTION(lowercase_word));
|
||||||
register_key_input_callback({ 'u', Key::Alt }, EDITOR_INTERNAL_FUNCTION(uppercase_word));
|
register_key_input_callback(Key { 'u', Key::Alt }, EDITOR_INTERNAL_FUNCTION(uppercase_word));
|
||||||
register_key_input_callback({ 't', Key::Alt }, EDITOR_INTERNAL_FUNCTION(transpose_words));
|
register_key_input_callback(Key { 't', Key::Alt }, EDITOR_INTERNAL_FUNCTION(transpose_words));
|
||||||
}
|
}
|
||||||
|
|
||||||
Editor::Editor(Configuration configuration)
|
Editor::Editor(Configuration configuration)
|
||||||
|
@ -253,20 +256,15 @@ void Editor::register_key_input_callback(const KeyBinding& binding)
|
||||||
dbg() << "LibLine: Unknown internal function '" << binding.binding << "'";
|
dbg() << "LibLine: Unknown internal function '" << binding.binding << "'";
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
return register_key_input_callback(binding.key, move(internal_function));
|
return register_key_input_callback(binding.keys, move(internal_function));
|
||||||
}
|
}
|
||||||
|
|
||||||
return register_key_input_callback(binding.key, [binding = String(binding.binding)](auto& editor) {
|
return register_key_input_callback(binding.keys, [binding = String(binding.binding)](auto& editor) {
|
||||||
editor.insert(binding);
|
editor.insert(binding);
|
||||||
return false;
|
return false;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void Editor::register_key_input_callback(Key key, Function<bool(Editor&)> callback)
|
|
||||||
{
|
|
||||||
m_key_callbacks.set(key, make<KeyCallback>(move(callback)));
|
|
||||||
}
|
|
||||||
|
|
||||||
static size_t code_point_length_in_utf8(u32 code_point)
|
static size_t code_point_length_in_utf8(u32 code_point)
|
||||||
{
|
{
|
||||||
if (code_point <= 0x7f)
|
if (code_point <= 0x7f)
|
||||||
|
@ -556,13 +554,9 @@ void Editor::handle_interrupt_event()
|
||||||
{
|
{
|
||||||
m_was_interrupted = false;
|
m_was_interrupted = false;
|
||||||
|
|
||||||
auto cb = m_key_callbacks.get(ctrl('C'));
|
m_callback_machine.interrupted(*this);
|
||||||
if (cb.has_value()) {
|
if (!m_callback_machine.should_process_last_pressed_key())
|
||||||
if (!cb.value()->callback(*this)) {
|
return;
|
||||||
// Oh well.
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fprintf(stderr, "^C");
|
fprintf(stderr, "^C");
|
||||||
fflush(stderr);
|
fflush(stderr);
|
||||||
|
@ -657,12 +651,7 @@ void Editor::handle_read_event()
|
||||||
continue;
|
continue;
|
||||||
default: {
|
default: {
|
||||||
m_state = InputState::Free;
|
m_state = InputState::Free;
|
||||||
auto cb = m_key_callbacks.get({ code_point, Key::Alt });
|
m_callback_machine.key_pressed(*this, { code_point, Key::Alt });
|
||||||
if (cb.has_value()) {
|
|
||||||
if (!cb.value()->callback(*this)) {
|
|
||||||
// There's nothing interesting to do here.
|
|
||||||
}
|
|
||||||
}
|
|
||||||
cleanup_suggestions();
|
cleanup_suggestions();
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -772,12 +761,10 @@ void Editor::handle_read_event()
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto cb = m_key_callbacks.get(code_point);
|
m_callback_machine.key_pressed(*this, code_point);
|
||||||
if (cb.has_value()) {
|
if (!m_callback_machine.should_process_last_pressed_key())
|
||||||
if (!cb.value()->callback(*this)) {
|
continue;
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
m_search_offset = 0; // reset search offset on any key
|
m_search_offset = 0; // reset search offset on any key
|
||||||
|
|
||||||
if (code_point == '\t' || reverse_tab) {
|
if (code_point == '\t' || reverse_tab) {
|
||||||
|
|
|
@ -41,6 +41,7 @@
|
||||||
#include <LibCore/DirIterator.h>
|
#include <LibCore/DirIterator.h>
|
||||||
#include <LibCore/Notifier.h>
|
#include <LibCore/Notifier.h>
|
||||||
#include <LibCore/Object.h>
|
#include <LibCore/Object.h>
|
||||||
|
#include <LibLine/KeyCallbackMachine.h>
|
||||||
#include <LibLine/Span.h>
|
#include <LibLine/Span.h>
|
||||||
#include <LibLine/StringMetrics.h>
|
#include <LibLine/StringMetrics.h>
|
||||||
#include <LibLine/Style.h>
|
#include <LibLine/Style.h>
|
||||||
|
@ -53,33 +54,8 @@
|
||||||
|
|
||||||
namespace Line {
|
namespace Line {
|
||||||
|
|
||||||
struct Key {
|
|
||||||
enum Modifier : int {
|
|
||||||
None = 0,
|
|
||||||
Alt = 1,
|
|
||||||
};
|
|
||||||
|
|
||||||
int modifiers { None };
|
|
||||||
unsigned key { 0 };
|
|
||||||
|
|
||||||
Key(unsigned c)
|
|
||||||
: modifiers(None)
|
|
||||||
, key(c) {};
|
|
||||||
|
|
||||||
Key(unsigned c, int modifiers)
|
|
||||||
: modifiers(modifiers)
|
|
||||||
, key(c)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator==(const Key& other) const
|
|
||||||
{
|
|
||||||
return other.key == key && other.modifiers == modifiers;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
struct KeyBinding {
|
struct KeyBinding {
|
||||||
Key key;
|
Vector<Key> keys;
|
||||||
enum class Kind {
|
enum class Kind {
|
||||||
InternalFunction,
|
InternalFunction,
|
||||||
Insertion,
|
Insertion,
|
||||||
|
@ -171,7 +147,8 @@ public:
|
||||||
const Vector<String>& history() const { return m_history; }
|
const Vector<String>& history() const { return m_history; }
|
||||||
|
|
||||||
void register_key_input_callback(const KeyBinding&);
|
void register_key_input_callback(const KeyBinding&);
|
||||||
void register_key_input_callback(Key, Function<bool(Editor&)> callback);
|
void register_key_input_callback(Vector<Key> keys, Function<bool(Editor&)> callback) { m_callback_machine.register_key_input_callback(move(keys), move(callback)); }
|
||||||
|
void register_key_input_callback(Key key, Function<bool(Editor&)> callback) { register_key_input_callback(Vector<Key> { key }, move(callback)); }
|
||||||
|
|
||||||
static StringMetrics actual_rendered_string_metrics(const StringView&);
|
static StringMetrics actual_rendered_string_metrics(const StringView&);
|
||||||
static StringMetrics actual_rendered_string_metrics(const Utf32View&);
|
static StringMetrics actual_rendered_string_metrics(const Utf32View&);
|
||||||
|
@ -283,14 +260,6 @@ private:
|
||||||
// FIXME: Port to Core::Property
|
// FIXME: Port to Core::Property
|
||||||
void save_to(JsonObject&);
|
void save_to(JsonObject&);
|
||||||
|
|
||||||
struct KeyCallback {
|
|
||||||
KeyCallback(Function<bool(Editor&)> cb)
|
|
||||||
: callback(move(cb))
|
|
||||||
{
|
|
||||||
}
|
|
||||||
Function<bool(Editor&)> callback;
|
|
||||||
};
|
|
||||||
|
|
||||||
void handle_interrupt_event();
|
void handle_interrupt_event();
|
||||||
void handle_read_event();
|
void handle_read_event();
|
||||||
|
|
||||||
|
@ -460,7 +429,7 @@ private:
|
||||||
};
|
};
|
||||||
TabDirection m_tab_direction { TabDirection::Forward };
|
TabDirection m_tab_direction { TabDirection::Forward };
|
||||||
|
|
||||||
HashMap<Key, NonnullOwnPtr<KeyCallback>> m_key_callbacks;
|
KeyCallbackMachine m_callback_machine;
|
||||||
|
|
||||||
struct termios m_termios {
|
struct termios m_termios {
|
||||||
};
|
};
|
||||||
|
@ -500,13 +469,3 @@ private:
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace AK {
|
|
||||||
|
|
||||||
template<>
|
|
||||||
struct Traits<Line::Key> : public GenericTraits<Line::Key> {
|
|
||||||
static constexpr bool is_trivial() { return true; }
|
|
||||||
static unsigned hash(Line::Key k) { return pair_int_hash(k.key, k.modifiers); }
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
111
Libraries/LibLine/KeyCallbackMachine.cpp
Normal file
111
Libraries/LibLine/KeyCallbackMachine.cpp
Normal file
|
@ -0,0 +1,111 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2020, the SerenityOS developers.
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
||||||
|
* list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||||
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||||
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||||
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||||
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <LibLine/Editor.h>
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
constexpr u32 ctrl(char c) { return c & 0x3f; }
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace Line {
|
||||||
|
|
||||||
|
void KeyCallbackMachine::register_key_input_callback(Vector<Key> keys, Function<bool(Editor&)> callback)
|
||||||
|
{
|
||||||
|
m_key_callbacks.set(keys, make<KeyCallback>(move(callback)));
|
||||||
|
}
|
||||||
|
|
||||||
|
void KeyCallbackMachine::key_pressed(Editor& editor, Key key)
|
||||||
|
{
|
||||||
|
#ifdef CALLBACK_MACHINE_DEBUG
|
||||||
|
dbgln("Key<{}, {}> pressed, seq_length={}, {} things in the matching vector", key.key, key.modifiers, m_sequence_length, m_current_matching_keys.size());
|
||||||
|
#endif
|
||||||
|
if (m_sequence_length == 0) {
|
||||||
|
ASSERT(m_current_matching_keys.is_empty());
|
||||||
|
|
||||||
|
for (auto& it : m_key_callbacks) {
|
||||||
|
if (it.key.first() == key)
|
||||||
|
m_current_matching_keys.append(it.key);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_current_matching_keys.is_empty()) {
|
||||||
|
m_should_process_this_key = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
++m_sequence_length;
|
||||||
|
Vector<Vector<Key>> old_macthing_keys;
|
||||||
|
swap(m_current_matching_keys, old_macthing_keys);
|
||||||
|
|
||||||
|
for (auto& okey : old_macthing_keys) {
|
||||||
|
if (okey.size() < m_sequence_length)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (okey[m_sequence_length - 1] == key)
|
||||||
|
m_current_matching_keys.append(okey);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_current_matching_keys.is_empty()) {
|
||||||
|
// Insert any keys that were captured
|
||||||
|
if (!old_macthing_keys.is_empty()) {
|
||||||
|
auto& keys = old_macthing_keys.first();
|
||||||
|
for (size_t i = 0; i < m_sequence_length - 1; ++i)
|
||||||
|
editor.insert(keys[i].key);
|
||||||
|
}
|
||||||
|
m_sequence_length = 0;
|
||||||
|
m_should_process_this_key = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef CALLBACK_MACHINE_DEBUG
|
||||||
|
dbgln("seq_length={}, matching vector:", m_sequence_length);
|
||||||
|
for (auto& key : m_current_matching_keys) {
|
||||||
|
for (auto& k : key)
|
||||||
|
dbgln(" {}, {}", k.key, k.modifiers);
|
||||||
|
dbgln("");
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
m_should_process_this_key = false;
|
||||||
|
for (auto& key : m_current_matching_keys) {
|
||||||
|
if (key.size() == m_sequence_length) {
|
||||||
|
m_should_process_this_key = m_key_callbacks.get(key).value()->callback(editor);
|
||||||
|
m_sequence_length = 0;
|
||||||
|
m_current_matching_keys.clear();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void KeyCallbackMachine::interrupted(Editor& editor)
|
||||||
|
{
|
||||||
|
m_sequence_length = 0;
|
||||||
|
m_current_matching_keys.clear();
|
||||||
|
if (auto callback = m_key_callbacks.get({ ctrl('C') }); callback.has_value())
|
||||||
|
m_should_process_this_key = callback.value()->callback(editor);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
112
Libraries/LibLine/KeyCallbackMachine.h
Normal file
112
Libraries/LibLine/KeyCallbackMachine.h
Normal file
|
@ -0,0 +1,112 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2020, the SerenityOS developers.
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
||||||
|
* list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||||
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||||
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||||
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||||
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <AK/Function.h>
|
||||||
|
#include <AK/HashMap.h>
|
||||||
|
#include <AK/String.h>
|
||||||
|
#include <AK/Vector.h>
|
||||||
|
|
||||||
|
namespace Line {
|
||||||
|
|
||||||
|
class Editor;
|
||||||
|
|
||||||
|
struct Key {
|
||||||
|
enum Modifier : int {
|
||||||
|
None = 0,
|
||||||
|
Alt = 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
int modifiers { None };
|
||||||
|
unsigned key { 0 };
|
||||||
|
|
||||||
|
Key(unsigned c)
|
||||||
|
: modifiers(None)
|
||||||
|
, key(c) {};
|
||||||
|
|
||||||
|
Key(unsigned c, int modifiers)
|
||||||
|
: modifiers(modifiers)
|
||||||
|
, key(c)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
bool operator==(const Key& other) const
|
||||||
|
{
|
||||||
|
return other.key == key && other.modifiers == modifiers;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool operator!=(const Key& other) const
|
||||||
|
{
|
||||||
|
return !(*this == other);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct KeyCallback {
|
||||||
|
KeyCallback(Function<bool(Editor&)> cb)
|
||||||
|
: callback(move(cb))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
Function<bool(Editor&)> callback;
|
||||||
|
};
|
||||||
|
|
||||||
|
class KeyCallbackMachine {
|
||||||
|
public:
|
||||||
|
void register_key_input_callback(Vector<Key>, Function<bool(Editor&)> callback);
|
||||||
|
void key_pressed(Editor&, Key);
|
||||||
|
void interrupted(Editor&);
|
||||||
|
bool should_process_last_pressed_key() const { return m_should_process_this_key; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
HashMap<Vector<Key>, NonnullOwnPtr<KeyCallback>> m_key_callbacks;
|
||||||
|
Vector<Vector<Key>> m_current_matching_keys;
|
||||||
|
size_t m_sequence_length { 0 };
|
||||||
|
bool m_should_process_this_key { true };
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace AK {
|
||||||
|
|
||||||
|
template<>
|
||||||
|
struct Traits<Line::Key> : public GenericTraits<Line::Key> {
|
||||||
|
static constexpr bool is_trivial() { return true; }
|
||||||
|
static unsigned hash(Line::Key k) { return pair_int_hash(k.key, k.modifiers); }
|
||||||
|
};
|
||||||
|
|
||||||
|
template<>
|
||||||
|
struct Traits<Vector<Line::Key>> : public GenericTraits<Vector<Line::Key>> {
|
||||||
|
static constexpr bool is_trivial() { return false; }
|
||||||
|
static unsigned hash(const Vector<Line::Key>& ks)
|
||||||
|
{
|
||||||
|
unsigned h = 0;
|
||||||
|
for (auto& k : ks)
|
||||||
|
h ^= Traits<Line::Key>::hash(k);
|
||||||
|
return h;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue