2018-08-15 18:17:17 +00:00
|
|
|
#include "modules/sway/window.hpp"
|
2019-05-18 23:44:45 +00:00
|
|
|
#include <spdlog/spdlog.h>
|
2018-08-15 18:17:17 +00:00
|
|
|
|
2019-04-19 09:09:06 +00:00
|
|
|
namespace waybar::modules::sway {
|
|
|
|
|
|
|
|
Window::Window(const std::string& id, const Bar& bar, const Json::Value& config)
|
2019-06-28 12:16:09 +00:00
|
|
|
: ALabel(config, "window", id, "{}", 0, true), bar_(bar), windowId_(-1) {
|
2019-06-11 12:01:55 +00:00
|
|
|
ipc_.subscribe(R"(["window","workspace"])");
|
2019-04-19 09:09:06 +00:00
|
|
|
ipc_.signal_event.connect(sigc::mem_fun(*this, &Window::onEvent));
|
|
|
|
ipc_.signal_cmd.connect(sigc::mem_fun(*this, &Window::onCmd));
|
2019-04-25 14:47:51 +00:00
|
|
|
// Get Initial focused window
|
|
|
|
getTree();
|
2018-08-20 12:50:45 +00:00
|
|
|
// Launch worker
|
|
|
|
worker();
|
|
|
|
}
|
|
|
|
|
2019-05-07 11:43:48 +00:00
|
|
|
void Window::onEvent(const struct Ipc::ipc_response& res) { getTree(); }
|
2019-04-19 09:09:06 +00:00
|
|
|
|
2019-04-24 10:37:24 +00:00
|
|
|
void Window::onCmd(const struct Ipc::ipc_response& res) {
|
2019-05-09 08:30:54 +00:00
|
|
|
try {
|
2019-06-17 09:39:45 +00:00
|
|
|
std::lock_guard<std::mutex> lock(mutex_);
|
2019-06-15 12:57:52 +00:00
|
|
|
auto payload = parser_.parse(res.payload);
|
2019-08-31 17:50:56 +00:00
|
|
|
auto output = payload["output"].isString() ? payload["output"].asString() : "";
|
2019-06-14 09:27:40 +00:00
|
|
|
std::tie(app_nb_, windowId_, window_, app_id_) = getFocusedNode(payload["nodes"], output);
|
2019-06-04 15:34:00 +00:00
|
|
|
dp.emit();
|
2019-05-09 08:30:54 +00:00
|
|
|
} catch (const std::exception& e) {
|
2019-05-18 23:44:45 +00:00
|
|
|
spdlog::error("Window: {}", e.what());
|
2019-04-25 14:47:51 +00:00
|
|
|
}
|
2019-04-19 09:09:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Window::worker() {
|
2018-08-16 12:29:41 +00:00
|
|
|
thread_ = [this] {
|
2018-08-15 18:17:17 +00:00
|
|
|
try {
|
2019-04-19 09:09:06 +00:00
|
|
|
ipc_.handleEvent();
|
2018-08-15 18:17:17 +00:00
|
|
|
} catch (const std::exception& e) {
|
2019-05-18 23:44:45 +00:00
|
|
|
spdlog::error("Window: {}", e.what());
|
2018-08-15 18:17:17 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-04-19 09:09:06 +00:00
|
|
|
auto Window::update() -> void {
|
2019-06-04 15:34:00 +00:00
|
|
|
if (!old_app_id_.empty()) {
|
|
|
|
bar_.window.get_style_context()->remove_class(old_app_id_);
|
|
|
|
}
|
|
|
|
if (app_nb_ == 0) {
|
|
|
|
bar_.window.get_style_context()->remove_class("solo");
|
|
|
|
if (!bar_.window.get_style_context()->has_class("empty")) {
|
|
|
|
bar_.window.get_style_context()->add_class("empty");
|
|
|
|
}
|
|
|
|
} else if (app_nb_ == 1) {
|
|
|
|
bar_.window.get_style_context()->remove_class("empty");
|
|
|
|
if (!bar_.window.get_style_context()->has_class("solo")) {
|
|
|
|
bar_.window.get_style_context()->add_class("solo");
|
|
|
|
}
|
|
|
|
if (!app_id_.empty() && !bar_.window.get_style_context()->has_class(app_id_)) {
|
|
|
|
bar_.window.get_style_context()->add_class(app_id_);
|
|
|
|
old_app_id_ = app_id_;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
bar_.window.get_style_context()->remove_class("solo");
|
|
|
|
bar_.window.get_style_context()->remove_class("empty");
|
|
|
|
}
|
2018-11-21 19:49:09 +00:00
|
|
|
label_.set_markup(fmt::format(format_, window_));
|
2019-02-22 10:35:26 +00:00
|
|
|
if (tooltipEnabled()) {
|
|
|
|
label_.set_tooltip_text(window_);
|
|
|
|
}
|
2018-08-15 18:17:17 +00:00
|
|
|
}
|
|
|
|
|
2019-04-25 14:47:51 +00:00
|
|
|
std::tuple<std::size_t, int, std::string, std::string> Window::getFocusedNode(
|
2019-06-14 09:27:40 +00:00
|
|
|
const Json::Value& nodes, std::string& output) {
|
|
|
|
for (auto const& node : nodes) {
|
2019-06-14 08:57:22 +00:00
|
|
|
if (node["output"].isString()) {
|
|
|
|
output = node["output"].asString();
|
|
|
|
}
|
2019-06-14 09:27:40 +00:00
|
|
|
if (node["focused"].asBool() && (node["type"] == "con" || node["type"] == "floating_con")) {
|
2019-06-14 08:57:22 +00:00
|
|
|
if ((!config_["all-outputs"].asBool() && output == bar_.output->name) ||
|
2019-04-25 14:47:51 +00:00
|
|
|
config_["all-outputs"].asBool()) {
|
|
|
|
auto app_id = node["app_id"].isString() ? node["app_id"].asString()
|
|
|
|
: node["window_properties"]["instance"].asString();
|
2019-06-14 09:27:40 +00:00
|
|
|
return {nodes.size(),
|
2019-04-25 14:47:51 +00:00
|
|
|
node["id"].asInt(),
|
|
|
|
Glib::Markup::escape_text(node["name"].asString()),
|
|
|
|
app_id};
|
|
|
|
}
|
2018-08-16 12:29:41 +00:00
|
|
|
}
|
2019-06-14 09:27:40 +00:00
|
|
|
auto [nb, id, name, app_id] = getFocusedNode(node["nodes"], output);
|
|
|
|
if (id > -1 && !name.empty()) {
|
|
|
|
return {nb, id, name, app_id};
|
|
|
|
}
|
|
|
|
// Search for floating node
|
|
|
|
std::tie(nb, id, name, app_id) = getFocusedNode(node["floating_nodes"], output);
|
2018-09-18 19:16:35 +00:00
|
|
|
if (id > -1 && !name.empty()) {
|
2019-04-25 14:47:51 +00:00
|
|
|
return {nb, id, name, app_id};
|
2018-08-16 12:29:41 +00:00
|
|
|
}
|
2018-08-15 18:17:17 +00:00
|
|
|
}
|
2019-04-25 14:47:51 +00:00
|
|
|
return {0, -1, "", ""};
|
2018-08-15 18:17:17 +00:00
|
|
|
}
|
|
|
|
|
2019-04-25 14:47:51 +00:00
|
|
|
void Window::getTree() {
|
2018-08-15 18:17:17 +00:00
|
|
|
try {
|
2019-04-19 09:09:06 +00:00
|
|
|
ipc_.sendCmd(IPC_GET_TREE);
|
2019-04-18 15:52:00 +00:00
|
|
|
} catch (const std::exception& e) {
|
2019-05-18 23:44:45 +00:00
|
|
|
spdlog::error("Window: {}", e.what());
|
2018-08-15 18:17:17 +00:00
|
|
|
}
|
|
|
|
}
|
2019-04-19 09:09:06 +00:00
|
|
|
|
2019-05-11 18:36:10 +00:00
|
|
|
} // namespace waybar::modules::sway
|