Qt如何枚举屏幕?

cbu*_*art 6 c++ windows qt multiple-monitors qt5

今天我发现Qt枚举screen(QGuiApplication::screens)的顺序与Windows(EnumDisplayMonitors)中的顺序不同.

这种差异背后的逻辑是什么,以便在混合使用Windows API和Qt时将其考虑在内?例如,如果需要在屏幕#2中显示某些内容(使用Windows枚举).

这里是我用来测试的代码(也可以在GitHub中使用):

#include <qapplication.h>
#include <qdebug.h>
#include <qscreen.h>
#include <Windows.h>
#include <iostream>

std::ostream& operator<<(std::ostream& of, const RECT& rect)
{
  return of << "RECT(" << rect.left << ", " << rect.top << " " << (rect.right - rect.left) << "x" << (rect.bottom - rect.top) << ")";
}

BOOL CALLBACK printMonitorInfoByHandle(HMONITOR hMonitor, HDC hdcMonitor, LPRECT lprcMonitor, LPARAM dwData)
{
  auto index = (int*)dwData;
  std::cout << ++*index << " " << *lprcMonitor << std::endl;
  return TRUE;
}

int main(int argc, char* argv[])
{
  QApplication a(argc, argv);

  qDebug() << "*** Qt screens ***";
  const auto screens = qApp->screens();
  for (int ii = 0; ii < screens.count(); ++ii) {
    qDebug() << ii + 1 << screens[ii]->geometry();
  }

  qDebug() << "*** Windows monitors ***";
  int index = 0;
  EnumDisplayMonitors(NULL, NULL, printMonitorInfoByHandle, (LPARAM)&index);

  return 0;
}
Run Code Online (Sandbox Code Playgroud)

我的显示配置是从左到右,2(1280x1024),3(1920x1080),1(1920x1080),是我的主屏幕3.

结果:

*** Qt screens ***
1 QRect(0,0 1920x1080)
2 QRect(1920,233 1920x1080)
3 QRect(-1280,47 1280x1024)
*** Windows monitors ***
1 RECT(1920, 233 1920x1080)
2 RECT(-1280, 47 1280x1024)
3 RECT(0, 0 1920x1080)
Run Code Online (Sandbox Code Playgroud)

cbu*_*art 5

就我在不同系统中看到的而言,EnumDisplayMonitors按照显示设置中定义的顺序返回显示器,同时QGuiApplication::screens始终在第一个位置显示主屏幕(实际上,QGuiApplication::primaryScreen只需这样做:返回第一个元素)。

查看源代码,在Windows Qt中也使用了 EnumDisplayMonitors功能,但基本上将主屏幕移动到第一个位置(它实际上是在第一个位置插入主屏幕,同时在列表末尾插入任何其他监视器)。

因此,主屏幕将在第一个位置,索引低于主屏幕的屏幕将移动一个位置,而其余屏幕将与索引匹配。


作为旁注,从代码的注释中可以看出,如果在应用程序执行期间主屏幕发生了变化,Qt 将无法报告该变化。

请注意,此策略的副作用是无法更改 Qt 报告的主屏幕,除非我们想删除所有现有屏幕并在主屏幕更改时再次添加它们。