Implement INCR transfers
Incremental transfers are required to be supported. This commit adds that necessary support to awesomeWM. Signed-off-by: Uli Schlachter <psychon@znc.in>
This commit is contained in:
parent
8ad29950f5
commit
0d2c6fc5d1
|
@ -64,5 +64,6 @@ XSEL_DATA
|
||||||
WM_TAKE_FOCUS
|
WM_TAKE_FOCUS
|
||||||
AWESOME_CLIENT_ORDER
|
AWESOME_CLIENT_ORDER
|
||||||
AWESOME_SELECTION_ATOM
|
AWESOME_SELECTION_ATOM
|
||||||
|
INCR
|
||||||
_XKB_RULES_NAMES
|
_XKB_RULES_NAMES
|
||||||
_MOTIF_WM_HINTS
|
_MOTIF_WM_HINTS
|
||||||
|
|
|
@ -89,6 +89,22 @@ luaA_selection_getter_new(lua_State *L)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
selection_transfer_finished(lua_State *L, int ud)
|
||||||
|
{
|
||||||
|
selection_getter_t *selection = lua_touserdata(L, ud);
|
||||||
|
|
||||||
|
/* Unreference the selection object; it's dead */
|
||||||
|
lua_pushliteral(L, REGISTRY_GETTER_TABLE_INDEX);
|
||||||
|
lua_rawget(L, LUA_REGISTRYINDEX);
|
||||||
|
luaL_unref(L, -1, selection->ref);
|
||||||
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
selection->ref = LUA_NOREF;
|
||||||
|
|
||||||
|
luaA_object_emit_signal(L, ud, "data_end", 0);
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
selection_handle_selectionnotify(lua_State *L, int ud, xcb_atom_t property)
|
selection_handle_selectionnotify(lua_State *L, int ud, xcb_atom_t property)
|
||||||
{
|
{
|
||||||
|
@ -99,34 +115,37 @@ selection_handle_selectionnotify(lua_State *L, int ud, xcb_atom_t property)
|
||||||
|
|
||||||
if (property != XCB_NONE)
|
if (property != XCB_NONE)
|
||||||
{
|
{
|
||||||
|
xcb_change_window_attributes(globalconf.connection, selection->window,
|
||||||
|
XCB_CW_EVENT_MASK, (uint32_t[]) { XCB_EVENT_MASK_PROPERTY_CHANGE });
|
||||||
|
|
||||||
xcb_get_property_reply_t *property_r = xcb_get_property_reply(globalconf.connection,
|
xcb_get_property_reply_t *property_r = xcb_get_property_reply(globalconf.connection,
|
||||||
xcb_get_property(globalconf.connection, true, selection->window, AWESOME_SELECTION_ATOM,
|
xcb_get_property(globalconf.connection, true, selection->window, AWESOME_SELECTION_ATOM,
|
||||||
XCB_GET_PROPERTY_TYPE_ANY, 0, 0xffffffff), NULL);
|
XCB_GET_PROPERTY_TYPE_ANY, 0, 0xffffffff), NULL);
|
||||||
|
|
||||||
if (property_r)
|
if (property_r)
|
||||||
{
|
{
|
||||||
|
if (property_r->type == INCR)
|
||||||
|
{
|
||||||
|
/* This is an incremental transfer. The above GetProperty had
|
||||||
|
* delete=true. This indicates to the other end that the
|
||||||
|
* transfer should start now. Right now we only get an estimate
|
||||||
|
* of the size of the data to be transferred, which we ignore.
|
||||||
|
*/
|
||||||
|
p_delete(&property_r);
|
||||||
|
return;
|
||||||
|
}
|
||||||
lua_pushlstring(L, xcb_get_property_value(property_r), xcb_get_property_value_length(property_r));
|
lua_pushlstring(L, xcb_get_property_value(property_r), xcb_get_property_value_length(property_r));
|
||||||
luaA_object_emit_signal(L, ud, "data", 1);
|
luaA_object_emit_signal(L, ud, "data", 1);
|
||||||
p_delete(&property_r);
|
p_delete(&property_r);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
luaA_object_emit_signal(L, ud, "data_end", 0);
|
selection_transfer_finished(L, ud);
|
||||||
|
|
||||||
/* Now unreference the selection object; it's dead */
|
|
||||||
lua_pushliteral(L, REGISTRY_GETTER_TABLE_INDEX);
|
|
||||||
lua_rawget(L, LUA_REGISTRYINDEX);
|
|
||||||
luaL_unref(L, -1, selection->ref);
|
|
||||||
lua_pop(L, 1);
|
|
||||||
|
|
||||||
selection->ref = LUA_NOREF;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
static int
|
||||||
event_handle_selectionnotify(xcb_selection_notify_event_t *ev)
|
selection_getter_find_by_window(lua_State *L, xcb_window_t window)
|
||||||
{
|
{
|
||||||
lua_State *L = globalconf_get_lua_State();
|
|
||||||
|
|
||||||
/* Iterate over all active selection getters */
|
/* Iterate over all active selection getters */
|
||||||
lua_pushliteral(L, REGISTRY_GETTER_TABLE_INDEX);
|
lua_pushliteral(L, REGISTRY_GETTER_TABLE_INDEX);
|
||||||
lua_rawget(L, LUA_REGISTRYINDEX);
|
lua_rawget(L, LUA_REGISTRYINDEX);
|
||||||
|
@ -134,11 +153,11 @@ event_handle_selectionnotify(xcb_selection_notify_event_t *ev)
|
||||||
while (lua_next(L, -2) != 0) {
|
while (lua_next(L, -2) != 0) {
|
||||||
if (lua_type(L, -1) == LUA_TUSERDATA) {
|
if (lua_type(L, -1) == LUA_TUSERDATA) {
|
||||||
selection_getter_t *selection = lua_touserdata(L, -1);
|
selection_getter_t *selection = lua_touserdata(L, -1);
|
||||||
if (ev->requestor == selection->window) {
|
if (selection->window == window) {
|
||||||
/* Found the right selection */
|
/* Found the right selection, remove table and key */
|
||||||
selection_handle_selectionnotify(L, -1, ev->property);
|
lua_remove(L, -2);
|
||||||
lua_pop(L, 2);
|
lua_remove(L, -2);
|
||||||
break;
|
return 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/* Remove the value, leaving only the key */
|
/* Remove the value, leaving only the key */
|
||||||
|
@ -146,6 +165,57 @@ event_handle_selectionnotify(xcb_selection_notify_event_t *ev)
|
||||||
}
|
}
|
||||||
/* Remove the getter table */
|
/* Remove the getter table */
|
||||||
lua_pop(L, 1);
|
lua_pop(L, 1);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
property_handle_awesome_selection_atom(uint8_t state, xcb_window_t window)
|
||||||
|
{
|
||||||
|
lua_State *L = globalconf_get_lua_State();
|
||||||
|
|
||||||
|
if (state != XCB_PROPERTY_NEW_VALUE)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (selection_getter_find_by_window(L, window) == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
selection_getter_t *selection = lua_touserdata(L, -1);
|
||||||
|
|
||||||
|
xcb_get_property_reply_t *property_r = xcb_get_property_reply(globalconf.connection,
|
||||||
|
xcb_get_property(globalconf.connection, true, selection->window, AWESOME_SELECTION_ATOM,
|
||||||
|
XCB_GET_PROPERTY_TYPE_ANY, 0, 0xffffffff), NULL);
|
||||||
|
|
||||||
|
if (property_r)
|
||||||
|
{
|
||||||
|
if (property_r->value_len > 0)
|
||||||
|
{
|
||||||
|
lua_pushlstring(L, xcb_get_property_value(property_r), xcb_get_property_value_length(property_r));
|
||||||
|
luaA_object_emit_signal(L, -2, "data", 1);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Transfer finished */
|
||||||
|
selection_transfer_finished(L, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
p_delete(&property_r);
|
||||||
|
}
|
||||||
|
|
||||||
|
lua_pop(L, 1);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
event_handle_selectionnotify(xcb_selection_notify_event_t *ev)
|
||||||
|
{
|
||||||
|
lua_State *L = globalconf_get_lua_State();
|
||||||
|
|
||||||
|
if (selection_getter_find_by_window(L, ev->requestor) == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
selection_handle_selectionnotify(L, -1, ev->property);
|
||||||
|
lua_pop(L, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
|
|
@ -27,6 +27,7 @@
|
||||||
|
|
||||||
void selection_getter_class_setup(lua_State*);
|
void selection_getter_class_setup(lua_State*);
|
||||||
void event_handle_selectionnotify(xcb_selection_notify_event_t*);
|
void event_handle_selectionnotify(xcb_selection_notify_event_t*);
|
||||||
|
int property_handle_awesome_selection_atom(uint8_t, xcb_window_t);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
// vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80
|
// vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80
|
||||||
|
|
|
@ -25,6 +25,7 @@
|
||||||
#include "ewmh.h"
|
#include "ewmh.h"
|
||||||
#include "objects/client.h"
|
#include "objects/client.h"
|
||||||
#include "objects/drawin.h"
|
#include "objects/drawin.h"
|
||||||
|
#include "objects/selection_getter.h"
|
||||||
#include "xwindow.h"
|
#include "xwindow.h"
|
||||||
|
|
||||||
#include <xcb/xcb_atom.h>
|
#include <xcb/xcb_atom.h>
|
||||||
|
@ -524,6 +525,9 @@ property_handle_propertynotify(xcb_property_notify_event_t *ev)
|
||||||
/* background change */
|
/* background change */
|
||||||
HANDLE(_XROOTPMAP_ID, property_handle_xrootpmap_id)
|
HANDLE(_XROOTPMAP_ID, property_handle_xrootpmap_id)
|
||||||
|
|
||||||
|
/* selection transfers */
|
||||||
|
HANDLE(AWESOME_SELECTION_ATOM, property_handle_awesome_selection_atom)
|
||||||
|
|
||||||
/* If nothing was found, return */
|
/* If nothing was found, return */
|
||||||
END;
|
END;
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue