awesome/objects/selection_watcher.c

200 lines
7.0 KiB
C

/*
* selection_watcher.h - selection change watcher
*
* Copyright © 2019 Uli Schlachter <psychon@znc.in>
*
* 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 "objects/selection_watcher.h"
#include "common/luaobject.h"
#include "globalconf.h"
#include <xcb/xfixes.h>
#define REGISTRY_WATCHER_TABLE_INDEX "awesome_selection_watchers"
typedef struct selection_watcher_t
{
LUA_OBJECT_HEADER
/** Is this watcher currently active and watching? Used as reference with luaL_ref */
int active_ref;
/** Atom identifying the selection to watch */
xcb_atom_t selection;
/** Window used for watching */
xcb_window_t window;
} selection_watcher_t;
static lua_class_t selection_watcher_class;
LUA_OBJECT_FUNCS(selection_watcher_class, selection_watcher_t, selection_watcher)
void
event_handle_xfixes_selection_notify(xcb_generic_event_t *ev)
{
xcb_xfixes_selection_notify_event_t *e = (void *) ev;
lua_State *L = globalconf_get_lua_State();
/* Iterate over all active selection watchers */
lua_pushliteral(L, REGISTRY_WATCHER_TABLE_INDEX);
lua_rawget(L, LUA_REGISTRYINDEX);
lua_pushnil(L);
while (lua_next(L, -2) != 0) {
if (lua_type(L, -1) == LUA_TUSERDATA) {
selection_watcher_t *selection = lua_touserdata(L, -1);
if (selection->selection == e->selection && selection->window == e->window) {
lua_pushboolean(L, e->owner != XCB_NONE);
luaA_object_emit_signal(L, -2, "selection_changed", 1);
}
}
/* Remove the watcher */
lua_pop(L, 1);
}
/* Remove watcher table */
lua_pop(L, 1);
}
/** Create a new selection watcher object.
* \param L The Lua VM state.
* \return The number of elements pushed on the stack.
*/
static int
luaA_selection_watcher_new(lua_State *L)
{
size_t name_length;
const char *name;
xcb_intern_atom_reply_t *reply;
selection_watcher_t *selection;
name = luaL_checklstring(L, 2, &name_length);
selection = (void *) selection_watcher_class.allocator(L);
selection->active_ref = LUA_NOREF;
selection->window = XCB_NONE;
/* Get the atom identifying the selection to watch */
reply = xcb_intern_atom_reply(globalconf.connection,
xcb_intern_atom_unchecked(globalconf.connection, false, name_length, name),
NULL);
if (reply) {
selection->selection = reply->atom;
p_delete(&reply);
}
return 1;
}
static int
luaA_selection_watcher_set_active(lua_State *L, selection_watcher_t *selection)
{
bool b = luaA_checkboolean(L, -1);
bool is_active = selection->active_ref != LUA_NOREF;
if(b != is_active)
{
if (b)
{
/* Selection becomes active */
/* Create a window for it */
if (selection->window == XCB_NONE)
selection->window = xcb_generate_id(globalconf.connection);
xcb_create_window(globalconf.connection, globalconf.screen->root_depth,
selection->window, globalconf.screen->root, -1, -1, 1, 1, 0,
XCB_COPY_FROM_PARENT, globalconf.screen->root_visual,
0, NULL);
/* Start watching for selection changes */
if (globalconf.have_xfixes)
{
xcb_xfixes_select_selection_input(globalconf.connection, selection->window, selection->selection,
XCB_XFIXES_SELECTION_EVENT_MASK_SET_SELECTION_OWNER |
XCB_XFIXES_SELECTION_EVENT_MASK_SELECTION_WINDOW_DESTROY |
XCB_XFIXES_SELECTION_EVENT_MASK_SELECTION_CLIENT_CLOSE);
} else {
luaA_warn(L, "X11 server does not support the XFixes extension; cannot watch selections");
}
/* Reference the selection watcher. For this, first get the tracking
* table out of the registry. */
lua_pushliteral(L, REGISTRY_WATCHER_TABLE_INDEX);
lua_rawget(L, LUA_REGISTRYINDEX);
/* Then actually get the reference */
lua_pushvalue(L, -3 - 1);
selection->active_ref = luaL_ref(L, -2);
/* And pop the tracking table again */
lua_pop(L, 1);
} else {
/* Stop watching and destroy the window */
if (globalconf.have_xfixes)
xcb_xfixes_select_selection_input(globalconf.connection, selection->window, selection->selection, 0);
xcb_destroy_window(globalconf.connection, selection->window);
/* Unreference the selection object */
lua_pushliteral(L, REGISTRY_WATCHER_TABLE_INDEX);
lua_rawget(L, LUA_REGISTRYINDEX);
luaL_unref(L, -1, selection->active_ref);
lua_pop(L, 1);
selection->active_ref = LUA_NOREF;
}
luaA_object_emit_signal(L, -3, "property::active", 0);
}
return 0;
}
static int
luaA_selection_watcher_get_active(lua_State *L, selection_watcher_t *selection)
{
lua_pushboolean(L, selection->active_ref != LUA_NOREF);
return 1;
}
void
selection_watcher_class_setup(lua_State *L)
{
static const struct luaL_Reg selection_watcher_methods[] =
{
LUA_CLASS_METHODS(selection_watcher)
{ "__call", luaA_selection_watcher_new },
{ NULL, NULL }
};
static const struct luaL_Reg selection_watcher_meta[] = {
LUA_OBJECT_META(selection_watcher)
LUA_CLASS_META
{ NULL, NULL }
};
/* Reference a table in the registry that tracks active watchers. This code
* does debug.getregistry()[REGISTRY_WATCHER_TABLE_INDEX] = {}
*/
lua_pushliteral(L, REGISTRY_WATCHER_TABLE_INDEX);
lua_newtable(L);
lua_rawset(L, LUA_REGISTRYINDEX);
luaA_class_setup(L, &selection_watcher_class, "selection_watcher", NULL,
(lua_class_allocator_t) selection_watcher_new, NULL, NULL,
luaA_class_index_miss_property, luaA_class_newindex_miss_property,
selection_watcher_methods, selection_watcher_meta);
luaA_class_add_property(&selection_watcher_class, "active",
(lua_class_propfunc_t) luaA_selection_watcher_set_active,
(lua_class_propfunc_t) luaA_selection_watcher_get_active,
(lua_class_propfunc_t) luaA_selection_watcher_set_active);
}
// vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80