2007-12-27 17:27:20 +01:00
|
|
|
/*
|
|
|
|
* ewmh.c - EWMH support functions
|
|
|
|
*
|
2009-01-02 11:58:52 +01:00
|
|
|
* Copyright © 2007-2009 Julien Danjou <julien@danjou.info>
|
2007-12-27 17:27:20 +01:00
|
|
|
*
|
|
|
|
* 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-06-18 08:55:31 +02:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
#include <xcb/xcb.h>
|
|
|
|
#include <xcb/xcb_atom.h>
|
2007-12-27 17:27:20 +01:00
|
|
|
|
|
|
|
#include "ewmh.h"
|
2009-09-14 20:20:27 +02:00
|
|
|
#include "objects/tag.h"
|
2007-12-28 18:31:31 +01:00
|
|
|
#include "screen.h"
|
2009-09-14 20:22:12 +02:00
|
|
|
#include "objects/client.h"
|
2009-09-14 20:26:33 +02:00
|
|
|
#include "objects/widget.h"
|
2009-09-14 20:31:27 +02:00
|
|
|
#include "objects/wibox.h"
|
2009-08-17 16:38:56 +02:00
|
|
|
#include "luaa.h"
|
2008-06-30 18:25:01 +02:00
|
|
|
#include "common/atoms.h"
|
2008-12-02 17:14:23 +01:00
|
|
|
#include "common/buffer.h"
|
2009-04-17 16:52:25 +02:00
|
|
|
#include "common/xutil.h"
|
2007-12-27 17:27:20 +01:00
|
|
|
|
2007-12-28 18:31:31 +01:00
|
|
|
#define _NET_WM_STATE_REMOVE 0
|
|
|
|
#define _NET_WM_STATE_ADD 1
|
|
|
|
#define _NET_WM_STATE_TOGGLE 2
|
|
|
|
|
2009-10-08 14:08:22 +02:00
|
|
|
/** Update client EWMH hints.
|
|
|
|
* \param c The client.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
ewmh_client_update_hints(lua_State *L)
|
|
|
|
{
|
|
|
|
client_t *c = luaA_checkudata(L, 1, &client_class);
|
|
|
|
xcb_atom_t state[10]; /* number of defined state atoms */
|
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
if(c->modal)
|
|
|
|
state[i++] = _NET_WM_STATE_MODAL;
|
|
|
|
if(c->fullscreen)
|
|
|
|
state[i++] = _NET_WM_STATE_FULLSCREEN;
|
|
|
|
if(c->maximized_vertical)
|
|
|
|
state[i++] = _NET_WM_STATE_MAXIMIZED_VERT;
|
|
|
|
if(c->maximized_horizontal)
|
|
|
|
state[i++] = _NET_WM_STATE_MAXIMIZED_HORZ;
|
|
|
|
if(c->sticky)
|
|
|
|
state[i++] = _NET_WM_STATE_STICKY;
|
|
|
|
if(c->skip_taskbar)
|
|
|
|
state[i++] = _NET_WM_STATE_SKIP_TASKBAR;
|
|
|
|
if(c->above)
|
|
|
|
state[i++] = _NET_WM_STATE_ABOVE;
|
|
|
|
if(c->below)
|
|
|
|
state[i++] = _NET_WM_STATE_BELOW;
|
|
|
|
if(c->minimized)
|
|
|
|
state[i++] = _NET_WM_STATE_HIDDEN;
|
|
|
|
if(c->urgent)
|
|
|
|
state[i++] = _NET_WM_STATE_DEMANDS_ATTENTION;
|
|
|
|
|
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2010-08-08 18:31:07 +02:00
|
|
|
c->window, _NET_WM_STATE, XCB_ATOM_ATOM, 32, i, state);
|
2009-10-08 14:08:22 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
ewmh_signal_on_client_new(lua_State *L)
|
|
|
|
{
|
2009-10-09 20:39:55 +02:00
|
|
|
luaA_object_connect_signal(L, 1, "property::modal" , ewmh_client_update_hints);
|
|
|
|
luaA_object_connect_signal(L, 1, "property::fullscreen" , ewmh_client_update_hints);
|
|
|
|
luaA_object_connect_signal(L, 1, "property::maximized_horizontal" , ewmh_client_update_hints);
|
|
|
|
luaA_object_connect_signal(L, 1, "property::maximized_vertical" , ewmh_client_update_hints);
|
|
|
|
luaA_object_connect_signal(L, 1, "property::sticky" , ewmh_client_update_hints);
|
|
|
|
luaA_object_connect_signal(L, 1, "property::skip_taskbar" , ewmh_client_update_hints);
|
|
|
|
luaA_object_connect_signal(L, 1, "property::above" , ewmh_client_update_hints);
|
|
|
|
luaA_object_connect_signal(L, 1, "property::below" , ewmh_client_update_hints);
|
|
|
|
luaA_object_connect_signal(L, 1, "property::minimized" , ewmh_client_update_hints);
|
|
|
|
luaA_object_connect_signal(L, 1, "property::urgent" , ewmh_client_update_hints);
|
2009-10-08 14:08:22 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-02-13 10:57:42 +01:00
|
|
|
/** Update the desktop geometry.
|
|
|
|
*/
|
|
|
|
static void
|
2010-08-16 13:47:40 +02:00
|
|
|
ewmh_update_desktop_geometry(void)
|
2009-02-13 10:57:42 +01:00
|
|
|
{
|
2010-08-16 13:47:40 +02:00
|
|
|
area_t geom = screen_area_get(&globalconf.screens.tab[0], false);
|
2009-02-13 10:57:42 +01:00
|
|
|
uint32_t sizes[] = { geom.width, geom.height };
|
|
|
|
|
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2010-08-16 14:10:58 +02:00
|
|
|
globalconf.screen->root,
|
2010-08-08 18:31:07 +02:00
|
|
|
_NET_DESKTOP_GEOMETRY, XCB_ATOM_CARDINAL, 32, countof(sizes), sizes);
|
2009-02-13 10:57:42 +01:00
|
|
|
}
|
|
|
|
|
2009-10-08 14:49:41 +02:00
|
|
|
static int
|
|
|
|
ewmh_update_net_active_window(lua_State *L)
|
|
|
|
{
|
|
|
|
xcb_window_t win;
|
|
|
|
|
2010-08-17 15:52:49 +02:00
|
|
|
if(globalconf.client_focus)
|
|
|
|
win = globalconf.client_focus->window;
|
2009-10-08 14:49:41 +02:00
|
|
|
else
|
|
|
|
win = XCB_NONE;
|
|
|
|
|
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2010-08-16 14:10:58 +02:00
|
|
|
globalconf.screen->root,
|
2010-08-08 18:31:07 +02:00
|
|
|
_NET_ACTIVE_WINDOW, XCB_ATOM_WINDOW, 32, 1, &win);
|
2009-10-08 14:49:41 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-08 14:53:46 +02:00
|
|
|
static int
|
|
|
|
ewmh_update_net_client_list(lua_State *L)
|
|
|
|
{
|
|
|
|
xcb_window_t *wins = p_alloca(xcb_window_t, globalconf.clients.len);
|
|
|
|
|
|
|
|
int n = 0;
|
|
|
|
foreach(client, globalconf.clients)
|
|
|
|
wins[n++] = (*client)->window;
|
|
|
|
|
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
|
|
|
globalconf.screen->root,
|
|
|
|
_NET_CLIENT_LIST, WINDOW, 32, n, wins);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-12-27 17:27:20 +01:00
|
|
|
void
|
2010-08-16 13:47:40 +02:00
|
|
|
ewmh_init(void)
|
2007-12-27 17:27:20 +01:00
|
|
|
{
|
2008-06-17 21:25:23 +02:00
|
|
|
xcb_window_t father;
|
2010-08-16 14:10:58 +02:00
|
|
|
xcb_screen_t *xscreen = globalconf.screen;
|
2008-06-30 18:25:01 +02:00
|
|
|
xcb_atom_t atom[] =
|
|
|
|
{
|
2008-06-30 18:55:14 +02:00
|
|
|
_NET_SUPPORTED,
|
|
|
|
_NET_SUPPORTING_WM_CHECK,
|
2009-04-04 13:04:19 +02:00
|
|
|
_NET_STARTUP_ID,
|
2008-06-30 18:55:14 +02:00
|
|
|
_NET_CLIENT_LIST,
|
|
|
|
_NET_CLIENT_LIST_STACKING,
|
|
|
|
_NET_NUMBER_OF_DESKTOPS,
|
|
|
|
_NET_CURRENT_DESKTOP,
|
|
|
|
_NET_DESKTOP_NAMES,
|
|
|
|
_NET_ACTIVE_WINDOW,
|
2009-02-13 10:57:42 +01:00
|
|
|
_NET_DESKTOP_GEOMETRY,
|
2008-06-30 18:55:14 +02:00
|
|
|
_NET_CLOSE_WINDOW,
|
|
|
|
_NET_WM_NAME,
|
2008-09-03 22:15:14 +02:00
|
|
|
_NET_WM_STRUT_PARTIAL,
|
2008-06-30 18:55:14 +02:00
|
|
|
_NET_WM_ICON_NAME,
|
|
|
|
_NET_WM_VISIBLE_ICON_NAME,
|
|
|
|
_NET_WM_DESKTOP,
|
|
|
|
_NET_WM_WINDOW_TYPE,
|
|
|
|
_NET_WM_WINDOW_TYPE_DESKTOP,
|
|
|
|
_NET_WM_WINDOW_TYPE_DOCK,
|
2008-11-30 01:47:24 +01:00
|
|
|
_NET_WM_WINDOW_TYPE_TOOLBAR,
|
|
|
|
_NET_WM_WINDOW_TYPE_MENU,
|
|
|
|
_NET_WM_WINDOW_TYPE_UTILITY,
|
2008-06-30 18:55:14 +02:00
|
|
|
_NET_WM_WINDOW_TYPE_SPLASH,
|
|
|
|
_NET_WM_WINDOW_TYPE_DIALOG,
|
2009-02-13 11:25:38 +01:00
|
|
|
_NET_WM_WINDOW_TYPE_DROPDOWN_MENU,
|
|
|
|
_NET_WM_WINDOW_TYPE_POPUP_MENU,
|
|
|
|
_NET_WM_WINDOW_TYPE_TOOLTIP,
|
|
|
|
_NET_WM_WINDOW_TYPE_NOTIFICATION,
|
|
|
|
_NET_WM_WINDOW_TYPE_COMBO,
|
|
|
|
_NET_WM_WINDOW_TYPE_DND,
|
2008-11-30 01:47:24 +01:00
|
|
|
_NET_WM_WINDOW_TYPE_NORMAL,
|
2008-06-30 18:55:14 +02:00
|
|
|
_NET_WM_ICON,
|
|
|
|
_NET_WM_PID,
|
|
|
|
_NET_WM_STATE,
|
|
|
|
_NET_WM_STATE_STICKY,
|
|
|
|
_NET_WM_STATE_SKIP_TASKBAR,
|
|
|
|
_NET_WM_STATE_FULLSCREEN,
|
2008-11-25 11:54:38 +01:00
|
|
|
_NET_WM_STATE_MAXIMIZED_HORZ,
|
|
|
|
_NET_WM_STATE_MAXIMIZED_VERT,
|
2008-06-30 18:55:14 +02:00
|
|
|
_NET_WM_STATE_ABOVE,
|
|
|
|
_NET_WM_STATE_BELOW,
|
|
|
|
_NET_WM_STATE_MODAL,
|
|
|
|
_NET_WM_STATE_HIDDEN,
|
|
|
|
_NET_WM_STATE_DEMANDS_ATTENTION
|
2008-06-30 18:25:01 +02:00
|
|
|
};
|
|
|
|
int i;
|
2007-12-27 17:27:20 +01:00
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2010-08-08 18:31:07 +02:00
|
|
|
xscreen->root, _NET_SUPPORTED, XCB_ATOM_ATOM, 32,
|
2008-06-30 18:25:01 +02:00
|
|
|
countof(atom), atom);
|
2008-06-17 21:25:23 +02:00
|
|
|
|
|
|
|
/* create our own window */
|
|
|
|
father = xcb_generate_id(globalconf.connection);
|
|
|
|
xcb_create_window(globalconf.connection, xscreen->root_depth,
|
|
|
|
father, xscreen->root, -1, -1, 1, 1, 0,
|
|
|
|
XCB_COPY_FROM_PARENT, xscreen->root_visual, 0, NULL);
|
|
|
|
|
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2010-08-08 18:31:07 +02:00
|
|
|
xscreen->root, _NET_SUPPORTING_WM_CHECK, XCB_ATOM_WINDOW, 32,
|
2008-06-17 22:12:30 +02:00
|
|
|
1, &father);
|
2008-06-17 21:25:23 +02:00
|
|
|
|
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2010-08-08 18:31:07 +02:00
|
|
|
father, _NET_SUPPORTING_WM_CHECK, XCB_ATOM_WINDOW, 32,
|
2008-06-17 22:12:30 +02:00
|
|
|
1, &father);
|
2008-06-17 22:47:31 +02:00
|
|
|
|
2008-06-18 08:55:31 +02:00
|
|
|
/* set the window manager name */
|
2008-06-17 22:47:31 +02:00
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2008-06-30 18:55:14 +02:00
|
|
|
father, _NET_WM_NAME, UTF8_STRING, 8, 7, "awesome");
|
2008-06-18 08:55:31 +02:00
|
|
|
|
|
|
|
/* set the window manager PID */
|
|
|
|
i = getpid();
|
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2010-08-08 18:31:07 +02:00
|
|
|
father, _NET_WM_PID, XCB_ATOM_CARDINAL, 32, 1, &i);
|
2009-02-13 10:57:42 +01:00
|
|
|
|
2010-08-16 13:47:40 +02:00
|
|
|
ewmh_update_desktop_geometry();
|
2009-10-08 14:08:22 +02:00
|
|
|
|
2009-10-09 20:39:55 +02:00
|
|
|
luaA_class_connect_signal(globalconf.L, &client_class, "new", ewmh_signal_on_client_new);
|
|
|
|
luaA_class_connect_signal(globalconf.L, &client_class, "focus", ewmh_update_net_active_window);
|
|
|
|
luaA_class_connect_signal(globalconf.L, &client_class, "unfocus", ewmh_update_net_active_window);
|
|
|
|
luaA_class_connect_signal(globalconf.L, &client_class, "manage", ewmh_update_net_client_list);
|
|
|
|
luaA_class_connect_signal(globalconf.L, &client_class, "unmanage", ewmh_update_net_client_list);
|
2007-12-27 18:01:36 +01:00
|
|
|
}
|
2007-12-27 17:27:20 +01:00
|
|
|
|
2008-06-17 21:47:57 +02:00
|
|
|
/** Set the client list in stacking order, bottom to top.
|
|
|
|
*/
|
|
|
|
void
|
2010-08-16 13:47:40 +02:00
|
|
|
ewmh_update_net_client_list_stacking(void)
|
2008-06-17 21:47:57 +02:00
|
|
|
{
|
|
|
|
int n = 0;
|
2009-04-17 23:26:26 +02:00
|
|
|
xcb_window_t *wins = p_alloca(xcb_window_t, globalconf.stack.len);
|
2008-06-17 21:47:57 +02:00
|
|
|
|
2009-04-17 23:26:26 +02:00
|
|
|
foreach(client, globalconf.stack)
|
2010-08-16 13:47:40 +02:00
|
|
|
wins[n++] = (*client)->window;
|
2008-06-17 21:47:57 +02:00
|
|
|
|
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2010-08-16 14:10:58 +02:00
|
|
|
globalconf.screen->root,
|
2010-08-08 18:31:07 +02:00
|
|
|
_NET_CLIENT_LIST_STACKING, XCB_ATOM_WINDOW, 32, n, wins);
|
2008-06-17 21:47:57 +02:00
|
|
|
}
|
|
|
|
|
2007-12-27 18:18:12 +01:00
|
|
|
void
|
2010-08-16 13:47:40 +02:00
|
|
|
ewmh_update_net_numbers_of_desktop(void)
|
2007-12-27 18:18:12 +01:00
|
|
|
{
|
2010-08-16 13:47:40 +02:00
|
|
|
uint32_t count = globalconf.screens.tab[0].tags.len;
|
2007-12-27 18:18:12 +01:00
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2010-08-16 14:10:58 +02:00
|
|
|
globalconf.screen->root,
|
2010-08-08 18:31:07 +02:00
|
|
|
_NET_NUMBER_OF_DESKTOPS, XCB_ATOM_CARDINAL, 32, 1, &count);
|
2007-12-27 18:18:12 +01:00
|
|
|
}
|
|
|
|
|
2007-12-27 18:42:20 +01:00
|
|
|
void
|
2010-08-16 13:47:40 +02:00
|
|
|
ewmh_update_net_current_desktop(void)
|
2007-12-27 18:42:20 +01:00
|
|
|
{
|
2010-08-16 13:47:40 +02:00
|
|
|
uint32_t idx = tags_get_first_selected_index(&globalconf.screens.tab[0]);
|
2007-12-27 18:42:20 +01:00
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2010-08-16 14:10:58 +02:00
|
|
|
globalconf.screen->root,
|
2010-08-08 18:31:07 +02:00
|
|
|
_NET_CURRENT_DESKTOP, XCB_ATOM_CARDINAL, 32, 1, &idx);
|
2007-12-27 18:42:20 +01:00
|
|
|
}
|
|
|
|
|
2007-12-27 19:11:18 +01:00
|
|
|
void
|
2010-08-16 13:47:40 +02:00
|
|
|
ewmh_update_net_desktop_names(void)
|
2007-12-27 19:11:18 +01:00
|
|
|
{
|
2008-06-23 17:37:19 +02:00
|
|
|
buffer_t buf;
|
2007-12-27 19:11:18 +01:00
|
|
|
|
2008-06-23 17:37:19 +02:00
|
|
|
buffer_inita(&buf, BUFSIZ);
|
|
|
|
|
2010-08-16 13:47:40 +02:00
|
|
|
foreach(tag, globalconf.screens.tab[0].tags)
|
2007-12-27 19:11:18 +01:00
|
|
|
{
|
2009-07-29 17:38:48 +02:00
|
|
|
buffer_adds(&buf, tag_get_name(*tag));
|
2008-06-23 17:37:19 +02:00
|
|
|
buffer_addc(&buf, '\0');
|
2007-12-27 19:11:18 +01:00
|
|
|
}
|
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2010-08-16 14:10:58 +02:00
|
|
|
globalconf.screen->root,
|
2008-06-30 18:55:14 +02:00
|
|
|
_NET_DESKTOP_NAMES, UTF8_STRING, 8, buf.len, buf.s);
|
2008-06-23 17:37:19 +02:00
|
|
|
buffer_wipe(&buf);
|
2007-12-27 19:33:01 +01:00
|
|
|
}
|
|
|
|
|
2007-12-27 22:39:29 +01:00
|
|
|
static void
|
2008-04-11 11:35:11 +02:00
|
|
|
ewmh_process_state_atom(client_t *c, xcb_atom_t state, int set)
|
2007-12-27 22:39:29 +01:00
|
|
|
{
|
2009-08-17 17:02:45 +02:00
|
|
|
luaA_object_push(globalconf.L, c);
|
|
|
|
|
2008-06-30 18:55:14 +02:00
|
|
|
if(state == _NET_WM_STATE_STICKY)
|
2007-12-27 22:39:29 +01:00
|
|
|
{
|
2008-08-21 16:29:53 +02:00
|
|
|
if(set == _NET_WM_STATE_REMOVE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_sticky(globalconf.L, -1, false);
|
2008-08-21 16:29:53 +02:00
|
|
|
else if(set == _NET_WM_STATE_ADD)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_sticky(globalconf.L, -1, true);
|
2008-09-18 09:44:51 +02:00
|
|
|
else if(set == _NET_WM_STATE_TOGGLE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_sticky(globalconf.L, -1, !c->sticky);
|
2007-12-27 22:39:29 +01:00
|
|
|
}
|
2008-06-30 18:55:14 +02:00
|
|
|
else if(state == _NET_WM_STATE_SKIP_TASKBAR)
|
2008-01-03 13:15:15 +01:00
|
|
|
{
|
|
|
|
if(set == _NET_WM_STATE_REMOVE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_skip_taskbar(globalconf.L, -1, false);
|
2008-01-03 13:15:15 +01:00
|
|
|
else if(set == _NET_WM_STATE_ADD)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_skip_taskbar(globalconf.L, -1, true);
|
2008-09-18 09:44:51 +02:00
|
|
|
else if(set == _NET_WM_STATE_TOGGLE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_skip_taskbar(globalconf.L, -1, !c->skip_taskbar);
|
2008-01-03 13:15:15 +01:00
|
|
|
}
|
2008-06-30 18:55:14 +02:00
|
|
|
else if(state == _NET_WM_STATE_FULLSCREEN)
|
2007-12-28 18:31:31 +01:00
|
|
|
{
|
|
|
|
if(set == _NET_WM_STATE_REMOVE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_fullscreen(globalconf.L, -1, false);
|
2008-01-06 14:40:23 +01:00
|
|
|
else if(set == _NET_WM_STATE_ADD)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_fullscreen(globalconf.L, -1, true);
|
2008-09-18 09:44:51 +02:00
|
|
|
else if(set == _NET_WM_STATE_TOGGLE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_fullscreen(globalconf.L, -1, !c->fullscreen);
|
2007-12-28 18:31:31 +01:00
|
|
|
}
|
2008-11-25 11:54:38 +01:00
|
|
|
else if(state == _NET_WM_STATE_MAXIMIZED_HORZ)
|
|
|
|
{
|
|
|
|
if(set == _NET_WM_STATE_REMOVE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_maximized_horizontal(globalconf.L, -1, false);
|
2008-11-25 11:54:38 +01:00
|
|
|
else if(set == _NET_WM_STATE_ADD)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_maximized_horizontal(globalconf.L, -1, true);
|
2008-11-25 11:54:38 +01:00
|
|
|
else if(set == _NET_WM_STATE_TOGGLE)
|
2009-10-21 11:41:20 +02:00
|
|
|
client_set_maximized_horizontal(globalconf.L, -1, !c->maximized_horizontal);
|
2008-11-25 11:54:38 +01:00
|
|
|
}
|
|
|
|
else if(state == _NET_WM_STATE_MAXIMIZED_VERT)
|
|
|
|
{
|
|
|
|
if(set == _NET_WM_STATE_REMOVE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_maximized_vertical(globalconf.L, -1, false);
|
2008-11-25 11:54:38 +01:00
|
|
|
else if(set == _NET_WM_STATE_ADD)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_maximized_vertical(globalconf.L, -1, true);
|
2008-11-25 11:54:38 +01:00
|
|
|
else if(set == _NET_WM_STATE_TOGGLE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_maximized_vertical(globalconf.L, -1, !c->maximized_vertical);
|
2008-11-25 11:54:38 +01:00
|
|
|
}
|
2008-06-30 18:55:14 +02:00
|
|
|
else if(state == _NET_WM_STATE_ABOVE)
|
2008-04-08 19:52:59 +02:00
|
|
|
{
|
|
|
|
if(set == _NET_WM_STATE_REMOVE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_above(globalconf.L, -1, false);
|
2008-04-08 19:52:59 +02:00
|
|
|
else if(set == _NET_WM_STATE_ADD)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_above(globalconf.L, -1, true);
|
2008-09-18 09:44:51 +02:00
|
|
|
else if(set == _NET_WM_STATE_TOGGLE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_above(globalconf.L, -1, !c->above);
|
2008-04-08 19:52:59 +02:00
|
|
|
}
|
2008-06-30 18:55:14 +02:00
|
|
|
else if(state == _NET_WM_STATE_BELOW)
|
2008-04-08 19:52:59 +02:00
|
|
|
{
|
|
|
|
if(set == _NET_WM_STATE_REMOVE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_below(globalconf.L, -1, false);
|
2008-04-08 19:52:59 +02:00
|
|
|
else if(set == _NET_WM_STATE_ADD)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_below(globalconf.L, -1, true);
|
2008-09-18 09:44:51 +02:00
|
|
|
else if(set == _NET_WM_STATE_TOGGLE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_below(globalconf.L, -1, !c->below);
|
2008-04-08 19:52:59 +02:00
|
|
|
}
|
2008-06-30 18:55:14 +02:00
|
|
|
else if(state == _NET_WM_STATE_MODAL)
|
2008-06-13 14:41:10 +02:00
|
|
|
{
|
|
|
|
if(set == _NET_WM_STATE_REMOVE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_modal(globalconf.L, -1, false);
|
2008-06-13 14:41:10 +02:00
|
|
|
else if(set == _NET_WM_STATE_ADD)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_modal(globalconf.L, -1, true);
|
2008-09-18 09:44:51 +02:00
|
|
|
else if(set == _NET_WM_STATE_TOGGLE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_modal(globalconf.L, -1, !c->modal);
|
2008-06-13 14:41:10 +02:00
|
|
|
}
|
2008-06-30 18:55:14 +02:00
|
|
|
else if(state == _NET_WM_STATE_HIDDEN)
|
2008-06-13 17:33:59 +02:00
|
|
|
{
|
|
|
|
if(set == _NET_WM_STATE_REMOVE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_minimized(globalconf.L, -1, false);
|
2008-06-13 17:33:59 +02:00
|
|
|
else if(set == _NET_WM_STATE_ADD)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_minimized(globalconf.L, -1, true);
|
2008-09-18 09:44:51 +02:00
|
|
|
else if(set == _NET_WM_STATE_TOGGLE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_minimized(globalconf.L, -1, !c->minimized);
|
2008-06-13 17:33:59 +02:00
|
|
|
}
|
2008-06-30 18:55:14 +02:00
|
|
|
else if(state == _NET_WM_STATE_DEMANDS_ATTENTION)
|
2008-06-17 22:57:54 +02:00
|
|
|
{
|
|
|
|
if(set == _NET_WM_STATE_REMOVE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_urgent(globalconf.L, -1, false);
|
2008-06-17 22:57:54 +02:00
|
|
|
else if(set == _NET_WM_STATE_ADD)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_urgent(globalconf.L, -1, true);
|
2008-09-18 09:44:51 +02:00
|
|
|
else if(set == _NET_WM_STATE_TOGGLE)
|
2009-08-17 17:02:45 +02:00
|
|
|
client_set_urgent(globalconf.L, -1, !c->urgent);
|
2008-06-17 22:57:54 +02:00
|
|
|
}
|
2010-08-03 11:25:54 +02:00
|
|
|
|
|
|
|
lua_pop(globalconf.L, 1);
|
2007-12-27 22:39:29 +01:00
|
|
|
}
|
|
|
|
|
2008-06-14 18:12:16 +02:00
|
|
|
int
|
2008-03-21 16:50:17 +01:00
|
|
|
ewmh_process_client_message(xcb_client_message_event_t *ev)
|
2007-12-27 20:49:38 +01:00
|
|
|
{
|
2008-04-11 11:35:11 +02:00
|
|
|
client_t *c;
|
2007-12-28 22:16:27 +01:00
|
|
|
|
2008-06-30 18:55:14 +02:00
|
|
|
if(ev->type == _NET_CURRENT_DESKTOP)
|
2010-08-16 13:47:40 +02:00
|
|
|
tag_view_only_byindex(&globalconf.screens.tab[0], ev->data.data32[0]);
|
2008-06-30 18:55:14 +02:00
|
|
|
else if(ev->type == _NET_CLOSE_WINDOW)
|
2007-12-27 22:39:29 +01:00
|
|
|
{
|
2008-06-05 09:25:38 +02:00
|
|
|
if((c = client_getbywin(ev->window)))
|
2007-12-27 20:49:38 +01:00
|
|
|
client_kill(c);
|
2007-12-27 22:39:29 +01:00
|
|
|
}
|
2008-06-30 18:55:14 +02:00
|
|
|
else if(ev->type == _NET_WM_DESKTOP)
|
2008-06-18 00:08:04 +02:00
|
|
|
{
|
|
|
|
if((c = client_getbywin(ev->window)))
|
|
|
|
{
|
2009-04-17 16:14:09 +02:00
|
|
|
tag_array_t *tags = &c->screen->tags;
|
2008-06-23 17:37:19 +02:00
|
|
|
|
2008-06-18 00:08:04 +02:00
|
|
|
if(ev->data.data32[0] == 0xffffffff)
|
2009-08-17 17:02:45 +02:00
|
|
|
c->sticky = true;
|
2008-06-18 00:08:04 +02:00
|
|
|
else
|
2008-06-23 17:37:19 +02:00
|
|
|
for(int i = 0; i < tags->len; i++)
|
|
|
|
if((int)ev->data.data32[0] == i)
|
2009-04-11 11:45:55 +02:00
|
|
|
{
|
2009-06-29 11:02:13 +02:00
|
|
|
luaA_object_push(globalconf.L, tags->tab[i]);
|
2009-04-11 11:45:55 +02:00
|
|
|
tag_client(c);
|
|
|
|
}
|
2008-06-18 00:08:04 +02:00
|
|
|
else
|
2008-06-23 17:37:19 +02:00
|
|
|
untag_client(c, tags->tab[i]);
|
2008-06-18 00:08:04 +02:00
|
|
|
}
|
|
|
|
}
|
2008-06-30 18:55:14 +02:00
|
|
|
else if(ev->type == _NET_WM_STATE)
|
2007-12-27 22:39:29 +01:00
|
|
|
{
|
2008-06-05 09:25:38 +02:00
|
|
|
if((c = client_getbywin(ev->window)))
|
2007-12-27 22:39:29 +01:00
|
|
|
{
|
2008-03-21 16:50:17 +01:00
|
|
|
ewmh_process_state_atom(c, (xcb_atom_t) ev->data.data32[1], ev->data.data32[0]);
|
|
|
|
if(ev->data.data32[2])
|
|
|
|
ewmh_process_state_atom(c, (xcb_atom_t) ev->data.data32[2],
|
|
|
|
ev->data.data32[0]);
|
2007-12-27 22:39:29 +01:00
|
|
|
}
|
|
|
|
}
|
2008-12-15 10:04:46 +01:00
|
|
|
else if(ev->type == _NET_ACTIVE_WINDOW)
|
|
|
|
{
|
|
|
|
if((c = client_getbywin(ev->window)))
|
2009-04-07 14:20:22 +02:00
|
|
|
client_focus(c);
|
2008-12-15 10:04:46 +01:00
|
|
|
}
|
2008-06-14 18:12:16 +02:00
|
|
|
|
|
|
|
return 0;
|
2007-12-27 22:39:29 +01:00
|
|
|
}
|
|
|
|
|
2009-01-02 11:55:46 +01:00
|
|
|
/** Update the client active desktop.
|
|
|
|
* This is "wrong" since it can be on several tags, but EWMH has a strict view
|
|
|
|
* of desktop system so just take the first tag.
|
|
|
|
* \param c The client.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
ewmh_client_update_desktop(client_t *c)
|
|
|
|
{
|
|
|
|
int i;
|
2009-04-17 16:14:09 +02:00
|
|
|
tag_array_t *tags = &c->screen->tags;
|
2009-01-02 11:55:46 +01:00
|
|
|
|
|
|
|
for(i = 0; i < tags->len; i++)
|
|
|
|
if(is_client_tagged(c, tags->tab[i]))
|
|
|
|
{
|
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2010-08-08 18:31:07 +02:00
|
|
|
c->window, _NET_WM_DESKTOP, XCB_ATOM_CARDINAL, 32, 1, &i);
|
2009-01-02 11:55:46 +01:00
|
|
|
return;
|
|
|
|
}
|
2010-06-13 08:57:31 +02:00
|
|
|
/* It doesn't have any tags, remove the property */
|
|
|
|
xcb_delete_property(globalconf.connection, c->window, _NET_WM_DESKTOP);
|
2009-01-02 11:55:46 +01:00
|
|
|
}
|
|
|
|
|
2009-02-07 19:40:07 +01:00
|
|
|
/** Update the client struts.
|
2009-08-25 04:55:10 +02:00
|
|
|
* \param window The window to update the struts for.
|
|
|
|
* \param strut The strut type to update the window with.
|
2009-02-07 19:40:07 +01:00
|
|
|
*/
|
|
|
|
void
|
2009-08-18 12:01:44 +02:00
|
|
|
ewmh_update_strut(xcb_window_t window, strut_t *strut)
|
2009-02-07 19:40:07 +01:00
|
|
|
{
|
2009-10-01 16:36:46 +02:00
|
|
|
if(window)
|
2009-02-07 19:40:07 +01:00
|
|
|
{
|
2009-10-01 16:36:46 +02:00
|
|
|
const uint32_t state[] =
|
|
|
|
{
|
|
|
|
strut->left,
|
|
|
|
strut->right,
|
|
|
|
strut->top,
|
|
|
|
strut->bottom,
|
|
|
|
strut->left_start_y,
|
|
|
|
strut->left_end_y,
|
|
|
|
strut->right_start_y,
|
|
|
|
strut->right_end_y,
|
|
|
|
strut->top_start_x,
|
|
|
|
strut->top_end_x,
|
|
|
|
strut->bottom_start_x,
|
|
|
|
strut->bottom_end_x
|
|
|
|
};
|
|
|
|
|
|
|
|
xcb_change_property(globalconf.connection, XCB_PROP_MODE_REPLACE,
|
2010-08-08 18:31:07 +02:00
|
|
|
window, _NET_WM_STRUT_PARTIAL, XCB_ATOM_CARDINAL, 32, countof(state), state);
|
2009-10-01 16:36:46 +02:00
|
|
|
}
|
2009-02-07 19:40:07 +01:00
|
|
|
}
|
|
|
|
|
2008-09-18 13:51:10 +02:00
|
|
|
void
|
|
|
|
ewmh_client_check_hints(client_t *c)
|
2007-12-27 22:39:29 +01:00
|
|
|
{
|
2008-03-21 16:50:17 +01:00
|
|
|
xcb_atom_t *state;
|
|
|
|
void *data = NULL;
|
2008-06-23 17:37:19 +02:00
|
|
|
int desktop;
|
2008-06-18 00:08:04 +02:00
|
|
|
xcb_get_property_cookie_t c0, c1, c2;
|
2008-03-21 16:50:17 +01:00
|
|
|
xcb_get_property_reply_t *reply;
|
|
|
|
|
|
|
|
/* Send the GetProperty requests which will be processed later */
|
2009-08-17 17:02:45 +02:00
|
|
|
c0 = xcb_get_property_unchecked(globalconf.connection, false, c->window,
|
2008-06-30 18:55:14 +02:00
|
|
|
_NET_WM_DESKTOP, XCB_GET_PROPERTY_TYPE_ANY, 0, 1);
|
2008-06-18 00:08:04 +02:00
|
|
|
|
2009-08-17 17:02:45 +02:00
|
|
|
c1 = xcb_get_property_unchecked(globalconf.connection, false, c->window,
|
2010-08-08 18:31:07 +02:00
|
|
|
_NET_WM_STATE, XCB_ATOM_ATOM, 0, UINT32_MAX);
|
2008-03-21 16:50:17 +01:00
|
|
|
|
2009-08-17 17:02:45 +02:00
|
|
|
c2 = xcb_get_property_unchecked(globalconf.connection, false, c->window,
|
2010-08-08 18:31:07 +02:00
|
|
|
_NET_WM_WINDOW_TYPE, XCB_ATOM_ATOM, 0, UINT32_MAX);
|
2008-03-21 16:50:17 +01:00
|
|
|
|
2008-06-18 00:08:04 +02:00
|
|
|
reply = xcb_get_property_reply(globalconf.connection, c0, NULL);
|
|
|
|
if(reply && reply->value_len && (data = xcb_get_property_value(reply)))
|
|
|
|
{
|
2009-04-17 16:14:09 +02:00
|
|
|
tag_array_t *tags = &c->screen->tags;
|
2008-06-23 17:37:19 +02:00
|
|
|
|
2008-06-18 00:08:04 +02:00
|
|
|
desktop = *(uint32_t *) data;
|
|
|
|
if(desktop == -1)
|
2009-08-17 17:02:45 +02:00
|
|
|
c->sticky = true;
|
2010-06-13 09:03:31 +02:00
|
|
|
else if (desktop >= 0 && desktop < tags->len)
|
2008-06-23 17:37:19 +02:00
|
|
|
for(int i = 0; i < tags->len; i++)
|
2008-06-18 00:08:04 +02:00
|
|
|
if(desktop == i)
|
2009-04-11 11:45:55 +02:00
|
|
|
{
|
2009-06-29 11:02:13 +02:00
|
|
|
luaA_object_push(globalconf.L, tags->tab[i]);
|
2009-04-11 11:45:55 +02:00
|
|
|
tag_client(c);
|
|
|
|
}
|
2008-06-18 00:08:04 +02:00
|
|
|
else
|
2008-06-23 17:37:19 +02:00
|
|
|
untag_client(c, tags->tab[i]);
|
2010-06-13 09:03:31 +02:00
|
|
|
else
|
|
|
|
/* Value out of bounds, just give it the first tag */
|
|
|
|
if (tags->len > 0)
|
|
|
|
{
|
|
|
|
luaA_object_push(globalconf.L, tags->tab[0]);
|
|
|
|
tag_client(c);
|
|
|
|
}
|
2008-06-18 00:08:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
p_delete(&reply);
|
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
reply = xcb_get_property_reply(globalconf.connection, c1, NULL);
|
|
|
|
if(reply && (data = xcb_get_property_value(reply)))
|
2007-12-28 20:48:29 +01:00
|
|
|
{
|
2008-03-21 16:50:17 +01:00
|
|
|
state = (xcb_atom_t *) data;
|
2009-08-19 16:40:16 +02:00
|
|
|
for(int i = 0; i < xcb_get_property_value_length(reply) / ssizeof(xcb_atom_t); i++)
|
2007-12-28 20:48:29 +01:00
|
|
|
ewmh_process_state_atom(c, state[i], _NET_WM_STATE_ADD);
|
|
|
|
}
|
2007-12-27 22:39:29 +01:00
|
|
|
|
2008-03-21 16:50:17 +01:00
|
|
|
p_delete(&reply);
|
|
|
|
|
|
|
|
reply = xcb_get_property_reply(globalconf.connection, c2, NULL);
|
|
|
|
if(reply && (data = xcb_get_property_value(reply)))
|
2007-12-28 20:48:29 +01:00
|
|
|
{
|
2008-03-21 16:50:17 +01:00
|
|
|
state = (xcb_atom_t *) data;
|
2009-08-19 16:40:16 +02:00
|
|
|
for(int i = 0; i < xcb_get_property_value_length(reply) / ssizeof(xcb_atom_t); i++)
|
2008-09-03 14:59:18 +02:00
|
|
|
if(state[i] == _NET_WM_WINDOW_TYPE_DESKTOP)
|
2008-10-02 14:07:42 +02:00
|
|
|
c->type = MAX(c->type, WINDOW_TYPE_DESKTOP);
|
2008-09-03 14:59:18 +02:00
|
|
|
else if(state[i] == _NET_WM_WINDOW_TYPE_DIALOG)
|
2008-10-02 14:07:42 +02:00
|
|
|
c->type = MAX(c->type, WINDOW_TYPE_DIALOG);
|
2008-09-03 14:59:18 +02:00
|
|
|
else if(state[i] == _NET_WM_WINDOW_TYPE_SPLASH)
|
2008-10-02 14:07:42 +02:00
|
|
|
c->type = MAX(c->type, WINDOW_TYPE_SPLASH);
|
2008-09-03 14:59:18 +02:00
|
|
|
else if(state[i] == _NET_WM_WINDOW_TYPE_DOCK)
|
2008-10-02 14:07:42 +02:00
|
|
|
c->type = MAX(c->type, WINDOW_TYPE_DOCK);
|
2008-11-30 01:47:24 +01:00
|
|
|
else if(state[i] == _NET_WM_WINDOW_TYPE_MENU)
|
|
|
|
c->type = MAX(c->type, WINDOW_TYPE_MENU);
|
|
|
|
else if(state[i] == _NET_WM_WINDOW_TYPE_TOOLBAR)
|
|
|
|
c->type = MAX(c->type, WINDOW_TYPE_TOOLBAR);
|
|
|
|
else if(state[i] == _NET_WM_WINDOW_TYPE_UTILITY)
|
|
|
|
c->type = MAX(c->type, WINDOW_TYPE_UTILITY);
|
2007-12-28 20:48:29 +01:00
|
|
|
}
|
2008-03-21 16:50:17 +01:00
|
|
|
|
|
|
|
p_delete(&reply);
|
2007-12-27 20:49:38 +01:00
|
|
|
}
|
|
|
|
|
2009-02-07 19:40:07 +01:00
|
|
|
/** Process the WM strut of a client.
|
2008-09-03 22:15:14 +02:00
|
|
|
* \param c The client.
|
2009-07-27 06:20:00 +02:00
|
|
|
* \param strut_r (Optional) An existing reply.
|
2008-09-03 22:15:14 +02:00
|
|
|
*/
|
|
|
|
void
|
2010-08-09 13:04:08 +02:00
|
|
|
ewmh_process_client_strut(client_t *c)
|
2008-09-03 22:15:14 +02:00
|
|
|
{
|
|
|
|
void *data;
|
2010-08-09 13:04:08 +02:00
|
|
|
xcb_get_property_reply_t *strut_r;
|
2008-09-03 22:15:14 +02:00
|
|
|
|
2010-08-09 13:04:08 +02:00
|
|
|
xcb_get_property_cookie_t strut_q = xcb_get_property_unchecked(globalconf.connection, false, c->window,
|
|
|
|
_NET_WM_STRUT_PARTIAL, XCB_ATOM_CARDINAL, 0, 12);
|
|
|
|
strut_r = xcb_get_property_reply(globalconf.connection, strut_q, NULL);
|
2008-09-03 22:15:14 +02:00
|
|
|
|
|
|
|
if(strut_r
|
|
|
|
&& strut_r->value_len
|
|
|
|
&& (data = xcb_get_property_value(strut_r)))
|
|
|
|
{
|
|
|
|
uint32_t *strut = data;
|
|
|
|
|
|
|
|
if(c->strut.left != strut[0]
|
|
|
|
|| c->strut.right != strut[1]
|
|
|
|
|| c->strut.top != strut[2]
|
|
|
|
|| c->strut.bottom != strut[3]
|
|
|
|
|| c->strut.left_start_y != strut[4]
|
|
|
|
|| c->strut.left_end_y != strut[5]
|
|
|
|
|| c->strut.right_start_y != strut[6]
|
|
|
|
|| c->strut.right_end_y != strut[7]
|
|
|
|
|| c->strut.top_start_x != strut[8]
|
|
|
|
|| c->strut.top_end_x != strut[9]
|
|
|
|
|| c->strut.bottom_start_x != strut[10]
|
|
|
|
|| c->strut.bottom_end_x != strut[11])
|
|
|
|
{
|
|
|
|
c->strut.left = strut[0];
|
|
|
|
c->strut.right = strut[1];
|
|
|
|
c->strut.top = strut[2];
|
|
|
|
c->strut.bottom = strut[3];
|
|
|
|
c->strut.left_start_y = strut[4];
|
|
|
|
c->strut.left_end_y = strut[5];
|
|
|
|
c->strut.right_start_y = strut[6];
|
|
|
|
c->strut.right_end_y = strut[7];
|
|
|
|
c->strut.top_start_x = strut[8];
|
|
|
|
c->strut.top_end_x = strut[9];
|
|
|
|
c->strut.bottom_start_x = strut[10];
|
|
|
|
c->strut.bottom_end_x = strut[11];
|
|
|
|
|
2009-08-18 11:52:37 +02:00
|
|
|
luaA_object_push(globalconf.L, c);
|
|
|
|
luaA_object_emit_signal(globalconf.L, -1, "property::struts", 0);
|
|
|
|
lua_pop(globalconf.L, 1);
|
2008-09-03 22:15:14 +02:00
|
|
|
}
|
|
|
|
}
|
2008-09-16 14:09:56 +02:00
|
|
|
|
2010-08-09 13:04:08 +02:00
|
|
|
p_delete(&strut_r);
|
2008-09-03 22:15:14 +02:00
|
|
|
}
|
|
|
|
|
2008-08-13 02:02:19 +02:00
|
|
|
/** Send request to get NET_WM_ICON (EWMH)
|
2008-07-29 16:44:57 +02:00
|
|
|
* \param w The window.
|
2008-08-13 02:02:19 +02:00
|
|
|
* \return The cookie associated with the request.
|
|
|
|
*/
|
|
|
|
xcb_get_property_cookie_t
|
|
|
|
ewmh_window_icon_get_unchecked(xcb_window_t w)
|
|
|
|
{
|
|
|
|
return xcb_get_property_unchecked(globalconf.connection, false, w,
|
2010-08-08 18:31:07 +02:00
|
|
|
_NET_WM_ICON, XCB_ATOM_CARDINAL, 0, UINT32_MAX);
|
2008-08-13 02:02:19 +02:00
|
|
|
}
|
|
|
|
|
2010-08-09 13:04:08 +02:00
|
|
|
static int
|
2008-09-16 14:09:56 +02:00
|
|
|
ewmh_window_icon_from_reply(xcb_get_property_reply_t *r)
|
2008-01-03 19:02:23 +01:00
|
|
|
{
|
2008-03-27 12:08:31 +01:00
|
|
|
uint32_t *data;
|
2009-05-12 11:08:43 +02:00
|
|
|
uint64_t len;
|
2008-03-21 16:50:17 +01:00
|
|
|
|
2010-08-08 18:31:07 +02:00
|
|
|
if(!r || r->type != XCB_ATOM_CARDINAL || r->format != 32 || r->length < 2)
|
2009-05-12 11:05:34 +02:00
|
|
|
return 0;
|
2008-01-03 19:02:23 +01:00
|
|
|
|
2009-05-12 11:05:34 +02:00
|
|
|
data = (uint32_t *) xcb_get_property_value(r);
|
2009-05-12 11:08:43 +02:00
|
|
|
if (!data)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* Check that the property is as long as it should be, handling integer
|
|
|
|
* overflow. <uint32_t> times <another uint32_t casted to uint64_t> always
|
|
|
|
* fits into an uint64_t and thus this multiplication cannot overflow.
|
|
|
|
*/
|
|
|
|
len = data[0] * (uint64_t) data[1];
|
|
|
|
if (!data[0] || !data[1] || len > r->length - 2)
|
2009-05-12 11:05:34 +02:00
|
|
|
return 0;
|
|
|
|
|
2009-11-10 10:48:23 +01:00
|
|
|
return image_new_from_argb32(globalconf.L, data[0], data[1], data + 2);
|
2008-09-16 14:09:56 +02:00
|
|
|
}
|
2008-01-03 19:02:23 +01:00
|
|
|
|
2008-09-16 14:09:56 +02:00
|
|
|
/** Get NET_WM_ICON.
|
|
|
|
* \param cookie The cookie.
|
2009-04-09 11:38:56 +02:00
|
|
|
* \return The number of elements on stack.
|
2008-09-16 14:09:56 +02:00
|
|
|
*/
|
2009-04-09 11:38:56 +02:00
|
|
|
int
|
2008-09-16 14:09:56 +02:00
|
|
|
ewmh_window_icon_get_reply(xcb_get_property_cookie_t cookie)
|
|
|
|
{
|
|
|
|
xcb_get_property_reply_t *r = xcb_get_property_reply(globalconf.connection, cookie, NULL);
|
2009-04-09 11:38:56 +02:00
|
|
|
int ret = ewmh_window_icon_from_reply(r);
|
2008-09-16 14:09:56 +02:00
|
|
|
p_delete(&r);
|
2009-04-09 11:38:56 +02:00
|
|
|
return ret;
|
2008-01-03 19:02:23 +01:00
|
|
|
}
|
|
|
|
|
2007-12-27 17:27:20 +01:00
|
|
|
// vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80
|