Applied patch [1] with unicode support in clipboard code from Ilya Konstantinov

[1] https://sourceforge.net/tracker/?func=detail&atid=381349&aid=1394324&group_id=24366


git-svn-id: svn://svn.code.sf.net/p/rdesktop/code/trunk/rdesktop@1036 423420c4-83ab-492f-b58f-81f9feb106b5
This commit is contained in:
Erik Forsberg 2006-01-02 15:55:59 +00:00
parent 6dc26eef0b
commit da3ab666b3

510
xclip.c
View File

@ -32,7 +32,23 @@
http://msdn.microsoft.com/library/en-us/winui/winui/windowsuserinterface/dataexchange/clipboard/clipboardformats.asp http://msdn.microsoft.com/library/en-us/winui/winui/windowsuserinterface/dataexchange/clipboard/clipboardformats.asp
*/ */
#define NUM_TARGETS 6 #ifdef HAVE_ICONV
#ifdef HAVE_LANGINFO_H
#ifdef HAVE_ICONV_H
#include <langinfo.h>
#include <iconv.h>
#define USE_UNICODE_CLIPBOARD
#endif
#endif
#endif
#ifdef USE_UNICODE_CLIPBOARD
#define RDP_CF_TEXT CF_UNICODETEXT
#else
#define RDP_CF_TEXT CF_TEXT
#endif
#define MAX_TARGETS 7
extern Display *g_display; extern Display *g_display;
extern Window g_wnd; extern Window g_wnd;
@ -63,6 +79,7 @@ static Atom rdesktop_clipboard_target_atom;
denoting all Windows native clipboard formats it offers via denoting all Windows native clipboard formats it offers via
requests of the _RDESKTOP_CLIPBOARD_FORMATS target. */ requests of the _RDESKTOP_CLIPBOARD_FORMATS target. */
static Atom rdesktop_clipboard_formats_atom; static Atom rdesktop_clipboard_formats_atom;
static Atom format_string_atom, format_utf8_string_atom, format_unicode_atom;
/* Atom of the INCR clipboard type (see ICCCM on "INCR Properties") */ /* Atom of the INCR clipboard type (see ICCCM on "INCR Properties") */
static Atom incr_atom; static Atom incr_atom;
/* Stores the last "selection request" (= another X client requesting clipboard data from us). /* Stores the last "selection request" (= another X client requesting clipboard data from us).
@ -70,8 +87,17 @@ static Atom incr_atom;
When we receive the response from the RDP server (asynchronously), this variable gives us When we receive the response from the RDP server (asynchronously), this variable gives us
the context to proceed. */ the context to proceed. */
static XSelectionRequestEvent selection_request; static XSelectionRequestEvent selection_request;
/* Denotes we have a pending selection request. */
static Bool has_selection_request;
/* Stores the clipboard format (CF_TEXT, CF_UNICODETEXT etc.) requested in the last
CLIPDR_DATA_REQUEST (= the RDP server requesting clipboard data from us).
When we receive this data from whatever X client offering it, this variable gives us
the context to proceed.
*/
static int rdp_clipboard_request_format;
/* Array of offered clipboard targets that will be sent to fellow X clients upon a TARGETS request. */ /* Array of offered clipboard targets that will be sent to fellow X clients upon a TARGETS request. */
static Atom targets[NUM_TARGETS]; static Atom targets[MAX_TARGETS];
static int num_targets;
/* Denotes that this client currently holds the PRIMARY selection. */ /* Denotes that this client currently holds the PRIMARY selection. */
static int have_primary = 0; static int have_primary = 0;
/* Denotes that an rdesktop (not this rdesktop) is owning the selection, /* Denotes that an rdesktop (not this rdesktop) is owning the selection,
@ -80,6 +106,8 @@ static int rdesktop_is_selection_owner = 0;
/* Denotes that an INCR ("chunked") transfer is in progress. */ /* Denotes that an INCR ("chunked") transfer is in progress. */
static int g_waiting_for_INCR = 0; static int g_waiting_for_INCR = 0;
/* Denotes the target format of the ongoing INCR ("chunked") transfer. */
static Atom g_incr_target = 0;
/* Buffers an INCR transfer. */ /* Buffers an INCR transfer. */
static uint8 *g_clip_buffer = 0; static uint8 *g_clip_buffer = 0;
/* Denotes the size of g_clip_buffer. */ /* Denotes the size of g_clip_buffer. */
@ -103,6 +131,42 @@ crlf2lf(uint8 * data, uint32 * length)
*length = dst - data; *length = dst - data;
} }
#ifdef USE_UNICODE_CLIPBOARD
/* Translate LF to CR-LF. To do this, we must allocate more memory.
The returned string is null-terminated, as required by CF_UNICODETEXT.
The size is updated. */
static uint8 *
utf16_lf2crlf(uint8 * data, uint32 * size)
{
uint8 *result;
uint16 *inptr, *outptr;
/* Worst case: Every char is LF */
result = xmalloc((*size * 2) + 2);
if (result == NULL)
return NULL;
inptr = (uint16*)data;
outptr = (uint16*)result;
/* Check for a reversed BOM */
Bool swap_endianess = (*inptr == 0xfffe);
while ((uint8*)inptr < data + *size)
{
uint16 uvalue = *inptr;
if (swap_endianess)
uvalue = ((uvalue << 8) & 0xff00) + (uvalue >> 8);
if (uvalue == 0x0a)
*outptr++ = swap_endianess ? 0x0d00 : 0x0d;
*outptr++ = *inptr++;
}
*outptr++ = 0; /* null termination */
*size = (uint8*)outptr - result;
return result;
}
#else
/* Translate LF to CR-LF. To do this, we must allocate more memory. /* Translate LF to CR-LF. To do this, we must allocate more memory.
The length is updated. */ The length is updated. */
static uint8 * static uint8 *
@ -129,7 +193,7 @@ lf2crlf(uint8 * data, uint32 * length)
return result; return result;
} }
#endif
static void static void
xclip_provide_selection(XSelectionRequestEvent * req, Atom type, unsigned int format, uint8 * data, xclip_provide_selection(XSelectionRequestEvent * req, Atom type, unsigned int format, uint8 * data,
@ -151,8 +215,163 @@ xclip_provide_selection(XSelectionRequestEvent * req, Atom type, unsigned int fo
XSendEvent(g_display, req->requestor, False, NoEventMask, &xev); XSendEvent(g_display, req->requestor, False, NoEventMask, &xev);
} }
/* Replies a clipboard requestor, telling that we're unable to satisfy his request for whatever reason.
This has the benefit of finalizing the clipboard negotiation and thus not leaving our requestor
lingering (and, potentially, stuck). */
static void
xclip_refuse_selection(XSelectionRequestEvent * req)
{
XEvent xev;
xev.xselection.type = SelectionNotify;
xev.xselection.serial = 0;
xev.xselection.send_event = True;
xev.xselection.requestor = req->requestor;
xev.xselection.selection = req->selection;
xev.xselection.target = req->target;
xev.xselection.property = None;
xev.xselection.time = req->time;
XSendEvent(g_display, req->requestor, False, NoEventMask, &xev);
}
/* Wrapper for cliprdr_send_data which also cleans the request state. */
static void
helper_cliprdr_send_response(uint8 * data, uint32 length)
{
if (rdp_clipboard_request_format != 0)
{
cliprdr_send_data(data, length);
rdp_clipboard_request_format = 0;
if (!rdesktop_is_selection_owner)
cliprdr_send_simple_native_format_announce(RDP_CF_TEXT);
}
}
/* Last resort, when we have to provide clipboard data but for whatever
reason couldn't get any.
*/
static void
helper_cliprdr_send_empty_response()
{
helper_cliprdr_send_response(NULL, 0);
}
/* Replies with clipboard data to RDP, converting it from the target format
to the expected RDP format as necessary. Returns true if data was sent.
*/
static Bool
xclip_send_data_with_convert(uint8* source, size_t source_size, Atom target)
{
#ifdef USE_UNICODE_CLIPBOARD
if (target == format_string_atom ||
target == format_unicode_atom ||
target == format_utf8_string_atom)
{
if (rdp_clipboard_request_format != RDP_CF_TEXT)
return False;
/* Make an attempt to convert any string we send to Unicode.
We don't know what the RDP server's ANSI Codepage is, or how to convert
to it, so using CF_TEXT is not safe (and is unnecessary, since all
WinNT versions are Unicode-minded).
*/
size_t unicode_buffer_size;
char* unicode_buffer;
iconv_t cd;
if (target == format_string_atom)
{
char* locale_charset = nl_langinfo(CODESET);
cd = iconv_open(WINDOWS_CODEPAGE, locale_charset);
if (cd == (iconv_t)-1)
{
DEBUG_CLIPBOARD(("Locale charset %s not found in iconv. Unable to convert clipboard text.\n", locale_charset));
return False;
}
unicode_buffer_size = source_size * 4;
}
else if (target == format_unicode_atom)
{
cd = iconv_open(WINDOWS_CODEPAGE, "UCS-2");
if (cd == (iconv_t)-1)
{
return False;
}
unicode_buffer_size = source_size;
}
else if (target == format_utf8_string_atom)
{
cd = iconv_open(WINDOWS_CODEPAGE, "UTF-8");
if (cd == (iconv_t)-1)
{
return False;
}
/* UTF-8 is guaranteed to be less or equally compact
as UTF-16 for all Unicode chars >=2 bytes.
*/
unicode_buffer_size = source_size * 2;
}
else
{
return False;
}
unicode_buffer = xmalloc(unicode_buffer_size);
size_t unicode_buffer_size_remaining = unicode_buffer_size;
char* unicode_buffer_remaining = unicode_buffer;
char* data_remaining = (char*)source;
size_t data_size_remaining = source_size;
iconv(cd, &data_remaining, &data_size_remaining, &unicode_buffer_remaining, &unicode_buffer_size_remaining);
iconv_close(cd);
/* translate linebreaks */
uint32 translated_data_size = unicode_buffer_size - unicode_buffer_size_remaining;
uint8* translated_data = utf16_lf2crlf((uint8*)unicode_buffer, &translated_data_size);
if (translated_data != NULL)
{
DEBUG_CLIPBOARD(("Sending Unicode string of %d bytes\n", translated_data_size));
cliprdr_send_data(translated_data, translated_data_size);
xfree(translated_data); /* Not the same thing as XFree! */
}
xfree(unicode_buffer);
return True;
}
#else
if (target == format_string_atom)
{
uint8 *translated_data;
uint32 length = source_size;
if (rdp_clipboard_request_format != RDP_CF_TEXT)
return False;
DEBUG_CLIPBOARD(("Translating linebreaks before sending data\n"));
translated_data = lf2crlf(source, &length);
if (translated_data != NULL)
{
cliprdr_send_data(translated_data, length);
xfree(translated_data); /* Not the same thing as XFree! */
}
return True;
}
#endif
else if (target == rdesktop_clipboard_formats_atom)
{
helper_cliprdr_send_response(source, source_size + 1);
return True;
}
else
{
return False;
}
}
/* This function is called for SelectionNotify events. /* This function is called for SelectionNotify events.
The SelectionNotify message is sent from the clipboard owner to the requestor The SelectionNotify event is sent from the clipboard owner to the requestor
after his request was satisfied. after his request was satisfied.
If this function is called, we're the requestor side. */ If this function is called, we're the requestor side. */
#ifndef MAKE_PROTO #ifndef MAKE_PROTO
@ -161,7 +380,7 @@ xclip_handle_SelectionNotify(XSelectionEvent * event)
{ {
unsigned long nitems, bytes_left; unsigned long nitems, bytes_left;
XWindowAttributes wa; XWindowAttributes wa;
Atom type, best_target, text_target; Atom type;
Atom *supported_targets; Atom *supported_targets;
int res, i, format; int res, i, format;
uint8 *data; uint8 *data;
@ -187,7 +406,6 @@ xclip_handle_SelectionNotify(XSelectionEvent * event)
goto fail; goto fail;
} }
if (type == incr_atom) if (type == incr_atom)
{ {
DEBUG_CLIPBOARD(("Received INCR.\n")); DEBUG_CLIPBOARD(("Received INCR.\n"));
@ -199,6 +417,7 @@ xclip_handle_SelectionNotify(XSelectionEvent * event)
} }
XDeleteProperty(g_display, g_wnd, rdesktop_clipboard_target_atom); XDeleteProperty(g_display, g_wnd, rdesktop_clipboard_target_atom);
XFree(data); XFree(data);
g_incr_target = event->target;
g_waiting_for_INCR = 1; g_waiting_for_INCR = 1;
return; return;
} }
@ -208,61 +427,85 @@ xclip_handle_SelectionNotify(XSelectionEvent * event)
/* Negotiate target format */ /* Negotiate target format */
if (event->target == targets_atom) if (event->target == targets_atom)
{ {
/* FIXME: We should choose format here based on what the server wanted */ /* Determine the best of text targets that we have available:
best_target = XA_STRING; Prefer UTF8_STRING > text/unicode (unspecified encoding) > STRING
(ignore TEXT and COMPOUND_TEXT because we don't have code to handle them)
*/
int text_target_satisfaction = 0;
Atom best_text_target = 0; /* measures how much we're satisfied with what we found */
if (type != None) if (type != None)
{ {
supported_targets = (Atom *) data; supported_targets = (Atom *) data;
text_target = XInternAtom(g_display, "TEXT", False);
for (i = 0; i < nitems; i++) for (i = 0; i < nitems; i++)
{ {
DEBUG_CLIPBOARD(("Target %d: %s\n", i, DEBUG_CLIPBOARD(("Target %d: %s\n", i,
XGetAtomName(g_display, supported_targets[i]))); XGetAtomName(g_display, supported_targets[i])));
if (supported_targets[i] == text_target) if (supported_targets[i] == format_string_atom)
{ {
DEBUG_CLIPBOARD(("Other party supports TEXT, choosing that as best_target\n")); if (text_target_satisfaction < 1)
best_target = text_target; {
break; DEBUG_CLIPBOARD(("Other party supports STRING, choosing that as best_target\n"));
best_text_target = supported_targets[i];
text_target_satisfaction = 1;
} }
} }
XFree(data); #ifdef USE_UNICODE_CLIPBOARD
else if (supported_targets[i] == format_unicode_atom)
{
if (text_target_satisfaction < 2)
{
DEBUG_CLIPBOARD(("Other party supports text/unicode, choosing that as best_target\n"));
best_text_target = supported_targets[i];
text_target_satisfaction = 2;
}
}
else if (supported_targets[i] == format_utf8_string_atom)
{
if (text_target_satisfaction < 3)
{
DEBUG_CLIPBOARD(("Other party supports UTF8_STRING, choosing that as best_target\n"));
best_text_target = supported_targets[i];
text_target_satisfaction = 3;
}
}
#endif
}
} }
XConvertSelection(g_display, primary_atom, best_target, /* Kickstarting the next step in the process of satisfying RDP's
rdesktop_clipboard_target_atom, g_wnd, event->time); clipboard request -- specifically, requesting the actual clipboard data.
*/
if (best_text_target != 0)
{
XConvertSelection(g_display, clipboard_atom, best_text_target, rdesktop_clipboard_target_atom, g_wnd, event->time);
return; return;
} }
/* Translate linebreaks, but only if not getting data from
other rdesktop instance */
if (event->target != rdesktop_clipboard_formats_atom)
{
uint8 *translated_data;
uint32 length = nitems;
DEBUG_CLIPBOARD(("Translating linebreaks before sending data\n"));
translated_data = lf2crlf(data, &length);
cliprdr_send_data(translated_data, length + 1);
xfree(translated_data); /* Not the same thing as XFree! */
}
else else
{ {
cliprdr_send_data(data, nitems + 1); DEBUG_CLIPBOARD(("Unable to find a textual target to satisfy RDP clipboard text request\n"));
goto fail;
} }
}
else
{
if (!xclip_send_data_with_convert(data, nitems, event->target))
{
goto fail;
}
}
XFree(data); XFree(data);
if (!rdesktop_is_selection_owner)
cliprdr_send_simple_native_format_announce(CF_TEXT);
return; return;
fail: fail:
XDeleteProperty(g_display, g_wnd, rdesktop_clipboard_target_atom); XDeleteProperty(g_display, g_wnd, rdesktop_clipboard_target_atom);
XFree(data); XFree(data);
cliprdr_send_data(NULL, 0); helper_cliprdr_send_empty_response();
} }
/* This function is called for SelectionRequest events. /* This function is called for SelectionRequest events.
The SelectionRequest message is sent from the requestor to the clipboard owner The SelectionRequest event is sent from the requestor to the clipboard owner
to request clipboard data. to request clipboard data.
*/ */
void void
@ -281,7 +524,7 @@ xclip_handle_SelectionRequest(XSelectionRequestEvent * event)
if (event->target == targets_atom) if (event->target == targets_atom)
{ {
xclip_provide_selection(event, XA_ATOM, 32, (uint8 *) & targets, NUM_TARGETS); xclip_provide_selection(event, XA_ATOM, 32, (uint8 *) & targets, num_targets);
return; return;
} }
else if (event->target == timestamp_atom) else if (event->target == timestamp_atom)
@ -289,28 +532,70 @@ xclip_handle_SelectionRequest(XSelectionRequestEvent * event)
xclip_provide_selection(event, XA_INTEGER, 32, (uint8 *) & g_last_gesturetime, 1); xclip_provide_selection(event, XA_INTEGER, 32, (uint8 *) & g_last_gesturetime, 1);
return; return;
} }
else if (event->target == rdesktop_clipboard_formats_atom) else
{ {
/* All the following targets require an async operation with the RDP server
and currently we don't do X clipboard request queueing so we can only
handle one such request at a time. */
if (has_selection_request)
{
DEBUG_CLIPBOARD(("Error: Another clipboard request was already sent to the RDP server and not yet responded. Refusing this request.\n"));
xclip_refuse_selection(event);
return;
}
if (event->target == rdesktop_clipboard_formats_atom)
{
/* Before the requestor makes a request for the _RDESKTOP_CLIPBOARD_FORMATS target,
he should declare requestor[_RDESKTOP_CLIPBOARD_TARGET] = CF_SOMETHING.
Otherwise, we default to RDP_CF_TEXT.
*/
res = XGetWindowProperty(g_display, event->requestor, res = XGetWindowProperty(g_display, event->requestor,
rdesktop_clipboard_target_atom, 0, 1, True, XA_INTEGER, rdesktop_clipboard_target_atom, 0, 1, True, XA_INTEGER,
&type, &format, &nitems, &bytes_left, &prop_return); &type, &format, &nitems, &bytes_left, &prop_return);
wanted_format = (uint32 *) prop_return; wanted_format = (uint32 *) prop_return;
format = (res == Success) ? *wanted_format : CF_TEXT; format = (res == Success) ? *wanted_format : RDP_CF_TEXT;
/* FIXME: Need to free returned data? */ XFree(prop_return);
}
else if (event->target == format_string_atom ||
event->target == XA_STRING)
{
/* STRING and XA_STRING are defined to be ISO8859-1 */
format = CF_TEXT;
}
else if (event->target == format_utf8_string_atom)
{
#ifdef USE_UNICODE_CLIPBOARD
format = CF_UNICODETEXT;
#else
DEBUG_CLIPBOARD(("Requested target unavailable due to lack of Unicode support. (It was not in TARGETS, so why did you ask for it?!)\n"));
xclip_refuse_selection(event);
return;
#endif
}
else if (event->target == format_unicode_atom)
{
/* Assuming text/unicode to be UTF-16 */
format = CF_UNICODETEXT;
} }
else else
{ {
format = CF_TEXT; DEBUG_CLIPBOARD(("Requested target unavailable. (It was not in TARGETS, so why did you ask for it?!)\n"));
xclip_refuse_selection(event);
return;
} }
cliprdr_send_data_request(format); cliprdr_send_data_request(format);
selection_request = *event; selection_request = *event;
/* wait for data */ has_selection_request = True;
return; /* wait for data */
}
} }
/* When this rdesktop holds ownership over the clipboard, it means the clipboard data /* While this rdesktop holds ownership over the clipboard, it means the clipboard data
is offered by the RDP server (and when its pasted inside RDP, there's no network is offered by the RDP server (and when it is pasted inside RDP, there's no network
roundtrip). This event symbolizes this rdesktop lost onwership of the clipboard roundtrip).
This event (SelectionClear) symbolizes this rdesktop lost onwership of the clipboard
to some other X client. We should find out what clipboard formats this other to some other X client. We should find out what clipboard formats this other
client offers and announce that to RDP. */ client offers and announce that to RDP. */
void void
@ -319,7 +604,11 @@ xclip_handle_SelectionClear(void)
DEBUG_CLIPBOARD(("xclip_handle_SelectionClear\n")); DEBUG_CLIPBOARD(("xclip_handle_SelectionClear\n"));
have_primary = 0; have_primary = 0;
XDeleteProperty(g_display, DefaultRootWindow(g_display), rdesktop_clipboard_formats_atom); XDeleteProperty(g_display, DefaultRootWindow(g_display), rdesktop_clipboard_formats_atom);
cliprdr_send_simple_native_format_announce(CF_TEXT); /* FIXME:
Without XFIXES, we cannot reliably know the formats offered by the
new owner of the X11 clipboard, so we just lie about him
offering RDP_CF_TEXT. */
cliprdr_send_simple_native_format_announce(RDP_CF_TEXT);
} }
/* Called when any property changes in our window or the root window. */ /* Called when any property changes in our window or the root window. */
@ -340,6 +629,8 @@ xclip_handle_PropertyNotify(XPropertyEvent * event)
while (bytes_left > 0) while (bytes_left > 0)
{ {
/* Unlike the specification, we don't set the 'delete' arugment to True
since we slurp the INCR's chunks in even-smaller chunks of 4096 bytes. */
if ((XGetWindowProperty if ((XGetWindowProperty
(g_display, g_wnd, rdesktop_clipboard_target_atom, offset, 4096L, (g_display, g_wnd, rdesktop_clipboard_target_atom, offset, 4096L,
False, AnyPropertyType, &type, &format, &nitems, &bytes_left, False, AnyPropertyType, &type, &format, &nitems, &bytes_left,
@ -351,6 +642,7 @@ xclip_handle_PropertyNotify(XPropertyEvent * event)
if (nitems == 0) if (nitems == 0)
{ {
/* INCR transfer finished */
XGetWindowAttributes(g_display, g_wnd, &wa); XGetWindowAttributes(g_display, g_wnd, &wa);
XSelectInput(g_display, g_wnd, XSelectInput(g_display, g_wnd,
(wa.your_event_mask ^ PropertyChangeMask)); (wa.your_event_mask ^ PropertyChangeMask));
@ -359,51 +651,38 @@ xclip_handle_PropertyNotify(XPropertyEvent * event)
if (g_clip_buflen > 0) if (g_clip_buflen > 0)
{ {
cliprdr_send_data(g_clip_buffer, g_clip_buflen + 1); if (!xclip_send_data_with_convert(g_clip_buffer, g_clip_buflen, g_incr_target))
{
if (!rdesktop_is_selection_owner) helper_cliprdr_send_empty_response();
cliprdr_send_simple_native_format_announce(CF_TEXT); }
xfree(g_clip_buffer); xfree(g_clip_buffer);
g_clip_buffer = 0; g_clip_buffer = NULL;
g_clip_buflen = 0; g_clip_buflen = 0;
} }
} }
else else
{ {
uint8 *translated_data; /* Another chunk in the INCR transfer */
uint32 length = nitems; offset += (nitems / 4); /* offset at which to begin the next slurp */
uint8 *tmp; g_clip_buffer = xrealloc(g_clip_buffer, g_clip_buflen + nitems);
memcpy(g_clip_buffer + g_clip_buflen, data, nitems);
offset += (length / 4); g_clip_buflen += nitems;
DEBUG_CLIPBOARD(("Translating linebreaks before sending data\n"));
translated_data = lf2crlf(data, &length);
tmp = xmalloc(length + g_clip_buflen);
strncpy((char *) tmp, (char *) g_clip_buffer, g_clip_buflen);
xfree(g_clip_buffer);
strncpy((char *) (tmp + g_clip_buflen), (char *) translated_data,
length);
xfree(translated_data);
g_clip_buffer = tmp;
g_clip_buflen += length;
XFree(data); XFree(data);
} }
} }
XDeleteProperty(g_display, g_wnd, rdesktop_clipboard_target_atom); XDeleteProperty(g_display, g_wnd, rdesktop_clipboard_target_atom);
return;
} }
if (event->atom != rdesktop_clipboard_formats_atom) if ((event->atom == rdesktop_clipboard_formats_atom) &&
return; (event->window == DefaultRootWindow(g_display)) &&
!have_primary /* not interested in our own events */)
if (have_primary) /* from us */ {
return;
if (event->state == PropertyNewValue) if (event->state == PropertyNewValue)
{ {
DEBUG_CLIPBOARD(("xclip_handle_PropertyNotify: getting fellow rdesktop formats\n"));
res = XGetWindowProperty(g_display, DefaultRootWindow(g_display), res = XGetWindowProperty(g_display, DefaultRootWindow(g_display),
rdesktop_clipboard_formats_atom, 0, rdesktop_clipboard_formats_atom, 0,
XMaxRequestSize(g_display), False, XA_STRING, &type, XMaxRequestSize(g_display), False, XA_STRING, &type,
@ -417,10 +696,11 @@ xclip_handle_PropertyNotify(XPropertyEvent * event)
} }
} }
/* PropertyDelete, or XGetWindowProperty failed */ /* For some reason, we couldn't announce the native formats */
cliprdr_send_simple_native_format_announce(CF_TEXT); cliprdr_send_simple_native_format_announce(RDP_CF_TEXT);
rdesktop_is_selection_owner = 0; rdesktop_is_selection_owner = 0;
} }
}
#endif #endif
@ -449,12 +729,16 @@ ui_clip_format_announce(uint8 * data, uint32 length)
warning("Failed to aquire ownership of CLIPBOARD clipboard\n"); warning("Failed to aquire ownership of CLIPBOARD clipboard\n");
} }
/* Called when the RDP server responds with clipboard data (after we've requested it). */
void void
ui_clip_handle_data(uint8 * data, uint32 length) ui_clip_handle_data(uint8 * data, uint32 length)
{ {
if (selection_request.target != rdesktop_clipboard_formats_atom) BOOL free_data = False;
if (selection_request.target == format_string_atom ||
selection_request.target == XA_STRING)
{ {
/* We're expecting a CF_TEXT response */
uint8 *firstnull; uint8 *firstnull;
/* translate linebreaks */ /* translate linebreaks */
@ -467,8 +751,54 @@ ui_clip_handle_data(uint8 * data, uint32 length)
length = firstnull - data + 1; length = firstnull - data + 1;
} }
} }
#ifdef USE_UNICODE_CLIPBOARD
else if (selection_request.target == format_utf8_string_atom)
{
/* We're expecting a CF_UNICODETEXT response */
iconv_t cd = iconv_open("UTF-8", WINDOWS_CODEPAGE);
if (cd != (iconv_t)-1)
{
size_t utf8_length = length * 2;
char* utf8_data = malloc(utf8_length);
size_t utf8_length_remaining = utf8_length;
char* utf8_data_remaining = utf8_data;
char* data_remaining = (char*)data;
size_t length_remaining = (size_t)length;
if (utf8_data == NULL)
{
iconv_close(cd);
return;
}
iconv(cd, &data_remaining, &length_remaining, &utf8_data_remaining, &utf8_length_remaining);
iconv_close(cd);
free_data = True;
data = (uint8*)utf8_data;
length = utf8_length - utf8_length_remaining;
}
}
else if (selection_request.target == format_unicode_atom)
{
/* We're expecting a CF_UNICODETEXT response, so what we're
receiving matches our requirements and there's no need
for further conversions. */
}
#endif
else if (selection_request.target == rdesktop_clipboard_formats_atom)
{
/* Pass as-is */
}
else
{
xclip_refuse_selection(&selection_request);
has_selection_request = False;
return;
}
xclip_provide_selection(&selection_request, XA_STRING, 8, data, length - 1); xclip_provide_selection(&selection_request, selection_request.target, 8, data, length - 1);
has_selection_request = False;
if (free_data)
free(data);
} }
void void
@ -477,6 +807,7 @@ ui_clip_request_data(uint32 format)
Window selectionowner; Window selectionowner;
DEBUG_CLIPBOARD(("Request from server for format %d\n", format)); DEBUG_CLIPBOARD(("Request from server for format %d\n", format));
rdp_clipboard_request_format = format;
if (rdesktop_is_selection_owner) if (rdesktop_is_selection_owner)
{ {
@ -512,7 +843,7 @@ ui_clip_request_data(uint32 format)
void void
ui_clip_sync(void) ui_clip_sync(void)
{ {
cliprdr_send_simple_native_format_announce(CF_TEXT); cliprdr_send_simple_native_format_announce(RDP_CF_TEXT);
} }
@ -529,12 +860,19 @@ xclip_init(void)
rdesktop_clipboard_target_atom = rdesktop_clipboard_target_atom =
XInternAtom(g_display, "_RDESKTOP_CLIPBOARD_TARGET", False); XInternAtom(g_display, "_RDESKTOP_CLIPBOARD_TARGET", False);
incr_atom = XInternAtom(g_display, "INCR", False); incr_atom = XInternAtom(g_display, "INCR", False);
targets[0] = targets_atom; format_string_atom = XInternAtom(g_display, "STRING", False);
targets[1] = XInternAtom(g_display, "TEXT", False); format_utf8_string_atom = XInternAtom(g_display, "UTF8_STRING", False);
targets[2] = XInternAtom(g_display, "STRING", False); format_unicode_atom = XInternAtom(g_display, "text/unicode", False);
targets[3] = XInternAtom(g_display, "text/unicode", False); num_targets = 0;
targets[4] = XInternAtom(g_display, "TIMESTAMP", False); targets[num_targets++] = targets_atom;
targets[5] = XA_STRING; targets[num_targets++] = timestamp_atom;
targets[num_targets++] = rdesktop_clipboard_formats_atom;
targets[num_targets++] = format_string_atom;
#ifdef USE_UNICODE_CLIPBOARD
targets[num_targets++] = format_utf8_string_atom;
#endif
targets[num_targets++] = format_unicode_atom;
targets[num_targets++] = XA_STRING;
/* rdesktop sets _RDESKTOP_CLIPBOARD_FORMATS on the root window when acquiring the clipboard. /* rdesktop sets _RDESKTOP_CLIPBOARD_FORMATS on the root window when acquiring the clipboard.
Other interested rdesktops can use this to notify their server of the available formats. */ Other interested rdesktops can use this to notify their server of the available formats. */