summaryrefslogtreecommitdiff
path: root/src/keys.cc
blob: 6b2479ff6285ca97993eb345408c7089c3bf30b7 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
#include "keys.hh"

#include <unordered_map>

namespace Kakoune
{

Key canonicalize_ifn(Key key)
{
    if (key.key > 0 and key.key < 27)
    {
        assert(key.modifiers == Key::Modifiers::None);
        key.modifiers = Key::Modifiers::Control;
        key.key = key.key - 1 + 'a';
    }
    return key;
}

static std::unordered_map<String, Codepoint> keynamemap = {
    { "ret", '\r' },
    { "space", ' ' },
    { "esc", Key::Escape }
};

KeyList parse_keys(const String& str)
{
    KeyList result;
    for (ByteCount pos = 0; pos < str.length(); ++pos)
    {
        if (str[pos] == '<')
        {
            ByteCount end_pos = pos;
            while (end_pos < str.length() and str[end_pos] != '>')
                ++end_pos;

            if (end_pos < str.length())
            {
                Key::Modifiers modifier = Key::Modifiers::None;

                String keyname = str.substr(pos+1, end_pos - pos - 1);
                if (keyname.length() > 2)
                {
                    if (tolower(keyname[0]) == 'c' and keyname[1] == '-')
                    {
                        modifier = Key::Modifiers::Control;
                        keyname = keyname.substr(2_byte);
                    }
                    if (tolower(keyname[0]) == 'a' and keyname[1] == '-')
                    {
                        modifier = Key::Modifiers::Alt;
                        keyname = keyname.substr(2_byte);
                    }
                }
                if (keyname.length() == 1)
                {
                    result.push_back(Key{ modifier, Codepoint(keyname[0]) });
                    pos = end_pos;
                    continue;
                }
                auto it = keynamemap.find(keyname);
                if (it != keynamemap.end())
                {
                    Key key = canonicalize_ifn(Key{ modifier, it->second });
                    result.push_back(key);
                    pos = end_pos;
                    continue;
                }
            }
        }
        result.push_back({Key::Modifiers::None, Codepoint(str[pos])});
    }
    return result;
}

}