2009-02-14 02:33:31 +01:00
|
|
|
|
/*
|
2010-11-29 22:35:07 +01:00
|
|
|
|
* vim:ts=4:sw=4:expandtab
|
2009-02-14 02:33:31 +01:00
|
|
|
|
*
|
|
|
|
|
* i3 - an improved dynamic tiling window manager
|
2015-04-04 02:17:56 +02:00
|
|
|
|
* © 2009 Michael Stapelberg and contributors (see also: LICENSE)
|
2009-02-14 02:33:31 +01:00
|
|
|
|
*
|
2009-02-15 03:07:29 +01:00
|
|
|
|
* xcb.c: Helper functions for easier usage of XCB
|
|
|
|
|
*
|
2009-02-14 02:33:31 +01:00
|
|
|
|
*/
|
2010-03-27 15:25:51 +01:00
|
|
|
|
#include "all.h"
|
2009-02-13 19:04:45 +01:00
|
|
|
|
|
2009-03-27 15:24:52 +01:00
|
|
|
|
unsigned int xcb_numlock_mask;
|
2009-03-04 14:52:04 +01:00
|
|
|
|
|
2009-03-04 12:09:43 +01:00
|
|
|
|
/*
|
|
|
|
|
* Convenience wrapper around xcb_create_window which takes care of depth, generating an ID and checking
|
|
|
|
|
* for errors.
|
|
|
|
|
*
|
|
|
|
|
*/
|
2012-03-01 06:53:06 +01:00
|
|
|
|
xcb_window_t create_window(xcb_connection_t *conn, Rect dims,
|
2014-06-15 19:07:02 +02:00
|
|
|
|
uint16_t depth, xcb_visualid_t visual, uint16_t window_class,
|
|
|
|
|
enum xcursor_cursor_t cursor, bool map, uint32_t mask, uint32_t *values) {
|
2010-11-27 00:26:51 +01:00
|
|
|
|
xcb_window_t result = xcb_generate_id(conn);
|
|
|
|
|
|
2012-01-28 17:09:02 +01:00
|
|
|
|
/* If the window class is XCB_WINDOW_CLASS_INPUT_ONLY, we copy depth and
|
|
|
|
|
* visual id from the parent window. */
|
|
|
|
|
if (window_class == XCB_WINDOW_CLASS_INPUT_ONLY) {
|
|
|
|
|
depth = XCB_COPY_FROM_PARENT;
|
|
|
|
|
visual = XCB_COPY_FROM_PARENT;
|
|
|
|
|
}
|
2010-11-27 00:26:51 +01:00
|
|
|
|
|
2017-04-07 16:59:24 +02:00
|
|
|
|
xcb_void_cookie_t gc_cookie = xcb_create_window(conn,
|
|
|
|
|
depth,
|
|
|
|
|
result, /* the window id */
|
|
|
|
|
root, /* parent == root */
|
|
|
|
|
dims.x, dims.y, dims.width, dims.height, /* dimensions */
|
|
|
|
|
0, /* border = 0, we draw our own */
|
|
|
|
|
window_class,
|
|
|
|
|
visual,
|
|
|
|
|
mask,
|
|
|
|
|
values);
|
|
|
|
|
|
|
|
|
|
xcb_generic_error_t *error = xcb_request_check(conn, gc_cookie);
|
|
|
|
|
if (error != NULL) {
|
|
|
|
|
ELOG("Could not create window. Error code: %d.\n", error->error_code);
|
|
|
|
|
}
|
2010-11-27 00:26:51 +01:00
|
|
|
|
|
|
|
|
|
/* Set the cursor */
|
|
|
|
|
if (xcursor_supported) {
|
|
|
|
|
mask = XCB_CW_CURSOR;
|
|
|
|
|
values[0] = xcursor_get_cursor(cursor);
|
|
|
|
|
xcb_change_window_attributes(conn, result, mask, values);
|
|
|
|
|
} else {
|
2011-07-29 01:40:05 +02:00
|
|
|
|
xcb_cursor_t cursor_id = xcb_generate_id(conn);
|
2011-03-10 23:20:17 +01:00
|
|
|
|
i3Font cursor_font = load_font("cursor", false);
|
2010-11-27 00:26:51 +01:00
|
|
|
|
int xcb_cursor = xcursor_get_xcb_cursor(cursor);
|
2011-11-15 00:39:03 +01:00
|
|
|
|
xcb_create_glyph_cursor(conn, cursor_id, cursor_font.specific.xcb.id,
|
2014-06-15 19:07:02 +02:00
|
|
|
|
cursor_font.specific.xcb.id, xcb_cursor, xcb_cursor + 1, 0, 0, 0,
|
|
|
|
|
65535, 65535, 65535);
|
2010-01-26 11:06:10 +01:00
|
|
|
|
xcb_change_window_attributes(conn, result, XCB_CW_CURSOR, &cursor_id);
|
|
|
|
|
xcb_free_cursor(conn, cursor_id);
|
2010-11-27 00:26:51 +01:00
|
|
|
|
}
|
2009-03-04 15:28:50 +01:00
|
|
|
|
|
2010-11-27 00:26:51 +01:00
|
|
|
|
/* Map the window (= make it visible) */
|
|
|
|
|
if (map)
|
|
|
|
|
xcb_map_window(conn, result);
|
2009-02-16 03:28:07 +01:00
|
|
|
|
|
2010-11-27 00:26:51 +01:00
|
|
|
|
return result;
|
2009-02-16 03:28:07 +01:00
|
|
|
|
}
|
2009-02-24 00:30:04 +01:00
|
|
|
|
|
2009-04-11 22:37:48 +02:00
|
|
|
|
/*
|
|
|
|
|
* Generates a configure_notify_event with absolute coordinates (relative to the X root
|
|
|
|
|
* window, not to the client’s frame) for the given client.
|
|
|
|
|
*
|
|
|
|
|
*/
|
2010-05-31 23:00:36 +02:00
|
|
|
|
void fake_absolute_configure_notify(Con *con) {
|
2011-10-09 14:40:15 +02:00
|
|
|
|
xcb_rectangle_t absolute;
|
2010-11-29 22:35:07 +01:00
|
|
|
|
if (con->window == NULL)
|
|
|
|
|
return;
|
2009-04-11 22:37:48 +02:00
|
|
|
|
|
2010-11-29 22:35:07 +01:00
|
|
|
|
absolute.x = con->rect.x + con->window_rect.x;
|
|
|
|
|
absolute.y = con->rect.y + con->window_rect.y;
|
|
|
|
|
absolute.width = con->window_rect.width;
|
|
|
|
|
absolute.height = con->window_rect.height;
|
2009-04-11 22:37:48 +02:00
|
|
|
|
|
2011-01-21 21:01:02 +01:00
|
|
|
|
DLOG("fake rect = (%d, %d, %d, %d)\n", absolute.x, absolute.y, absolute.width, absolute.height);
|
|
|
|
|
|
2011-09-17 15:11:55 +02:00
|
|
|
|
fake_configure_notify(conn, absolute, con->window->id, con->border_width);
|
2009-04-11 22:37:48 +02:00
|
|
|
|
}
|
|
|
|
|
|
2011-03-18 17:07:56 +01:00
|
|
|
|
/*
|
|
|
|
|
* Sends the WM_TAKE_FOCUS ClientMessage to the given window
|
|
|
|
|
*
|
|
|
|
|
*/
|
2014-03-29 05:25:52 +01:00
|
|
|
|
void send_take_focus(xcb_window_t window, xcb_timestamp_t timestamp) {
|
2011-07-31 19:33:56 +02:00
|
|
|
|
/* Every X11 event is 32 bytes long. Therefore, XCB will copy 32 bytes.
|
|
|
|
|
* In order to properly initialize these bytes, we allocate 32 bytes even
|
|
|
|
|
* though we only need less for an xcb_configure_notify_event_t */
|
2015-08-03 11:50:13 +02:00
|
|
|
|
void *event = scalloc(32, 1);
|
2011-07-31 19:33:56 +02:00
|
|
|
|
xcb_client_message_event_t *ev = event;
|
2011-03-18 17:07:56 +01:00
|
|
|
|
|
2011-07-31 19:33:56 +02:00
|
|
|
|
ev->response_type = XCB_CLIENT_MESSAGE;
|
|
|
|
|
ev->window = window;
|
|
|
|
|
ev->type = A_WM_PROTOCOLS;
|
|
|
|
|
ev->format = 32;
|
|
|
|
|
ev->data.data32[0] = A_WM_TAKE_FOCUS;
|
2014-03-29 05:25:52 +01:00
|
|
|
|
ev->data.data32[1] = timestamp;
|
2011-03-18 17:07:56 +01:00
|
|
|
|
|
|
|
|
|
DLOG("Sending WM_TAKE_FOCUS to the client\n");
|
2014-06-15 19:07:02 +02:00
|
|
|
|
xcb_send_event(conn, false, window, XCB_EVENT_MASK_NO_EVENT, (char *)ev);
|
2011-07-31 19:34:55 +02:00
|
|
|
|
free(event);
|
2011-03-18 17:07:56 +01:00
|
|
|
|
}
|
|
|
|
|
|
2010-02-28 20:35:30 +01:00
|
|
|
|
/*
|
|
|
|
|
* Configures the given window to have the size/position specified by given rect
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
void xcb_set_window_rect(xcb_connection_t *conn, xcb_window_t window, Rect r) {
|
2010-11-29 22:35:07 +01:00
|
|
|
|
xcb_void_cookie_t cookie;
|
|
|
|
|
cookie = xcb_configure_window(conn, window,
|
2014-06-15 19:07:02 +02:00
|
|
|
|
XCB_CONFIG_WINDOW_X |
|
|
|
|
|
XCB_CONFIG_WINDOW_Y |
|
|
|
|
|
XCB_CONFIG_WINDOW_WIDTH |
|
|
|
|
|
XCB_CONFIG_WINDOW_HEIGHT,
|
|
|
|
|
&(r.x));
|
2010-11-29 22:35:07 +01:00
|
|
|
|
/* ignore events which are generated because we configured a window */
|
2011-07-10 23:44:13 +02:00
|
|
|
|
add_ignore_event(cookie.sequence, -1);
|
2010-02-28 20:35:30 +01:00
|
|
|
|
}
|
2010-04-17 18:26:46 +02:00
|
|
|
|
|
2015-04-18 21:09:03 +02:00
|
|
|
|
/*
|
|
|
|
|
* Returns the first supported _NET_WM_WINDOW_TYPE atom.
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
xcb_atom_t xcb_get_preferred_window_type(xcb_get_property_reply_t *reply) {
|
|
|
|
|
if (reply == NULL || xcb_get_property_value_length(reply) == 0)
|
|
|
|
|
return XCB_NONE;
|
|
|
|
|
|
|
|
|
|
xcb_atom_t *atoms;
|
|
|
|
|
if ((atoms = xcb_get_property_value(reply)) == NULL)
|
|
|
|
|
return XCB_NONE;
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < xcb_get_property_value_length(reply) / (reply->format / 8); i++) {
|
|
|
|
|
if (atoms[i] == A__NET_WM_WINDOW_TYPE_NORMAL ||
|
|
|
|
|
atoms[i] == A__NET_WM_WINDOW_TYPE_DIALOG ||
|
|
|
|
|
atoms[i] == A__NET_WM_WINDOW_TYPE_UTILITY ||
|
|
|
|
|
atoms[i] == A__NET_WM_WINDOW_TYPE_TOOLBAR ||
|
|
|
|
|
atoms[i] == A__NET_WM_WINDOW_TYPE_SPLASH ||
|
|
|
|
|
atoms[i] == A__NET_WM_WINDOW_TYPE_MENU ||
|
|
|
|
|
atoms[i] == A__NET_WM_WINDOW_TYPE_DROPDOWN_MENU ||
|
|
|
|
|
atoms[i] == A__NET_WM_WINDOW_TYPE_POPUP_MENU ||
|
2015-12-07 12:34:24 +01:00
|
|
|
|
atoms[i] == A__NET_WM_WINDOW_TYPE_TOOLTIP ||
|
|
|
|
|
atoms[i] == A__NET_WM_WINDOW_TYPE_NOTIFICATION) {
|
2015-04-18 21:09:03 +02:00
|
|
|
|
return atoms[i];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return XCB_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
2010-04-17 18:26:46 +02:00
|
|
|
|
/*
|
|
|
|
|
* Returns true if the given reply contains the given atom.
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
bool xcb_reply_contains_atom(xcb_get_property_reply_t *prop, xcb_atom_t atom) {
|
|
|
|
|
if (prop == NULL || xcb_get_property_value_length(prop) == 0)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
xcb_atom_t *atoms;
|
|
|
|
|
if ((atoms = xcb_get_property_value(prop)) == NULL)
|
|
|
|
|
return false;
|
|
|
|
|
|
2011-03-03 14:14:35 +01:00
|
|
|
|
for (int i = 0; i < xcb_get_property_value_length(prop) / (prop->format / 8); i++)
|
2010-04-17 18:26:46 +02:00
|
|
|
|
if (atoms[i] == atom)
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2011-08-11 21:57:22 +02:00
|
|
|
|
|
2011-10-10 16:53:57 +02:00
|
|
|
|
/*
|
|
|
|
|
* Set the cursor of the root window to the given cursor id.
|
|
|
|
|
* This function should only be used if xcursor_supported == false.
|
|
|
|
|
* Otherwise, use xcursor_set_root_cursor().
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
void xcb_set_root_cursor(int cursor) {
|
|
|
|
|
xcb_cursor_t cursor_id = xcb_generate_id(conn);
|
|
|
|
|
i3Font cursor_font = load_font("cursor", false);
|
|
|
|
|
int xcb_cursor = xcursor_get_xcb_cursor(cursor);
|
2011-11-15 00:39:03 +01:00
|
|
|
|
xcb_create_glyph_cursor(conn, cursor_id, cursor_font.specific.xcb.id,
|
2014-06-15 19:07:02 +02:00
|
|
|
|
cursor_font.specific.xcb.id, xcb_cursor, xcb_cursor + 1, 0, 0, 0,
|
|
|
|
|
65535, 65535, 65535);
|
2011-10-10 16:53:57 +02:00
|
|
|
|
xcb_change_window_attributes(conn, root, XCB_CW_CURSOR, &cursor_id);
|
|
|
|
|
xcb_free_cursor(conn, cursor_id);
|
|
|
|
|
xcb_flush(conn);
|
|
|
|
|
}
|
2012-03-01 06:53:06 +01:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Get depth of visual specified by visualid
|
|
|
|
|
*
|
|
|
|
|
*/
|
2014-06-15 19:07:02 +02:00
|
|
|
|
uint16_t get_visual_depth(xcb_visualid_t visual_id) {
|
2012-03-01 06:53:06 +01:00
|
|
|
|
xcb_depth_iterator_t depth_iter;
|
|
|
|
|
|
|
|
|
|
depth_iter = xcb_screen_allowed_depths_iterator(root_screen);
|
|
|
|
|
for (; depth_iter.rem; xcb_depth_next(&depth_iter)) {
|
|
|
|
|
xcb_visualtype_iterator_t visual_iter;
|
|
|
|
|
|
|
|
|
|
visual_iter = xcb_depth_visuals_iterator(depth_iter.data);
|
|
|
|
|
for (; visual_iter.rem; xcb_visualtype_next(&visual_iter)) {
|
|
|
|
|
if (visual_id == visual_iter.data->visual_id) {
|
|
|
|
|
return depth_iter.data->depth;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2015-11-17 12:50:06 +01:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Get visual type specified by visualid
|
|
|
|
|
*
|
|
|
|
|
*/
|
Migrate i3 rendering to cairo.
This patch migrates all decoration rendering of i3 to cairo. Using the
compile switch CAIRO_SUPPORT, rendering can be switched back to the
previous XCB behavior, just like with the previous migration to cairo
in i3bar.
This patch also fixes a bug in draw_util.c where copying one surface
to another would use incorrect coordinates if the source coordinates
are not 0, 0.
Furthermore, this patch implicitly fixes some minor issues in the
decoration rendering which would be ignored previously due to the fact
that errors would only show up in the event queue, but not cause the
rendering code path to crash. One example is zero-height pixmaps which
are not allowed. Using cairo, these would cause i3 to instantly segfault,
so this patch avoids this.
Lastly, this patch annotates other issues found but not fixed in this patch
using TODO comments, e.g., the zero-height check not working correctly
and the comment that it should probably work the same way for zero-width
pixmaps.
relates to #1278
2015-11-16 21:26:06 +01:00
|
|
|
|
xcb_visualtype_t *get_visualtype_by_id(xcb_visualid_t visual_id) {
|
|
|
|
|
xcb_depth_iterator_t depth_iter;
|
|
|
|
|
|
|
|
|
|
depth_iter = xcb_screen_allowed_depths_iterator(root_screen);
|
|
|
|
|
for (; depth_iter.rem; xcb_depth_next(&depth_iter)) {
|
|
|
|
|
xcb_visualtype_iterator_t visual_iter;
|
|
|
|
|
|
|
|
|
|
visual_iter = xcb_depth_visuals_iterator(depth_iter.data);
|
|
|
|
|
for (; visual_iter.rem; xcb_visualtype_next(&visual_iter)) {
|
|
|
|
|
if (visual_id == visual_iter.data->visual_id) {
|
|
|
|
|
return visual_iter.data;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2012-03-01 06:53:06 +01:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Get visualid with specified depth
|
|
|
|
|
*
|
|
|
|
|
*/
|
2014-06-15 19:07:02 +02:00
|
|
|
|
xcb_visualid_t get_visualid_by_depth(uint16_t depth) {
|
2012-03-01 06:53:06 +01:00
|
|
|
|
xcb_depth_iterator_t depth_iter;
|
|
|
|
|
|
|
|
|
|
depth_iter = xcb_screen_allowed_depths_iterator(root_screen);
|
|
|
|
|
for (; depth_iter.rem; xcb_depth_next(&depth_iter)) {
|
|
|
|
|
if (depth_iter.data->depth != depth)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
xcb_visualtype_iterator_t visual_iter;
|
|
|
|
|
|
|
|
|
|
visual_iter = xcb_depth_visuals_iterator(depth_iter.data);
|
|
|
|
|
if (!visual_iter.rem)
|
|
|
|
|
continue;
|
|
|
|
|
return visual_iter.data->visual_id;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2015-09-14 13:34:15 +02:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Add an atom to a list of atoms the given property defines.
|
|
|
|
|
* This is useful, for example, for manipulating _NET_WM_STATE.
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
void xcb_add_property_atom(xcb_connection_t *conn, xcb_window_t window, xcb_atom_t property, xcb_atom_t atom) {
|
|
|
|
|
xcb_change_property(conn, XCB_PROP_MODE_APPEND, window, property, XCB_ATOM_ATOM, 32, 1, (uint32_t[]){atom});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Remove an atom from a list of atoms the given property defines without
|
|
|
|
|
* removing any other potentially set atoms. This is useful, for example, for
|
|
|
|
|
* manipulating _NET_WM_STATE.
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
void xcb_remove_property_atom(xcb_connection_t *conn, xcb_window_t window, xcb_atom_t property, xcb_atom_t atom) {
|
|
|
|
|
xcb_grab_server(conn);
|
|
|
|
|
|
|
|
|
|
xcb_get_property_reply_t *reply =
|
|
|
|
|
xcb_get_property_reply(conn,
|
|
|
|
|
xcb_get_property(conn, false, window, property, XCB_GET_PROPERTY_TYPE_ANY, 0, 4096), NULL);
|
|
|
|
|
if (reply == NULL || xcb_get_property_value_length(reply) == 0)
|
|
|
|
|
goto release_grab;
|
|
|
|
|
xcb_atom_t *atoms = xcb_get_property_value(reply);
|
|
|
|
|
if (atoms == NULL) {
|
|
|
|
|
goto release_grab;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
int num = 0;
|
|
|
|
|
const int current_size = xcb_get_property_value_length(reply) / (reply->format / 8);
|
|
|
|
|
xcb_atom_t values[current_size];
|
|
|
|
|
for (int i = 0; i < current_size; i++) {
|
|
|
|
|
if (atoms[i] != atom)
|
|
|
|
|
values[num++] = atoms[i];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
xcb_change_property(conn, XCB_PROP_MODE_REPLACE, window, property, XCB_ATOM_ATOM, 32, num, values);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
release_grab:
|
|
|
|
|
FREE(reply);
|
|
|
|
|
xcb_ungrab_server(conn);
|
|
|
|
|
}
|
2015-11-11 20:21:26 +01:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Grab the specified buttons on a window when managing it.
|
|
|
|
|
*
|
|
|
|
|
*/
|
2016-04-13 19:45:57 +02:00
|
|
|
|
void xcb_grab_buttons(xcb_connection_t *conn, xcb_window_t window, int *buttons) {
|
|
|
|
|
int i = 0;
|
|
|
|
|
while (buttons[i] > 0) {
|
2015-11-11 20:21:26 +01:00
|
|
|
|
xcb_grab_button(conn, false, window, XCB_EVENT_MASK_BUTTON_PRESS, XCB_GRAB_MODE_SYNC,
|
|
|
|
|
XCB_GRAB_MODE_ASYNC, root, XCB_NONE, buttons[i], XCB_BUTTON_MASK_ANY);
|
2016-04-13 19:45:57 +02:00
|
|
|
|
|
|
|
|
|
i++;
|
2015-11-11 20:21:26 +01:00
|
|
|
|
}
|
|
|
|
|
}
|