#include "components/builder.hpp" #include "components/logger.hpp" #include "components/config.hpp" #include "events/signal.hpp" #include "events/signal_emitter.hpp" POLYBAR_NS namespace sig_ev = signals::eventqueue; 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_name("module/" + name) , m_builder(make_unique(bar)) , m_formatter(make_unique(m_conf, m_name)) {} template module::~module() noexcept { m_log.trace("%s: Deconstructing", name()); for (auto&& thread_ : m_threads) { if (thread_.joinable()) { thread_.join(); } } if (m_mainthread.joinable()) { m_mainthread.join(); } } template string module::name() const { return m_name; } template bool module::running() const { return static_cast(m_enabled); } 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(sig_ev::process_check{}); } } template void module::halt(string error_message) { m_log.err("%s: %s", name(), error_message); m_log.warn("Stopping '%s'...", name()); stop(); } template void module::teardown() {} template string module::contents() { if (m_changed) { m_log.info("Rebuilding cache for '%s'...", name()); m_cache = CAST_MOD(Impl)->get_output(); m_changed = false; } return m_cache; } // }}} // module protected {{{ template void module::broadcast() { m_changed = true; m_sig.emit(sig_ev::process_broadcast{}); } template void module::idle() { CAST_MOD(Impl)->sleep(25ms); } template void module::sleep(chrono::duration sleep_duration) { std::unique_lock lck(m_sleeplock); m_sleephandler.wait_for(lck, sleep_duration); } 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); int i = 0; bool tag_built = true; for (auto&& tag : string_util::split(format->value, ' ')) { bool is_blankspace = tag.empty(); if (tag[0] == '<' && tag[tag.length() - 1] == '>') { if (i > 0) m_builder->space(format->spacing); if (!(tag_built = CONST_MOD(Impl).build(m_builder.get(), tag)) && i > 0) m_builder->remove_trailing_space(format->spacing); if (tag_built) i++; } else if (is_blankspace && tag_built) { m_builder->node(" "); } else if (!is_blankspace) { m_builder->node(tag); } } return format->decorate(m_builder.get(), m_builder->flush()); } // }}} } POLYBAR_NS_END