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

时间:2016-01-25 08:02:50

标签: windows winapi multiple-monitors

我希望获得一个监视器句柄(HMONITOR),该句柄可用于Windows多监视器A​​PI,用于通过索引连接到系统特定监视器 。例如,假设我有三台显示器连接到我的系统并构成我桌面的一部分;我想得到一个监控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));

1 个答案:

答案 0 :(得分:5)

很抱歉这么晚的回复,但也许有人会觉得这很有用。

多监视器A​​PI至少可以说是极简主义。获得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!
}