为指定的父窗口枚举子窗口、按钮
很早就写过类似spy++和查看密码窗口的东西,一直想给这个小东西再加点特别的。前段时间对软件安装注册发生了兴趣,有些软件如果你不输入正确注册码,那该死的“下一步”按钮就一直disable。这次我就让spy++彻底spy到底,把那个注册用的按钮置亮,让我轻松进入“下一步”,呵呵...。
我的想法是光标移到指定的窗口上后,探测这个窗口上到底有多少按钮,如果有,就将它们都Enable。在这里我不想讨论怎样具体实现这个功能,但你要知道的是想得到这些被disable窗口(按钮)的句柄是根本无法通过WindowFromPoint这个API函数得到的,GetWindow也不要妄想了。顺手查了下MSDN,看到EnumChildWindows可是个好东西,可以枚举一个父窗口的所有子窗口:
- BOOL EnumChildWindows(
- HWND hWndParent, // handle to parent window // 父窗口句柄
- WNDENUMPROC lpEnumFunc, // callback function // 回调函数的地址
- LPARAM lParam // application-defined value // 你自已定义的参数
- );
就这么简单,让我们再定义一个回调函数,像下面这样:
- BOOL CALLBACK EnumChildProc(
- HWND hwnd, // handle to child window
- LPARAM lParam // application-defined value
- );
注意:这个回调函数要么是类的静态函数,要么就是一个全局的函数。
--------------------------------
在调用EnumChildWindows 这个函数时,直到调用到最个一个子窗口被枚举或回调函数返回一个false,否则将一直枚举下去。有了上面的知识,我想你应该知道怎么做了。有了回调函数的概念及上面的例子,我们可以继续了。其实想要找到一个标题已知的窗口句柄,用一个API函数就可以了:FindWindow.其函数原形是:
- function FindWindow(lpClassName, lpWindowName: PChar): HWND; stdcall;
- lpClassName:窗口类名.如果只知道标题,可以为空.窗口类名可以用很多工具获得.如winsignt32.
lpWindowName:窗口标题.
调用方式举例:
- var wndhwnd:HWND;
- wndhwnd:=FindWindow(nil,'某窗口标题');
- if wndhwnd<>0 then file://找到此窗口句柄.
- begin
- xxxxx
- end
- else begin
- MessageBox(self.handle,'没找到该窗口句柄','提示',0);
- end;
有了这个窗口句柄,就离我们的初始目的不远了:控制其他窗体上的窗口控件.同样,首先要得到其他窗体上窗口控件的句柄.我们用这个API函数:EnumChildWindows.其函数原形是:
- function EnumChildWindows(hWndParent: HWND;
- lpEnumFunc: TFNWndEnumProc;
- lParam: LPARAM): BOOL; stdcall;
这个函数和EnumWindow函数很有些想象.其作用也很相似.它的功能就是列举窗口句柄为hWndParent的窗体上所有的窗口控件的句柄.同样也是以回调函数参数的形式给出的. 我们再举一个实际的例子,来说明这个函数的用法.程序的功能是让用户输入一个窗口标题,然后调用FindWindow函数找到此窗口句柄.通过这个句柄,我们在一个Memo里显示该窗口上所有的窗口控件.同样先编写回调函数.
- function EnumChildWndProc(AhWnd:LongInt;
- AlParam:lParam):boolean;stdcall;
- var
- WndClassName: array[0..254] of Char;
- WndCaption: array[0..254] of Char;
- begin
- GetClassName(AhWnd,wndClassName,254);
- GetWindowText(aHwnd,WndCaption,254);
- with form1.memo1 do
- begin
- lines.add( string(wndClassName));
- lines.add( string(wndCaption));
- lines.add('-------');
- end;
- result:=true;
- end;
然后在一事件里调用EnumChildWindows函数.
- procedure TForm1.Button1Click(Sender: TObject);
- var
- hWnd:LongInt;
- begin
- memo1.Lines.Clear;
- Memo1.Lines.Add(Edit1.Text+' 有如下控件类名称');
- hWnd:=FindWindow(nil,pchar(Edit1.Text));
- if hWnd<>0 then
- begin
- EnumChildWindows(hWnd,@EnumChildWndProc,0);
- end
- else MessageBox(self.handle,'没找到该窗口句柄','提示',0);
- end;
程序清单如下:
- unit Unit1;
- interface
- uses
- Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
- Dialogs, StdCtrls;
- type
- TForm1 = class(TForm)
- Memo1: TMemo; //用来显示找到的控件
- Label1: TLabel;
- Edit1: TEdit; //输入标题.
- Button1: TButton;
- procedure Button1Click(Sender: TObject);
- private
- { Private declarations }
- public
- { Public declarations }
- end;
- var
- Form1: TForm1;
- function EnumChildWndProc(AhWnd:LongInt;
- AlParam:lParam):boolean;stdcall;
- implementation
- {$R *.dfm}
- function EnumChildWndProc(AhWnd:LongInt;
- AlParam:lParam):boolean;stdcall;
- var
- WndClassName: array[0..254] of Char;
- WndCaption: array[0..254] of Char;
- begin
- GetClassName(AhWnd,wndClassName,254);
- GetWindowText(aHwnd,WndCaption,254);
- with form1.memo1 do
- begin
- lines.add( string(wndClassName));
- lines.add( string(wndCaption));
- lines.add('-------');
- end;
- result:=true;
- end;
- procedure TForm1.Button1Click(Sender: TObject);
- var
- hWnd:LongInt;
- begin
- memo1.Lines.Clear;
- Memo1.Lines.Add(Edit1.Text+' 有如下控件类名称');
- hWnd:=FindWindow(nil,pchar(Edit1.Text));
- if hWnd<>0 then
- begin
- EnumChildWindows(hWnd,@EnumChildWndProc,0);
- end
- else MessageBox(self.handle,'没找到该窗口句柄','提示',0);
- end;
- end.
有了控件句柄,我们当然就可以随心所欲了.比如:
- SendMessage(hWnd,WM_SETTEXT,0,LongInt(Pchar('sdafdsf')));
就可以给控件发送文本.其他还可以发送不同的消息可以做很多事情.但是,有很大一个问题:假设一个窗体上有很多相同的控件,并且根本没办法区分他们,即使我们能找到所有的控件句柄,我们又不能区分到底哪个是我们想要的,同样是干着急.我想了很长时间,后来在大富翁里找到了答案,只要用到一个小技巧,就可以解决了.
Trackback: http://tb.blog.youkuaiyun.com/TrackBack.aspx?PostId=588250