2011-12-20 23:21:11 +04:00
|
|
|
#include "keys.hh"
|
2013-04-09 22:05:40 +04:00
|
|
|
|
2013-01-30 22:03:11 +04:00
|
|
|
#include "utils.hh"
|
2011-12-20 23:21:11 +04:00
|
|
|
|
|
|
|
namespace Kakoune
|
|
|
|
{
|
|
|
|
|
2012-09-03 21:21:11 +04:00
|
|
|
Key canonicalize_ifn(Key key)
|
|
|
|
{
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2014-03-20 23:52:11 +04:00
|
|
|
using KeyAndName = std::pair<const char*, Codepoint>;
|
2014-04-03 04:19:49 +04:00
|
|
|
static const 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 },
|
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;
|
2012-10-11 02:41:48 +04:00
|
|
|
for (ByteCount pos = 0; pos < str.length(); ++pos)
|
2011-12-20 23:21:11 +04:00
|
|
|
{
|
|
|
|
if (str[pos] == '<')
|
|
|
|
{
|
2012-10-11 02:41:48 +04:00
|
|
|
ByteCount end_pos = pos;
|
2011-12-20 23:21:11 +04:00
|
|
|
while (end_pos < str.length() and str[end_pos] != '>')
|
|
|
|
++end_pos;
|
|
|
|
|
|
|
|
if (end_pos < str.length())
|
|
|
|
{
|
2012-01-30 02:49:14 +04:00
|
|
|
Key::Modifiers modifier = Key::Modifiers::None;
|
|
|
|
|
2014-04-20 14:27:59 +04:00
|
|
|
auto keyname = str.substr(pos+1, end_pos - pos - 1);
|
2012-01-30 02:49:14 +04:00
|
|
|
if (keyname.length() > 2)
|
|
|
|
{
|
|
|
|
if (tolower(keyname[0]) == 'c' and keyname[1] == '-')
|
|
|
|
{
|
|
|
|
modifier = Key::Modifiers::Control;
|
2012-10-11 02:41:48 +04:00
|
|
|
keyname = keyname.substr(2_byte);
|
2012-01-30 02:49:14 +04:00
|
|
|
}
|
|
|
|
if (tolower(keyname[0]) == 'a' and keyname[1] == '-')
|
|
|
|
{
|
2012-08-21 22:03:18 +04:00
|
|
|
modifier = Key::Modifiers::Alt;
|
2012-10-11 02:41:48 +04:00
|
|
|
keyname = keyname.substr(2_byte);
|
2012-01-30 02:49:14 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (keyname.length() == 1)
|
|
|
|
{
|
2012-10-09 21:15:05 +04:00
|
|
|
result.push_back(Key{ modifier, Codepoint(keyname[0]) });
|
2012-01-30 02:49:14 +04:00
|
|
|
pos = end_pos;
|
|
|
|
continue;
|
|
|
|
}
|
2013-01-30 22:03:11 +04:00
|
|
|
auto it = find_if(keynamemap, [&keyname](const KeyAndName& item)
|
|
|
|
{ return item.first == keyname; });
|
2014-03-20 23:52:11 +04:00
|
|
|
if (it != end(keynamemap))
|
2011-12-20 23:21:11 +04:00
|
|
|
{
|
2012-09-03 21:21:11 +04:00
|
|
|
Key key = canonicalize_ifn(Key{ modifier, it->second });
|
|
|
|
result.push_back(key);
|
2011-12-20 23:21:11 +04:00
|
|
|
pos = end_pos;
|
|
|
|
continue;
|
|
|
|
}
|
2013-11-09 15:12:55 +04:00
|
|
|
if ((keyname[0] == 'f' or keyname[0] == 'F') and
|
|
|
|
keyname.length() <= 3)
|
|
|
|
{
|
|
|
|
|
|
|
|
int val = 0;
|
|
|
|
for (auto i = 1_byte; i < keyname.length(); ++i)
|
|
|
|
{
|
|
|
|
char c = keyname[i];
|
|
|
|
if (c >= '0' and c <= '9')
|
|
|
|
val = val*10 + c - '0';
|
|
|
|
else
|
|
|
|
{
|
|
|
|
val = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (val >= 1 and val <= 12)
|
|
|
|
{
|
|
|
|
result.push_back(Key{ modifier, Key::F1 + (val - 1) });
|
|
|
|
pos = end_pos;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2011-12-20 23:21:11 +04:00
|
|
|
}
|
|
|
|
}
|
2012-10-09 21:15:05 +04:00
|
|
|
result.push_back({Key::Modifiers::None, Codepoint(str[pos])});
|
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
|
|
|
{
|
|
|
|
bool named = false;
|
|
|
|
String res;
|
|
|
|
auto it = find_if(keynamemap, [&key](const KeyAndName& item)
|
|
|
|
{ return item.second == key.key; });
|
2014-03-20 23:52:11 +04:00
|
|
|
if (it != end(keynamemap))
|
2013-01-30 22:03:11 +04:00
|
|
|
{
|
|
|
|
named = true;
|
|
|
|
res = it->first;
|
|
|
|
}
|
2013-11-09 15:12:55 +04:00
|
|
|
else if (key.key >= Key::F1 and key.key < Key::F12)
|
|
|
|
{
|
|
|
|
named = true;
|
|
|
|
res = "F" + to_string((int)(Codepoint)key.key - (int)(Codepoint)Key::F1 + 1);
|
|
|
|
}
|
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)
|
|
|
|
res = '<' + res + '>';
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2011-12-20 23:21:11 +04:00
|
|
|
}
|