Cleanup tray position handling
This commit is contained in:
parent
3244b10ce3
commit
ba0e478026
7 changed files with 116 additions and 91 deletions
|
@ -101,9 +101,6 @@ namespace signals {
|
|||
} // namespace ui
|
||||
|
||||
namespace ui_tray {
|
||||
struct mapped_clients : public detail::value_signal<mapped_clients, unsigned int> {
|
||||
using base_type::base_type;
|
||||
};
|
||||
struct tray_width_change : public detail::value_signal<tray_width_change, unsigned int> {
|
||||
using base_type::base_type;
|
||||
};
|
||||
|
|
|
@ -35,7 +35,6 @@ namespace signals {
|
|||
struct update_geometry;
|
||||
} // namespace ui
|
||||
namespace ui_tray {
|
||||
struct mapped_clients;
|
||||
struct tray_width_change;
|
||||
struct tray_pos_change;
|
||||
}
|
||||
|
|
|
@ -13,7 +13,7 @@ class connection;
|
|||
|
||||
class tray_client {
|
||||
public:
|
||||
explicit tray_client(connection& conn, xcb_window_t win, unsigned int w, unsigned int h);
|
||||
explicit tray_client(connection& conn, xcb_window_t win, size s);
|
||||
tray_client(const tray_client& c) = delete;
|
||||
tray_client& operator=(tray_client& c) = delete;
|
||||
|
||||
|
@ -55,8 +55,7 @@ class tray_client {
|
|||
|
||||
bool m_mapped{false};
|
||||
|
||||
unsigned int m_width;
|
||||
unsigned int m_height;
|
||||
size m_size;
|
||||
};
|
||||
|
||||
POLYBAR_NS_END
|
||||
|
|
|
@ -43,14 +43,41 @@ struct tray_settings {
|
|||
alignment align{alignment::NONE};
|
||||
bool running{false};
|
||||
|
||||
int orig_x{0};
|
||||
int orig_y{0};
|
||||
unsigned int configured_w{0U};
|
||||
unsigned int configured_slots{0U};
|
||||
unsigned int width{0U};
|
||||
/**
|
||||
* Tray window position.
|
||||
*
|
||||
* Relative to the bar window
|
||||
* TODO make relative to inner area
|
||||
*/
|
||||
position pos{0, 0};
|
||||
|
||||
/**
|
||||
* Tray offset in pixels.
|
||||
*/
|
||||
position offset{0, 0};
|
||||
|
||||
/**
|
||||
* Current dimensions of the tray window.
|
||||
*/
|
||||
size win_size{0, 0};
|
||||
|
||||
/**
|
||||
* Dimensions for client windows.
|
||||
*/
|
||||
size client_size{0, 0};
|
||||
|
||||
/**
|
||||
* Number of clients currently mapped.
|
||||
*/
|
||||
int num_clients{0};
|
||||
|
||||
// This is the width of the bar window
|
||||
// TODO directly read from bar_settings
|
||||
unsigned int width_max{0U};
|
||||
unsigned int height{0U};
|
||||
unsigned int height_fill{0U};
|
||||
|
||||
/**
|
||||
* Number of pixels added between tray icons
|
||||
*/
|
||||
unsigned int spacing{0U};
|
||||
rgba background{};
|
||||
rgba foreground{};
|
||||
|
@ -116,7 +143,8 @@ class tray_manager : public xpp::event::sink<evt::expose, evt::visibility_notify
|
|||
shared_ptr<tray_client> find_client(const xcb_window_t& win) const;
|
||||
void remove_client(shared_ptr<tray_client>& client, bool reconfigure = true);
|
||||
void remove_client(xcb_window_t win, bool reconfigure = true);
|
||||
unsigned int mapped_clients() const;
|
||||
int mapped_clients() const;
|
||||
bool has_mapped_clients() const;
|
||||
|
||||
void handle(const evt::expose& evt) override;
|
||||
void handle(const evt::visibility_notify& evt) override;
|
||||
|
|
|
@ -393,9 +393,9 @@ void bar::parse(string&& data, bool force) {
|
|||
|
||||
auto rect = m_opts.inner_area();
|
||||
|
||||
if (m_tray && !m_tray->settings().detached && m_tray->settings().configured_slots && !m_tray->settings().adaptive) {
|
||||
if (m_tray && !m_tray->settings().detached && m_tray->settings().num_clients > 0 && !m_tray->settings().adaptive) {
|
||||
auto trayalign = m_tray->settings().align;
|
||||
auto traywidth = m_tray->settings().configured_w;
|
||||
auto traywidth = m_tray->settings().win_size.w;
|
||||
if (trayalign == alignment::LEFT) {
|
||||
rect.x += traywidth;
|
||||
rect.width -= traywidth;
|
||||
|
|
|
@ -8,19 +8,18 @@
|
|||
|
||||
POLYBAR_NS
|
||||
|
||||
tray_client::tray_client(connection& conn, xcb_window_t win, unsigned int w, unsigned int h)
|
||||
: m_connection(conn), m_window(win), m_width(w), m_height(h) {}
|
||||
tray_client::tray_client(connection& conn, xcb_window_t win, size s) : m_connection(conn), m_window(win), m_size(s) {}
|
||||
|
||||
tray_client::~tray_client() {
|
||||
xembed::unembed(m_connection, window(), m_connection.root());
|
||||
}
|
||||
|
||||
unsigned int tray_client::width() const {
|
||||
return m_width;
|
||||
return m_size.w;
|
||||
}
|
||||
|
||||
unsigned int tray_client::height() const {
|
||||
return m_height;
|
||||
return m_size.h;
|
||||
}
|
||||
|
||||
void tray_client::clear_window() const {
|
||||
|
@ -92,8 +91,8 @@ void tray_client::reconfigure(int x, int y) const {
|
|||
unsigned int configure_values[7];
|
||||
xcb_params_configure_window_t configure_params{};
|
||||
|
||||
XCB_AUX_ADD_PARAM(&configure_mask, &configure_params, width, m_width);
|
||||
XCB_AUX_ADD_PARAM(&configure_mask, &configure_params, height, m_height);
|
||||
XCB_AUX_ADD_PARAM(&configure_mask, &configure_params, width, m_size.w);
|
||||
XCB_AUX_ADD_PARAM(&configure_mask, &configure_params, height, m_size.h);
|
||||
XCB_AUX_ADD_PARAM(&configure_mask, &configure_params, x, x);
|
||||
XCB_AUX_ADD_PARAM(&configure_mask, &configure_params, y, y);
|
||||
|
||||
|
@ -113,8 +112,8 @@ void tray_client::configure_notify(int x, int y) const {
|
|||
notify.above_sibling = 0;
|
||||
notify.x = x;
|
||||
notify.y = y;
|
||||
notify.width = m_width;
|
||||
notify.height = m_height;
|
||||
notify.width = m_size.w;
|
||||
notify.height = m_size.h;
|
||||
notify.border_width = 0;
|
||||
|
||||
unsigned int mask{XCB_EVENT_MASK_STRUCTURE_NOTIFY};
|
||||
|
|
|
@ -92,46 +92,40 @@ void tray_manager::setup() {
|
|||
return;
|
||||
}
|
||||
|
||||
m_opts.detached = conf.get(bs, "tray-detached", false);
|
||||
m_opts.height = m_bar_opts.size.h;
|
||||
m_opts.height -= m_bar_opts.borders.at(edge::BOTTOM).size;
|
||||
m_opts.height -= m_bar_opts.borders.at(edge::TOP).size;
|
||||
m_opts.height_fill = m_opts.height;
|
||||
auto inner_area = m_bar_opts.inner_area();
|
||||
m_opts.win_size.w = 0;
|
||||
m_opts.win_size.h = inner_area.height;
|
||||
|
||||
if (m_opts.height % 2 != 0) {
|
||||
m_opts.height--;
|
||||
}
|
||||
m_opts.detached = conf.get(bs, "tray-detached", false);
|
||||
unsigned int client_height = inner_area.height;
|
||||
|
||||
auto maxsize = conf.get<unsigned int>(bs, "tray-maxsize", 16);
|
||||
if (m_opts.height > maxsize) {
|
||||
m_opts.spacing += (m_opts.height - maxsize) / 2;
|
||||
m_opts.height = maxsize;
|
||||
if (client_height > maxsize) {
|
||||
m_opts.spacing += (client_height - maxsize) / 2;
|
||||
client_height = maxsize;
|
||||
}
|
||||
|
||||
m_opts.width_max = m_bar_opts.size.w;
|
||||
m_opts.width = m_opts.height;
|
||||
m_opts.orig_y = m_bar_opts.borders.at(edge::TOP).size;
|
||||
m_opts.client_size = {client_height, client_height};
|
||||
|
||||
// Apply user-defined scaling
|
||||
auto scale = conf.get(bs, "tray-scale", 1.0);
|
||||
m_opts.width *= scale;
|
||||
m_opts.height_fill *= scale;
|
||||
m_opts.client_size.w *= scale;
|
||||
m_opts.win_size.h *= scale;
|
||||
|
||||
auto inner_area = m_bar_opts.inner_area();
|
||||
|
||||
switch (m_opts.align) {
|
||||
case alignment::NONE:
|
||||
break;
|
||||
case alignment::LEFT:
|
||||
m_opts.orig_x = inner_area.x;
|
||||
break;
|
||||
case alignment::CENTER:
|
||||
m_opts.orig_x = inner_area.x + inner_area.width / 2 - m_opts.width / 2;
|
||||
break;
|
||||
case alignment::RIGHT:
|
||||
m_opts.orig_x = inner_area.x + inner_area.width;
|
||||
break;
|
||||
}
|
||||
m_opts.pos.x = inner_area.x + [&]() -> int {
|
||||
switch (m_opts.align) {
|
||||
case alignment::LEFT:
|
||||
return 0;
|
||||
case alignment::CENTER:
|
||||
return inner_area.width / 2 - m_opts.client_size.w / 2;
|
||||
case alignment::RIGHT:
|
||||
return inner_area.width;
|
||||
default:
|
||||
return 0;
|
||||
}
|
||||
}();
|
||||
m_opts.pos.y = inner_area.y;
|
||||
|
||||
if (conf.has(bs, "tray-transparent")) {
|
||||
m_log.warn("tray-transparent is deprecated, the tray always uses pseudo-transparency. Please remove it.");
|
||||
|
@ -164,10 +158,13 @@ void tray_manager::setup() {
|
|||
max_y = inner_area.height;
|
||||
}
|
||||
|
||||
m_opts.orig_x += units_utils::percentage_with_offset_to_pixel(offset_x, max_x, m_bar_opts.dpi_x);
|
||||
m_opts.orig_y += units_utils::percentage_with_offset_to_pixel(offset_y, max_y, m_bar_opts.dpi_y);
|
||||
m_opts.offset.x = units_utils::percentage_with_offset_to_pixel(offset_x, max_x, m_bar_opts.dpi_x);
|
||||
m_opts.offset.y = units_utils::percentage_with_offset_to_pixel(offset_y, max_y, m_bar_opts.dpi_y);
|
||||
|
||||
m_opts.bar_window = bar_opts.window;
|
||||
m_opts.pos.x += m_opts.offset.x;
|
||||
m_opts.pos.y += m_opts.offset.y;
|
||||
|
||||
m_opts.bar_window = m_bar_opts.window;
|
||||
|
||||
// Activate the tray manager
|
||||
query_atom();
|
||||
|
@ -264,8 +261,8 @@ void tray_manager::deactivate(bool clear_selection) {
|
|||
m_tray = 0;
|
||||
m_pixmap = 0;
|
||||
m_gc = 0;
|
||||
m_opts.configured_w = 0;
|
||||
m_opts.configured_slots = 0;
|
||||
m_opts.win_size.w = 0;
|
||||
m_opts.num_clients = 0;
|
||||
m_acquired_selection = false;
|
||||
m_mapped = false;
|
||||
|
||||
|
@ -299,7 +296,7 @@ void tray_manager::reconfigure() {
|
|||
m_log.err("Failed to reconfigure tray background (%s)", err.what());
|
||||
}
|
||||
|
||||
m_opts.configured_slots = mapped_clients();
|
||||
m_opts.num_clients = mapped_clients();
|
||||
guard.unlock();
|
||||
refresh_window();
|
||||
m_connection.flush();
|
||||
|
@ -318,17 +315,17 @@ void tray_manager::reconfigure_window() {
|
|||
return;
|
||||
}
|
||||
|
||||
auto clients = mapped_clients();
|
||||
if (!clients && m_mapped) {
|
||||
bool has_clients = has_mapped_clients();
|
||||
if (!has_clients && m_mapped) {
|
||||
m_log.trace("tray: Reconfigure window / unmap");
|
||||
m_connection.unmap_window_checked(m_tray);
|
||||
} else if (clients && !m_mapped && !m_hidden) {
|
||||
} else if (has_clients && !m_mapped && !m_hidden) {
|
||||
m_log.trace("tray: Reconfigure window / map");
|
||||
m_connection.map_window_checked(m_tray);
|
||||
}
|
||||
|
||||
auto width = calculate_w();
|
||||
auto x = calculate_x(width);
|
||||
m_opts.win_size.w = width;
|
||||
|
||||
if (m_opts.transparent) {
|
||||
xcb_rectangle_t rect{0, 0, calculate_w(), calculate_h()};
|
||||
|
@ -336,6 +333,7 @@ void tray_manager::reconfigure_window() {
|
|||
}
|
||||
|
||||
if (width > 0) {
|
||||
auto x = calculate_x(width);
|
||||
m_log.trace("tray: New window values, width=%d, x=%d", width, x);
|
||||
|
||||
unsigned int mask = 0;
|
||||
|
@ -347,8 +345,6 @@ void tray_manager::reconfigure_window() {
|
|||
connection::pack_values(mask, ¶ms, values);
|
||||
m_connection.configure_window_checked(m_tray, mask, values);
|
||||
}
|
||||
|
||||
m_opts.configured_w = width;
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -366,7 +362,7 @@ void tray_manager::reconfigure_clients() {
|
|||
client->ensure_state();
|
||||
client->reconfigure(x, calculate_client_y());
|
||||
|
||||
x += m_opts.width + m_opts.spacing;
|
||||
x += m_opts.client_size.w + m_opts.spacing;
|
||||
} catch (const xpp::x::error::window& err) {
|
||||
remove_client(client, false);
|
||||
}
|
||||
|
@ -441,10 +437,10 @@ void tray_manager::refresh_window() {
|
|||
|
||||
m_connection.flush();
|
||||
|
||||
if (!mapped_clients()) {
|
||||
m_opts.configured_w = 0;
|
||||
if (has_mapped_clients()) {
|
||||
m_opts.win_size.w = width;
|
||||
} else {
|
||||
m_opts.configured_w = width;
|
||||
m_opts.win_size.w = 0;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -696,7 +692,7 @@ void tray_manager::track_selection_owner(xcb_window_t owner) {
|
|||
void tray_manager::process_docking_request(xcb_window_t win) {
|
||||
m_log.info("Processing docking request from '%s' (%s)", ewmh_util::get_wm_name(win), m_connection.id(win));
|
||||
|
||||
m_clients.emplace_back(std::make_shared<tray_client>(m_connection, win, m_opts.width, m_opts.height));
|
||||
m_clients.emplace_back(std::make_shared<tray_client>(m_connection, win, m_opts.client_size));
|
||||
auto& client = m_clients.back();
|
||||
|
||||
try {
|
||||
|
@ -753,11 +749,11 @@ void tray_manager::process_docking_request(xcb_window_t win) {
|
|||
* Calculate x position of tray window
|
||||
*/
|
||||
int tray_manager::calculate_x(unsigned int width) const {
|
||||
auto x = m_opts.orig_x;
|
||||
auto x = m_opts.pos.x;
|
||||
if (m_opts.align == alignment::RIGHT) {
|
||||
x -= ((m_opts.width + m_opts.spacing) * m_clients.size() + m_opts.spacing);
|
||||
x -= ((m_opts.client_size.w + m_opts.spacing) * m_clients.size() + m_opts.spacing);
|
||||
} else if (m_opts.align == alignment::CENTER) {
|
||||
x -= (width / 2) - (m_opts.width / 2);
|
||||
x -= (width / 2) - (m_opts.client_size.w / 2);
|
||||
}
|
||||
return x;
|
||||
}
|
||||
|
@ -766,7 +762,7 @@ int tray_manager::calculate_x(unsigned int width) const {
|
|||
* Calculate y position of tray window
|
||||
*/
|
||||
int tray_manager::calculate_y() const {
|
||||
return m_opts.orig_y;
|
||||
return m_opts.pos.y;
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -778,7 +774,7 @@ unsigned short int tray_manager::calculate_w() const {
|
|||
for (auto&& client : m_clients) {
|
||||
if (client->mapped()) {
|
||||
count++;
|
||||
width += m_opts.spacing + m_opts.width;
|
||||
width += m_opts.spacing + m_opts.client_size.w;
|
||||
}
|
||||
}
|
||||
return count ? width : 0;
|
||||
|
@ -788,7 +784,7 @@ unsigned short int tray_manager::calculate_w() const {
|
|||
* Calculate height of tray window
|
||||
*/
|
||||
unsigned short int tray_manager::calculate_h() const {
|
||||
return m_opts.height_fill;
|
||||
return m_opts.win_size.h;
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -797,7 +793,7 @@ unsigned short int tray_manager::calculate_h() const {
|
|||
int tray_manager::calculate_client_x(const xcb_window_t& win) {
|
||||
for (unsigned int i = 0; i < m_clients.size(); i++) {
|
||||
if (m_clients[i]->match(win)) {
|
||||
return m_opts.spacing + m_opts.width * i;
|
||||
return m_opts.spacing + m_opts.client_size.w * i;
|
||||
}
|
||||
}
|
||||
return m_opts.spacing;
|
||||
|
@ -807,7 +803,7 @@ int tray_manager::calculate_client_x(const xcb_window_t& win) {
|
|||
* Calculate y position of client window
|
||||
*/
|
||||
int tray_manager::calculate_client_y() {
|
||||
return (m_opts.height_fill - m_opts.height) / 2;
|
||||
return (m_opts.win_size.h - m_opts.client_size.h) / 2;
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -852,8 +848,8 @@ void tray_manager::remove_client(xcb_window_t win, bool reconfigure) {
|
|||
/**
|
||||
* Get number of mapped clients
|
||||
*/
|
||||
unsigned int tray_manager::mapped_clients() const {
|
||||
unsigned int mapped_clients = 0;
|
||||
int tray_manager::mapped_clients() const {
|
||||
int mapped_clients = 0;
|
||||
|
||||
for (auto&& client : m_clients) {
|
||||
if (client->mapped()) {
|
||||
|
@ -864,6 +860,16 @@ unsigned int tray_manager::mapped_clients() const {
|
|||
return mapped_clients;
|
||||
}
|
||||
|
||||
bool tray_manager::has_mapped_clients() const {
|
||||
for (auto&& client : m_clients) {
|
||||
if (client->mapped()) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* Event callback : XCB_EXPOSE
|
||||
*/
|
||||
|
@ -1052,11 +1058,9 @@ void tray_manager::handle(const evt::map_notify& evt) {
|
|||
m_log.trace("tray: Received map_notify");
|
||||
m_log.trace("tray: Set client mapped");
|
||||
find_client(evt->window)->mapped(true);
|
||||
unsigned int clientcount{mapped_clients()};
|
||||
if (clientcount > m_opts.configured_slots) {
|
||||
if (mapped_clients() > m_opts.num_clients) {
|
||||
reconfigure();
|
||||
}
|
||||
m_sig.emit(signals::ui_tray::mapped_clients{move(clientcount)});
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1072,7 +1076,6 @@ void tray_manager::handle(const evt::unmap_notify& evt) {
|
|||
m_log.trace("tray: Received unmap_notify");
|
||||
m_log.trace("tray: Set client unmapped");
|
||||
find_client(evt->window)->mapped(false);
|
||||
m_sig.emit(signals::ui_tray::mapped_clients{mapped_clients()});
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1083,7 +1086,7 @@ void tray_manager::handle(const evt::unmap_notify& evt) {
|
|||
*/
|
||||
bool tray_manager::on(const signals::ui::visibility_change& evt) {
|
||||
bool visible{evt.cast()};
|
||||
unsigned int clients{mapped_clients()};
|
||||
bool has_clients = has_mapped_clients();
|
||||
|
||||
m_log.trace("tray: visibility_change (state=%i, activated=%i, mapped=%i, hidden=%i)", visible,
|
||||
static_cast<bool>(m_activated), static_cast<bool>(m_mapped), static_cast<bool>(m_hidden));
|
||||
|
@ -1092,11 +1095,11 @@ bool tray_manager::on(const signals::ui::visibility_change& evt) {
|
|||
|
||||
if (!m_activated) {
|
||||
return false;
|
||||
} else if (!m_hidden && !m_mapped && clients) {
|
||||
} else if (!m_hidden && !m_mapped && has_clients) {
|
||||
m_connection.map_window(m_tray);
|
||||
} else if ((!clients || m_hidden) && m_mapped) {
|
||||
} else if ((!has_clients || m_hidden) && m_mapped) {
|
||||
m_connection.unmap_window(m_tray);
|
||||
} else if (m_mapped && !m_hidden && clients) {
|
||||
} else if (m_mapped && !m_hidden && has_clients) {
|
||||
redraw_window();
|
||||
}
|
||||
|
||||
|
@ -1120,7 +1123,7 @@ bool tray_manager::on(const signals::ui::update_background&) {
|
|||
}
|
||||
|
||||
bool tray_manager::on(const signals::ui_tray::tray_pos_change& evt) {
|
||||
m_opts.orig_x = m_bar_opts.inner_area(true).x + evt.cast();
|
||||
m_opts.pos.x = m_bar_opts.inner_area(true).x + evt.cast();
|
||||
reconfigure_window();
|
||||
|
||||
return true;
|
||||
|
|
Loading…
Reference in a new issue