2 * FreeRDP: A Remote Desktop Protocol Client
3 * X11 Clipboard Redirection
5 * Copyright 2010-2011 Vic Lee
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this file except in compliance with the License.
9 * You may obtain a copy of the License at
11 * http://www.apache.org/licenses/LICENSE-2.0
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
22 #include <X11/Xatom.h>
23 #include <freerdp/utils/event.h>
24 #include <freerdp/utils/stream.h>
25 #include <freerdp/utils/unicode.h>
26 #include <freerdp/plugins/cliprdr.h>
28 #include "xf_cliprdr.h"
30 typedef struct clipboard_format_mapping clipboardFormatMapping;
31 struct clipboard_format_mapping
37 typedef struct clipboard_context clipboardContext;
38 struct clipboard_context
40 rdpChannels* channels;
46 clipboardFormatMapping format_mappings[20];
47 int num_format_mappings;
49 /* server->client data */
56 uint32 data_alt_format;
60 /* client->server data */
72 void xf_cliprdr_init(xfInfo* xfi, rdpChannels* chanman)
78 cb = xnew(clipboardContext);
79 xfi->clipboard_context = cb;
81 cb->channels = chanman;
82 cb->request_index = -1;
84 cb->root_window = DefaultRootWindow(xfi->display);
85 cb->clipboard_atom = XInternAtom(xfi->display, "CLIPBOARD", false);
87 if (cb->clipboard_atom == None)
89 DEBUG_WARN("unable to get CLIPBOARD atom");
93 cb->property_atom = XInternAtom(xfi->display, "_FREERDP_CLIPRDR", false);
94 cb->identity_atom = XInternAtom(xfi->display, "_FREERDP_CLIPRDR_ID", false);
96 XChangeProperty(xfi->display, xfi->drawable, cb->identity_atom,
97 XA_INTEGER, 32, PropModeReplace, (uint8*) &id, 1);
99 XSelectInput(xfi->display, cb->root_window, PropertyChangeMask);
102 cb->format_mappings[n].target_format = XInternAtom(xfi->display, "_FREERDP_RAW", false);
103 cb->format_mappings[n].format_id = CB_FORMAT_RAW;
106 cb->format_mappings[n].target_format = XInternAtom(xfi->display, "UTF8_STRING", false);
107 cb->format_mappings[n].format_id = CB_FORMAT_UNICODETEXT;
110 cb->format_mappings[n].target_format = XA_STRING;
111 cb->format_mappings[n].format_id = CB_FORMAT_TEXT;
114 cb->format_mappings[n].target_format = XInternAtom(xfi->display, "image/png", false);
115 cb->format_mappings[n].format_id = CB_FORMAT_PNG;
118 cb->format_mappings[n].target_format = XInternAtom(xfi->display, "image/jpeg", false);
119 cb->format_mappings[n].format_id = CB_FORMAT_JPEG;
122 cb->format_mappings[n].target_format = XInternAtom(xfi->display, "image/gif", false);
123 cb->format_mappings[n].format_id = CB_FORMAT_GIF;
126 cb->format_mappings[n].target_format = XInternAtom(xfi->display, "image/bmp", false);
127 cb->format_mappings[n].format_id = CB_FORMAT_DIB;
130 cb->format_mappings[n].target_format = XInternAtom(xfi->display, "text/html", false);
131 cb->format_mappings[n].format_id = CB_FORMAT_HTML;
133 cb->num_format_mappings = n + 1;
134 cb->targets[0] = XInternAtom(xfi->display, "TIMESTAMP", false);
135 cb->targets[1] = XInternAtom(xfi->display, "TARGETS", false);
138 cb->incr_atom = XInternAtom(xfi->display, "INCR", false);
141 void xf_cliprdr_uninit(xfInfo* xfi)
143 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
150 xfree(cb->incr_data);
152 xfi->clipboard_context = NULL;
156 static uint8* lf2crlf(uint8* data, int* size)
165 out_size = (*size) * 2 + 1;
166 outbuf = (uint8*) xzalloc(out_size);
169 in_end = data + (*size);
186 *size = out - outbuf;
191 static void crlf2lf(uint8* data, int* size)
200 in_end = data + (*size);
213 static void be2le(uint8* data, int size)
228 static boolean xf_cliprdr_is_self_owned(xfInfo* xfi)
233 int format, result = 0;
234 unsigned long length, bytes_left;
235 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
237 cb->owner = XGetSelectionOwner(xfi->display, cb->clipboard_atom);
239 if (cb->owner != None)
241 result = XGetWindowProperty(xfi->display, cb->owner,
242 cb->identity_atom, 0, 4, 0, XA_INTEGER,
243 &type, &format, &length, &bytes_left, (uint8**) &pid);
252 if ((cb->owner == None) || (cb->owner == xfi->drawable))
255 if (result != Success)
258 return (id ? true : false);
261 static int xf_cliprdr_select_format_by_id(clipboardContext* cb, uint32 format_id)
265 for (i = 0; i < cb->num_format_mappings; i++)
267 if (cb->format_mappings[i].format_id == format_id)
274 static int xf_cliprdr_select_format_by_atom(clipboardContext* cb, Atom target)
279 if (cb->formats == NULL)
282 for (i = 0; i < cb->num_format_mappings; i++)
284 if (cb->format_mappings[i].target_format != target)
287 if (cb->format_mappings[i].format_id == CB_FORMAT_RAW)
290 for (j = 0; j < cb->num_formats; j++)
292 if (cb->format_mappings[i].format_id == cb->formats[j])
300 static void xf_cliprdr_send_raw_format_list(xfInfo* xfi)
305 unsigned long length, bytes_left;
306 RDP_CB_FORMAT_LIST_EVENT* event;
307 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
309 result = XGetWindowProperty(xfi->display, cb->root_window,
310 cb->property_atom, 0, 3600, 0, XA_STRING,
311 &type, &format, &length, &bytes_left, (uint8**) &format_data);
313 if (result != Success)
315 DEBUG_WARN("XGetWindowProperty failed");
318 DEBUG_X11_CLIPRDR("format=%d len=%d bytes_left=%d", format, (int) length, (int) bytes_left);
320 event = (RDP_CB_FORMAT_LIST_EVENT*) freerdp_event_new(RDP_EVENT_CLASS_CLIPRDR,
321 RDP_EVENT_TYPE_CB_FORMAT_LIST, NULL, NULL);
323 event->raw_format_data = (uint8*) xmalloc(length);
324 memcpy(event->raw_format_data, format_data, length);
325 event->raw_format_data_size = length;
328 freerdp_channels_send_event(cb->channels, (RDP_EVENT*) event);
331 static void xf_cliprdr_send_null_format_list(xfInfo* xfi)
333 RDP_CB_FORMAT_LIST_EVENT* event;
334 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
336 event = (RDP_CB_FORMAT_LIST_EVENT*) freerdp_event_new(RDP_EVENT_CLASS_CLIPRDR,
337 RDP_EVENT_TYPE_CB_FORMAT_LIST, NULL, NULL);
339 event->num_formats = 0;
341 freerdp_channels_send_event(cb->channels, (RDP_EVENT*) event);
344 static void xf_cliprdr_send_supported_format_list(xfInfo* xfi)
347 RDP_CB_FORMAT_LIST_EVENT* event;
348 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
350 event = (RDP_CB_FORMAT_LIST_EVENT*) freerdp_event_new(RDP_EVENT_CLASS_CLIPRDR,
351 RDP_EVENT_TYPE_CB_FORMAT_LIST, NULL, NULL);
353 event->formats = (uint32*) xmalloc(sizeof(uint32) * cb->num_format_mappings);
354 event->num_formats = cb->num_format_mappings;
356 for (i = 0; i < cb->num_format_mappings; i++)
357 event->formats[i] = cb->format_mappings[i].format_id;
359 freerdp_channels_send_event(cb->channels, (RDP_EVENT*) event);
362 static void xf_cliprdr_send_format_list(xfInfo* xfi)
364 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
366 if (xf_cliprdr_is_self_owned(xfi))
368 xf_cliprdr_send_raw_format_list(xfi);
370 else if (cb->owner == None)
372 xf_cliprdr_send_null_format_list(xfi);
374 else if (cb->owner != xfi->drawable)
376 /* Request the owner for TARGETS, and wait for SelectionNotify event */
377 XConvertSelection(xfi->display, cb->clipboard_atom,
378 cb->targets[1], cb->property_atom, xfi->drawable, CurrentTime);
382 static void xf_cliprdr_send_data_request(xfInfo* xfi, uint32 format)
384 RDP_CB_DATA_REQUEST_EVENT* event;
385 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
387 event = (RDP_CB_DATA_REQUEST_EVENT*) freerdp_event_new(RDP_EVENT_CLASS_CLIPRDR,
388 RDP_EVENT_TYPE_CB_DATA_REQUEST, NULL, NULL);
390 event->format = format;
392 freerdp_channels_send_event(cb->channels, (RDP_EVENT*) event);
395 static void xf_cliprdr_send_data_response(xfInfo* xfi, uint8* data, int size)
397 RDP_CB_DATA_RESPONSE_EVENT* event;
398 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
400 event = (RDP_CB_DATA_RESPONSE_EVENT*) freerdp_event_new(RDP_EVENT_CLASS_CLIPRDR,
401 RDP_EVENT_TYPE_CB_DATA_RESPONSE, NULL, NULL);
406 freerdp_channels_send_event(cb->channels, (RDP_EVENT*) event);
409 static void xf_cliprdr_send_null_data_response(xfInfo* xfi)
411 xf_cliprdr_send_data_response(xfi, NULL, 0);
414 static void xf_cliprdr_process_cb_monitor_ready_event(xfInfo* xfi)
416 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
418 xf_cliprdr_send_format_list(xfi);
422 static void xf_cliprdr_process_cb_data_request_event(xfInfo* xfi, RDP_CB_DATA_REQUEST_EVENT* event)
425 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
427 DEBUG_X11_CLIPRDR("format %d", event->format);
429 if (xf_cliprdr_is_self_owned(xfi))
433 XChangeProperty(xfi->display, xfi->drawable, cb->property_atom,
434 XA_INTEGER, 32, PropModeReplace, (uint8*) &event->format, 1);
438 i = xf_cliprdr_select_format_by_id(cb, event->format);
443 DEBUG_X11_CLIPRDR("unsupported format requested");
444 xf_cliprdr_send_null_data_response(xfi);
448 cb->request_index = i;
450 DEBUG_X11_CLIPRDR("target=%d", (int) cb->format_mappings[i].target_format);
452 XConvertSelection(xfi->display, cb->clipboard_atom,
453 cb->format_mappings[i].target_format, cb->property_atom,
454 xfi->drawable, CurrentTime);
455 XFlush(xfi->display);
456 /* After this point, we expect a SelectionNotify event from the clipboard owner. */
460 static void xf_cliprdr_get_requested_targets(xfInfo* xfi)
467 unsigned long length, bytes_left;
468 RDP_CB_FORMAT_LIST_EVENT* event;
469 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
471 XGetWindowProperty(xfi->display, xfi->drawable, cb->property_atom,
473 &atom, &format, &length, &bytes_left, &data);
475 DEBUG_X11_CLIPRDR("type=%d format=%d length=%d bytes_left=%d",
476 (int) atom, format, (int) length, (int) bytes_left);
480 event = (RDP_CB_FORMAT_LIST_EVENT*) freerdp_event_new(RDP_EVENT_CLASS_CLIPRDR,
481 RDP_EVENT_TYPE_CB_FORMAT_LIST, NULL, NULL);
483 event->formats = (uint32*) xmalloc(sizeof(uint32) * cb->num_format_mappings);
485 for (i = 0; i < length; i++)
487 atom = ((Atom*) data)[i];
488 DEBUG_X11("atom %d", (int) atom);
489 for (j = 0; j < cb->num_format_mappings; j++)
491 if (cb->format_mappings[j].target_format == atom)
493 DEBUG_X11("found format %d for atom %d",
494 cb->format_mappings[j].format_id, (int)atom);
495 event->formats[num++] = cb->format_mappings[j].format_id;
500 event->num_formats = num;
503 freerdp_channels_send_event(cb->channels, (RDP_EVENT*) event);
510 xf_cliprdr_send_null_format_list(xfi);
514 static uint8* xf_cliprdr_process_requested_raw(uint8* data, int* size)
518 outbuf = (uint8*) xmalloc(*size);
519 memcpy(outbuf, data, *size);
523 static uint8* xf_cliprdr_process_requested_unicodetext(uint8* data, int* size)
530 inbuf = lf2crlf(data, size);
532 uniconv = freerdp_uniconv_new();
533 outbuf = (uint8*) freerdp_uniconv_out(uniconv, (char*) inbuf, &out_size);
534 freerdp_uniconv_free(uniconv);
538 *size = (int) out_size + 2;
543 static uint8* xf_cliprdr_process_requested_text(uint8* data, int* size)
547 outbuf = lf2crlf(data, size);
552 static uint8* xf_cliprdr_process_requested_dib(uint8* data, int* size)
556 /* length should be at least BMP header (14) + sizeof(BITMAPINFOHEADER) */
559 DEBUG_X11_CLIPRDR("bmp length %d too short", *size);
564 outbuf = (uint8*) xzalloc(*size);
565 memcpy(outbuf, data + 14, *size);
570 static uint8* xf_cliprdr_process_requested_html(uint8* data, int* size)
581 if ((uint8) data[0] == 0xFE && (uint8) data[1] == 0xFF)
586 if ((uint8) data[0] == 0xFF && (uint8) data[1] == 0xFE)
588 uniconv = freerdp_uniconv_new();
589 inbuf = (uint8*) freerdp_uniconv_in(uniconv, data + 2, *size - 2);
590 freerdp_uniconv_free(uniconv);
595 inbuf = xzalloc(*size + 1);
596 memcpy(inbuf, data, *size);
599 outbuf = (uint8*) xzalloc(*size + 200);
600 strcpy((char*) outbuf,
602 "StartHTML:0000000000\r\n"
603 "EndHTML:0000000000\r\n"
604 "StartFragment:0000000000\r\n"
605 "EndFragment:0000000000\r\n");
607 in = (uint8*) strstr((char*) inbuf, "<body");
610 in = (uint8*) strstr((char*) inbuf, "<BODY");
613 snprintf(num, sizeof(num), "%010lu", (unsigned long) strlen((char*) outbuf));
614 memcpy(outbuf + 23, num, 10);
617 strcat((char*) outbuf, "<HTML><BODY>");
619 strcat((char*) outbuf, "<!--StartFragment-->");
621 snprintf(num, sizeof(num), "%010lu", (unsigned long) strlen((char*) outbuf));
622 memcpy(outbuf + 69, num, 10);
623 strcat((char*) outbuf, (char*) inbuf);
625 snprintf(num, sizeof(num), "%010lu", (unsigned long) strlen((char*) outbuf));
626 memcpy(outbuf + 93, num, 10);
627 strcat((char*) outbuf, "<!--EndFragment-->");
630 strcat((char*) outbuf, "</BODY></HTML>");
633 snprintf(num, sizeof(num), "%010lu", (unsigned long) strlen((char*) outbuf));
634 memcpy(outbuf + 43, num, 10);
636 *size = strlen((char*) outbuf) + 1;
642 static void xf_cliprdr_process_requested_data(xfInfo* xfi, boolean has_data, uint8* data, int size)
645 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
647 if (cb->incr_starts && has_data)
650 if (!has_data || data == NULL)
652 xf_cliprdr_send_null_data_response(xfi);
656 switch (cb->format_mappings[cb->request_index].format_id)
662 outbuf = xf_cliprdr_process_requested_raw(data, &size);
665 case CB_FORMAT_UNICODETEXT:
666 outbuf = xf_cliprdr_process_requested_unicodetext(data, &size);
670 outbuf = xf_cliprdr_process_requested_text(data, &size);
674 outbuf = xf_cliprdr_process_requested_dib(data, &size);
678 outbuf = xf_cliprdr_process_requested_html(data, &size);
687 xf_cliprdr_send_data_response(xfi, outbuf, size);
689 xf_cliprdr_send_null_data_response(xfi);
691 /* Resend the format list, otherwise the server won't request again for the next paste */
692 xf_cliprdr_send_format_list(xfi);
695 static boolean xf_cliprdr_get_requested_data(xfInfo* xfi, Atom target)
700 boolean has_data = false;
701 unsigned long length, bytes_left, dummy;
702 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
704 if ((cb->request_index < 0) ||
705 (cb->format_mappings[cb->request_index].target_format != target))
707 DEBUG_X11_CLIPRDR("invalid target");
708 xf_cliprdr_send_null_data_response(xfi);
712 XGetWindowProperty(xfi->display, xfi->drawable,
713 cb->property_atom, 0, 0, 0, target,
714 &type, &format, &length, &bytes_left, &data);
716 DEBUG_X11_CLIPRDR("type=%d format=%d bytes=%d request_index=%d",
717 (int) type, format, (int) bytes_left, cb->request_index);
724 if (bytes_left <= 0 && !cb->incr_starts)
726 DEBUG_X11("no data");
728 else if (type == cb->incr_atom)
730 DEBUG_X11("INCR started");
731 cb->incr_starts = true;
734 xfree(cb->incr_data);
735 cb->incr_data = NULL;
737 cb->incr_data_length = 0;
738 /* Data will be followed in PropertyNotify event */
746 data = cb->incr_data;
747 cb->incr_data = NULL;
748 bytes_left = cb->incr_data_length;
749 cb->incr_data_length = 0;
751 DEBUG_X11("INCR finished");
754 else if (XGetWindowProperty(xfi->display, xfi->drawable,
755 cb->property_atom, 0, bytes_left, 0, target,
756 &type, &format, &length, &dummy, &data) == Success)
760 bytes_left = length * format / 8;
761 DEBUG_X11("%d bytes", (int)bytes_left);
762 cb->incr_data = (uint8*) xrealloc(cb->incr_data, cb->incr_data_length + bytes_left);
763 memcpy(cb->incr_data + cb->incr_data_length, data, bytes_left);
764 cb->incr_data_length += bytes_left;
772 DEBUG_X11_CLIPRDR("XGetWindowProperty failed");
775 XDeleteProperty(xfi->display, xfi->drawable, cb->property_atom);
777 xf_cliprdr_process_requested_data(xfi, has_data, data, (int) bytes_left);
785 static void xf_cliprdr_append_target(clipboardContext* cb, Atom target)
789 if (cb->num_targets >= sizeof(cb->targets) / sizeof(Atom))
792 for (i = 0; i < cb->num_targets; i++)
794 if (cb->targets[i] == target)
798 cb->targets[cb->num_targets++] = target;
801 static void xf_cliprdr_provide_targets(xfInfo* xfi, XEvent* respond)
803 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
805 if (respond->xselection.property != None)
807 XChangeProperty(xfi->display,
808 respond->xselection.requestor,
809 respond->xselection.property,
810 XA_ATOM, 32, PropModeReplace,
811 (uint8*) cb->targets, cb->num_targets);
815 static void xf_cliprdr_provide_data(xfInfo* xfi, XEvent* respond)
817 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
819 if (respond->xselection.property != None)
821 XChangeProperty(xfi->display,
822 respond->xselection.requestor,
823 respond->xselection.property,
824 respond->xselection.target, 8, PropModeReplace,
825 (uint8*) cb->data, cb->data_length);
829 static void xf_cliprdr_process_cb_format_list_event(xfInfo* xfi, RDP_CB_FORMAT_LIST_EVENT* event)
832 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
843 cb->formats = event->formats;
844 cb->num_formats = event->num_formats;
845 event->formats = NULL;
846 event->num_formats = 0;
849 for (i = 0; i < cb->num_formats; i++)
851 for (j = 0; j < cb->num_format_mappings; j++)
853 if (cb->formats[i] == cb->format_mappings[j].format_id)
855 DEBUG_X11("announce format#%d : %d", i, cb->formats[i]);
856 xf_cliprdr_append_target(cb, cb->format_mappings[j].target_format);
861 XSetSelectionOwner(xfi->display, cb->clipboard_atom, xfi->drawable, CurrentTime);
862 if (event->raw_format_data)
864 XChangeProperty(xfi->display, cb->root_window, cb->property_atom,
865 XA_STRING, 8, PropModeReplace,
866 event->raw_format_data, event->raw_format_data_size);
869 XFlush(xfi->display);
872 static void xf_cliprdr_process_text(clipboardContext* cb, uint8* data, int size)
874 cb->data = (uint8*) xmalloc(size);
875 memcpy(cb->data, data, size);
876 cb->data_length = size;
877 crlf2lf(cb->data, &cb->data_length);
880 static void xf_cliprdr_process_unicodetext(clipboardContext* cb, uint8* data, int size)
884 uniconv = freerdp_uniconv_new();
885 cb->data = (uint8*) freerdp_uniconv_in(uniconv, data, size);
886 freerdp_uniconv_free(uniconv);
887 cb->data_length = strlen((char*) cb->data);
888 crlf2lf(cb->data, &cb->data_length);
891 static void xf_cliprdr_process_dib(clipboardContext* cb, uint8* data, int size)
898 /* size should be at least sizeof(BITMAPINFOHEADER) */
901 DEBUG_X11_CLIPRDR("dib size %d too short", size);
906 stream_attach(s, data, size);
908 stream_read_uint16(s, bpp);
909 stream_read_uint32(s, ncolors);
910 offset = 14 + 40 + (bpp <= 8 ? (ncolors == 0 ? (1 << bpp) : ncolors) * 4 : 0);
914 DEBUG_X11_CLIPRDR("offset=%d bpp=%d ncolors=%d", offset, bpp, ncolors);
916 s = stream_new(14 + size);
917 stream_write_uint8(s, 'B');
918 stream_write_uint8(s, 'M');
919 stream_write_uint32(s, 14 + size);
920 stream_write_uint32(s, 0);
921 stream_write_uint32(s, offset);
922 stream_write(s, data, size);
924 cb->data = stream_get_head(s);
925 cb->data_length = stream_get_length(s);
930 static void xf_cliprdr_process_html(clipboardContext* cb, uint8* data, int size)
937 start_str = strstr((char*) data, "StartHTML:");
938 end_str = strstr((char*) data, "EndHTML:");
939 if (start_str == NULL || end_str == NULL)
941 DEBUG_X11_CLIPRDR("invalid HTML clipboard format");
944 start = atoi(start_str + 10);
945 end = atoi(end_str + 8);
946 if (start > size || end > size || start >= end)
948 DEBUG_X11_CLIPRDR("invalid HTML offset");
952 cb->data = (uint8*) xmalloc(size - start + 1);
953 memcpy(cb->data, data + start, end - start);
954 cb->data_length = end - start;
955 crlf2lf(cb->data, &cb->data_length);
958 static void xf_cliprdr_process_cb_data_response_event(xfInfo* xfi, RDP_CB_DATA_RESPONSE_EVENT* event)
960 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
962 DEBUG_X11_CLIPRDR("size=%d", event->size);
964 if (cb->respond == NULL)
966 DEBUG_X11_CLIPRDR("unexpected data");
970 if (event->size == 0)
972 cb->respond->xselection.property = None;
981 switch (cb->data_format)
987 cb->data = event->data;
988 cb->data_length = event->size;
994 xf_cliprdr_process_text(cb, event->data, event->size);
997 case CB_FORMAT_UNICODETEXT:
998 xf_cliprdr_process_unicodetext(cb, event->data, event->size);
1002 xf_cliprdr_process_dib(cb, event->data, event->size);
1005 case CB_FORMAT_HTML:
1006 xf_cliprdr_process_html(cb, event->data, event->size);
1010 cb->respond->xselection.property = None;
1013 xf_cliprdr_provide_data(xfi, cb->respond);
1016 XSendEvent(xfi->display, cb->respond->xselection.requestor, 0, 0, cb->respond);
1017 XFlush(xfi->display);
1022 void xf_process_cliprdr_event(xfInfo* xfi, RDP_EVENT* event)
1024 switch (event->event_type)
1026 case RDP_EVENT_TYPE_CB_MONITOR_READY:
1027 xf_cliprdr_process_cb_monitor_ready_event(xfi);
1030 case RDP_EVENT_TYPE_CB_FORMAT_LIST:
1031 xf_cliprdr_process_cb_format_list_event(xfi, (RDP_CB_FORMAT_LIST_EVENT*) event);
1034 case RDP_EVENT_TYPE_CB_DATA_REQUEST:
1035 xf_cliprdr_process_cb_data_request_event(xfi, (RDP_CB_DATA_REQUEST_EVENT*) event);
1038 case RDP_EVENT_TYPE_CB_DATA_RESPONSE:
1039 xf_cliprdr_process_cb_data_response_event(xfi, (RDP_CB_DATA_RESPONSE_EVENT*) event);
1043 DEBUG_X11_CLIPRDR("unknown event type %d", event->event_type);
1048 boolean xf_cliprdr_process_selection_notify(xfInfo* xfi, XEvent* xevent)
1050 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
1052 if (xevent->xselection.target == cb->targets[1])
1054 if (xevent->xselection.property == None)
1056 DEBUG_X11_CLIPRDR("owner not support TARGETS. sending all format.");
1057 xf_cliprdr_send_supported_format_list(xfi);
1061 xf_cliprdr_get_requested_targets(xfi);
1068 return xf_cliprdr_get_requested_data(xfi, xevent->xselection.target);
1072 boolean xf_cliprdr_process_selection_request(xfInfo* xfi, XEvent* xevent)
1081 boolean delay_respond;
1082 unsigned long length, bytes_left;
1083 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
1085 DEBUG_X11_CLIPRDR("target=%d", (int) xevent->xselectionrequest.target);
1087 if (xevent->xselectionrequest.owner != xfi->drawable)
1089 DEBUG_X11_CLIPRDR("not owner");
1093 delay_respond = false;
1094 respond = xnew(XEvent);
1095 respond->xselection.property = None;
1096 respond->xselection.type = SelectionNotify;
1097 respond->xselection.display = xevent->xselectionrequest.display;
1098 respond->xselection.requestor = xevent->xselectionrequest.requestor;
1099 respond->xselection.selection = xevent->xselectionrequest.selection;
1100 respond->xselection.target = xevent->xselectionrequest.target;
1101 respond->xselection.time = xevent->xselectionrequest.time;
1103 if (xevent->xselectionrequest.target == cb->targets[0]) /* TIMESTAMP */
1106 DEBUG_X11_CLIPRDR("target: TIMESTAMP (unimplemented)");
1108 else if (xevent->xselectionrequest.target == cb->targets[1]) /* TARGETS */
1110 /* Someone else requests our available formats */
1111 DEBUG_X11_CLIPRDR("target: TARGETS");
1112 respond->xselection.property = xevent->xselectionrequest.property;
1113 xf_cliprdr_provide_targets(xfi, respond);
1117 DEBUG_X11_CLIPRDR("target: other");
1119 i = xf_cliprdr_select_format_by_atom(cb, xevent->xselectionrequest.target);
1121 if (i >= 0 && xevent->xselectionrequest.requestor != xfi->drawable)
1123 format = cb->format_mappings[i].format_id;
1124 alt_format = format;
1125 if (format == CB_FORMAT_RAW)
1127 if (XGetWindowProperty(xfi->display, xevent->xselectionrequest.requestor,
1128 cb->property_atom, 0, 4, 0, XA_INTEGER,
1129 &type, &fmt, &length, &bytes_left, &data) != Success)
1131 DEBUG_X11_CLIPRDR("XGetWindowProperty failed");
1135 memcpy(&alt_format, data, 4);
1139 DEBUG_X11_CLIPRDR("provide format 0x%04x alt_format 0x%04x", format, alt_format);
1140 if ((cb->data != 0) && (format == cb->data_format) && (alt_format == cb->data_alt_format))
1142 /* Cached clipboard data available. Send it now */
1143 respond->xselection.property = xevent->xselectionrequest.property;
1144 xf_cliprdr_provide_data(xfi, respond);
1146 else if (cb->respond)
1148 DEBUG_X11_CLIPRDR("duplicated request");
1153 * Send clipboard data request to the server.
1154 * Response will be postponed after receiving the data
1162 respond->xselection.property = xevent->xselectionrequest.property;
1163 cb->respond = respond;
1164 cb->data_format = format;
1165 cb->data_alt_format = alt_format;
1166 delay_respond = true;
1168 xf_cliprdr_send_data_request(xfi, alt_format);
1173 if (delay_respond == false)
1175 XSendEvent(xfi->display, xevent->xselectionrequest.requestor, 0, 0, respond);
1176 XFlush(xfi->display);
1183 boolean xf_cliprdr_process_selection_clear(xfInfo* xfi, XEvent* xevent)
1185 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
1187 if (xf_cliprdr_is_self_owned(xfi))
1190 XDeleteProperty(xfi->display, cb->root_window, cb->property_atom);
1195 boolean xf_cliprdr_process_property_notify(xfInfo* xfi, XEvent* xevent)
1197 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
1199 if (xevent->xproperty.atom != cb->property_atom)
1200 return false; /* Not cliprdr-related */
1202 if (xevent->xproperty.window == cb->root_window)
1204 DEBUG_X11_CLIPRDR("root window PropertyNotify");
1205 xf_cliprdr_send_format_list(xfi);
1207 else if (xevent->xproperty.window == xfi->drawable &&
1208 xevent->xproperty.state == PropertyNewValue &&
1209 cb->incr_starts && cb->request_index >= 0)
1211 DEBUG_X11_CLIPRDR("cliprdr window PropertyNotify");
1212 xf_cliprdr_get_requested_data(xfi,
1213 cb->format_mappings[cb->request_index].target_format);
1219 void xf_cliprdr_check_owner(xfInfo* xfi)
1222 clipboardContext* cb = (clipboardContext*) xfi->clipboard_context;
1226 owner = XGetSelectionOwner(xfi->display, cb->clipboard_atom);
1228 if (cb->owner != owner)
1231 xf_cliprdr_send_format_list(xfi);