13

我想监控X11下所有打开的窗口。目前,我这样做如下:

  1. 最初通过从根窗口递归调用 XQueryTree 来遍历整个树
  2. 监听整个桌面的子结构变化:XSelectInput( display, root_window, SubstructureNotifyMask | PropertyChangeMask )
  3. 处理所有 MapNotify、UnmapNotify 和 DestroyNotify 事件,在进程中更新我自己的窗口列表

我主要担心第1点。在递归过程中,XQueryTree会被多次调用。有什么办法可以确保树在此期间不会改变?换句话说,在某个时间点获取整棵树的“快照”?

另外,我注意到在某些 X11 系统下,并非所有事件都正确到达。例如,当在桌面上打开一个新窗口时,该窗口的 MapNotify 可能永远不会到达我的监控应用程序。怎么会这样?是否有可能在到达之前被丢弃?

更新:

我编写了一个小程序来监视根窗口上的 X 事件(见下文)。现在,当我运行这个程序并启动和退出 xcalc 时,我得到以下输出:

Reparented: 0x4a0005b to 0x1001e40
Mapped    : 0x1001e40
Destroyed : 0x1001e40

而已。我从来没有收到关于真实窗口(0x4a0005b)被破坏的通知。甚至没有被映射!谁能告诉我为什么不呢?SubStructureNotifyMask 是否只导致发送直接子窗口的事件而不是整个子树?

顺便说一句,这显然不会在 Compiz 运行时发生。然后不进行重新父项:

Mapped    : 0x4a0005b
Mapped    : 0x4e00233
Destroyed : 0x4a0005b
Destroyed : 0x4e00233

监控程序来源:

#include <X11/Xlib.h>
#include <cstdio>

int main()
{
    Display *display;
    Window rootwin;

    display = XOpenDisplay( NULL );
    rootwin = DefaultRootWindow( display );
    XSelectInput( display, rootwin, SubstructureNotifyMask );

    XEvent event;

    while ( 1 ) {
        XNextEvent( display, &event );
        if ( event.type == MapNotify ) {
            XMapEvent *mapevent = (XMapEvent *)&event;
            printf( "Mapped    : 0x%x\n", (unsigned int)(mapevent->window) );
        }
        if ( event.type == DestroyNotify ) {
            XDestroyWindowEvent *destroywindowevent = (XDestroyWindowEvent *)&event;
            printf( "Destroyed : 0x%x\n", (unsigned int)(destroywindowevent->window) );
        }
        if ( event.type == ReparentNotify ) {
            XReparentEvent *reparentevent = (XReparentEvent *)&event;
            printf( "Reparented: 0x%x to 0x%x\n", (unsigned int)(reparentevent->window), (unsigned int)(reparentevent->parent) );
        }
    }

    return 0;
}
4

3 回答 3

16

看看xwininfo

您可能还喜欢xpropxspy获取更多信息。

更新:是的。尝试使用and 或者xwininfo-root所有窗口都参与进来。-tree-children

并且可以使用 跟踪更改xprop -spy

于 2009-06-19T14:17:50.087 回答
3

我相信抓取 X 服务器 (XGrabServer(3)) 将阻止对窗口层次结构的更改。虽然它有点重,所以你应该只在你真的需要它时才这样做。

有关遍历窗口层次结构、构建树、使用窗口事件使其保持最新以及忽略由于竞争而不可避免的 X 协议错误的代码示例,请参见文件src/VBox/Additions/x11/VBoxClient /seamless-x11.cpp在 VirtualBox 的源代码中。

于 2010-03-17T11:53:04.187 回答
1

X11 是一个远程协议。这意味着当您向 X 服务器查询任何信息时,您总是会得到自己的副本。当 X 服务器更新其内部数据结构时,您的副本永远不会改变。

这意味着当您遍历树时,树不会突然改变,但是当您使用其中的信息(例如检查窗口)时,该信息可能是陈旧的(有人可能已经关闭了窗口)。这就是为什么您需要进行适当的错误处理的原因。

于 2013-08-15T07:54:03.847 回答