1
0
Fork 0
mirror of https://github.com/polybar/polybar.git synced 2024-11-11 13:50:56 -05:00
polybar/src/components/builder.cpp

578 lines
11 KiB
C++
Raw Normal View History

#include "components/builder.hpp"
2016-11-25 07:55:15 -05:00
#include <utility>
#include "drawtypes/label.hpp"
#include "utils/actions.hpp"
2016-12-26 11:06:28 -05:00
#include "utils/color.hpp"
2016-11-02 15:22:45 -04:00
#include "utils/string.hpp"
2016-12-26 11:06:28 -05:00
#include "utils/time.hpp"
2016-11-19 00:22:44 -05:00
POLYBAR_NS
2016-11-02 15:22:45 -04:00
using namespace tags;
2016-12-30 22:32:11 -05:00
builder::builder(const bar_settings& bar) : m_bar(bar) {
reset();
}
void builder::reset() {
/* Add all values as keys so that we never have to check if a key exists in
* the map
*/
m_tags.clear();
2016-12-26 22:03:46 -05:00
m_tags[syntaxtag::A] = 0;
m_tags[syntaxtag::B] = 0;
m_tags[syntaxtag::F] = 0;
m_tags[syntaxtag::T] = 0;
m_tags[syntaxtag::R] = 0;
2016-12-26 22:03:46 -05:00
m_tags[syntaxtag::o] = 0;
m_tags[syntaxtag::u] = 0;
m_tags[syntaxtag::P] = 0;
2016-12-26 11:06:28 -05:00
m_colors.clear();
2016-12-26 11:06:28 -05:00
m_colors[syntaxtag::B] = string();
m_colors[syntaxtag::F] = string();
m_colors[syntaxtag::o] = string();
m_colors[syntaxtag::u] = string();
m_attrs.clear();
m_attrs[attribute::NONE] = false;
m_attrs[attribute::UNDERLINE] = false;
m_attrs[attribute::OVERLINE] = false;
m_output.clear();
m_fontindex = 1;
2016-12-26 04:37:14 -05:00
}
/**
* Flush contents of the builder and return built string
*
* This will also close any unclosed tags
*/
2016-11-02 15:22:45 -04:00
string builder::flush() {
2016-11-25 07:55:15 -05:00
if (m_tags[syntaxtag::B]) {
background_close();
2016-11-25 07:55:15 -05:00
}
if (m_tags[syntaxtag::F]) {
color_close();
2016-11-25 07:55:15 -05:00
}
if (m_tags[syntaxtag::T]) {
font_close();
2016-11-25 07:55:15 -05:00
}
if (m_tags[syntaxtag::o]) {
overline_color_close();
2016-11-25 07:55:15 -05:00
}
if (m_tags[syntaxtag::u]) {
underline_color_close();
2016-11-25 07:55:15 -05:00
}
if (m_attrs[attribute::UNDERLINE]) {
underline_close();
2016-11-25 07:55:15 -05:00
}
if (m_attrs[attribute::OVERLINE]) {
overline_close();
2016-11-25 07:55:15 -05:00
}
2016-11-02 15:22:45 -04:00
2016-11-24 22:10:26 -05:00
while (m_tags[syntaxtag::A]) {
action_close();
2016-11-24 22:10:26 -05:00
}
2016-12-26 22:03:46 -05:00
string output{m_output};
2016-11-02 15:22:45 -04:00
reset();
2016-11-02 15:22:45 -04:00
return output;
2016-11-02 15:22:45 -04:00
}
/**
* Insert raw text string
*/
2016-12-26 11:06:28 -05:00
void builder::append(string text) {
m_output.reserve(text.size());
m_output += move(text);
2016-11-02 15:22:45 -04:00
}
/**
* Insert text node
*
* This will also parse raw syntax tags
*/
void builder::node(string str) {
if (str.empty()) {
return;
}
append(move(str));
2016-11-02 15:22:45 -04:00
}
/**
* Insert text node with specific font index
*
* \see builder::node
*/
void builder::node(string str, int font_index) {
2016-11-02 15:22:45 -04:00
font(font_index);
node(move(str));
2016-11-02 15:22:45 -04:00
font_close();
}
/**
* Insert tags for given label
*/
void builder::node(const label_t& label) {
2016-11-25 07:55:15 -05:00
if (!label || !*label) {
2016-11-02 15:22:45 -04:00
return;
2016-11-25 07:55:15 -05:00
}
2016-11-02 15:22:45 -04:00
auto text = label->get();
2016-11-02 15:22:45 -04:00
if (label->m_margin.left > 0) {
space(label->m_margin.left);
2016-11-25 07:55:15 -05:00
}
2016-11-02 15:22:45 -04:00
if (label->m_overline.has_color()) {
2016-11-02 15:22:45 -04:00
overline(label->m_overline);
2016-11-25 07:55:15 -05:00
}
if (label->m_underline.has_color()) {
2016-11-02 15:22:45 -04:00
underline(label->m_underline);
2016-11-25 07:55:15 -05:00
}
2016-11-02 15:22:45 -04:00
if (label->m_background.has_color()) {
background(label->m_background);
2016-11-25 07:55:15 -05:00
}
if (label->m_foreground.has_color()) {
color(label->m_foreground);
2016-11-25 07:55:15 -05:00
}
2016-11-02 15:22:45 -04:00
if (label->m_padding.left > 0) {
space(label->m_padding.left);
2016-11-25 07:55:15 -05:00
}
2016-11-02 15:22:45 -04:00
node(text, label->m_font);
2016-11-02 15:22:45 -04:00
if (label->m_padding.right > 0) {
space(label->m_padding.right);
2016-11-25 07:55:15 -05:00
}
2016-11-02 15:22:45 -04:00
if (label->m_background.has_color()) {
background_close();
2016-11-25 07:55:15 -05:00
}
if (label->m_foreground.has_color()) {
color_close();
2016-11-25 07:55:15 -05:00
}
2016-11-02 15:22:45 -04:00
if (label->m_underline.has_color()) {
underline_close();
2016-11-25 07:55:15 -05:00
}
if (label->m_overline.has_color()) {
overline_close();
2016-11-25 07:55:15 -05:00
}
2016-11-02 15:22:45 -04:00
if (label->m_margin.right > 0) {
space(label->m_margin.right);
2016-11-25 07:55:15 -05:00
}
2016-11-02 15:22:45 -04:00
}
/**
* Repeat text string n times
*/
void builder::node_repeat(const string& str, size_t n) {
string text;
2016-12-26 11:06:28 -05:00
text.reserve(str.size() * n);
while (n--) {
text += str;
}
node(text);
}
/**
* Repeat label contents n times
*/
void builder::node_repeat(const label_t& label, size_t n) {
string text;
2016-12-26 11:06:28 -05:00
string label_text{label->get()};
text.reserve(label_text.size() * n);
while (n--) {
2016-12-26 11:06:28 -05:00
text += label_text;
}
label_t tmp{new label_t::element_type{text}};
tmp->replace_defined_values(label);
node(tmp);
}
/**
* Insert tag that will offset the contents by given pixels
*/
2016-11-02 15:22:45 -04:00
void builder::offset(int pixels) {
2016-11-25 07:55:15 -05:00
if (pixels == 0) {
return;
2016-11-25 07:55:15 -05:00
}
tag_open(syntaxtag::O, to_string(pixels));
2016-11-02 15:22:45 -04:00
}
/**
* Insert spaces
*/
2016-12-26 11:06:28 -05:00
void builder::space(size_t width) {
if (width) {
m_output.append(width, ' ');
} else {
space();
}
2016-12-26 11:06:28 -05:00
}
void builder::space() {
m_output.append(m_bar.spacing, ' ');
2016-11-02 15:22:45 -04:00
}
/**
* Remove trailing space
*/
2016-12-26 11:06:28 -05:00
void builder::remove_trailing_space(size_t len) {
2016-12-30 22:32:11 -05:00
if (len == 0_z || len > m_output.size()) {
2016-11-02 15:22:45 -04:00
return;
} else if (m_output.substr(m_output.size() - len) == string(len, ' ')) {
2016-12-26 11:06:28 -05:00
m_output.erase(m_output.size() - len);
2016-11-25 07:55:15 -05:00
}
2016-11-02 15:22:45 -04:00
}
2016-12-26 11:06:28 -05:00
void builder::remove_trailing_space() {
remove_trailing_space(m_bar.spacing);
}
2016-11-02 15:22:45 -04:00
/**
* Insert tag to alter the current font index
*/
2016-11-02 15:22:45 -04:00
void builder::font(int index) {
if (index == 0) {
return;
}
2016-11-02 15:22:45 -04:00
m_fontindex = index;
tag_open(syntaxtag::T, to_string(index));
2016-11-02 15:22:45 -04:00
}
/**
* Insert tag to reset the font index
*/
void builder::font_close() {
2016-11-02 15:22:45 -04:00
m_fontindex = 1;
tag_close(syntaxtag::T);
2016-11-02 15:22:45 -04:00
}
/**
* Insert tag to alter the current background color
*/
void builder::background(rgba color) {
color = color.try_apply_alpha_to(m_bar.background);
2016-11-02 15:22:45 -04:00
auto hex = color_util::simplify_hex(color);
m_colors[syntaxtag::B] = hex;
tag_open(syntaxtag::B, hex);
2016-11-02 15:22:45 -04:00
}
/**
* Insert tag to reset the background color
*/
void builder::background_close() {
2016-12-26 11:06:28 -05:00
m_colors[syntaxtag::B].clear();
tag_close(syntaxtag::B);
2016-11-02 15:22:45 -04:00
}
/**
* Insert tag to alter the current foreground color
*/
void builder::color(rgba color) {
color = color.try_apply_alpha_to(m_bar.foreground);
2016-11-02 15:22:45 -04:00
auto hex = color_util::simplify_hex(color);
m_colors[syntaxtag::F] = hex;
tag_open(syntaxtag::F, hex);
2016-11-02 15:22:45 -04:00
}
/**
* Insert tag to reset the foreground color
*/
void builder::color_close() {
2016-12-26 11:06:28 -05:00
m_colors[syntaxtag::F].clear();
tag_close(syntaxtag::F);
2016-11-02 15:22:45 -04:00
}
/**
* Insert tag to alter the current overline/underline color
*/
void builder::line_color(const rgba& color) {
overline_color(color);
underline_color(color);
2016-11-02 15:22:45 -04:00
}
/**
* Close overline/underline color tag
*/
void builder::line_color_close() {
overline_color_close();
underline_color_close();
2016-11-02 15:22:45 -04:00
}
/**
* Insert tag to alter the current overline color
*/
void builder::overline_color(rgba color) {
auto hex = color_util::simplify_hex(color);
m_colors[syntaxtag::o] = hex;
tag_open(syntaxtag::o, hex);
tag_open(attribute::OVERLINE);
}
/**
* Close underline color tag
*/
void builder::overline_color_close() {
2016-12-26 11:06:28 -05:00
m_colors[syntaxtag::o].clear();
tag_close(syntaxtag::o);
}
/**
* Insert tag to alter the current underline color
*/
void builder::underline_color(rgba color) {
auto hex = color_util::simplify_hex(color);
m_colors[syntaxtag::u] = hex;
tag_open(syntaxtag::u, hex);
tag_open(attribute::UNDERLINE);
}
/**
* Close underline color tag
*/
void builder::underline_color_close() {
tag_close(syntaxtag::u);
2016-12-26 11:06:28 -05:00
m_colors[syntaxtag::u].clear();
}
/**
* Insert tag to enable the overline attribute
*/
void builder::overline(const rgba& color) {
if (color.has_color()) {
overline_color(color);
2016-11-25 07:55:15 -05:00
} else {
tag_open(attribute::OVERLINE);
2016-11-25 07:55:15 -05:00
}
2016-11-02 15:22:45 -04:00
}
/**
* Close overline attribute tag
*/
void builder::overline_close() {
tag_close(attribute::OVERLINE);
2016-11-02 15:22:45 -04:00
}
/**
* Insert tag to enable the underline attribute
*/
void builder::underline(const rgba& color) {
if (color.has_color()) {
underline_color(color);
2016-11-25 07:55:15 -05:00
} else {
tag_open(attribute::UNDERLINE);
2016-11-25 07:55:15 -05:00
}
2016-11-02 15:22:45 -04:00
}
/**
* Close underline attribute tag
*/
void builder::underline_close() {
tag_close(attribute::UNDERLINE);
2016-11-02 15:22:45 -04:00
}
/**
* Add a polybar control tag
*/
void builder::control(controltag tag) {
string str;
switch (tag) {
case controltag::R:
str = "R";
break;
default:
break;
}
2019-08-06 13:35:07 -04:00
if (!str.empty()) {
tag_open(syntaxtag::P, str);
}
}
/**
* Open action tag with the given action string
*
* The action string is escaped, if needed.
*/
void builder::action(mousebtn index, string action) {
if (!action.empty()) {
action = string_util::replace_all(action, ":", "\\:");
tag_open(syntaxtag::A, to_string(static_cast<int>(index)) + ":" + action + ":");
2016-11-25 07:55:15 -05:00
}
}
2016-11-02 15:22:45 -04:00
/**
* Open action tag for the action of the given module
*/
void builder::action(mousebtn btn, const modules::module_interface& module, string action_name, string data) {
action(btn, actions_util::get_action_string(module, action_name, data));
}
/**
* Wrap label in action tag
*/
void builder::action(mousebtn index, string action_name, const label_t& label) {
if (label && *label) {
action(index, action_name);
node(label);
tag_close(syntaxtag::A);
}
2016-11-02 15:22:45 -04:00
}
/**
* Wrap label in module action tag
*/
void builder::action(
mousebtn btn, const modules::module_interface& module, string action_name, string data, const label_t& label) {
action(btn, actions_util::get_action_string(module, action_name, data), label);
}
/**
* Close command tag
*/
void builder::action_close() {
tag_close(syntaxtag::A);
2016-11-02 15:22:45 -04:00
}
/**
* Insert directive to change value of given tag
*/
2016-11-25 07:55:15 -05:00
void builder::tag_open(syntaxtag tag, const string& value) {
2016-11-24 22:10:26 -05:00
m_tags[tag]++;
switch (tag) {
case syntaxtag::A:
append("%{A" + value + "}");
break;
case syntaxtag::F:
append("%{F" + value + "}");
break;
case syntaxtag::B:
append("%{B" + value + "}");
break;
case syntaxtag::T:
append("%{T" + value + "}");
break;
case syntaxtag::u:
append("%{u" + value + "}");
break;
case syntaxtag::o:
append("%{o" + value + "}");
break;
case syntaxtag::R:
append("%{R}");
break;
case syntaxtag::O:
append("%{O" + value + "}");
break;
case syntaxtag::P:
append("%{P" + value + "}");
break;
case syntaxtag::l:
append("%{l}");
break;
case syntaxtag::c:
append("%{c}");
break;
case syntaxtag::r:
append("%{r}");
break;
}
}
/**
* Insert directive to use given attribute unless already set
*/
void builder::tag_open(attribute attr) {
if (m_attrs[attr]) {
return;
2016-11-25 07:55:15 -05:00
}
m_attrs[attr] = true;
switch (attr) {
case attribute::NONE:
break;
case attribute::UNDERLINE:
append("%{+u}");
break;
case attribute::OVERLINE:
append("%{+o}");
break;
}
}
/**
* Insert directive to reset given tag if it's open and closable
*/
void builder::tag_close(syntaxtag tag) {
if (!m_tags[tag]) {
return;
2016-11-25 07:55:15 -05:00
}
m_tags[tag]--;
switch (tag) {
case syntaxtag::A:
append("%{A}");
break;
case syntaxtag::F:
append("%{F-}");
break;
case syntaxtag::B:
append("%{B-}");
break;
case syntaxtag::T:
append("%{T-}");
break;
case syntaxtag::u:
append("%{u-}");
break;
case syntaxtag::o:
append("%{o-}");
break;
case syntaxtag::R:
case syntaxtag::P:
case syntaxtag::O:
case syntaxtag::l:
case syntaxtag::c:
case syntaxtag::r:
break;
}
}
/**
* Insert directive to remove given attribute if set
*/
void builder::tag_close(attribute attr) {
if (!m_attrs[attr]) {
return;
2016-11-25 07:55:15 -05:00
}
m_attrs[attr] = false;
switch (attr) {
case attribute::NONE:
break;
case attribute::UNDERLINE:
append("%{-u}");
break;
case attribute::OVERLINE:
append("%{-o}");
break;
}
2016-11-02 15:22:45 -04:00
}
2016-11-19 00:22:44 -05:00
POLYBAR_NS_END