#include #include "components/builder.hpp" #include "components/config.hpp" #include "components/logger.hpp" #include "events/signal.hpp" #include "events/signal_emitter.hpp" #include "modules/meta/base.hpp" #include "utils/action_router.hpp" POLYBAR_NS namespace modules { // module public {{{ template module::module(const bar_settings bar, string name) : m_sig(signal_emitter::make()) , m_bar(bar) , m_log(logger::make()) , m_conf(config::make()) , m_router(make_unique()) , m_name("module/" + name) , m_name_raw(name) , m_builder(make_unique(bar)) , m_formatter(make_unique(m_conf, m_name)) , m_handle_events(m_conf.get(m_name, "handle-events", true)) , m_visible(!m_conf.get(m_name, "hidden", false)) { m_router->register_action(EVENT_MODULE_TOGGLE, [this]() { action_module_toggle(); }); m_router->register_action(EVENT_MODULE_SHOW, [this]() { action_module_show(); }); m_router->register_action(EVENT_MODULE_HIDE, [this]() { action_module_hide(); }); } template module::~module() noexcept { m_log.trace("%s: Deconstructing", name()); if (running()) { /* * We can't stop in the destructor because we have to call the subclasses which at this point already have been * destructed. */ m_log.err("%s: Module was not stopped before deconstructing.", name()); } } template string module::name() const { return m_name; } template string module::name_raw() const { return m_name_raw; } template string module::type() const { return string(Impl::TYPE); } template bool module::running() const { return static_cast(m_enabled); } template void module::join() { for (auto&& thread_ : m_threads) { if (thread_.joinable()) { thread_.join(); } } if (m_mainthread.joinable()) { m_mainthread.join(); } } template bool module::visible() const { return static_cast(m_visible); } template void module::stop() { if (!static_cast(m_enabled)) { return; } m_log.info("%s: Stopping", name()); m_enabled = false; std::lock(m_buildlock, m_updatelock); std::lock_guard guard_a(m_buildlock, std::adopt_lock); std::lock_guard guard_b(m_updatelock, std::adopt_lock); { CAST_MOD(Impl)->wakeup(); CAST_MOD(Impl)->teardown(); m_sig.emit(signals::eventqueue::check_state{}); } } template void module::halt(string error_message) { m_log.err("%s: %s", name(), error_message); m_log.notice("Stopping '%s'...", name()); stop(); } template void module::teardown() {} template string module::contents() { if (m_changed) { m_log.info("%s: Rebuilding cache", name()); m_cache = CAST_MOD(Impl)->get_output(); // Make sure builder is really empty m_builder->flush(); if (!m_cache.empty()) { // Add a reset tag after the module m_builder->control(tags::controltag::R); m_cache += m_builder->flush(); } m_changed = false; } return m_cache; } template bool module::input(const string& name, const string& data) { if (!m_router->has_action(name)) { return false; } try { m_router->invoke(name, data); } catch (const exception& err) { m_log.err("%s: Failed to handle command '%s' with data '%s' (%s)", this->name(), name, data, err.what()); } return true; } // }}} // module protected {{{ template void module::broadcast() { m_changed = true; m_sig.emit(signals::eventqueue::notify_change{}); } template void module::idle() { if (running()) { CAST_MOD(Impl)->sleep(25ms); } } template void module::sleep(chrono::duration sleep_duration) { if (running()) { std::unique_lock lck(m_sleeplock); m_sleephandler.wait_for(lck, sleep_duration); } } template template void module::sleep_until(chrono::time_point point) { if (running()) { std::unique_lock lck(m_sleeplock); m_sleephandler.wait_until(lck, point); } } template void module::wakeup() { m_log.trace("%s: Release sleep lock", name()); m_sleephandler.notify_all(); } template string module::get_format() const { return DEFAULT_FORMAT; } template string module::get_output() { std::lock_guard guard(m_buildlock); auto format_name = CONST_MOD(Impl).get_format(); auto format = m_formatter->get(format_name); bool no_tag_built{true}; bool fake_no_tag_built{false}; bool tag_built{false}; auto mingap = std::max(1_z, format->spacing); size_t start, end; string value{format->value}; while ((start = value.find('<')) != string::npos && (end = value.find('>', start)) != string::npos) { if (start > 0) { if (no_tag_built) { // If no module tag has been built we do not want to add // whitespace defined between the format tags, but we do still // want to output other non-tag content auto trimmed = string_util::ltrim(value.substr(0, start), ' '); if (!trimmed.empty()) { fake_no_tag_built = false; m_builder->node(move(trimmed)); } } else { m_builder->node(value.substr(0, start)); } value.erase(0, start); end -= start; start = 0; } string tag{value.substr(start, end + 1)}; if (tag.empty()) { continue; } else if (tag[0] == '<' && tag[tag.size() - 1] == '>') { if (!no_tag_built) m_builder->space(format->spacing); else if (fake_no_tag_built) no_tag_built = false; if (!(tag_built = CONST_MOD(Impl).build(m_builder.get(), tag)) && !no_tag_built) m_builder->remove_trailing_space(mingap); if (tag_built) no_tag_built = false; } value.erase(0, tag.size()); } if (!value.empty()) { m_builder->append(value); } return format->decorate(&*m_builder, m_builder->flush()); } template void module::set_visible(bool value) { m_log.notice("%s: Visibility changed (state=%s)", m_name, value ? "shown" : "hidden"); m_visible = value; broadcast(); } template void module::action_module_toggle() { set_visible(!m_visible); } template void module::action_module_show() { set_visible(true); } template void module::action_module_hide() { set_visible(false); } // }}} } // namespace modules POLYBAR_NS_END