/* * common/xutil.c - X-related useful functions * * Copyright © 2007-2008 Julien Danjou * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. * */ #include "common/util.h" #include #include #include #include "common/xutil.h" #include "common/atoms.h" #include "common/tokenize.h" /** Get the string value of an atom. * \param conn X connection. * \param w Window. * \param atom The atom. * \param text Buffer to fill. * \param len Length of the filled buffer. * \return True on sucess, false on failure. */ bool xutil_text_prop_get(xcb_connection_t *conn, xcb_window_t w, xcb_atom_t atom, char **text, ssize_t *len) { xcb_get_text_property_reply_t reply; p_clear(&reply, 1); if(!xcb_get_text_property_reply(conn, xcb_get_text_property_unchecked(conn, w, atom), &reply, NULL) || !reply.name_len || reply.format != 8) { xcb_get_text_property_reply_wipe(&reply); return false; } if(text && len) { /* Check whether the returned property value is just an ascii * string, an UTF-8 string or just some random multibyte in any other * encoding. */ if(reply.encoding == STRING || reply.encoding == UTF8_STRING || reply.encoding == COMPOUND_TEXT) { *text = p_new(char, reply.name_len + 1); /* Use memcpy() because the property name is not be \0 * terminated */ memcpy(*text, reply.name, reply.name_len); (*text)[reply.name_len] = '\0'; *len = reply.name_len; } else { *text = NULL; *len = 0; } } xcb_get_text_property_reply_wipe(&reply); return true; } /* Number of different errors */ #define ERRORS_NBR 256 void xutil_error_handler_catch_all_set(xcb_event_handlers_t *evenths, xcb_generic_error_handler_t handler, void *data) { int err_num; for(err_num = 0; err_num < ERRORS_NBR; err_num++) xcb_event_set_error_handler(evenths, err_num, handler, data); } uint16_t xutil_key_mask_fromstr(const char *keyname, size_t len) { switch(a_tokenize(keyname, len)) { case A_TK_SHIFT: return XCB_MOD_MASK_SHIFT; case A_TK_LOCK: return XCB_MOD_MASK_LOCK; case A_TK_CTRL: case A_TK_CONTROL: return XCB_MOD_MASK_CONTROL; case A_TK_MOD1: return XCB_MOD_MASK_1; case A_TK_MOD2: return XCB_MOD_MASK_2; case A_TK_MOD3: return XCB_MOD_MASK_3; case A_TK_MOD4: return XCB_MOD_MASK_4; case A_TK_MOD5: return XCB_MOD_MASK_5; /* this is misnamed but correct */ case A_TK_ANY: return XCB_BUTTON_MASK_ANY; default: return XCB_NO_SYMBOL; } } /** Permit to use mouse with many more buttons */ #ifndef XCB_BUTTON_INDEX_6 #define XCB_BUTTON_INDEX_6 6 #endif #ifndef XCB_BUTTON_INDEX_7 #define XCB_BUTTON_INDEX_7 7 #endif #ifndef XCB_BUTTON_INDEX_8 #define XCB_BUTTON_INDEX_8 8 #endif #ifndef XCB_BUTTON_INDEX_9 #define XCB_BUTTON_INDEX_9 9 #endif /** Link a name to a mouse button symbol */ typedef struct { int id; unsigned int button; } mouse_button_t; /** Lookup for a mouse button from its index. * \param button Mouse button index. * \return Mouse button or 0 if not found. */ uint8_t xutil_button_fromint(int button) { /** List of button name and corresponding X11 mask codes */ static const mouse_button_t mouse_button_list[] = { { 1, XCB_BUTTON_INDEX_1 }, { 2, XCB_BUTTON_INDEX_2 }, { 3, XCB_BUTTON_INDEX_3 }, { 4, XCB_BUTTON_INDEX_4 }, { 5, XCB_BUTTON_INDEX_5 }, { 6, XCB_BUTTON_INDEX_6 }, { 7, XCB_BUTTON_INDEX_7 }, { 8, XCB_BUTTON_INDEX_8 }, { 9, XCB_BUTTON_INDEX_9 } }; if(button >= 1 && button <= countof(mouse_button_list)) return mouse_button_list[button - 1].button; return 0; } /** Convert a root window a physical screen ID. * \param connection The connection to the X server. * \param root Root window. * \return A physical screen number. */ int xutil_root2screen(xcb_connection_t *connection, xcb_window_t root) { xcb_screen_iterator_t iter; int phys_screen = 0; for(iter = xcb_setup_roots_iterator(xcb_get_setup(connection)); iter.rem && iter.data->root != root; xcb_screen_next(&iter), ++phys_screen); return phys_screen; } // vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80