Delphi指针,数组,句柄

时间:2013-07-12 17:09:09

标签: delphi pointers enums delphi-xe4 window-handles

我无法理解使用自定义记录的指针并创建记录数组,然后指向该记录,特别是我想要实现的是使用给定的类名为每个顶级窗口句柄创建记录,意思是有超过1个,对于每个窗口,我使用EnumChildWindow来获取子窗口的句柄。我想记录每个窗口并将其传递给带有标题的组合框并使记录成为那个项目,所以我可以通过选择它来访问以后记录的每个项目。

我的问题是我对指针的处理,我在第一个添加任何一个数组记录的实例上继续访问被拒绝。

在这里休息

Param[Form1.iEnumWin].MainHwnd:= aHwnd;

这是我正在使用的完整代码,以便人们可以更好地了解我想要做的事情。

implementation

{$R *.dfm}
type
  TMyEnumParam = record
   sTitle: String;
   MainHwnd: Hwnd;
   InTxtHwnd: Hwnd;
   OutTxtHwnd: Hwnd;
   NickListHwnd: Hwnd;
end;
 PMyEnumParam = ^TMyEnumParam;

 type
 ATMyEnumParam = Array[0..9] of PMyEnumParam;
 PATMyEnumParam = ^ATMyEnumParam;

{ Get the window Title based on Hwnd }
function GetWindowTitle(HWND: HWND): string;
begin
  SetLength(Result, 255);
  SetLength(Result, GetWindowText(HWND, PChar(Result), 255));
end;

{ Get the Classname based on Hwnd }
function GetWindowClass(HWND: HWND): string;
begin
  SetLength(Result, 255);
  SetLength(Result, GetClassName(HWND, PChar(Result), 255));
end;

{ EnumChildWidows Callback Add to our records }
Function EnumChildProc(aHwnd: Hwnd; Param: PMyEnumParam): Boolean; stdcall;
begin
     if ((GetDlgCtrlID(aHwnd) = 202) and (isWindowVisible(aHwnd) = True)) then
      Param.InTxtHwnd:= aHwnd;

     if ((GetDlgCtrlID(aHwnd) = 203) and (isWindowVisible(aHwnd) = True)) then
       Param.OutTxtHwnd:= aHwnd;

     if ((GetDlgCtrlID(aHwnd) = 1789) and (isWindowVisible(aHwnd) = True)) then
      Param.NickListHwnd:= aHwnd;

      Result:= True;
end;

{ EnumWindow fill our array of records for each window }
function EnumWindowsProc(aHwnd: HWND; Param: PATMyEnumParam): BOOL; stdcall;
begin
  Result := True;
  if GetWindowClass(aHwnd) = 'DlgGroupChat Window Class' then
  begin
   Param[Form1.iEnumWin].MainHwnd:= aHwnd;
   Param[Form1.iEnumWin].sTitle:=  GetWindowTitle(aHwnd);
   EnumChildWindows(aHwnd, @EnumChildProc, LParam(@Param[Form1.iEnumWin]));
   Form1.cbbRooms.AddItem(Param[Form1.iEnumWin].sTitle, TObject(Param[form1.iEnumWin]));
   inc(Form1.iEnumWin);
  end;
end;

{ On change display room Title for each item }
procedure TForm1.cbbRoomsChange(Sender: TObject);
var
  i: Integer;
  aHwnd: PMyEnumParam;
begin
  i := cbbRooms.ItemIndex;
  if cbbRooms.ItemIndex <> -1 then
  begin
    aHwnd:=  PMyEnumParam(cbbRooms.Items.Objects[i]);
    if aHwnd.MainHwnd > 0 then
    begin
     ShowMessage(aHwnd.sTitle);
    end;
  end;

end;

{ Call EnumWindows and fill our array records }
procedure TForm1.FormCreate(Sender: TObject);
var
 arInfo: PATMyEnumParam;
begin
  iEnumWin:= 0;
  EnumWindows(@EnumWindowsProc, LParam(@arInfo));
end;

如果有人能指出(没有双关语)我正朝着正确的方向前进,我将不胜感激。

1 个答案:

答案 0 :(得分:4)

您的代码有很多问题。这是一个非详尽的清单:

  1. 您没有为数组分配任何存储空间。
  2. 您将^PATMyEnumParam传递给EnumWindows,然后在回调中投放到PATMyEnumParam
  3. 您的数组是固定长度的,并且您不会尝试处理对数组的超出界限。
  4. 但是你最大的问题是你的代码试图在行走之前运行。它具有完整的复杂性和您需要的所有功能。然而,您还无法成功拨打EnumWindows

    我最重要的建议不是细节,而是解决问题的一般性。首先编写一段简单的代码。明白它。然后加强它。

    因此,在这方面,以下是如何调用EnumerateWindows

    program EnumWindowsDemo_17620346;
    
    {$APPTYPE CONSOLE}
    
    uses
      System.SysUtils, Winapi.Windows, Generics.Collections;
    
    type
      TWindowInfo = record
        Handle: HWND;
        // expand with more fields in due course
      end;
    
    function EnumWindowProc(hwnd: HWND; lParam: LPARAM): BOOL; stdcall;
    var
      WindowList: TList<TWindowInfo>;
      WindowInfo: TWindowInfo;
    begin
      WindowList := TList<TWindowInfo>(lParam);
      WindowInfo.Handle := hwnd;
      WindowList.Add(WindowInfo);
      Result := True;
    end;
    
    procedure Main;
    var
      WindowList: TList<TWindowInfo>;
      WindowInfo: TWindowInfo;
    begin
      WindowList := TList<TWindowInfo>.Create;
      try
        EnumWindows(@EnumWindowProc, LPARAM(WindowList));
        for WindowInfo in WindowList do
          Writeln(WindowInfo.Handle);
      finally
        WindowList.Free;
      end;
    end;
    
    begin
      Main;
      Readln;
    end.
    

    从这里开始你可以扩展这个概念,因为所有棘手的部分都已经处理好了。特别是指针,转换和内存管理。

相关问题