/* * 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 <julien@danjou.info> * @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