blob: 726b24bf7974983c820e9df87f6c51c2ae1e8a81 (
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
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
|
#include "config.hpp"
#include <string>
// toml++ is consumed via its PREBUILT subproject library, which is compiled with
// exceptions enabled (toml::v3::ex::parse). We therefore use the throwing parse
// and catch toml::parse_error HERE so a syntax error becomes a clean
// parse_error result, never an exception escaping into activate() (the brief's
// hard rule). Matching the lib's exception mode avoids the toml::v3::noex link
// mismatch that TOML_EXCEPTIONS=0 in this TU would cause.
#include <toml++/toml.hpp>
namespace unbox::ext_keybindings::config {
auto load_from_string(std::string_view toml_text) -> LoadResult {
LoadResult result;
toml::table root;
try {
root = toml::parse(toml_text);
} catch (const toml::parse_error& err) {
result.parse_error = true;
std::string msg = "unbox.toml parse error: ";
msg += std::string(err.description());
result.warnings.push_back(std::move(msg));
return result;
}
const toml::node* keybind_node = root.get("keybind");
if (keybind_node == nullptr) {
// No [[keybind]] at all: not an error, just zero bindings. The glue
// falls back to defaults.
result.warnings.emplace_back("unbox.toml has no [[keybind]] entries");
return result;
}
const toml::array* entries = keybind_node->as_array();
if (entries == nullptr) {
result.warnings.emplace_back("'keybind' must be an array of tables ([[keybind]])");
return result;
}
std::size_t idx = 0;
for (const toml::node& node : *entries) {
const std::string where = "keybind #" + std::to_string(idx);
++idx;
const toml::table* entry = node.as_table();
if (entry == nullptr) {
result.warnings.push_back(where + ": not a table");
continue;
}
// keys (required, string).
const toml::node* keys_node = entry->get("keys");
if (keys_node == nullptr || !keys_node->is_string()) {
result.warnings.push_back(where + ": missing or non-string 'keys'");
continue;
}
const std::string keys = keys_node->value<std::string>().value();
// action (required, string).
const toml::node* action_node = entry->get("action");
if (action_node == nullptr || !action_node->is_string()) {
result.warnings.push_back(where + ": missing or non-string 'action'");
continue;
}
const std::string action_str = action_node->value<std::string>().value();
const auto action = policy::action_from_string(action_str);
if (!action) {
result.warnings.push_back(where + ": unknown action '" + action_str + "'");
continue;
}
// command (required iff action == spawn; string when present).
std::string command;
const toml::node* command_node = entry->get("command");
if (command_node != nullptr) {
if (!command_node->is_string()) {
result.warnings.push_back(where + ": 'command' must be a string");
continue;
}
command = command_node->value<std::string>().value();
}
if (*action == policy::Action::spawn && command.empty()) {
result.warnings.push_back(where + ": action 'spawn' requires a non-empty 'command'");
continue;
}
// combo (validated last so a bad combo skips with a clear message).
const auto combo = policy::parse_combo(keys);
if (!combo) {
result.warnings.push_back(where + ": malformed key combo '" + keys + "'");
continue;
}
result.bindings.push_back(policy::Binding{
.combo = *combo, .action = *action, .command = std::move(command)});
}
return result;
}
auto reload_bindings(const std::vector<policy::Binding>& current, std::string_view toml_text)
-> ReloadDecision {
ReloadDecision decision;
LoadResult loaded = load_from_string(toml_text);
decision.warnings = std::move(loaded.warnings);
// KEEP-OLD on a syntax error or on a parse that produced no usable bindings
// (an empty file, a [[keybind]]-less doc, or one where every entry was
// skipped). Either way the live table must remain the user's working keys.
if (loaded.parse_error || loaded.bindings.empty()) {
decision.bindings = current; // unchanged copy
decision.swapped = false;
return decision;
}
// SUCCESS: the new table replaces the live one (the swap).
decision.bindings = std::move(loaded.bindings);
decision.swapped = true;
return decision;
}
} // namespace unbox::ext_keybindings::config
|