Implement INCR mechanism for Linux clipboard
Allows pasting from x11 clipboard to receive data incrementally, which is required when handling data size > 256KB.
This commit is contained in:
parent
d395f70828
commit
6b97901d4d
2 changed files with 105 additions and 25 deletions
|
@ -465,59 +465,138 @@ Bool DisplayServerX11::_predicate_clipboard_selection(Display *display, XEvent *
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Bool DisplayServerX11::_predicate_clipboard_incr(Display *display, XEvent *event, XPointer arg) {
|
||||||
|
if (event->type == PropertyNotify && event->xproperty.state == PropertyNewValue) {
|
||||||
|
return True;
|
||||||
|
} else {
|
||||||
|
return False;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
String DisplayServerX11::_clipboard_get_impl(Atom p_source, Window x11_window, Atom target) const {
|
String DisplayServerX11::_clipboard_get_impl(Atom p_source, Window x11_window, Atom target) const {
|
||||||
String ret;
|
String ret;
|
||||||
|
|
||||||
Atom type;
|
|
||||||
Atom selection = XA_PRIMARY;
|
|
||||||
int format, result;
|
|
||||||
unsigned long len, bytes_left, dummy;
|
|
||||||
unsigned char *data;
|
|
||||||
Window selection_owner = XGetSelectionOwner(x11_display, p_source);
|
Window selection_owner = XGetSelectionOwner(x11_display, p_source);
|
||||||
|
|
||||||
if (selection_owner == x11_window) {
|
if (selection_owner == x11_window) {
|
||||||
return internal_clipboard;
|
return internal_clipboard;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (selection_owner != None) {
|
if (selection_owner != None) {
|
||||||
{
|
// Block events polling while processing selection events.
|
||||||
// Block events polling while processing selection events.
|
MutexLock mutex_lock(events_mutex);
|
||||||
MutexLock mutex_lock(events_mutex);
|
|
||||||
|
|
||||||
XConvertSelection(x11_display, p_source, target, selection,
|
Atom selection = XA_PRIMARY;
|
||||||
x11_window, CurrentTime);
|
XConvertSelection(x11_display, p_source, target, selection,
|
||||||
|
x11_window, CurrentTime);
|
||||||
|
|
||||||
XFlush(x11_display);
|
XFlush(x11_display);
|
||||||
|
|
||||||
// Blocking wait for predicate to be True
|
// Blocking wait for predicate to be True and remove the event from the queue.
|
||||||
// and remove the event from the queue.
|
XEvent event;
|
||||||
XEvent event;
|
XIfEvent(x11_display, &event, _predicate_clipboard_selection, (XPointer)&x11_window);
|
||||||
XIfEvent(x11_display, &event, _predicate_clipboard_selection, (XPointer)&x11_window);
|
|
||||||
}
|
|
||||||
|
|
||||||
//
|
// Do not get any data, see how much data is there.
|
||||||
// Do not get any data, see how much data is there
|
Atom type;
|
||||||
//
|
int format, result;
|
||||||
|
unsigned long len, bytes_left, dummy;
|
||||||
|
unsigned char *data;
|
||||||
XGetWindowProperty(x11_display, x11_window,
|
XGetWindowProperty(x11_display, x11_window,
|
||||||
selection, // Tricky..
|
selection, // Tricky..
|
||||||
0, 0, // offset - len
|
0, 0, // offset - len
|
||||||
0, // Delete 0==FALSE
|
0, // Delete 0==FALSE
|
||||||
AnyPropertyType, //flag
|
AnyPropertyType, // flag
|
||||||
&type, // return type
|
&type, // return type
|
||||||
&format, // return format
|
&format, // return format
|
||||||
&len, &bytes_left, //that
|
&len, &bytes_left, // data length
|
||||||
&data);
|
&data);
|
||||||
// DATA is There
|
|
||||||
if (bytes_left > 0) {
|
if (data) {
|
||||||
|
XFree(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type == XInternAtom(x11_display, "INCR", 0)) {
|
||||||
|
// Data is going to be received incrementally.
|
||||||
|
DEBUG_LOG_X11("INCR selection started.\n");
|
||||||
|
|
||||||
|
LocalVector<uint8_t> incr_data;
|
||||||
|
uint32_t data_size = 0;
|
||||||
|
bool success = false;
|
||||||
|
|
||||||
|
// Delete INCR property to notify the owner.
|
||||||
|
XDeleteProperty(x11_display, x11_window, type);
|
||||||
|
|
||||||
|
// Process events from the queue.
|
||||||
|
bool done = false;
|
||||||
|
while (!done) {
|
||||||
|
if (!_wait_for_events()) {
|
||||||
|
// Error or timeout, abort.
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Non-blocking wait for next event and remove it from the queue.
|
||||||
|
XEvent ev;
|
||||||
|
while (XCheckIfEvent(x11_display, &ev, _predicate_clipboard_incr, nullptr)) {
|
||||||
|
result = XGetWindowProperty(x11_display, x11_window,
|
||||||
|
selection, // selection type
|
||||||
|
0, LONG_MAX, // offset - len
|
||||||
|
True, // delete property to notify the owner
|
||||||
|
AnyPropertyType, // flag
|
||||||
|
&type, // return type
|
||||||
|
&format, // return format
|
||||||
|
&len, &bytes_left, // data length
|
||||||
|
&data);
|
||||||
|
|
||||||
|
DEBUG_LOG_X11("PropertyNotify: len=%lu, format=%i\n", len, format);
|
||||||
|
|
||||||
|
if (result == Success) {
|
||||||
|
if (data && (len > 0)) {
|
||||||
|
uint32_t prev_size = incr_data.size();
|
||||||
|
if (prev_size == 0) {
|
||||||
|
// First property contains initial data size.
|
||||||
|
unsigned long initial_size = *(unsigned long *)data;
|
||||||
|
incr_data.resize(initial_size);
|
||||||
|
} else {
|
||||||
|
// New chunk, resize to be safe and append data.
|
||||||
|
incr_data.resize(MAX(data_size + len, prev_size));
|
||||||
|
memcpy(incr_data.ptr() + data_size, data, len);
|
||||||
|
data_size += len;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Last chunk, process finished.
|
||||||
|
done = true;
|
||||||
|
success = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
printf("Failed to get selection data chunk.\n");
|
||||||
|
done = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (data) {
|
||||||
|
XFree(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (done) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (success && (data_size > 0)) {
|
||||||
|
ret.parse_utf8((const char *)incr_data.ptr(), data_size);
|
||||||
|
}
|
||||||
|
} else if (bytes_left > 0) {
|
||||||
|
// Data is ready and can be processed all at once.
|
||||||
result = XGetWindowProperty(x11_display, x11_window,
|
result = XGetWindowProperty(x11_display, x11_window,
|
||||||
selection, 0, bytes_left, 0,
|
selection, 0, bytes_left, 0,
|
||||||
AnyPropertyType, &type, &format,
|
AnyPropertyType, &type, &format,
|
||||||
&len, &dummy, &data);
|
&len, &dummy, &data);
|
||||||
|
|
||||||
if (result == Success) {
|
if (result == Success) {
|
||||||
ret.parse_utf8((const char *)data);
|
ret.parse_utf8((const char *)data);
|
||||||
} else {
|
} else {
|
||||||
printf("FAIL\n");
|
printf("Failed to get selection data.\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (data) {
|
if (data) {
|
||||||
XFree(data);
|
XFree(data);
|
||||||
}
|
}
|
||||||
|
|
|
@ -270,6 +270,7 @@ class DisplayServerX11 : public DisplayServer {
|
||||||
|
|
||||||
static Bool _predicate_all_events(Display *display, XEvent *event, XPointer arg);
|
static Bool _predicate_all_events(Display *display, XEvent *event, XPointer arg);
|
||||||
static Bool _predicate_clipboard_selection(Display *display, XEvent *event, XPointer arg);
|
static Bool _predicate_clipboard_selection(Display *display, XEvent *event, XPointer arg);
|
||||||
|
static Bool _predicate_clipboard_incr(Display *display, XEvent *event, XPointer arg);
|
||||||
static Bool _predicate_clipboard_save_targets(Display *display, XEvent *event, XPointer arg);
|
static Bool _predicate_clipboard_save_targets(Display *display, XEvent *event, XPointer arg);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
Loading…
Reference in a new issue