#include "components/builder.hpp" #include "components/config.hpp" #include "components/logger.hpp" #include "events/signal.hpp" #include "events/signal_emitter.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_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(signals::eventqueue::check_state{}); } } 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("%s: Rebuilding cache", 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(signals::eventqueue::notify_change{}); } 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 prevtag{true}; auto mingap = std::max(1UL, 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) { m_builder->node(value.substr(0, start)); value.erase(0, start); end -= start; start = 0; } string tag{value.substr(start, end + 1)}; if (tag[0] == '<' && tag[tag.size() - 1] == '>') { if (i > 0) m_builder->space(format->spacing); if (!(prevtag = CONST_MOD(Impl).build(m_builder.get(), tag)) && i > 0) m_builder->remove_trailing_space(mingap); if (prevtag) i++; } value.erase(0, tag.size()); } if (!value.empty()) { m_builder->append(value); } return format->decorate(&*m_builder, m_builder->flush()); } // }}} } POLYBAR_NS_END