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

Posted

技术标签:

【中文标题】如何从显示设备名称中获取 HMONITOR 句柄?【英文标题】:How can I get an HMONITOR handle from a display device name? 【发布时间】:2016-01-25 08:02:50 【问题描述】:

我想获得一个监视器句柄 (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));

【问题讨论】:

【参考方案1】:

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

多显示器 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!

【讨论】:

以上是关于如何从显示设备名称中获取 HMONITOR 句柄?的主要内容,如果未能解决你的问题,请参考以下文章

已知窗体的句柄,如何显示出此窗体 C#.ne

获取蓝牙低功耗设备的设备句柄

Matlab:如何获取图形句柄中的所有轴句柄?

如何在 C++ 中按名称获取进程句柄?

通过镜像名称获取某个进程的进程句柄

我如何在 Windows 中获取正在运行的程序的窗口句柄 [关闭]