diff options
Diffstat (limited to 'x11vnc/xevents.c')
-rw-r--r-- | x11vnc/xevents.c | 188 |
1 files changed, 150 insertions, 38 deletions
diff --git a/x11vnc/xevents.c b/x11vnc/xevents.c index 3e9401b..69c332f 100644 --- a/x11vnc/xevents.c +++ b/x11vnc/xevents.c @@ -19,6 +19,7 @@ int sync_tod_delay = 3; void initialize_vnc_connect_prop(void); void initialize_x11vnc_remote_prop(void); +void initialize_clipboard_atom(void); void spawn_grab_buster(void); void sync_tod_with_servertime(void); void check_keycode_state(void); @@ -45,11 +46,25 @@ void initialize_x11vnc_remote_prop(void) { x11vnc_remote_prop = XInternAtom(dpy, "X11VNC_REMOTE", False); } +void initialize_clipboard_atom(void) { + clipboard_atom = XInternAtom(dpy, "CLIPBOARD", False); + if (clipboard_atom == None) { + if (! quiet) rfbLog("could not find atom CLIPBOARD\n"); + if (watch_clipboard) { + watch_clipboard = 0; + } + if (set_clipboard) { + set_clipboard = 0; + } + } +} + static void initialize_xevents(void) { static int did_xselect_input = 0; static int did_xcreate_simple_window = 0; static int did_vnc_connect_prop = 0; static int did_x11vnc_remote_prop = 0; + static int did_clipboard_atom = 0; static int did_xfixes = 0; static int did_xdamage = 0; static int did_xrandr = 0; @@ -90,6 +105,10 @@ static void initialize_xevents(void) { kill(run_gui_pid, SIGUSR1); run_gui_pid = 0; } + if (!did_clipboard_atom) { + initialize_clipboard_atom(); + did_clipboard_atom = 1; + } if (xfixes_present && use_xfixes && !did_xfixes) { initialize_xfixes(); did_xfixes = 1; @@ -662,13 +681,13 @@ void check_xevents(void) { XEvent xev; int tmp, have_clients = 0; static int sent_some_sel = 0; - static time_t last_request = 0; static time_t last_call = 0; static time_t last_bell = 0; static time_t last_init_check = 0; static time_t last_sync = 0; static time_t last_time_sync = 0; time_t now = time(0); + static double last_request = 0.0; if (raw_fb && ! dpy) return; /* raw_fb hack */ @@ -798,56 +817,119 @@ void check_xevents(void) { #endif /* check for our PRIMARY request notification: */ - if (watch_primary) { + if (watch_primary || watch_clipboard) { + int doprimary = 1, doclipboard = 2, which, own; + double delay = 1.0; + Atom atom; + char *req; + if (XCheckTypedEvent(dpy, SelectionNotify, &xev)) { if (xev.type == SelectionNotify && xev.xselection.requestor == selwin && - xev.xselection.selection == XA_PRIMARY && xev.xselection.property != None && xev.xselection.target == XA_STRING) { - - /* go retrieve PRIMARY and check it */ - if (now > last_client + sel_waittime - || sent_some_sel) { - selection_send(&xev); + Atom s = xev.xselection.selection; + if (s == XA_PRIMARY || s == clipboard_atom) { + /* go retrieve it and check it */ + if (now > last_client + sel_waittime + || sent_some_sel) { + selection_send(&xev); + } } } } - if (now > last_request) { + /* + * Every second or so, request PRIMARY or CLIPBOARD, + * unless we already own it or there is no owner or we + * have no clients. + * TODO: even at this low rate we should look into + * and performance problems in odds cases (large text, + * modem, etc.) + */ + which = 0; + if (watch_primary && watch_clipboard && ! own_clipboard && + ! own_primary) { + delay = 0.6; + } + if (dnow() > last_request + delay) { /* - * Every second or two, request PRIMARY, unless we - * already own it or there is no owner or we have - * no clients. - * TODO: even at this low rate we should look into - * and performance problems in odds cases, etc. + * It is not a good idea to do both at the same + * time so we must choose one: */ - last_request = now; - if (! own_selection && have_clients && - XGetSelectionOwner(dpy, XA_PRIMARY) != None) { - XConvertSelection(dpy, XA_PRIMARY, XA_STRING, - XA_STRING, selwin, CurrentTime); + if (watch_primary && watch_clipboard) { + static int count = 0; + if (own_clipboard) { + which = doprimary; + } else if (own_primary) { + which = doclipboard; + } else if (count++ % 3 == 0) { + which = doclipboard; + } else { + which = doprimary; + } + } else if (watch_primary) { + which = doprimary; + } else if (watch_clipboard) { + which = doclipboard; + } + last_request = dnow(); + } + atom = None; + req = "none"; + if (which == doprimary) { + own = own_primary; + atom = XA_PRIMARY; + req = "PRIMARY"; + } else if (which == doclipboard) { + own = own_clipboard; + atom = clipboard_atom; + req = "CLIPBOARD"; + } + if (which != 0 && ! own && have_clients && + XGetSelectionOwner(dpy, atom) != None) { + XConvertSelection(dpy, atom, XA_STRING, XA_STRING, + selwin, CurrentTime); + if (debug_sel) { + rfbLog("request %s\n", req); } } } - if (own_selection) { - /* we own PRIMARY, see if someone requested it: */ + if (own_primary || own_clipboard) { + /* we own PRIMARY or CLIPBOARD, see if someone requested it: */ if (XCheckTypedEvent(dpy, SelectionRequest, &xev)) { - if (xev.type == SelectionRequest && + if (own_primary && xev.type == SelectionRequest && xev.xselectionrequest.selection == XA_PRIMARY) { - selection_request(&xev); + selection_request(&xev, "PRIMARY"); + } + if (own_clipboard && xev.type == SelectionRequest && + xev.xselectionrequest.selection == clipboard_atom) { + selection_request(&xev, "CLIPBOARD"); } } - /* we own PRIMARY, see if we no longer own it: */ + /* we own PRIMARY or CLIPBOARD, see if we no longer own it: */ if (XCheckTypedEvent(dpy, SelectionClear, &xev)) { - if (xev.type == SelectionClear && + if (own_primary && xev.type == SelectionClear && xev.xselectionclear.selection == XA_PRIMARY) { - - own_selection = 0; - if (xcut_str) { - free(xcut_str); - xcut_str = NULL; + own_primary = 0; + if (xcut_str_primary) { + free(xcut_str_primary); + xcut_str_primary = NULL; + } + if (debug_sel) { + rfbLog("Released PRIMARY.\n"); + } + } + if (own_clipboard && xev.type == SelectionClear && + xev.xselectionclear.selection == clipboard_atom) { + own_clipboard = 0; + if (xcut_str_clipboard) { + free(xcut_str_clipboard); + xcut_str_clipboard = NULL; + } + if (debug_sel) { + rfbLog("Released CLIPBOARD.\n"); } } } @@ -986,21 +1068,51 @@ void xcut_receive(char *text, int len, rfbClientPtr cl) { X_LOCK; /* associate this text with PRIMARY (and SECONDARY...) */ - if (! own_selection) { - own_selection = 1; + if (set_primary && ! own_primary) { + own_primary = 1; /* we need to grab the PRIMARY selection */ XSetSelectionOwner(dpy, XA_PRIMARY, selwin, CurrentTime); XFlush(dpy); + if (debug_sel) { + rfbLog("Own PRIMARY.\n"); + } + } + + if (set_clipboard && ! own_clipboard && clipboard_atom != None) { + own_clipboard = 1; + /* we need to grab the CLIPBOARD selection */ + XSetSelectionOwner(dpy, clipboard_atom, selwin, CurrentTime); + XFlush(dpy); + if (debug_sel) { + rfbLog("Own CLIPBOARD.\n"); + } } /* duplicate the text string for our own use. */ - if (xcut_str != NULL) { - free(xcut_str); - xcut_str = NULL; + if (set_primary) { + if (xcut_str_primary != NULL) { + free(xcut_str_primary); + xcut_str_primary = NULL; + } + xcut_str_primary = (char *) malloc((size_t) (len+1)); + strncpy(xcut_str_primary, text, len); + xcut_str_primary[len] = '\0'; /* make sure null terminated */ + if (debug_sel) { + rfbLog("Set PRIMARY '%s'\n", xcut_str_primary); + } + } + if (set_clipboard) { + if (xcut_str_clipboard != NULL) { + free(xcut_str_clipboard); + xcut_str_clipboard = NULL; + } + xcut_str_clipboard = (char *) malloc((size_t) (len+1)); + strncpy(xcut_str_clipboard, text, len); + xcut_str_clipboard[len] = '\0'; /* make sure null terminated */ + if (debug_sel) { + rfbLog("Set CLIPBOARD '%s'\n", xcut_str_clipboard); + } } - xcut_str = (char *) malloc((size_t) (len+1)); - strncpy(xcut_str, text, len); - xcut_str[len] = '\0'; /* make sure null terminated */ /* copy this text to CUT_BUFFER0 as well: */ XChangeProperty(dpy, rootwin, XA_CUT_BUFFER0, XA_STRING, 8, |