awesome/selection.c

182 lines
5.6 KiB
C

/*
* selection.c - Selection handling
*
* Copyright © 2009 Julien Danjou <julien@danjou.info>
* Copyright © 2009 Gregor Best <farhaven@googlemail.com>
*
* 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.
*
*/
/** awesome selection (clipboard) API
* @author Julien Danjou &lt;julien@danjou.info&gt;
* @copyright 2008-2009 Julien Danjou
* @module selection
*/
#include "selection.h"
#include "globalconf.h"
#include "common/atoms.h"
#include "event.h"
#include "xwindow.h"
#include <xcb/xcb_atom.h>
#include <xcb/xcb_event.h>
/** Get the selection (clipboard) content.
*
* @return A string with the selection (clipboard) content.
* @staticfct selection
*/
static xcb_window_t selection_window = XCB_NONE;
/** Get the current X selection buffer.
* \param L The Lua VM state.
* \return The number of elements pushed on stack.
* \luastack
* \lreturn A string with the current X selection buffer.
*/
static int
luaA_selection_get(lua_State *L)
{
luaA_deprecate(L, "selection.getter(\"PRIMARY\", \"UTF8_STRING\")");
if(selection_window == XCB_NONE)
{
xcb_screen_t *screen = globalconf.screen;
uint32_t mask = XCB_CW_BACK_PIXEL | XCB_CW_OVERRIDE_REDIRECT | XCB_CW_EVENT_MASK;
uint32_t values[] = { screen->black_pixel, 1, XCB_EVENT_MASK_PROPERTY_CHANGE };
selection_window = xcb_generate_id(globalconf.connection);
xcb_create_window(globalconf.connection, screen->root_depth, selection_window, screen->root,
0, 0, 1, 1, 0, XCB_COPY_FROM_PARENT, screen->root_visual,
mask, values);
xwindow_set_class_instance(selection_window);
xwindow_set_name_static(selection_window, "Awesome selection window");
}
xcb_convert_selection(globalconf.connection, selection_window,
XCB_ATOM_PRIMARY, UTF8_STRING, XSEL_DATA, globalconf.timestamp);
xcb_flush(globalconf.connection);
xcb_generic_event_t *event;
while(true)
{
event = xcb_wait_for_event(globalconf.connection);
if(!event)
return 0;
if(XCB_EVENT_RESPONSE_TYPE(event) != XCB_SELECTION_NOTIFY)
{
/* \todo Eventually, this may be rewritten with adding a static
* buffer, then a event handler for XCB_SELECTION_NOTIFY, then call
* xcb_event_poll_for_event_loop() and awesome_refresh(),
* then check if some static buffer has been filled with data.
* If yes, that'd be the xsel data, otherwise, re-loop.
* Anyway that's still brakes the socket or D-Bus, so maybe using
* ev_loop() would be even better.
*/
event_handle(event);
p_delete(&event);
awesome_refresh();
continue;
}
xcb_selection_notify_event_t *event_notify =
(xcb_selection_notify_event_t *) event;
if(event_notify->selection == XCB_ATOM_PRIMARY
&& event_notify->property != XCB_NONE)
{
xcb_icccm_get_text_property_reply_t prop;
xcb_get_property_cookie_t cookie =
xcb_icccm_get_text_property(globalconf.connection,
event_notify->requestor,
event_notify->property);
if(xcb_icccm_get_text_property_reply(globalconf.connection,
cookie, &prop, NULL))
{
lua_pushlstring(L, prop.name, prop.name_len);
xcb_icccm_get_text_property_reply_wipe(&prop);
xcb_delete_property(globalconf.connection,
event_notify->requestor,
event_notify->property);
p_delete(&event);
return 1;
}
else
break;
}
}
p_delete(&event);
return 0;
}
static void
move_global_to_table(lua_State *L, int index, const char *global_name, const char *local_name)
{
index = luaA_absindex(L, index);
/* Get the global */
lua_getglobal(L, global_name);
assert(!lua_isnil(L, -1));
/* Save it locally */
lua_setfield(L, index, local_name);
/* Set the global to nil */
lua_pushnil(L);
lua_setglobal(L, global_name);
}
void
selection_setup(lua_State *L)
{
/* This table will be the "selection" global */
lua_newtable(L);
/* Setup a metatable */
lua_newtable(L);
/* metatable.__index = metatable */
lua_pushvalue(L, -1);
lua_setfield(L, -2, "__index");
/* Set some more fields */
lua_pushcfunction(L, luaA_selection_get);
lua_setfield(L, -2, "__call");
move_global_to_table(L, -2, "selection_acquire", "acquire");
move_global_to_table(L, -2, "selection_getter", "getter");
move_global_to_table(L, -2, "selection_watcher", "watcher");
/* Set the metatable */
lua_setmetatable(L, -2);
/* Set the "selection" global */
lua_setglobal(L, "selection");
}
// vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80