2016-11-02 19:22:45 +00:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <X11/X.h>
|
|
|
|
#include <X11/Xlib-xcb.h>
|
|
|
|
#include <xcb/xcb.h>
|
|
|
|
#include <iomanip>
|
|
|
|
|
|
|
|
#include "common.hpp"
|
2016-12-05 19:41:00 +00:00
|
|
|
#include "utils/factory.hpp"
|
2016-11-30 09:06:16 +00:00
|
|
|
#include "x11/extensions.hpp"
|
|
|
|
#include "x11/registry.hpp"
|
2016-11-02 19:22:45 +00:00
|
|
|
#include "x11/types.hpp"
|
2016-12-05 19:41:00 +00:00
|
|
|
#include "x11/xutils.hpp"
|
2016-11-02 19:22:45 +00:00
|
|
|
|
2016-11-19 05:22:44 +00:00
|
|
|
POLYBAR_NS
|
2016-11-02 19:22:45 +00:00
|
|
|
|
2016-11-30 09:06:16 +00:00
|
|
|
using xpp_connection = xpp::connection<XPP_EXTENSION_LIST>;
|
2016-11-02 19:22:45 +00:00
|
|
|
|
|
|
|
class connection : public xpp_connection {
|
|
|
|
public:
|
2016-12-09 08:40:46 +00:00
|
|
|
using make_type = connection&;
|
|
|
|
static make_type make();
|
2016-12-09 08:02:47 +00:00
|
|
|
|
2016-12-05 19:41:00 +00:00
|
|
|
explicit connection(xcb_connection_t* conn) : connection(conn, 0) {}
|
2016-12-03 15:44:08 +00:00
|
|
|
explicit connection(xcb_connection_t* conn, int connection_fd)
|
|
|
|
: xpp_connection(conn), m_connection_fd(connection_fd) {}
|
2016-11-02 19:22:45 +00:00
|
|
|
|
|
|
|
connection& operator=(const connection&) {
|
|
|
|
return *this;
|
|
|
|
}
|
2016-12-15 02:30:41 +00:00
|
|
|
connection(const connection& o) = delete;
|
2016-11-02 19:22:45 +00:00
|
|
|
|
|
|
|
virtual ~connection() {}
|
|
|
|
|
2016-11-26 23:46:41 +00:00
|
|
|
template <typename Event, uint32_t ResponseType>
|
|
|
|
void wait_for_response(function<bool(const Event&)> check_event) {
|
|
|
|
auto fd = get_file_descriptor();
|
|
|
|
fd_set fds;
|
|
|
|
FD_ZERO(&fds);
|
|
|
|
FD_SET(fd, &fds);
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
if (select(fd + 1, &fds, nullptr, nullptr, nullptr) > 0) {
|
|
|
|
shared_ptr<xcb_generic_event_t> evt;
|
|
|
|
|
|
|
|
if ((evt = poll_for_event()) && evt->response_type == ResponseType) {
|
|
|
|
if (check_event(reinterpret_cast<const xcb_map_notify_event_t&>(*(evt.get())))) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (connection_has_error()) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-02 19:22:45 +00:00
|
|
|
void preload_atoms();
|
|
|
|
void query_extensions();
|
|
|
|
|
|
|
|
string id(xcb_window_t w) const;
|
|
|
|
|
2016-12-03 15:44:31 +00:00
|
|
|
xcb_screen_t* screen(bool realloc = false);
|
2016-11-02 19:22:45 +00:00
|
|
|
|
2016-11-26 23:46:41 +00:00
|
|
|
void ensure_event_mask(xcb_window_t win, uint32_t event);
|
|
|
|
void clear_event_mask(xcb_window_t win);
|
|
|
|
|
2016-11-25 07:42:31 +00:00
|
|
|
shared_ptr<xcb_client_message_event_t> make_client_message(xcb_atom_t type, xcb_window_t target) const;
|
2016-11-02 19:22:45 +00:00
|
|
|
|
2016-11-25 12:55:15 +00:00
|
|
|
void send_client_message(const shared_ptr<xcb_client_message_event_t>& message, xcb_window_t target,
|
2016-11-02 19:22:45 +00:00
|
|
|
uint32_t event_mask = 0xFFFFFF, bool propagate = false) const;
|
|
|
|
|
2016-12-15 16:13:15 +00:00
|
|
|
xcb_visualtype_t* visual_type(xcb_screen_t* screen, int match_depth = 32);
|
2016-11-02 19:22:45 +00:00
|
|
|
|
|
|
|
static string error_str(int error_code);
|
|
|
|
|
|
|
|
void dispatch_event(const shared_ptr<xcb_generic_event_t>& evt) const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Attach sink to the registry */
|
|
|
|
template <typename Sink>
|
|
|
|
void attach_sink(Sink&& sink, registry::priority prio = 0) {
|
|
|
|
m_registry.attach(prio, forward<Sink>(sink));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Detach sink from the registry
|
|
|
|
*/
|
|
|
|
template <typename Sink>
|
|
|
|
void detach_sink(Sink&& sink, registry::priority prio = 0) {
|
|
|
|
m_registry.detach(prio, forward<Sink>(sink));
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
registry m_registry{*this};
|
2016-12-03 15:44:08 +00:00
|
|
|
xcb_screen_t* m_screen{nullptr};
|
|
|
|
int m_connection_fd{0};
|
2016-11-02 19:22:45 +00:00
|
|
|
};
|
|
|
|
|
2016-11-19 05:22:44 +00:00
|
|
|
POLYBAR_NS_END
|