xcb:EnterNotify后立即收到LeaveNotify

问题描述

我正在使用xcb库在Rust中编写用于学习的窗口管理器。我的代码和几个测试窗口(xterm实例)都在Xephyr会话中运行。我在根窗口上将事件掩码设置为

xproto::EVENT_MASK_SUBSTRUCTURE_REDIRECT
| xproto::EVENT_MASK_SUBSTRUCTURE_NOTIFY
| xproto::EVENT_MASK_POINTER_MOTION
| xproto::EVENT_MASK_LEAVE_WINDOW
| xproto::EVENT_MASK_ENTER_WINDOW
| xproto::EVENT_MASK_BUTTON_PRESS
| xproto::EVENT_MASK_PROPERTY_CHANGE
| xproto::EVENT_MASK_FOCUS_CHANGE

所有子窗口的事件掩码为

xproto::EVENT_MASK_ENTER_WINDOW
| xproto::EVENT_MASK_LEAVE_WINDOW
| xproto::EVENT_MASK_BUTTON_PRESS
| xproto::EVENT_MASK_PROPERTY_CHANGE
| xproto::EVENT_MASK_POINTER_MOTION
| xproto::EVENT_MASK_FOCUS_CHANGE
| xproto::EVENT_MASK_STRUCTURE_NOTIFY
| xproto::EVENT_MASK_EXPOSURE

当我将鼠标移到非根窗口上时,我使用EnterNotify事件来抓住该窗口上的鼠标按钮(以实现单击焦点),并放弃LeaveNotify上的位置。事件的顺序是:

  • 鼠标在根窗口上
  • 将鼠标移到非根窗口上
  • 为非根窗口接收EnterNotify
  • 非根窗口上的抓取按钮
  • 当鼠标仍位于非根窗口上方时,立即收到LeaveNotify
  • 由于LeaveNotify导致按钮无法抓取
  • 尝试单击非根窗口
  • 接收第二个LeaveNotify用于非根窗口
  • 在根窗口中收到EnterNotify,然后在根窗口上抓住鼠标按钮
  • ButtonPress发送到根窗口,尽管光标位于非根窗口上

我真的不确定这可能是什么原因;使用Google之类的工具并没有什么用。


对于以后可能偶然发现此问题的任何人,部分解决方案是:

  • 仅监听窗口输入事件以获取窗口上的按钮
  • 将根设为仅SUBSTRUCTURE_REDIRECT | SUBSTRUCTURE_NOTIFY | BUTTON_PRESS
  • 请勿将LEAVE_WINDOW屏蔽在非根窗口上,并且不要处理这些事件

我不想将此添加为答案,因为:

  1. 我不确定这是正确的方法。
  2. 我还没有对此进行彻底的测试。
  3. 它没有回答我的主要问题“为什么会收到这样的事件?”

解决方法

看一下协议参考手册。它(其中包括许多其他事情)描述了生成进入和离开事件的确切算法:https://www.x.org/releases/X11R7.6/doc/xproto/x11protocol.html#events:pointer_window

在您的特定情况下,我希望LeaveNotify事件具有mode: Grab,这意味着该窗口不再具有“正常”指针焦点,因为某些东西(您的程序)已捕获了输入

如果这不是答案,我建议您在xtrace / x11trace下运行您的WM(在基于Debian的发行版中作为软件包xtrace提供)。该程序将打印“通过”的所有X11通信。这可能有助于弄清发生了什么。

相关问答

错误1:Request method ‘DELETE‘ not supported 错误还原:...
错误1:启动docker镜像时报错:Error response from daemon:...
错误1:private field ‘xxx‘ is never assigned 按Alt...
报错如下,通过源不能下载,最后警告pip需升级版本 Requirem...