Xlib XGetWindowProperty返回零项

时间:2015-07-02 14:48:28

标签: 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);
}

在收到ClientMessage后调用get_property_value函数,这是处理事件的代码片段:

    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-------

1 个答案:

答案 0 :(得分:3)

我还没有足够的代表发表评论(我对此有点新意),但我一直在花一些时间自己摔跤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返回“输出到打印原子名称而不是二进制数据。无论如何,我将这个版本的代码指向我计算机上运行的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:

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

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