#include "modules/xbacklight.hpp" #include "utils/math.hpp" #include "x11/graphics.hpp" LEMONBUDDY_NS namespace modules { /** * Bootstrap the module by grabbing all required components */ void xbacklight_module::setup() { auto output = m_conf.get(name(), "output", m_bar.monitor->name); auto strict = m_conf.get(name(), "monitor-strict", false); // Grab a list of all outputs and try to find the one defined in the config for (auto&& mon : randr_util::get_monitors(m_connection, m_connection.root(), strict)) { if (mon->match(output, strict)) { m_output.swap(mon); break; } } // If we didn't get a match we stop the module if (!m_output) { throw module_error("No matching output found for \"" + output + "\", stopping module..."); } // Get flag to check if we should add scroll handlers for changing value GET_CONFIG_VALUE(name(), m_scroll, "enable-scroll"); // Query randr for the backlight max and min value try { auto& backlight = m_output->backlight; randr_util::get_backlight_range(m_connection, m_output, backlight); randr_util::get_backlight_value(m_connection, m_output, backlight); } catch (const exception& err) { throw module_error("No backlight data found for \"" + output + "\", stopping module..."); } // Create window that will proxy all RandR notify events if (!graphics_util::create_window(m_connection, &m_proxy, -1, -1, 1, 1)) { throw module_error("Failed to create event proxy"); } // Connect with the event registry and make sure we get // notified when a RandR output property gets modified m_connection.attach_sink(this, 1); m_connection.select_input_checked(m_proxy, XCB_RANDR_NOTIFY_MASK_OUTPUT_PROPERTY); // Add formats and elements m_formatter->add(DEFAULT_FORMAT, TAG_LABEL, {TAG_LABEL, TAG_BAR, TAG_RAMP}); if (m_formatter->has(TAG_LABEL)) m_label = load_optional_label(m_conf, name(), TAG_LABEL, "%percentage%"); if (m_formatter->has(TAG_BAR)) m_progressbar = load_progressbar(m_bar, m_conf, name(), TAG_BAR); if (m_formatter->has(TAG_RAMP)) m_ramp = load_ramp(m_conf, name(), TAG_RAMP); // Trigger the initial draw event update(); } /** * Handler for XCB_RANDR_NOTIFY events */ void xbacklight_module::handle(const evt::randr_notify& evt) { if (evt->subCode != XCB_RANDR_NOTIFY_OUTPUT_PROPERTY) return; else if (evt->u.op.status != XCB_PROPERTY_NEW_VALUE) return; else if (evt->u.op.window != m_proxy) return; else if (evt->u.op.output != m_output->output) return; else if (evt->u.op.atom != m_output->backlight.atom) return; else if (evt->u.op.timestamp <= m_timestamp) return; // Store the timestamp with a throttle offset (ms) m_timestamp = evt->u.op.timestamp + 50; // Fetch the new values update(); } /** * Query the RandR extension for the new values */ void xbacklight_module::update() { // Query for the new backlight value auto& bl = m_output->backlight; randr_util::get_backlight_value(m_connection, m_output, bl); m_percentage = math_util::percentage(bl.val, bl.min, bl.max); // Update label tokens if (m_label) { m_label->reset_tokens(); m_label->replace_token("%percentage%", to_string(m_percentage) + "%"); } // Emit a broadcast notification so that // the new data will be drawn to the bar broadcast(); } /** * Generate the module output */ string xbacklight_module::get_output() { if (m_scroll) { if (m_percentage < 100) m_builder->cmd(mousebtn::SCROLL_UP, EVENT_SCROLLUP); if (m_percentage > 0) m_builder->cmd(mousebtn::SCROLL_DOWN, EVENT_SCROLLDOWN); m_builder->node(static_module::get_output()); if (m_percentage < 100) m_builder->cmd_close(true); if (m_percentage > 0) m_builder->cmd_close(true); } else { m_builder->node(static_module::get_output()); } return m_builder->flush(); } /** * Output content as defined in the config */ bool xbacklight_module::build(builder* builder, string tag) const { if (tag == TAG_BAR) builder->node(m_progressbar->output(m_percentage)); else if (tag == TAG_RAMP) builder->node(m_ramp->get_by_percentage(m_percentage)); else if (tag == TAG_LABEL) builder->node(m_label); else return false; return true; } /** * Process scroll events by changing backlight value */ bool xbacklight_module::handle_event(string cmd) { int value_mod = 0; if (cmd == EVENT_SCROLLUP) { value_mod = 10; m_log.info("%s: Increasing value by %i%", name(), value_mod); } else if (cmd == EVENT_SCROLLDOWN) { value_mod = -10; m_log.info("%s: Decreasing value by %i%", name(), -value_mod); } else { return false; } try { const int new_perc = math_util::cap(m_percentage + value_mod, 0, 100); const int new_value = math_util::percentage_to_value(new_perc, m_output->backlight.max); const int values[1]{new_value}; m_connection.change_output_property_checked( m_output->output, m_output->backlight.atom, XCB_ATOM_INTEGER, 32, XCB_PROP_MODE_REPLACE, 1, values); } catch (const exception& err) { m_log.err("%s: %s", name(), err.what()); } return true; } } LEMONBUDDY_NS_END