2008-03-04 20:39:21 +01:00
|
|
|
/*
|
|
|
|
* common/xutil.c - X-related useful functions
|
|
|
|
*
|
|
|
|
* Copyright © 2007-2008 Julien Danjou <julien@danjou.info>
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2008-11-10 10:35:53 +01:00
|
|
|
#include "common/util.h"
|
2008-04-11 16:35:16 +02:00
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
#include <xcb/xcb.h>
|
|
|
|
#include <xcb/xcb_atom.h>
|
2008-09-09 16:14:36 +02:00
|
|
|
#include <xcb/xcb_icccm.h>
|
2008-03-04 20:39:21 +01:00
|
|
|
|
|
|
|
#include "common/xutil.h"
|
2008-06-30 18:55:14 +02:00
|
|
|
#include "common/atoms.h"
|
2008-12-03 15:23:10 +01:00
|
|
|
#include "common/tokenize.h"
|
2008-03-04 20:39:21 +01:00
|
|
|
|
2008-04-09 13:46:42 +02:00
|
|
|
/** Get the string value of an atom.
|
2008-07-01 14:44:19 +02:00
|
|
|
* \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.
|
2008-04-09 13:46:42 +02:00
|
|
|
*/
|
2008-03-21 16:50:17 +01:00
|
|
|
bool
|
2008-08-11 13:27:28 +02:00
|
|
|
xutil_text_prop_get(xcb_connection_t *conn, xcb_window_t w, xcb_atom_t atom,
|
|
|
|
char **text, ssize_t *len)
|
2008-03-04 20:39:21 +01:00
|
|
|
{
|
2008-09-09 16:14:36 +02:00
|
|
|
xcb_get_text_property_reply_t reply;
|
2008-09-15 17:07:29 +02:00
|
|
|
|
|
|
|
p_clear(&reply, 1);
|
|
|
|
|
2008-09-09 16:14:36 +02:00
|
|
|
if(!xcb_get_text_property_reply(conn,
|
|
|
|
xcb_get_text_property_unchecked(conn, w,
|
|
|
|
atom),
|
|
|
|
&reply, NULL) ||
|
|
|
|
!reply.name_len || reply.format != 8)
|
2008-03-28 22:50:06 +01:00
|
|
|
{
|
2008-09-09 16:14:36 +02:00
|
|
|
xcb_get_text_property_reply_wipe(&reply);
|
2008-03-21 16:50:17 +01:00
|
|
|
return false;
|
2008-03-28 22:50:06 +01:00
|
|
|
}
|
2008-03-04 20:39:21 +01:00
|
|
|
|
2008-08-31 08:55:15 +02:00
|
|
|
if(text && len)
|
2008-04-23 17:06:15 +02:00
|
|
|
{
|
2008-08-31 08:55:15 +02:00
|
|
|
/* Check whether the returned property value is just an ascii
|
2009-02-20 11:26:25 +01:00
|
|
|
* 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)
|
2008-08-31 08:55:15 +02:00
|
|
|
{
|
2008-09-09 16:14:36 +02:00
|
|
|
*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;
|
2008-08-31 08:55:15 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*text = NULL;
|
|
|
|
*len = 0;
|
|
|
|
}
|
2008-08-31 08:41:27 +02:00
|
|
|
}
|
2008-03-04 20:39:21 +01:00
|
|
|
|
2008-09-09 16:14:36 +02:00
|
|
|
xcb_get_text_property_reply_wipe(&reply);
|
2008-03-21 16:50:17 +01:00
|
|
|
return true;
|
2008-03-04 20:39:21 +01:00
|
|
|
}
|
|
|
|
|
2008-03-27 16:51:24 +01:00
|
|
|
/* Number of different errors */
|
|
|
|
#define ERRORS_NBR 256
|
|
|
|
|
|
|
|
void
|
2008-08-11 13:27:28 +02:00
|
|
|
xutil_error_handler_catch_all_set(xcb_event_handlers_t *evenths,
|
2008-03-27 16:51:24 +01:00
|
|
|
xcb_generic_error_handler_t handler,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
int err_num;
|
|
|
|
for(err_num = 0; err_num < ERRORS_NBR; err_num++)
|
2008-09-16 15:36:44 +02:00
|
|
|
xcb_event_set_error_handler(evenths, err_num, handler, data);
|
2008-03-27 16:51:24 +01:00
|
|
|
}
|
|
|
|
|
2009-04-15 14:03:17 +02:00
|
|
|
uint16_t
|
2008-10-01 11:09:57 +02:00
|
|
|
xutil_key_mask_fromstr(const char *keyname, size_t len)
|
2008-05-20 15:39:47 +02:00
|
|
|
{
|
2008-10-01 11:09:57 +02:00
|
|
|
switch(a_tokenize(keyname, len))
|
2008-05-20 15:39:47 +02:00
|
|
|
{
|
2008-10-01 11:09:57 +02:00
|
|
|
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;
|
2009-04-15 13:59:22 +02:00
|
|
|
/* this is misnamed but correct */
|
|
|
|
case A_TK_ANY: return XCB_BUTTON_MASK_ANY;
|
2008-10-01 11:09:57 +02:00
|
|
|
default: return XCB_NO_SYMBOL;
|
|
|
|
}
|
2008-05-20 15:39:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/** 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
|
2008-05-28 07:27:12 +02:00
|
|
|
#ifndef XCB_BUTTON_INDEX_8
|
2008-05-28 07:31:09 +02:00
|
|
|
#define XCB_BUTTON_INDEX_8 8
|
2008-05-28 07:27:12 +02:00
|
|
|
#endif
|
|
|
|
#ifndef XCB_BUTTON_INDEX_9
|
2008-05-28 07:31:09 +02:00
|
|
|
#define XCB_BUTTON_INDEX_9 9
|
2008-05-28 07:27:12 +02:00
|
|
|
#endif
|
2008-05-20 15:39:47 +02:00
|
|
|
|
|
|
|
/** 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.
|
|
|
|
*/
|
2009-04-15 14:05:45 +02:00
|
|
|
uint8_t
|
2008-05-20 15:39:47 +02:00
|
|
|
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 },
|
2008-05-28 10:15:58 +02:00
|
|
|
{ 7, XCB_BUTTON_INDEX_7 },
|
|
|
|
{ 8, XCB_BUTTON_INDEX_8 },
|
2008-05-28 07:27:12 +02:00
|
|
|
{ 9, XCB_BUTTON_INDEX_9 }
|
2008-05-20 15:39:47 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
if(button >= 1 && button <= countof(mouse_button_list))
|
|
|
|
return mouse_button_list[button - 1].button;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-10-06 11:42:02 +02:00
|
|
|
/** 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;
|
|
|
|
}
|
|
|
|
|
2008-03-04 20:39:21 +01:00
|
|
|
// vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80
|