diff options
Diffstat (limited to 'x11vnc/win_utils.c')
-rw-r--r-- | x11vnc/win_utils.c | 444 |
1 files changed, 444 insertions, 0 deletions
diff --git a/x11vnc/win_utils.c b/x11vnc/win_utils.c new file mode 100644 index 0000000..11efaca --- /dev/null +++ b/x11vnc/win_utils.c @@ -0,0 +1,444 @@ +/* -- win_utils.c -- */ + +#include "x11vnc.h" +#include "xinerama.h" +#include "winattr_t.h" +#include "cleanup.h" + +winattr_t *stack_list = NULL; +int stack_list_len = 0; +int stack_list_num = 0; + + +Window parent_window(Window win, char **name); +int valid_window(Window win, XWindowAttributes *attr_ret, int bequiet); +Bool xtranslate(Window src, Window dst, int src_x, int src_y, int *dst_x, + int *dst_y, Window *child, int bequiet); +int get_window_size(Window win, int *x, int *y); +void snapshot_stack_list(int free_only, double allowed_age); +void update_stack_list(void); +Window query_pointer(Window start); +int pick_windowid(unsigned long *num); +Window descend_pointer(int depth, Window start, char *name_info, int len); + + +Window parent_window(Window win, char **name) { + Window r, parent; + Window *list; + unsigned int nchild; + + if (name != NULL) { + *name = NULL; + } + + if (! XQueryTree(dpy, win, &r, &parent, &list, &nchild)) { + return None; + } + if (list) { + XFree(list); + } + if (parent && name) { + XFetchName(dpy, parent, name); + } + return parent; +} + +/* trapping utility to check for a valid window: */ +int valid_window(Window win, XWindowAttributes *attr_ret, int bequiet) { + XErrorHandler old_handler; + XWindowAttributes attr, *pattr; + int ok = 0; + + if (attr_ret == NULL) { + pattr = &attr; + } else { + pattr = attr_ret; + } + + if (win == None) { + return 0; + } + + trapped_xerror = 0; + old_handler = XSetErrorHandler(trap_xerror); + if (XGetWindowAttributes(dpy, win, pattr)) { + ok = 1; + } + if (trapped_xerror && trapped_xerror_event) { + if (! quiet && ! bequiet) { + rfbLog("valid_window: trapped XError: %s (0x%lx)\n", + xerror_string(trapped_xerror_event), win); + } + ok = 0; + } + XSetErrorHandler(old_handler); + trapped_xerror = 0; + + return ok; +} + +Bool xtranslate(Window src, Window dst, int src_x, int src_y, int *dst_x, + int *dst_y, Window *child, int bequiet) { + XErrorHandler old_handler; + Bool ok = False; + + trapped_xerror = 0; + old_handler = XSetErrorHandler(trap_xerror); + if (XTranslateCoordinates(dpy, src, dst, src_x, src_y, dst_x, + dst_y, child)) { + ok = True; + } + if (trapped_xerror && trapped_xerror_event) { + if (! quiet && ! bequiet) { + rfbLog("xtranslate: trapped XError: %s (0x%lx)\n", + xerror_string(trapped_xerror_event), src); + } + ok = False; + } + XSetErrorHandler(old_handler); + trapped_xerror = 0; + + return ok; +} + +int get_window_size(Window win, int *x, int *y) { + XWindowAttributes attr; + /* valid_window? */ + if (valid_window(win, &attr, 1)) { + *x = attr.width; + *y = attr.height; + return 1; + } else { + return 0; + } +} + +/* + * For use in the -wireframe stuff, save the stacking order of the direct + * children of the root window. Ideally done before we send ButtonPress + * to the X server. + */ +void snapshot_stack_list(int free_only, double allowed_age) { + static double last_snap = 0.0, last_free = 0.0; + double now; + int num, rc, i, j; + unsigned int ui; + Window r, w; + Window *list; + + if (! stack_list) { + stack_list = (winattr_t *) malloc(256*sizeof(winattr_t)); + stack_list_num = 0; + stack_list_len = 256; + } + + dtime0(&now); + if (free_only) { + /* we really don't free it, just reset to zero windows */ + stack_list_num = 0; + last_free = now; + return; + } + + if (stack_list_num && now < last_snap + allowed_age) { + return; + } + + stack_list_num = 0; + last_free = now; + + X_LOCK; + rc = XQueryTree(dpy, rootwin, &r, &w, &list, &ui); + num = (int) ui; + + if (! rc) { + stack_list_num = 0; + last_free = now; + last_snap = 0.0; + X_UNLOCK; + return; + } + + last_snap = now; + if (num > stack_list_len + blackouts) { + int n = 2*num; + free(stack_list); + stack_list = (winattr_t *) malloc(n*sizeof(winattr_t)); + stack_list_len = n; + } + j = 0; + for (i=0; i<num; i++) { + stack_list[j].win = list[i]; + stack_list[j].fetched = 0; + stack_list[j].valid = 0; + stack_list[j].time = now; + j++; + } + for (i=0; i<blackouts; i++) { + stack_list[j].win = 0x1; + stack_list[j].fetched = 1; + stack_list[j].valid = 1; + stack_list[j].x = blackr[i].x1; + stack_list[j].y = blackr[i].y1; + stack_list[j].width = blackr[i].x2 - blackr[i].x1; + stack_list[j].height = blackr[i].y2 - blackr[i].y1; + stack_list[j].time = now; + stack_list[j].map_state = IsViewable; + stack_list[j].rx = -1; + stack_list[j].ry = -1; + j++; + +if (0) fprintf(stderr, "blackr: %d %dx%d+%d+%d\n", i, + stack_list[j-1].width, stack_list[j-1].height, + stack_list[j-1].x, stack_list[j-1].y); + + } + stack_list_num = num + blackouts; + if (debug_wireframe > 1) { + fprintf(stderr, "snapshot_stack_list: num=%d len=%d\n", + stack_list_num, stack_list_len); + } + + XFree(list); + X_UNLOCK; +} + +void update_stack_list(void) { + int k; + double now; + XWindowAttributes attr; + + if (! stack_list) { + return; + } + if (! stack_list_num) { + return; + } + + dtime0(&now); + + for (k=0; k < stack_list_num; k++) { + Window win = stack_list[k].win; + if (win != None && win < 10) { + ; /* special, blackout */ + } else if (!valid_window(win, &attr, 1)) { + stack_list[k].valid = 0; + } else { + stack_list[k].valid = 1; + stack_list[k].x = attr.x; + stack_list[k].y = attr.y; + stack_list[k].width = attr.width; + stack_list[k].height = attr.height; + stack_list[k].depth = attr.depth; + stack_list[k].class = attr.class; + stack_list[k].backing_store = attr.backing_store; + stack_list[k].map_state = attr.map_state; + + /* root_x, root_y not used for stack_list usage: */ + stack_list[k].rx = -1; + stack_list[k].ry = -1; + } + stack_list[k].fetched = 1; + stack_list[k].time = now; + } +if (0) fprintf(stderr, "update_stack_list[%d]: %.4f %.4f\n", stack_list_num, now - x11vnc_start, dtime(&now)); +} + +Window query_pointer(Window start) { + Window r, c; + int rx, ry, wx, wy; + unsigned int mask; + if (start == None) { + start = rootwin; + } + if (XQueryPointer(dpy, start, &r, &c, &rx, &ry, &wx, &wy, &mask)) { + return c; + } else { + return None; + } +} + +int pick_windowid(unsigned long *num) { + char line[512]; + int ok = 0, n = 0, msec = 10, secmax = 15; + FILE *p; + + if (use_dpy) { + set_env("DISPLAY", use_dpy); + } + if (no_external_cmds) { + rfbLogEnable(1); + rfbLog("cannot run external commands in -nocmds mode:\n"); + rfbLog(" \"%s\"\n", "xwininfo"); + rfbLog(" exiting.\n"); + clean_up_exit(1); + } + p = popen("xwininfo", "r"); + + if (! p) { + return 0; + } + + fprintf(stderr, "\n"); + fprintf(stderr, " Please select the window for x11vnc to poll\n"); + fprintf(stderr, " by clicking the mouse in that window.\n"); + fprintf(stderr, "\n"); + + while (msec * n++ < 1000 * secmax) { + unsigned long tmp; + char *q; + fd_set set; + struct timeval tv; + + if (screen && screen->clientHead) { + /* they may be doing the pointer-pick thru vnc: */ + int nfds; + tv.tv_sec = 0; + tv.tv_usec = msec * 1000; + FD_ZERO(&set); + FD_SET(fileno(p), &set); + + nfds = select(fileno(p)+1, &set, NULL, NULL, &tv); + + if (nfds == 0 || nfds < 0) { + /* + * select timedout or error. + * note this rfbPE takes about 30ms too: + */ + rfbPE(-1); + XFlush(dpy); + continue; + } + } + + if (fgets(line, 512, p) == NULL) { + break; + } + q = strstr(line, " id: 0x"); + if (q) { + q += 5; + if (sscanf(q, "0x%lx ", &tmp) == 1) { + ok = 1; + *num = tmp; + fprintf(stderr, " Picked: 0x%lx\n\n", tmp); + break; + } + } + } + pclose(p); + return ok; +} + +Window descend_pointer(int depth, Window start, char *name_info, int len) { + Window r, c, clast; + int i, rx, ry, wx, wy; + int written = 0, filled = 0; + char *store = NULL; + unsigned int m; + static XClassHint *classhint = NULL; + static char *nm_cache = NULL; + static int nm_cache_len = 0; + static Window prev_start = None; + + if (! classhint) { + classhint = XAllocClassHint(); + } + + if (! nm_cache) { + nm_cache = (char *) malloc(1024); + nm_cache_len = 1024; + nm_cache[0] = '\0'; + } + if (name_info && nm_cache_len < len) { + if (nm_cache) { + free(nm_cache); + } + nm_cache_len = 2*len; + nm_cache = (char *) malloc(nm_cache_len); + } + + if (name_info) { + if (start != None && start == prev_start) { + store = NULL; + strncpy(name_info, nm_cache, len); + } else { + store = name_info; + name_info[0] = '\0'; + } + } + + if (start != None) { + c = start; + if (name_info) { + prev_start = start; + } + } else { + c = rootwin; + } + + for (i=0; i<depth; i++) { + clast = c; + if (store && ! filled) { + char *name; + if (XFetchName(dpy, clast, &name) && name != NULL) { + int l = strlen(name); + if (written + l+2 < len) { + strcat(store, "^^"); + written += 2; + strcat(store, name); + written += l; + } else { + filled = 1; + } + XFree(name); + } + } + if (store && classhint && ! filled) { + classhint->res_name = NULL; + classhint->res_class = NULL; + if (XGetClassHint(dpy, clast, classhint)) { + int l = 0; + if (classhint->res_class) { + l += strlen(classhint->res_class); + } + if (classhint->res_name) { + l += strlen(classhint->res_name); + } + if (written + l+4 < len) { + strcat(store, "##"); + if (classhint->res_class) { + strcat(store, + classhint->res_class); + } + strcat(store, "++"); + if (classhint->res_name) { + strcat(store, + classhint->res_name); + } + written += l+4; + } else { + filled = 1; + } + if (classhint->res_class) { + XFree(classhint->res_class); + } + if (classhint->res_name) { + XFree(classhint->res_name); + } + } + } + if (! XQueryPointer(dpy, c, &r, &c, &rx, &ry, &wx, &wy, &m)) { + break; + } + if (! c) { + break; + } + } + if (start != None && name_info) { + strncpy(nm_cache, name_info, nm_cache_len); + } + + return clast; +} + + |