c - Xlib XGetWindowProperty 返回零项

标签 c x11 xlib

我在阅读某些 XWindow ICCCM 属性时遇到问题。

问题实际上是当我尝试读取 _NET_WM_STATUS 属性时。
我正在使用的功能是:

int get_property_value(Display* display, Window window,char *propname, long max_length,
           unsigned long *nitems_return, unsigned char **prop_return){
    int result;
    Atom property;
    Atom actual_type_return;
    int actual_format_return;
    unsigned long bytes_after_return;
    unsigned char* prop_to_return;
    unsigned long n_items;
    printf("-----GET_PROPERTY_VALUE-------\n");
    printf("\tPropname: %s\n", propname);
    property = XInternAtom(display, propname, True);
    if(property==None){
      printf("\tWrong Atom\n");
      return;
    }

    result = XGetWindowProperty(display, window, property, 0,   /* long_offset */
            (~0L),  /* long_length */
            False,  /* delete */
            AnyPropertyType,    /* req_type */
            &actual_type_return,
            &actual_format_return,
            &n_items, &bytes_after_return, &prop_to_return);
    if (result != Success){
        printf("\tXGetWindowProperty failed\n");
        return (-1);
    }   else {
        printf("\tActual Type: %s\n", XGetAtomName(display,property));
        printf("\tProperty format: %d\n", actual_format_return);
        //printf("Actual property return: %s\n", XGetAtomName(display,actual_type_return));
        printf("\tByte after return: %ld\n", bytes_after_return);
        printf("\tnitems return: %d\n", n_items);
        printf("\tprop return: %s\n", prop_to_return);
    }
    printf("-----END OF GET_PROPERTY_VALUE-------\n");

    return (0);
}

get_property_value 函数在收到 ClientMessage 后被调用,这是处理事件的代码段:
    case ClientMessage:
        printf("ClientMessage\n");
        printf("Message: %s\n", XGetAtomName(display,local_event.xclient.message_type));
        unsigned long nitems_return;
        unsigned char *prop_return;
        get_property_value(display, local_event.xclient.window, XGetAtomName(display,local_event.xclient.message_type), 256, &nitems_return, (unsigned char **)&prop_return);
    break;

但是当我读取属性时,有时我会得到一个没有值的属性,这可能吗?
问题主要是当我在 ClientMessage 事件中尝试读取从 firefox 发送的 AtomProperties(我正在尝试读取 _NET_WM_STATE 值)时。

正如您从输出中看到的那样,正确读取了属性名称,但它似乎不包含任何项目。
ClientMessage
Message: _NET_WM_STATE
-----GET_PROPERTY_VALUE-------
    Propname: _NET_WM_STATE
    Actual Type: _NET_WM_STATE
    Property format: 0
    Byte after return: 0
    nitems return: 0
    prop return: (null)
-----END OF GET_PROPERTY_VALUE-------

最佳答案

我还没有足够的代表来发表评论(我对此有点陌生),但我自己花了一些时间与 Xlib 搏斗,我会尝试看看是否可以提供帮助。我写了一个包含你的代码的小程序:

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

int get_property_value(Display* display, Window window,char *propname, long max_length,
           unsigned long *nitems_return, unsigned char **prop_return){
    int result;
    Atom property;
    Atom actual_type_return;
    int actual_format_return;
    unsigned long bytes_after_return;
    unsigned char* prop_to_return;
    unsigned long n_items;
    printf("-----GET_PROPERTY_VALUE-------\n");
    printf("\tPropname: %s\n", propname);
    property = XInternAtom(display, propname, True);
    if(property==None){
      printf("\tWrong Atom\n");
      return;
    }

    result = XGetWindowProperty(display, window, property, 0,   /* long_offset */
            (~0L),  /* long_length */
            False,  /* delete */
            AnyPropertyType,    /* req_type */
            &actual_type_return,
            &actual_format_return,
            &n_items, &bytes_after_return, &prop_to_return);
    if (result != Success){
        printf("\tXGetWindowProperty failed\n");
        return (-1);
    }   else {
        printf("\tActual Type: %s\n", XGetAtomName(display,actual_type_return));
        printf("\tProperty format: %d\n", actual_format_return);
        //printf("Actual property return: %s\n", XGetAtomName(display,actual_type_return));
        printf("\tByte after return: %ld\n", bytes_after_return);
        printf("\tnitems return: %d\n", n_items);
        printf("\tprop return: %s %s\n", XGetAtomName(display,*(Atom*)prop_to_return), XGetAtomName(display,((Atom*)prop_to_return)[1]));
    }
    printf("-----END OF GET_PROPERTY_VALUE-------\n");

    return (0);
}

