是否可以知道剪贴板的来源(应用程序)?

是否可以知道剪贴板的来源(应用程序)?

我注意到,如果源应用程序(从中复制内容)关闭,有时剪贴板内容将变得不可用。

这让我想知道是否有可能知道源应用程序是什么(例如也许通过 PID)。

为什么?如果源应用程序是终端,我想找到终端的工作目录,以防复制的内容是相对路径,以构造文件的完整路径。

仅供参考,我目前正在使用 xclip 来确定剪贴板内容,例如

xclip -selection primary -t STRING -o 2> /dev/null

答案1

我编写了一个工具,它返回普通的应用程序名称(例如“Terminal”、“gedit”或“SmartGit”,这是我测试的)。大多数代码都是从@Harvey 那里无耻地窃取的这里

// gcc clipboard-owner.c -lX11 -o clipboard-owner

#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <X11/Xlib.h>
#include <X11/Xatom.h>

#define MAX_PROPERTY_VALUE_LEN 4096

typedef unsigned long ulong;

static char *get_property(Display *, Window, Atom , const char *, ulong *);

int main(void) {
    // Open the Display
    Display *display = XOpenDisplay(NULL);

    // Get the selection window
    Window selection_owner = XGetSelectionOwner(display, XA_PRIMARY);
    
    if(!selection_owner) {
        exit(0);
    } else {
          char *window_name = get_property(display, selection_owner, XA_STRING, "WM_NAME", NULL);
          printf("%s\n", window_name);
    }

    XCloseDisplay(display);
}

static char *get_property (Display *disp, Window win,
        Atom xa_prop_type, const char *prop_name, ulong *size) {
    Atom xa_prop_name;
    Atom xa_ret_type;
    int ret_format;
    ulong ret_nitems;
    ulong ret_bytes_after;
    ulong tmp_size;
    unsigned char *ret_prop;
    char *ret;

    xa_prop_name = XInternAtom(disp, prop_name, False);

    if (XGetWindowProperty(disp, win, xa_prop_name, 0,
            MAX_PROPERTY_VALUE_LEN / 4, False,
            xa_prop_type, &xa_ret_type, &ret_format,     
            &ret_nitems, &ret_bytes_after, &ret_prop) != Success) {
        printf("Cannot get %s property.\n", prop_name);
        return NULL;
    }

    if (xa_ret_type != xa_prop_type) {
        printf("Invalid type of %s property.\n", prop_name);
        XFree(ret_prop);
        return NULL;
    }

    /* null terminate the result to make string handling easier */
    tmp_size = (ret_format / 8) * ret_nitems;
    /* Correct 64 Architecture implementation of 32 bit data */
    if(ret_format==32) tmp_size *= sizeof(long)/4;
    ret = (char *)malloc(tmp_size + 1);
    memcpy(ret, ret_prop, tmp_size);
    ret[tmp_size] = '\0';

    if (size) {
        *size = tmp_size;
    }

    XFree(ret_prop);
    return ret;
}

相关内容