/[rdesktop]/sourceforge.net/trunk/rdesktop/xclip.c
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Diff of /sourceforge.net/trunk/rdesktop/xclip.c

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 468 by astrand, Mon Sep 15 07:59:36 2003 UTC revision 1026 by forsberg, Wed Nov 9 15:15:28 2005 UTC
# Line 36  static Atom targets[NUM_TARGETS]; Line 36  static Atom targets[NUM_TARGETS];
36  static int have_primary = 0;  static int have_primary = 0;
37  static int rdesktop_is_selection_owner = 0;  static int rdesktop_is_selection_owner = 0;
38    
39    static int g_waiting_for_INCR = 0;
40    static uint8 *g_clip_buffer = 0;
41    static uint32 g_clip_buflen = 0;
42    
43    /* Replace CR-LF to LF (well, rather removing all CR:s) This is done
44       in-place. The length is updated. Handles embedded nulls */
45    static void
46    crlf2lf(uint8 * data, uint32 * length)
47    {
48            uint8 *dst, *src;
49            src = dst = data;
50            while (src < data + *length)
51            {
52                    if (*src != '\x0d')
53                            *dst++ = *src;
54                    src++;
55            }
56            *length = dst - data;
57    }
58    
59    /* Translate LF to CR-LF. To do this, we must allocate more memory.  
60       The length is updated. */
61    static uint8 *
62    lf2crlf(uint8 * data, uint32 * length)
63    {
64            uint8 *result, *p, *o;
65    
66            /* Worst case: Every char is LF */
67            result = xmalloc(*length * 2);
68    
69            p = data;
70            o = result;
71    
72            while (p < data + *length)
73            {
74                    if (*p == '\x0a')
75                            *o++ = '\x0d';
76                    *o++ = *p++;
77            }
78            *length = o - result;
79    
80            /* Convenience */
81            *o++ = '\0';
82    
83            return result;
84    }
85    
86    
87  static void  static void
88  xclip_provide_selection(XSelectionRequestEvent * req, Atom type, unsigned int format, uint8 * data,  xclip_provide_selection(XSelectionRequestEvent * req, Atom type, unsigned int format, uint8 * data,
89                          uint32 length)                          uint32 length)
# Line 56  xclip_provide_selection(XSelectionReques Line 104  xclip_provide_selection(XSelectionReques
104          XSendEvent(g_display, req->requestor, False, NoEventMask, &xev);          XSendEvent(g_display, req->requestor, False, NoEventMask, &xev);
105  }  }
106    
107    #ifndef MAKE_PROTO
108  void  void
109  xclip_handle_SelectionNotify(XSelectionEvent * event)  xclip_handle_SelectionNotify(XSelectionEvent * event)
110  {  {
111          unsigned long nitems, bytes_left;          unsigned long nitems, bytes_left;
112            XWindowAttributes wa;
113          Atom type, best_target, text_target;          Atom type, best_target, text_target;
114          Atom *supported_targets;          Atom *supported_targets;
115          int res, i, format;          int res, i, format;
# Line 77  xclip_handle_SelectionNotify(XSelectionE Line 127  xclip_handle_SelectionNotify(XSelectionE
127                  goto fail;                  goto fail;
128    
129          res = XGetWindowProperty(g_display, g_wnd, rdesktop_clipboard_target_atom,          res = XGetWindowProperty(g_display, g_wnd, rdesktop_clipboard_target_atom,
130                                   0, XMaxRequestSize(g_display), True, AnyPropertyType,                                   0, XMaxRequestSize(g_display), False, AnyPropertyType,
131                                   &type, &format, &nitems, &bytes_left, &data);                                   &type, &format, &nitems, &bytes_left, &data);
132    
133          if (res != Success)          if (res != Success)
# Line 86  xclip_handle_SelectionNotify(XSelectionE Line 136  xclip_handle_SelectionNotify(XSelectionE
136                  goto fail;                  goto fail;
137          }          }
138    
139    
140            if (type == incr_atom)
141            {
142                    DEBUG_CLIPBOARD(("Received INCR.\n"));
143    
144                    XGetWindowAttributes(g_display, g_wnd, &wa);
145                    if ((wa.your_event_mask | PropertyChangeMask) != wa.your_event_mask)
146                    {
147                            XSelectInput(g_display, g_wnd, (wa.your_event_mask | PropertyChangeMask));
148                    }
149                    XDeleteProperty(g_display, g_wnd, rdesktop_clipboard_target_atom);
150                    XFree(data);
151                    g_waiting_for_INCR = 1;
152                    return;
153            }
154            
155            XDeleteProperty(g_display, g_wnd, rdesktop_clipboard_target_atom);
156                    
157            /* Negotiate target format */
158          if (event->target == targets_atom)          if (event->target == targets_atom)
159          {          {
160                  /* FIXME: We should choose format here based on what the server wanted */                  /* FIXME: We should choose format here based on what the server wanted */
# Line 102  xclip_handle_SelectionNotify(XSelectionE Line 171  xclip_handle_SelectionNotify(XSelectionE
171                                  {                                  {
172                                          DEBUG_CLIPBOARD(("Other party supports TEXT, choosing that as best_target\n"));                                          DEBUG_CLIPBOARD(("Other party supports TEXT, choosing that as best_target\n"));
173                                          best_target = text_target;                                          best_target = text_target;
174                                            break;
175                                  }                                  }
176                          }                          }
177                          XFree(data);                          XFree(data);
# Line 112  xclip_handle_SelectionNotify(XSelectionE Line 182  xclip_handle_SelectionNotify(XSelectionE
182                  return;                  return;
183          }          }
184    
185          if (type == incr_atom)          /* Translate linebreaks, but only if not getting data from
186               other rdesktop instance */
187            if (event->target != rdesktop_clipboard_formats_atom)
188            {
189                    uint8 *translated_data;
190                    uint32 length = nitems;
191    
192                    DEBUG_CLIPBOARD(("Translating linebreaks before sending data\n"));
193                    translated_data = lf2crlf(data, &length);
194                    cliprdr_send_data(translated_data, length + 1);
195                    xfree(translated_data); /* Not the same thing as XFree! */
196            }
197            else
198          {          {
199                  warning("We don't support INCR transfers at this time. Try cutting less data.\n");                  cliprdr_send_data(data, nitems + 1);
                 goto fail;  
200          }          }
   
         cliprdr_send_data(data, nitems + 1);  
