Skip to content

Commit

Permalink
Implement windows formating in sway/workspaces
Browse files Browse the repository at this point in the history
This implementation mimics to some extend the implementation of hyprland

Signed-off-by: Jo De Boeck <[email protected]>
  • Loading branch information
grimpy committed Dec 27, 2023
1 parent 41ebdc3 commit d284a07
Show file tree
Hide file tree
Showing 3 changed files with 96 additions and 12 deletions.
5 changes: 5 additions & 0 deletions include/modules/sway/workspaces.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
#include "client.hpp"
#include "modules/sway/ipc/client.hpp"
#include "util/json.hpp"
#include "util/regex_collection.hpp"

namespace waybar::modules::sway {

Expand All @@ -31,6 +32,7 @@ class Workspaces : public AModule, public sigc::trackable {
void onCmd(const struct Ipc::ipc_response&);
void onEvent(const struct Ipc::ipc_response&);
bool filterButtons();
bool hasFlag(const Json::Value&, const std::string);

Check warning on line 35 in include/modules/sway/workspaces.hpp

View workflow job for this annotation

GitHub Actions / build

include/modules/sway/workspaces.hpp:35:36 [readability-avoid-const-params-in-decls]

parameter 2 is const-qualified in the function declaration; const-qualification of parameters only has an effect in function definitions
Gtk::Button& addButton(const Json::Value&);
void onButtonReady(const Json::Value&, Gtk::Button&);
std::string getIcon(const std::string&, const Json::Value&);
Expand All @@ -44,6 +46,9 @@ class Workspaces : public AModule, public sigc::trackable {
std::vector<std::string> high_priority_named_;
std::vector<std::string> workspaces_order_;
Gtk::Box box_;
std::string m_format_window_seperator;

Check warning on line 49 in include/modules/sway/workspaces.hpp

View workflow job for this annotation

GitHub Actions / build

include/modules/sway/workspaces.hpp:49:15 [readability-identifier-naming]

invalid case style for private member 'm_format_window_seperator'
std::string m_windowRewriteDefault;
util::RegexCollection m_windowRewriteRules;
util::JsonParser parser_;
std::unordered_map<std::string, Gtk::Button> buttons_;
std::mutex mutex_;
Expand Down
32 changes: 32 additions & 0 deletions man/waybar-sway-workspaces.5.scd
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,23 @@ warp-on-scroll: ++
default: true ++
If set to false, you can scroll to cycle through workspaces without mouse warping being enabled. If set to true this behaviour is disabled.

*window-rewrite*: ++
typeof: object ++
Regex rules to map window class to an icon or preferred method of representation for a workspace's window.
Keys are the rules, while the values are the methods of representation.
Rules may specify `class<...>`, `title<...>`, or both in order to fine-tune the matching.

*window-rewrite-default*:
typeof: string ++
default: "?" ++
The default method of representation for a workspace's window. This will be used for windows whose classes do not match any of the rules in *window-rewrite*.

*format-window-separator*: ++
typeof: string ++
default: " " ++
The separator to be used between windows in a workspace.


# FORMAT REPLACEMENTS

*{value}*: Name of the workspace, as defined by sway.
Expand All @@ -94,6 +111,8 @@ warp-on-scroll: ++

*{output}*: Output where the workspace is located.

*{windows}*: Result from window-rewrite

# ICONS

Additional to workspace name matching, the following *format-icons* can be set.
Expand Down Expand Up @@ -143,6 +162,19 @@ n.b.: the list of outputs can be obtained from command line using *swaymsg -t ge
}
```

```
"sway/workspaces": {
"format": "<span size='larger'>{name}</span> {windows}",
"format-window-separator": " | ",
"window-rewrite-default": "{name}",
"window-format": "<span color='#e0e0e0'>{name}</span>",
"window-rewrite": {
"class<firefox>": "",
"class<kitty>": "k",
}
}
```

# Style

- *#workspaces button*
Expand Down
71 changes: 59 additions & 12 deletions src/modules/sway/workspaces.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -38,10 +38,25 @@ Workspaces::Workspaces(const std::string &id, const Bar &bar, const Json::Value
box_.get_style_context()->add_class(id);
}
event_box_.add(box_);
if (config_["format-window-separator"].isString()) {
m_format_window_seperator = config_["format-window-separator"].asString();
} else {
m_format_window_seperator = " ";
}
const Json::Value &windowRewrite = config["window-rewrite"];

const Json::Value &windowRewriteDefaultConfig = config["window-rewrite-default"];
m_windowRewriteDefault =
windowRewriteDefaultConfig.isString() ? windowRewriteDefaultConfig.asString() : "?";

m_windowRewriteRules = waybar::util::RegexCollection(
windowRewrite, m_windowRewriteDefault,
[this](std::string &window_rule) { return 1; });
ipc_.subscribe(R"(["workspace"])");
ipc_.subscribe(R"(["window"])");
ipc_.signal_event.connect(sigc::mem_fun(*this, &Workspaces::onEvent));
ipc_.signal_cmd.connect(sigc::mem_fun(*this, &Workspaces::onCmd));
ipc_.sendCmd(IPC_GET_WORKSPACES);
ipc_.sendCmd(IPC_GET_TREE);
if (config["enable-bar-scroll"].asBool()) {
auto &window = const_cast<Bar &>(bar_).window;
window.add_events(Gdk::SCROLL_MASK | Gdk::SMOOTH_SCROLL_MASK);
Expand All @@ -59,26 +74,30 @@ Workspaces::Workspaces(const std::string &id, const Bar &bar, const Json::Value

void Workspaces::onEvent(const struct Ipc::ipc_response &res) {
try {
ipc_.sendCmd(IPC_GET_WORKSPACES);
ipc_.sendCmd(IPC_GET_TREE);
} catch (const std::exception &e) {
spdlog::error("Workspaces: {}", e.what());
}
}

void Workspaces::onCmd(const struct Ipc::ipc_response &res) {
if (res.type == IPC_GET_WORKSPACES) {
if (res.type == IPC_GET_TREE) {
try {
{
std::lock_guard<std::mutex> lock(mutex_);
auto payload = parser_.parse(res.payload);
workspaces_.clear();
std::copy_if(payload.begin(), payload.end(), std::back_inserter(workspaces_),
std::vector<Json::Value> outputs;
std::copy_if(payload["nodes"].begin(), payload["nodes"].end(), std::back_inserter(outputs),
[&](const auto &workspace) {
return !config_["all-outputs"].asBool()
? workspace["output"].asString() == bar_.output->name
? workspace["name"].asString() == bar_.output->name
: true;
});

for (auto &output : outputs) {
std::copy(output["nodes"].begin(), output["nodes"].end(), std::back_inserter(workspaces_));
}
if (config_["persistent_workspaces"].isObject()) {
spdlog::warn(
"persistent_workspaces is deprecated. Please change config to use "
Expand Down Expand Up @@ -203,6 +222,20 @@ bool Workspaces::filterButtons() {
return needReorder;
}

bool Workspaces::hasFlag(const Json::Value& node, const std::string flag) {

Check warning on line 225 in src/modules/sway/workspaces.cpp

View workflow job for this annotation

GitHub Actions / build

src/modules/sway/workspaces.cpp:225:18 [readability-convert-member-functions-to-static]

method 'hasFlag' can be made static

Check warning on line 225 in src/modules/sway/workspaces.cpp

View workflow job for this annotation

GitHub Actions / build

src/modules/sway/workspaces.cpp:225:71 [performance-unnecessary-value-param]

the const qualified parameter 'flag' is copied for each invocation; consider making it a reference
bool result = false;
if (node[flag].asBool()) {
return true;
}
for (auto it = node["nodes"].begin(); it != node["nodes"].end(); ++it) {

Check warning on line 230 in src/modules/sway/workspaces.cpp

View workflow job for this annotation

GitHub Actions / build

src/modules/sway/workspaces.cpp:230:3 [modernize-loop-convert]

use range-based for loop instead
if ((*it)[flag].asBool()) {
result = true;
break;
}
}
return result;
}

auto Workspaces::update() -> void {
std::lock_guard<std::mutex> lock(mutex_);
bool needReorder = filterButtons();
Expand All @@ -212,22 +245,25 @@ auto Workspaces::update() -> void {
needReorder = true;
}
auto &button = bit == buttons_.end() ? addButton(*it) : bit->second;
if ((*it)["focused"].asBool()) {
if (needReorder) {
box_.reorder_child(button, it - workspaces_.begin());
}
if (hasFlag((*it), "focused")) {
button.get_style_context()->add_class("focused");
} else {
button.get_style_context()->remove_class("focused");
}
if ((*it)["visible"].asBool()) {
if (hasFlag((*it), "visible")) {
button.get_style_context()->add_class("visible");
} else {
button.get_style_context()->remove_class("visible");
}
if ((*it)["urgent"].asBool()) {
if (hasFlag((*it), "urgent")) {
button.get_style_context()->add_class("urgent");
} else {
button.get_style_context()->remove_class("urgent");
}
if ((*it)["target_output"].isString()) {
if (hasFlag((*it), "target_output")) {
button.get_style_context()->add_class("persistent");
} else {
button.get_style_context()->remove_class("persistent");
Expand All @@ -241,15 +277,26 @@ auto Workspaces::update() -> void {
} else {
button.get_style_context()->remove_class("current_output");
}
if (needReorder) {
box_.reorder_child(button, it - workspaces_.begin());
}
std::string output = (*it)["name"].asString();
std::string windows = "";
if (config_["window-format"].isString()) {
auto format = config_["window-format"].asString();
for (auto node = (*it)["nodes"].begin(); node != (*it)["nodes"].end(); ++node) {
std::string title = g_markup_escape_text((*node)["name"].asString().c_str(), -1);
std::string window_class = (*node)["app_id"].asString();

Check warning on line 286 in src/modules/sway/workspaces.cpp

View workflow job for this annotation

GitHub Actions / build

src/modules/sway/workspaces.cpp:286:21 [readability-identifier-naming]

invalid case style for variable 'window_class'
std::string windowReprKey = fmt::format("class<{}> title<{}>", window_class, title);
std::string window = m_windowRewriteRules.get(windowReprKey);
// allow result to have formatting
window = fmt::format(fmt::runtime(window), fmt::arg("name", title), fmt::arg("class", window_class));
windows = windows + window + m_format_window_seperator;

Check warning on line 291 in src/modules/sway/workspaces.cpp

View workflow job for this annotation

GitHub Actions / build

src/modules/sway/workspaces.cpp:291:9 [performance-inefficient-string-concatenation]

string concatenation results in allocation of unnecessary temporary strings; consider using 'operator+=' or 'string::append()' instead
}
}
if (config_["format"].isString()) {
auto format = config_["format"].asString();
output = fmt::format(fmt::runtime(format), fmt::arg("icon", getIcon(output, *it)),
fmt::arg("value", output), fmt::arg("name", trimWorkspaceName(output)),
fmt::arg("index", (*it)["num"].asString()),
fmt::arg("windows", windows.substr(0, windows.length() - m_format_window_seperator.length())),
fmt::arg("output", (*it)["output"].asString()));
}
if (!config_["disable-markup"].asBool()) {
Expand Down

0 comments on commit d284a07

Please sign in to comment.