Skip to content

Commit 84cc699

Browse files
committed
hyprland/language: Fix commas in keyboard names and layouts
Commit 29917fb ("Fix hyprland/language events not working with keyboard names with commas in them (#3224)") attempted to fix the problem, but introduced the opposite problem. The event data from Hyprland is "activelayout>>KEYBOARDNAME,LAYOUTNAME", but both KEYBOARDNAME and LAYOUTNAME may contain embedded commas, so the simple `find_first_of(',')` or `find_last_of(',')` solutions will break things for different users. This patch attempts to solve the problem by trying all combinations of keyboardname and layoutname until one makes sense, either by matching the keyboard name to the user-configured device name, or by matching the layoutname to the list of valid layouts.
1 parent 034760e commit 84cc699

File tree

1 file changed

+26
-5
lines changed

1 file changed

+26
-5
lines changed

src/modules/hyprland/language.cpp

+26-5
Original file line numberDiff line numberDiff line change
@@ -69,15 +69,36 @@ auto Language::update() -> void {
6969

7070
void Language::onEvent(const std::string& ev) {
7171
std::lock_guard<std::mutex> lg(mutex_);
72-
std::string kbName(begin(ev) + ev.find_last_of('>') + 1, begin(ev) + ev.find_first_of(','));
73-
auto layoutName = ev.substr(ev.find_last_of(',') + 1);
72+
73+
const static std::string eventPrefix("activelayout>>");
74+
75+
if (!ev.starts_with(eventPrefix))
76+
return;
77+
78+
std::string kbName;
79+
std::string layoutName;
80+
Layout layout;
81+
82+
// The format of this event data is "activelayout>>KEYBOARDNAME,LAYOUTNAME",
83+
// but both KEYBOARDNAME and LAYOUTNAME may contain embedded commas. Try to
84+
// split the string at each comma until we find either the configured
85+
// keyboard-name or a recognized layout name.
86+
for (size_t i = eventPrefix.length(); (i = ev.find_first_of(',', i)) != ev.npos; i++) {
87+
kbName = ev.substr(eventPrefix.length(), i - eventPrefix.length());
88+
layoutName = waybar::util::sanitize_string(ev.substr(i + 1));
89+
layout = getLayout(layoutName);
90+
91+
if (config_.isMember("keyboard-name") && kbName == config_["keyboard-name"].asString())
92+
break;
93+
94+
if (layout.full_name != "")
95+
break;
96+
}
7497

7598
if (config_.isMember("keyboard-name") && kbName != config_["keyboard-name"].asString())
7699
return; // ignore
77100

78-
layoutName = waybar::util::sanitize_string(layoutName);
79-
80-
layout_ = getLayout(layoutName);
101+
layout_ = layout;
81102

82103
spdlog::debug("hyprland language onevent with {}", layoutName);
83104

0 commit comments

Comments
 (0)