201          XFree(data);          XFree(data);
202    
203          if (!rdesktop_is_selection_owner)          if (!rdesktop_is_selection_owner)
204                  cliprdr_send_text_format_announce();                  cliprdr_send_simple_native_format_announce(CF_TEXT);
205          return;          return;
206    
207        fail:   fail:
208            XDeleteProperty(g_display, g_wnd, rdesktop_clipboard_target_atom);
209            XFree(data);
210          cliprdr_send_data(NULL, 0);          cliprdr_send_data(NULL, 0);
211  }  }
212    
# Line 178  xclip_handle_SelectionClear(void) Line 259  xclip_handle_SelectionClear(void)
259          DEBUG_CLIPBOARD(("xclip_handle_SelectionClear\n"));          DEBUG_CLIPBOARD(("xclip_handle_SelectionClear\n"));
260          have_primary = 0;          have_primary = 0;
261          XDeleteProperty(g_display, DefaultRootWindow(g_display), rdesktop_clipboard_formats_atom);          XDeleteProperty(g_display, DefaultRootWindow(g_display), rdesktop_clipboard_formats_atom);
262          cliprdr_send_text_format_announce();          cliprdr_send_simple_native_format_announce(CF_TEXT);
263  }  }
264    
265  void  void
266  xclip_handle_PropertyNotify(XPropertyEvent * event)  xclip_handle_PropertyNotify(XPropertyEvent * event)
267  {  {
268          unsigned long nitems, bytes_left;          unsigned long nitems;
269            unsigned long offset = 0;
270            unsigned long bytes_left = 1;
271          int format, res;          int format, res;
272            XWindowAttributes wa;
273          uint8 *data;          uint8 *data;
274          Atom type;          Atom type;
275    
276            if (event->state == PropertyNewValue && g_waiting_for_INCR)
277            {
278                    DEBUG_CLIPBOARD(("x_clip_handle_PropertyNotify: g_waiting_for_INCR != 0\n"));
279    
280                    while (bytes_left > 0) {
281                            if ((XGetWindowProperty(g_display, g_wnd, rdesktop_clipboard_target_atom, offset,
282                                                    4096L, False, AnyPropertyType,
283                                                    &type, &format, &nitems, &bytes_left, &data) != Success))
284                            {
285                                    XFree(data);
286                                    return;
287                            }
288    
289                            if (nitems == 0)
290                            {
291                                    XGetWindowAttributes(g_display, g_wnd, &wa);
292                                    XSelectInput(g_display, g_wnd, (wa.your_event_mask ^ PropertyChangeMask));
293                                    XFree(data);
294                                    g_waiting_for_INCR = 0;
295    
296                                    if (g_clip_buflen > 0)
297                                    {
298                                            cliprdr_send_data(g_clip_buffer, g_clip_buflen + 1);
299    
300                                            if (!rdesktop_is_selection_owner)
301                                                    cliprdr_send_simple_native_format_announce(CF_TEXT);
302    
303                                            xfree(g_clip_buffer);
304                                            g_clip_buffer = 0;
305                                            g_clip_buflen = 0;
306                                    }
307                            }
308                            else
309                            {
310                                    uint8 *translated_data;
311                                    uint32 length = nitems;
312                                    uint8 *tmp;
313    
314                                    offset += (length/4);
315                                    DEBUG_CLIPBOARD(("Translating linebreaks before sending data\n"));
316                                    translated_data = lf2crlf(data, &length);
317    
318                                    tmp = xmalloc(length + g_clip_buflen);
319                                    strncpy((char *) tmp, (char *) g_clip_buffer, g_clip_buflen);
320                                    xfree(g_clip_buffer);
321    
322                                    strncpy((char *) (tmp + g_clip_buflen), (char *) translated_data, length);
323                                    xfree(translated_data);
324    
325                                    g_clip_buffer = tmp;
326                                    g_clip_buflen += length;
327    
328                                    XFree(data);
329                            }
330                    }
331                    XDeleteProperty(g_display, g_wnd, rdesktop_clipboard_target_atom);
332            }
333    
334          if (event->atom != rdesktop_clipboard_formats_atom)          if (event->atom != rdesktop_clipboard_formats_atom)
335                  return;                  return;
336    
# Line 211  xclip_handle_PropertyNotify(XPropertyEve Line 353  xclip_handle_PropertyNotify(XPropertyEve
353          }          }
354    
355          /* PropertyDelete, or XGetWindowProperty failed */          /* PropertyDelete, or XGetWindowProperty failed */
356          cliprdr_send_text_format_announce();          cliprdr_send_simple_native_format_announce(CF_TEXT);
357          rdesktop_is_selection_owner = 0;          rdesktop_is_selection_owner = 0;
358  }  }
359    #endif
360    
361    
362  void  void
363  ui_clip_format_announce(char *data, uint32 length)  ui_clip_format_announce(uint8 * data, uint32 length)
364  {  {
365          XSetSelectionOwner(g_display, primary_atom, g_wnd, g_last_gesturetime);          XSetSelectionOwner(g_display, primary_atom, g_wnd, g_last_gesturetime);
366          if (XGetSelectionOwner(g_display, primary_atom) != g_wnd)          if (XGetSelectionOwner(g_display, primary_atom) != g_wnd)
# Line 238  ui_clip_format_announce(char *data, uint Line 381  ui_clip_format_announce(char *data, uint
381    
382    
383  void  void
384  ui_clip_handle_data(char *data, uint32 length)  ui_clip_handle_data(uint8 * data, uint32 length)
385  {  {
386            if (selection_request.target != rdesktop_clipboard_formats_atom)
387            {
388                    uint8 *firstnull;
389    
390                    /* translate linebreaks */
391                    crlf2lf(data, &length);
392    
393                    /* Only send data up to null byte, if any */
394                    firstnull = (uint8 *) strchr((char *) data, '\0');
395                    if (firstnull)
396                    {
397                            length = firstnull - data + 1;
398                    }
399            }
400    
401          xclip_provide_selection(&selection_request, XA_STRING, 8, data, length - 1);          xclip_provide_selection(&selection_request, XA_STRING, 8, data, length - 1);
402  }  }
403    
# Line 284  ui_clip_request_data(uint32 format) Line 442  ui_clip_request_data(uint32 format)
442  void  void
443  ui_clip_sync(void)  ui_clip_sync(void)
444  {  {
445          cliprdr_send_text_format_announce();          cliprdr_send_simple_native_format_announce(CF_TEXT);
446  }  }
447    
448    
# Line 303  xclip_init(void) Line 461  xclip_init(void)
461          incr_atom = XInternAtom(g_display, "INCR", False);          incr_atom = XInternAtom(g_display, "INCR", False);
462          targets[0] = targets_atom;          targets[0] = targets_atom;
463          targets[1] = XInternAtom(g_display, "TEXT", False);          targets[1] = XInternAtom(g_display, "TEXT", False);
464          targets[2] = XInternAtom(g_display, "UTF8_STRING", False);          targets[2] = XInternAtom(g_display, "STRING", False);
465          targets[3] = XInternAtom(g_display, "text/unicode", False);          targets[3] = XInternAtom(g_display, "text/unicode", False);
466          targets[4] = XInternAtom(g_display, "TIMESTAMP", False);          targets[4] = XInternAtom(g_display, "TIMESTAMP", False);
467          targets[5] = XA_STRING;          targets[5] = XA_STRING;

Legend:
Removed from v.468  
changed lines
  Added in v.1026

  ViewVC Help
Powered by ViewVC 1.1.26