int main(int argc, char** argv) {
    Display* dsp = XOpenDisplay(NULL);
    unsigned long nitems_return;
    unsigned char* prop_return;
    get_property_value(dsp, (Window)atoi(argv[1]), "_NET_WM_STATE", 100000, &nitems_return, &prop_return);
    return 0;
}

我确实更改了一些输出内容:我更改了“实际类型”输出以打印名称 actual_type_return而不是 property (因为这似乎是一个错字),我更改了“prop return”输出以打印原子名称而不是二进制数据。无论如何,我将这个版本的代码指向我计算机上运行的 Firefox 实例,这就是我得到的:
$ ./xproptest 60817587
-----GET_PROPERTY_VALUE-------
    Propname: _NET_WM_STATE
    Actual Type: ATOM
    Property format: 32
    Byte after return: 0
    nitems return: 2
    prop return: _NET_WM_STATE_MAXIMIZED_VERT _NET_WM_STATE_MAXIMIZED_HORZ
-----END OF GET_PROPERTY_VALUE-------

所以,好消息是你的代码实际上在我的机器上运行得很好。不过,我不确定为什么它对你的不起作用。当然,上面的输出是Firefox最大化时的;如果不是,则输出是这样的:
$ ./xproptest 60817587
-----GET_PROPERTY_VALUE-------
    Propname: _NET_WM_STATE
    Actual Type: ATOM
    Property format: 32
    Byte after return: 0
    nitems return: 0
    prop return:

…然后是关于我如何读取无效 Atom 的一些错误,因为上面的代码只是假设有两个。这是当属性为空且没有值时的正确输出;但是,请注意我的属性格式是 32,而你的属性格式是 0。从阅读手册来看,这似乎是成功调用 XGetWindowProperty 后获得 0 属性格式的唯一方法。是在不存在的属性上调用它。我不知道为什么你的 Firefox 没有 _NET_WM_STATE设置,不过。

总之,我的猜测是您的代码实际上运行良好(除了其中一个输出上的拼写错误),但是由于某种原因,您尝试读取的属性在您尝试读取的窗口上不存在它从。如果我是对的,如果你检查 actual_type_return 的值看完这个属性你应该会发现它是None .此外,您应该能够(如果您还没有)打印您试图在代码中读取的窗口 ID( window )并使用 xwininfoxprop-id标志,以检查您正在读取的窗口是否是您认为的那个窗口,以及它是否确实具有您要读取的属性。希望这将帮助您弄清楚实际问题是什么。

关于c - Xlib XGetWindowProperty 返回零项,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31187800/

相关文章:

c - 在C中生成从-n到n的随机数

c# - 我可以使用 Xcode 编写 Excel 插件吗?

c - 无限循环和指针

c - 为什么 XKeysymToKeycode() 将我的所有键都设为小写?

c++ - 将 QWidget 嵌入到 X11 窗口中

xlib - XChangeProperty() 总是失败

C宏打开或关闭引脚

c++ - 从 X11 请求图像,用 C 或 C++ 合成 WM

Python:文本覆盖在所有窗口顶部,包括 Linux 中的全屏

c - 如何在 Linux 中检测通过 GUI 登录的用户