1

我开始用 C++ 学习 Qt,并试图弄清楚事件机制是如何工作的。我有点理解它是如何在应用程序本身中工作的,以及 QApplication 如何处理事件并将它们排队以处理和通知对象/小部件。

我仍然有一些悬而未决的问题:

  1. 窗口系统如何通知 QApplication 本身有关事件的信息?
  2. 窗口系统如何知道我的小部件的内部命名,以便它可以告诉 QApplication 小部件“x”被单击、聚焦等。
  3. 如果窗口系统不知道小部件名称或没有对单击的小部件(或任何其他类型的事件)的引用,QApplication 如何确定需要通知哪个小部件。

在此先感谢您的帮助。

4

2 回答 2

2

在 X11 上,Qt 轮询 xlib (Qt4) 或 XCB (Qt5) 的事件。使用 xlib,这将是 XNextEvent() 函数和其他相关函数:

http://www.x.org/archive/X11R7.5/doc/man/man3/XMaskEvent.3.html

如果您通过 grep 查找“XNextEvent”的 Qt 源代码,您可以找到使用它的地方。例如,在 Qt 4.8.4 源代码中:

$ grep -r XNextEvent
工具/qvfb/x11keyfaker.cpp: XNextEvent(dpy, &event);
src/plugins/platforms/xlib/qxlibscreen.cpp: XNextEvent(mDisplay->nativeDisplay(), &event);
src/gui/kernel/qwidget_x11.cpp: XNextEvent(X11->display, &ev);
src/gui/kernel/qguieventdispatcher_glib.cpp: XNextEvent(X11->display, &event);
src/gui/kernel/qeventdispatcher_x11.cpp: XNextEvent(X11->display, &event);
src/gui/kernel/qapplication_x11.cpp: XNextEvent(X11->display, &nextEvent);

对于 XCB,有 xcb_wait_for_event()。

一旦 Qt 有了事件,它就可以开始通过小部件层次结构和 QML 分派它。

当然,这仅适用于 X11(即 Unix 和 Linux)。其他平台有不同的方式将事件传递给应用程序。

于 2013-02-18T06:40:00.663 回答
1

例如,在 Windows 中,QWidget 只是一个带有 hwnd 和所有其他相关内容的普通窗口。您甚至可以通过调用 EffectiveWinId() 方法来获取 hwnd。它使用标准的 Windows 消息循环来获取所有 Windows 事件的通知,您可以通过 QCoreApplication::setEventFilter 安装适当的过滤器来获得通知。但是 Qt 也有 QML 部分和其他非本地的东西,为了让它工作,它使用了一些内部簿记,这取决于“外星小部件”的性质

对于非外星小部件,上述所有内容都是正确的,而外星小部件则有所不同。Alien widgets 由 Qt 本身管理,它是一个 Qt 工作来识别应该传播什么 QWidget 动作以及动作是什么。Qt 从 4.4 开始默认使用外星小部件,但如果需要,它们可以很容易地被异化。

其他操作系统在实现上有所不同,但想法是相同的。

于 2013-02-18T06:45:45.593 回答