2011-12-20 23:21:11 +04:00
|
|
|
#include "keys.hh"
|
2013-04-09 22:05:40 +04:00
|
|
|
|
2014-12-23 16:34:21 +03:00
|
|
|
#include "containers.hh"
|
|
|
|
#include "exception.hh"
|
|
|
|
#include "string.hh"
|
2015-05-22 15:58:56 +03:00
|
|
|
#include "unit_tests.hh"
|
2014-06-04 14:51:20 +04:00
|
|
|
#include "utf8_iterator.hh"
|
2015-05-22 15:58:56 +03:00
|
|
|
#include "utils.hh"
|
2011-12-20 23:21:11 +04:00
|
|
|
|
|
|
|
namespace Kakoune
|
|
|
|
{
|
|
|
|
|
2015-08-18 02:19:14 +03:00
|
|
|
static Key canonicalize_ifn(Key key)
|
2012-09-03 21:21:11 +04:00
|
|
|
{
|
|
|
|
if (key.key > 0 and key.key < 27)
|
|
|
|
{
|
2013-04-09 22:04:11 +04:00
|
|
|
kak_assert(key.modifiers == Key::Modifiers::None);
|
2012-09-03 21:21:11 +04:00
|
|
|
key.modifiers = Key::Modifiers::Control;
|
|
|
|
key.key = key.key - 1 + 'a';
|
|
|
|
}
|
|
|
|
return key;
|
|
|
|
}
|
|
|
|
|
2015-08-18 02:19:14 +03:00
|
|
|
Optional<Codepoint> Key::codepoint() const
|
|
|
|
{
|
|
|
|
if (*this == ctrl('m'))
|
|
|
|
return '\n';
|
|
|
|
if (*this == ctrl('i'))
|
|
|
|
return '\t';
|
|
|
|
if (modifiers == Modifiers::None and key > 27 and
|
|
|
|
(key < 0xD800 or key > 0xDFFF)) // avoid surrogates
|
|
|
|
return key;
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2014-12-29 20:18:42 +03:00
|
|
|
struct KeyAndName { const char* name; Codepoint key; };
|
|
|
|
static constexpr KeyAndName keynamemap[] = {
|
2012-09-03 21:21:11 +04:00
|
|
|
{ "ret", '\r' },
|
2012-01-30 02:49:14 +04:00
|
|
|
{ "space", ' ' },
|
2013-02-19 16:50:27 +04:00
|
|
|
{ "tab", '\t' },
|
2013-04-02 20:41:45 +04:00
|
|
|
{ "lt", '<' },
|
|
|
|
{ "gt", '>' },
|
2013-02-19 16:50:27 +04:00
|
|
|
{ "backspace", Key::Backspace},
|
2013-01-30 22:03:11 +04:00
|
|
|
{ "esc", Key::Escape },
|
|
|
|
{ "up", Key::Up },
|
2013-02-18 21:58:07 +04:00
|
|
|
{ "down", Key::Down},
|
2013-02-19 16:50:27 +04:00
|
|
|
{ "left", Key::Left },
|
|
|
|
{ "right", Key::Right },
|
|
|
|
{ "pageup", Key::PageUp },
|
|
|
|
{ "pagedown", Key::PageDown },
|
|
|
|
{ "home", Key::Home },
|
|
|
|
{ "end", Key::End },
|
|
|
|
{ "backtab", Key::BackTab },
|
2014-05-25 20:41:28 +04:00
|
|
|
{ "del", Key::Delete },
|
2011-12-20 23:21:11 +04:00
|
|
|
};
|
|
|
|
|
2014-04-20 14:27:59 +04:00
|
|
|
KeyList parse_keys(StringView str)
|
2011-12-20 23:21:11 +04:00
|
|
|
{
|
|
|
|
KeyList result;
|
2014-11-09 16:02:01 +03:00
|
|
|
using Utf8It = utf8::iterator<const char*>;
|
2015-09-23 21:39:21 +03:00
|
|
|
for (Utf8It it{str.begin(), str}, str_end{str.end(), str}; it < str_end; ++it)
|
2011-12-20 23:21:11 +04:00
|
|
|
{
|
2014-11-09 16:02:01 +03:00
|
|
|
if (*it != '<')
|
2011-12-20 23:21:11 +04:00
|
|
|
{
|
2014-11-09 16:02:01 +03:00
|
|
|
result.push_back({Key::Modifiers::None, *it});
|
|
|
|
continue;
|
|
|
|
}
|
2011-12-20 23:21:11 +04:00
|
|
|
|
2015-03-27 16:18:52 +03:00
|
|
|
Utf8It end_it = std::find(it, str_end, '>');
|
2014-11-09 16:02:01 +03:00
|
|
|
if (end_it == str_end)
|
|
|
|
{
|
|
|
|
result.push_back({Key::Modifiers::None, *it});
|
|
|
|
continue;
|
|
|
|
}
|
2012-01-30 02:49:14 +04:00
|
|
|
|
2014-11-09 16:02:01 +03:00
|
|
|
Key::Modifiers modifier = Key::Modifiers::None;
|
2013-11-09 15:12:55 +04:00
|
|
|
|
2014-11-09 16:02:01 +03:00
|
|
|
StringView desc{it.base()+1, end_it.base()};
|
2015-03-10 22:33:46 +03:00
|
|
|
if (desc.length() > 2 and desc[1_byte] == '-')
|
2014-11-09 16:02:01 +03:00
|
|
|
{
|
2015-03-10 22:33:46 +03:00
|
|
|
switch(tolower(desc[0_byte]))
|
2014-11-09 16:02:01 +03:00
|
|
|
{
|
|
|
|
case 'c': modifier = Key::Modifiers::Control; break;
|
|
|
|
case 'a': modifier = Key::Modifiers::Alt; break;
|
|
|
|
default:
|
|
|
|
throw runtime_error("unable to parse modifier in " +
|
|
|
|
StringView{it.base(), end_it.base()+1});
|
2011-12-20 23:21:11 +04:00
|
|
|
}
|
2014-11-09 16:02:01 +03:00
|
|
|
desc = desc.substr(2_byte);
|
|
|
|
}
|
|
|
|
auto name_it = find_if(keynamemap, [&desc](const KeyAndName& item)
|
2014-12-29 20:18:42 +03:00
|
|
|
{ return item.name == desc; });
|
2014-11-09 16:02:01 +03:00
|
|
|
if (name_it != end(keynamemap))
|
2014-12-29 20:18:42 +03:00
|
|
|
result.push_back(canonicalize_ifn({ modifier, name_it->key }));
|
2014-11-09 16:02:01 +03:00
|
|
|
else if (desc.char_length() == 1)
|
|
|
|
result.push_back(Key{ modifier, desc[0_char] });
|
2015-03-10 22:33:46 +03:00
|
|
|
else if (tolower(desc[0_byte]) == 'f' and desc.length() <= 3)
|
2014-11-09 16:02:01 +03:00
|
|
|
{
|
|
|
|
int val = str_to_int(desc.substr(1_byte));
|
|
|
|
if (val >= 1 and val <= 12)
|
|
|
|
result.push_back(Key{ modifier, Key::F1 + (val - 1) });
|
|
|
|
else
|
|
|
|
throw runtime_error("Only F1 through F12 are supported");
|
2011-12-20 23:21:11 +04:00
|
|
|
}
|
2014-11-09 16:02:01 +03:00
|
|
|
else
|
|
|
|
throw runtime_error("Failed to parse " +
|
|
|
|
StringView{it.base(), end_it.base()+1});
|
|
|
|
|
|
|
|
it = end_it;
|
2011-12-20 23:21:11 +04:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2013-07-26 03:44:25 +04:00
|
|
|
String key_to_str(Key key)
|
2013-01-30 22:03:11 +04:00
|
|
|
{
|
2015-03-28 21:18:28 +03:00
|
|
|
if (key.modifiers & Key::Modifiers::MouseEvent)
|
|
|
|
return "<mouse event>";
|
|
|
|
|
2013-01-30 22:03:11 +04:00
|
|
|
bool named = false;
|
|
|
|
String res;
|
|
|
|
auto it = find_if(keynamemap, [&key](const KeyAndName& item)
|
2014-12-29 20:18:42 +03:00
|
|
|
{ return item.key == key.key; });
|
2014-03-20 23:52:11 +04:00
|
|
|
if (it != end(keynamemap))
|
2013-01-30 22:03:11 +04:00
|
|
|
{
|
|
|
|
named = true;
|
2014-12-29 20:18:42 +03:00
|
|
|
res = it->name;
|
2013-01-30 22:03:11 +04:00
|
|
|
}
|
2013-11-09 15:12:55 +04:00
|
|
|
else if (key.key >= Key::F1 and key.key < Key::F12)
|
|
|
|
{
|
|
|
|
named = true;
|
2014-11-09 14:42:06 +03:00
|
|
|
res = "F" + to_string((int)(key.key - Key::F1 + 1));
|
2013-11-09 15:12:55 +04:00
|
|
|
}
|
2013-01-30 22:03:11 +04:00
|
|
|
else
|
|
|
|
res = codepoint_to_str(key.key);
|
|
|
|
|
|
|
|
switch (key.modifiers)
|
|
|
|
{
|
|
|
|
case Key::Modifiers::Control: res = "c-" + res; named = true; break;
|
|
|
|
case Key::Modifiers::Alt: res = "a-" + res; named = true; break;
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
if (named)
|
2014-12-08 16:59:29 +03:00
|
|
|
res = StringView{'<'} + res + StringView{'>'};
|
2013-01-30 22:03:11 +04:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2015-05-22 15:58:56 +03:00
|
|
|
UnitTest test_keys{[]()
|
|
|
|
{
|
|
|
|
KeyList keys{
|
|
|
|
{ ' ' },
|
|
|
|
{ 'c' },
|
|
|
|
{ Key::Modifiers::Alt, 'j' },
|
|
|
|
{ Key::Modifiers::Control, 'r' }
|
|
|
|
};
|
|
|
|
String keys_as_str;
|
|
|
|
for (auto& key : keys)
|
|
|
|
keys_as_str += key_to_str(key);
|
|
|
|
auto parsed_keys = parse_keys(keys_as_str);
|
|
|
|
kak_assert(keys == parsed_keys);
|
|
|
|
}};
|
|
|
|
|
2011-12-20 23:21:11 +04:00
|
|
|
}
|