如何从显示设备名称获取 HMONITOR 句柄



我想获取一个监视器句柄(HMONITOR),该句柄可以与Windows多监视器API一起使用,用于通过索引连接到系统的特定监视器。例如,假设我有三台显示器连接到我的系统,并构成我的桌面的一部分;我想得到一个手柄来监控 3。

我已经知道如何通过调用 EnumDisplayDevices 函数按索引获取特定监视器的设备名称。例如:

HMONITOR MonitorFromIndex(int index /* (zero-indexed) */)
{
    DISPLAY_DEVICE dd;
    dd.cb = sizeof(dd);
    if (EnumDisplayDevices(NULL, index, &dd, 0) != FALSE)
    {
       // We found a match; make sure that it's part of the desktop.
       if ((dd.StateFlags & DISPLAY_DEVICE_ATTACHED_TO_DESKTOP) == DISPLAY_DEVICE_ATTACHED_TO_DESKTOP)
       {
          // Yup. Now we've got the name of the device:
          std::cout << dd.DeviceName << std::endl;
          // But how do I obtain an HMONITOR for this device?
          // ...
       }
    }
   return NULL;  // indicate failure
}

在上面的代码中,我们找到了所需设备的名称(dd.DeviceName)。我可以使用此名称通过调用 CreateDC 为该监视器创建 DC:

HDC hDC = CreateDC(dd.DeviceName, dd.DeviceName, NULL, NULL);

我可以通过调用EnumDisplaySettings来获取有关该监视器的信息:

DEVMODE dm;
dm.dmSize        = sizeof(dm);
dm.dmDriverExtra = 0;
if (EnumDisplaySettings(dd.DeviceName, ENUM_CURRENT_SETTINGS, &dm) != FALSE)
{
    std::cout << "The monitor supports " << dm.dmBitsPerPel << " bits per pixel." << std::endl;
}

这一切都很棒,但我想要那个显示器的手柄。我怎样才能得到它?

我尝试调用EnumDisplayMonitors,将句柄传递给我使用 CreateDC 创建的设备上下文,希望获得传递给回调函数的监视器的句柄,但没有这样的运气。回调函数从未被调用,EnumDisplayMonitors返回FALSE(未设置错误代码):

struct FoundMatch
{
   BOOL     found;
   HMONITOR hMonitor;
};
BOOL CALLBACK MonitorEnumProc(HMONITOR hMonitor, HDC, LPRECT, LPARAM dwData)
{
   FoundMatch* pfm = reinterpret_cast<FoundMatch*>(dwData);
   pfm->found    = TRUE;
   pfm->hMonitor = hMonitor;
   return FALSE;  // stop enumerating
}
// elsewhere, after getting the device name and using it to create a DC
FoundMatch fm;
fm.found    = FALSE;
fm.hMonitor = NULL;
BOOL result = EnumDisplayMonitors(hDC, NULL, MonitorEnumProc, reinterpret_cast<LPARAM>(&fm));

很抱歉回复这么晚,但也许有人会发现这很有用。

至少可以说,多显示器 API 确实是极简主义的。一旦你得到了你的dd.DeviceName,看来你必须经历EnumDisplayMonitors()枚举,直到你找到dd.DeviceNameMONITORINFOEX.szDevice的匹配。

MONITORINFOEX结构可以通过调用 GetMonitorInfo() 来获取。

下面是一个不可编译的 C++11 伪代码:

struct DataBag
{
    HMONITOR hmon;
    TCHAR* devname;
} bag;
bag.hmon = NULL;
bag.devname = &dd.DeviceName;
BOOL bRes = EnumDisplayMonitors(
    NULL, NULL,
    [](HMONITOR hMonitor, HDC hDC, LPRECT rc, LPARAM data) -> BOOL {
        auto& bag = *reinterpret_cast<DataBag*>(data);
        MONITORINFOEX mi;
        mi.cbSize = sizeof(mi);
        if (/* match bag.devname against mi.szDevice */ && GetMonitorInfo(hMonitor, &mi))
        {
            bag.hmon = hMonitor;
            return FALSE;
        }
        return TRUE;
    },
    reinterpret_cast<LPARAM>(&bag));
if (bRes && bag.hmon)
{
    // Monitor found!
}

最新更新