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-03-21 16:50:17 +01:00
|
|
|
/* strndup() */
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
|
|
|
|
#include <xcb/xcb.h>
|
|
|
|
#include <xcb/xcb_aux.h>
|
|
|
|
#include <xcb/xcb_atom.h>
|
|
|
|
#include <xcb/xinerama.h>
|
|
|
|
#include <xcb/xcb_keysyms.h>
|
2008-03-04 20:39:21 +01:00
|
|
|
|
|
|
|
#include "common/util.h"
|
|
|
|
#include "common/xutil.h"
|
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
bool
|
|
|
|
xgettextprop(xcb_connection_t *conn, xcb_window_t w, xcb_atom_t atom,
|
|
|
|
char *text, ssize_t textlen)
|
2008-03-04 20:39:21 +01:00
|
|
|
{
|
2008-03-21 16:50:17 +01:00
|
|
|
xcb_get_property_reply_t *name = NULL;
|
|
|
|
char *prop_val = NULL;
|
2008-03-04 20:39:21 +01:00
|
|
|
|
|
|
|
if(!text || !textlen)
|
2008-03-21 16:50:17 +01:00
|
|
|
return false;
|
2008-03-04 20:39:21 +01:00
|
|
|
|
|
|
|
text[0] = '\0';
|
2008-03-21 16:50:17 +01:00
|
|
|
name = xcb_get_property_reply(conn,
|
|
|
|
xcb_get_property_unchecked(conn, false,
|
|
|
|
w, atom,
|
|
|
|
XCB_GET_PROPERTY_TYPE_ANY,
|
|
|
|
0L, 1000000L),
|
|
|
|
NULL);
|
2008-03-04 20:39:21 +01:00
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
if(!name->value_len)
|
|
|
|
return false;
|
2008-03-04 20:39:21 +01:00
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
prop_val = (char *) xcb_get_property_value(name);
|
|
|
|
if(name->type == STRING)
|
2008-03-21 03:02:30 +01:00
|
|
|
a_strncpy(text, name->value_len + 1, prop_val, textlen - 1);
|
2008-03-04 20:39:21 +01:00
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
/* TODO: XCB doesn't provide a XmbTextPropertyToTextList(), check
|
|
|
|
* whether this code is correct (locales) */
|
|
|
|
else if(name->format == 8)
|
2008-03-21 03:02:30 +01:00
|
|
|
a_strncpy(text, name->value_len + 1, prop_val, textlen - 1);
|
2008-03-04 20:39:21 +01:00
|
|
|
|
|
|
|
text[textlen - 1] = '\0';
|
2008-03-21 16:50:17 +01:00
|
|
|
p_delete(&name);
|
2008-03-04 20:39:21 +01:00
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
return true;
|
2008-03-04 20:39:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int
|
2008-03-21 16:50:17 +01:00
|
|
|
xgetnumlockmask(xcb_connection_t *conn)
|
2008-03-04 20:39:21 +01:00
|
|
|
{
|
2008-03-21 16:50:17 +01:00
|
|
|
xcb_get_modifier_mapping_reply_t *modmap_r;
|
|
|
|
xcb_keycode_t *modmap;
|
|
|
|
xcb_key_symbols_t *keysyms = xcb_key_symbols_alloc(conn);
|
2008-03-04 20:39:21 +01:00
|
|
|
unsigned int mask = 0;
|
|
|
|
int i, j;
|
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
modmap_r = xcb_get_modifier_mapping_reply(conn,
|
|
|
|
xcb_get_modifier_mapping_unchecked(conn),
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
modmap = xcb_get_modifier_mapping_keycodes(modmap_r);
|
|
|
|
|
2008-03-04 20:39:21 +01:00
|
|
|
for(i = 0; i < 8; i++)
|
2008-03-21 16:50:17 +01:00
|
|
|
for(j = 0; j < modmap_r->keycodes_per_modifier; j++)
|
|
|
|
if(modmap[i * modmap_r->keycodes_per_modifier + j]
|
|
|
|
== xcb_key_symbols_get_keycode(keysyms, XK_Num_Lock))
|
2008-03-04 20:39:21 +01:00
|
|
|
mask = (1 << i);
|
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
p_delete(&modmap_r);
|
|
|
|
xcb_key_symbols_free(keysyms);
|
2008-03-04 20:39:21 +01:00
|
|
|
|
|
|
|
return mask;
|
|
|
|
}
|
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
/** Equivalent to 'XGetTransientForHint' which is actually a
|
|
|
|
* 'XGetWindowProperty' which gets the WM_TRANSIENT_FOR property of
|
|
|
|
* the specified window
|
|
|
|
*
|
|
|
|
* \param c X connection
|
|
|
|
* \param win get the property from this window
|
|
|
|
* \param prop_win returns the WM_TRANSIENT_FOR property of win
|
|
|
|
* \return return true if successfull
|
|
|
|
*/
|
|
|
|
bool
|
|
|
|
x_get_transient_for_hint(xcb_connection_t *c, xcb_window_t win,
|
|
|
|
xcb_window_t *prop_win)
|
|
|
|
{
|
|
|
|
xcb_get_property_reply_t *r;
|
|
|
|
|
|
|
|
/* Use checked because the error handler should not take care of
|
|
|
|
* this error as we only return a boolean */
|
|
|
|
r = xcb_get_property_reply(c,
|
|
|
|
xcb_get_property(c, false, win,
|
|
|
|
WM_TRANSIENT_FOR,
|
|
|
|
WINDOW, 0, 1),
|
|
|
|
NULL);
|
|
|
|
|
2008-03-22 16:16:51 +01:00
|
|
|
if(!r || r->type != WINDOW || r->format != 32 || r->length == 0)
|
|
|
|
{
|
|
|
|
*prop_win = XCB_NONE;
|
|
|
|
if(r)
|
|
|
|
p_delete(&r);
|
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
return false;
|
2008-03-22 16:16:51 +01:00
|
|
|
}
|
2008-03-21 16:50:17 +01:00
|
|
|
|
|
|
|
*prop_win = *((xcb_window_t *) xcb_get_property_value(r));
|
|
|
|
p_delete(&r);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
xinerama_is_active(xcb_connection_t *c)
|
|
|
|
{
|
|
|
|
bool ret;
|
|
|
|
xcb_xinerama_is_active_reply_t *r = NULL;
|
|
|
|
|
|
|
|
r = xcb_xinerama_is_active_reply(c, xcb_xinerama_is_active(c), NULL);
|
|
|
|
if(!r)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
ret = r->state;
|
|
|
|
p_delete(&r);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
xcb_window_t
|
|
|
|
root_window(xcb_connection_t *c, int screen_number)
|
|
|
|
{
|
|
|
|
return xcb_aux_get_screen(c, screen_number)->root;
|
|
|
|
}
|
|
|
|
|
|
|
|
xcb_atom_t
|
|
|
|
x_intern_atom(xcb_connection_t *c, const char *property)
|
|
|
|
{
|
|
|
|
xcb_atom_t atom;
|
|
|
|
xcb_intern_atom_reply_t *r_atom;
|
|
|
|
|
|
|
|
r_atom = xcb_intern_atom_reply(c,
|
|
|
|
xcb_intern_atom_unchecked(c, false, strlen(property), property),
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if(!r_atom)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
atom = r_atom->atom;
|
|
|
|
p_delete(&r_atom);
|
|
|
|
|
|
|
|
return atom;
|
|
|
|
}
|
|
|
|
|
|
|
|
class_hint_t *
|
|
|
|
x_get_class_hint(xcb_connection_t *conn, xcb_window_t win)
|
|
|
|
{
|
|
|
|
xcb_get_property_reply_t *r = NULL;
|
|
|
|
char *data = NULL;
|
|
|
|
|
|
|
|
int len_name, len_class;
|
|
|
|
|
|
|
|
class_hint_t *ch = p_new(class_hint_t, 1);
|
|
|
|
|
|
|
|
/* TODO: 2048? (BUFINC is declared as private in xcb.c) */
|
|
|
|
r = xcb_get_property_reply(conn,
|
|
|
|
xcb_get_property_unchecked(conn,
|
|
|
|
false, win, WM_CLASS,
|
|
|
|
STRING, 0L, 2048),
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if(!r || r->type != STRING || r->format == 8)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
data = xcb_get_property_value(r);
|
|
|
|
|
|
|
|
len_name = strlen((char *) data);
|
|
|
|
len_class = strlen((char *) (data + len_name + 1));
|
|
|
|
|
|
|
|
ch->res_name = strndup(data, len_name);
|
|
|
|
ch->res_class = strndup(data + len_name + 1, len_class);
|
|
|
|
|
|
|
|
p_delete(&r);
|
|
|
|
|
|
|
|
return ch;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
xutil_map_raised(xcb_connection_t *conn, xcb_window_t win)
|
|
|
|
{
|
|
|
|
const uint32_t map_raised_val = XCB_STACK_MODE_ABOVE;
|
|
|
|
|
|
|
|
xcb_configure_window(conn, win, XCB_CONFIG_WINDOW_STACK_MODE,
|
|
|
|
&map_raised_val);
|
|
|
|
|
|
|
|
xcb_map_window(conn, win);
|
|
|
|
}
|
|
|
|
|
2008-03-04 20:39:21 +01:00
|
|
|
// vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80
|