需要解决HMONITOR - > Windows中的deviceName(或deviceName - > HMONITOR)

时间:2013-04-01 15:27:09

标签: delphi winapi windows-7 delphi-7

编辑 - 请参阅结束时的更新

这适用于Delphi 7.0 Build 4.453

摘要

我需要能够从TMonitor对象(TScreen组件中的Monitors数组中的一个元素)获取Handle属性,该对象是一个HMONITOR,并将其转换为您在调用{{3}时使用的字符串}作为lpszDeviceName参数。

(我的最终目标是通过将已解析的lpszDeviceName传递给EnumDisplaySettings的调用,从给定的HMONITOR值获取设备设置列表。)

详细信息

如上所述,Screen.Monitors [x] .Handle属性的类型为HMONITOR,通常用于传递EnumDisplaySettings函数,该函数返回几何信息,但不返回lpszDeviceName。 (注意:有一个GetMonitorInfo结构有一个szDevice字段,但它似乎没有填入我的系统,即使我将cbSize字段设置为适当的大小)。

或者,如果我可以使用szDeviceName来获取等效的HMONITOR值,我可以将它插入到以下函数中,这将在比较中使用它(我已插入对虚函数的调用)在下面的代码中调用 hMonitorFromDeviceName 来指示它将如何使用。

function GetMonitorDeviceName(hmon : HMONITOR) : string;
var
  DispDev : TDisplayDevice;
  deviceName : string;
  nDeviceIndex : integer;
begin
  Result := '';

  FillChar(DispDev, sizeof(DispDev),0);
  DispDev.cb := sizeof(DispDev);

  nDeviceIndex := 0;
  while (EnumDisplayDevices(nil, nDeviceIndex, DispDev, 0)) do
  begin

     if ( hMonitorFromDeviceName(DispDev.DeviceString) = hmon ) then
     begin
        Result := StrPas(DispDev.DeviceString);
        exit;
     end;

     inc(nDeviceIndex);

  end;
end;

更新

感谢David Heffernan,我测试了他的解决方案,这是一个示例函数,用于从给定句柄中获取监视器名称:

function GetMonitorName(hmon : HMONITOR) : string;
type
  TMonitorInfoEx = record
    cbSize: DWORD;
    rcMonitor: TRect;
    rcWork: TRect;
    dwFlags: DWORD;
    szDevice: array[0..CCHDEVICENAME - 1] of AnsiChar;
end;
var
  DispDev : TDisplayDevice;
  deviceName : string;
   monInfo : TMonitorInfoEx;
begin
  Result := '';

  monInfo.cbSize := sizeof(monInfo);
  if GetMonitorInfo(hmon,@monInfo) then
  begin

    DispDev.cb := sizeof(DispDev);
     EnumDisplayDevices(@monInfo.szDevice, 0, DispDev, 0);
     Result := StrPas(DispDev.DeviceString);

  end;
end;

1 个答案:

答案 0 :(得分:5)

我认为您必须正确调用GetMonitorInfo。这段代码:

{$APPTYPE CONSOLE}

uses
  SysUtils, MultiMon, Windows, Forms;

var
  i: Integer;
  MonitorInfo: TMonitorInfoEx;
begin
  MonitorInfo.cbSize := SizeOf(MonitorInfo);
  for i := 0 to Screen.MonitorCount-1 do
  begin
    if not GetMonitorInfo(Screen.Monitors[i].Handle, @MonitorInfo) then
      RaiseLastOSError;
    Writeln(MonitorInfo.szDevice);
  end;
  Readln;
end.

在我的机器上生成此输出:

\\.\DISPLAY1
\\.\DISPLAY2

我怀疑您对GetMonitorInfo的调用在某种程度上失败了,也许您没有检查错误的返回值。


搜索过QualityCentral后,我怀疑你已经成为旧版Delphi中已知错误的受害者:QC#3239。报告已在版本10.0.2124.6661(Delphi 2006版本)中修复。


您的评论确认了此诊断。要解决此问题,您需要一个新的TMonitorInfoEx定义。这是一个适用于您的Unicode前Delphi的文件:

type
  TMonitorInfoEx = record
    cbSize: DWORD;
    rcMonitor: TRect;
    rcWork: TRect;
    dwFlags: DWORD;
    szDevice: array[0..CCHDEVICENAME - 1] of AnsiChar;
  end;

如果您将其添加到上面的代码中(当然是在声明变量之前),那么我相信它会解决您的问题。


有趣的是,即使在XE3中,这些结构也没有被正确翻译:QC#114460。不可否认,错误是相当温和的,因为它只会影响PMonitorInfoExATMonitorInfoExA,但是在尝试解决此问题时,错误引起了我的注意!