mirror of https://github.com/polybar/polybar.git
162 lines
4.6 KiB
C++
162 lines
4.6 KiB
C++
#include "modules/script.hpp"
|
|
|
|
#include "drawtypes/label.hpp"
|
|
#include "modules/meta/base.inl"
|
|
|
|
POLYBAR_NS
|
|
|
|
namespace modules {
|
|
script_module::script_module(const bar_settings& bar, string name_)
|
|
: module<script_module>(bar, move(name_))
|
|
, m_tail(m_conf.get(name(), "tail", false))
|
|
, m_interval_success(m_conf.get<script_runner::interval>(name(), "interval", m_tail ? 0s : 5s))
|
|
, m_interval_fail(m_conf.get<script_runner::interval>(name(), "interval-fail", m_interval_success))
|
|
, m_interval_if(m_conf.get<script_runner::interval>(name(), "interval-if", m_interval_success))
|
|
, m_runner([this](const auto& data) { handle_runner_update(data); }, m_conf.get(name(), "exec", ""s),
|
|
m_conf.get(name(), "exec-if", ""s), m_tail, m_interval_success, m_interval_fail,
|
|
m_conf.get_with_prefix(name(), "env-")) {
|
|
// Load configured click handlers
|
|
m_actions[mousebtn::LEFT] = m_conf.get(name(), "click-left", ""s);
|
|
m_actions[mousebtn::MIDDLE] = m_conf.get(name(), "click-middle", ""s);
|
|
m_actions[mousebtn::RIGHT] = m_conf.get(name(), "click-right", ""s);
|
|
m_actions[mousebtn::DOUBLE_LEFT] = m_conf.get(name(), "double-click-left", ""s);
|
|
m_actions[mousebtn::DOUBLE_MIDDLE] = m_conf.get(name(), "double-click-middle", ""s);
|
|
m_actions[mousebtn::DOUBLE_RIGHT] = m_conf.get(name(), "double-click-right", ""s);
|
|
m_actions[mousebtn::SCROLL_UP] = m_conf.get(name(), "scroll-up", ""s);
|
|
m_actions[mousebtn::SCROLL_DOWN] = m_conf.get(name(), "scroll-down", ""s);
|
|
|
|
// Setup formatting
|
|
m_formatter->add(DEFAULT_FORMAT, TAG_LABEL, {TAG_LABEL});
|
|
if (m_formatter->has(TAG_LABEL)) {
|
|
m_label = load_optional_label(m_conf, name(), TAG_LABEL, "%output%");
|
|
}
|
|
|
|
m_formatter->add_optional(FORMAT_FAIL, {TAG_LABEL_FAIL});
|
|
if (m_formatter->has(TAG_LABEL_FAIL)) {
|
|
m_label_fail = load_optional_label(m_conf, name(), TAG_LABEL_FAIL, "%output%");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Start the module worker
|
|
*/
|
|
void script_module::start() {
|
|
this->module::start();
|
|
m_mainthread = thread([&] {
|
|
try {
|
|
while (running()) {
|
|
script_runner::interval sleep_time;
|
|
if (m_runner.check_condition()) {
|
|
sleep_time = m_runner.process();
|
|
} else {
|
|
m_runner.clear_output();
|
|
sleep_time = std::max(m_interval_if, script_runner::interval(1s));
|
|
}
|
|
|
|
if (m_runner.is_stopping()) {
|
|
break;
|
|
}
|
|
|
|
sleep(sleep_time);
|
|
}
|
|
} catch (const exception& err) {
|
|
halt(err.what());
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Stop the module worker by terminating any running commands
|
|
*/
|
|
void script_module::stop() {
|
|
m_runner.stop();
|
|
wakeup();
|
|
|
|
module::stop();
|
|
}
|
|
|
|
/**
|
|
* Generate module output
|
|
*/
|
|
string script_module::get_format() const {
|
|
if (m_exit_status != 0 && m_conf.has(name(), FORMAT_FAIL)) {
|
|
return FORMAT_FAIL;
|
|
}
|
|
return DEFAULT_FORMAT;
|
|
}
|
|
|
|
string script_module::get_output() {
|
|
auto data = [this] {
|
|
std::lock_guard<std::mutex> lk(m_data_mutex);
|
|
return m_data;
|
|
}();
|
|
|
|
m_exit_status = data.exit_status;
|
|
|
|
if (data.output.empty()) {
|
|
return "";
|
|
}
|
|
|
|
if (m_label) {
|
|
m_label->reset_tokens();
|
|
m_label->replace_token("%output%", data.output);
|
|
}
|
|
|
|
if (m_label_fail) {
|
|
m_label_fail->reset_tokens();
|
|
m_label_fail->replace_token("%output%", data.output);
|
|
}
|
|
|
|
string cnt{to_string(data.counter)};
|
|
string output{module::get_output()};
|
|
|
|
for (const auto& a : m_actions) {
|
|
auto btn = a.first;
|
|
auto action = a.second;
|
|
|
|
if (!action.empty()) {
|
|
auto action_replaced = string_util::replace_all(action, "%counter%", cnt);
|
|
|
|
/*
|
|
* The pid token is only for tailed commands.
|
|
* If the command is not specified or running, replacement is unnecessary as well
|
|
*/
|
|
if (data.pid != -1) {
|
|
action_replaced = string_util::replace_all(action_replaced, "%pid%", to_string(data.pid));
|
|
}
|
|
m_builder->action(btn, action_replaced);
|
|
}
|
|
}
|
|
|
|
m_builder->node(output);
|
|
|
|
return m_builder->flush();
|
|
}
|
|
|
|
/**
|
|
* Output format tags
|
|
*/
|
|
bool script_module::build(builder* builder, const string& tag) const {
|
|
if (tag == TAG_LABEL) {
|
|
builder->node(m_label);
|
|
} else if (tag == TAG_LABEL_FAIL) {
|
|
builder->node(m_label_fail);
|
|
} else {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void script_module::handle_runner_update(const script_runner::data& data) {
|
|
{
|
|
std::lock_guard<std::mutex> lk(m_data_mutex);
|
|
m_data = data;
|
|
}
|
|
|
|
broadcast();
|
|
}
|
|
} // namespace modules
|
|
|
|
POLYBAR_NS_END
|