问题描述
我正在使用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
屏蔽在非根窗口上,并且不要处理这些事件
我不想将此添加为答案,因为:
- 我不确定这是正确的方法。
- 我还没有对此进行彻底的测试。
- 它没有回答我的主要问题“为什么会收到这样的事件?”
解决方法
看一下协议参考手册。它(其中包括许多其他事情)描述了生成进入和离开事件的确切算法:https://www.x.org/releases/X11R7.6/doc/xproto/x11protocol.html#events:pointer_window
在您的特定情况下,我希望LeaveNotify
事件具有mode: Grab
,这意味着该窗口不再具有“正常”指针焦点,因为某些东西(您的程序)已捕获了输入
如果这不是答案,我建议您在xtrace / x11trace下运行您的WM(在基于Debian的发行版中作为软件包xtrace
提供)。该程序将打印“通过”的所有X11通信。这可能有助于弄清发生了什么。