システム全体とイベントの通知を受け取るためのフックSetWindowsHookEx
を設定するために使用しようとしています。ドキュメントによると、 「フックプロシージャを、呼び出し元のスレッドと同じデスクトップで実行されているすべての既存のスレッドに関連付ける」必要がある最後の引数に0を渡します。また、私が見たすべての例と同様に、パラメーターのハンドルを( Delphiの)DLLに渡します。WH_SHELL
HSHELL_WINDOWCREATED
HSHELL_WINDOWDESTROYED
dwThreadId
HInstance
hMod
それでも、自分のアプリによって作成されたウィンドウの通知を受け取ることはありません。ほとんどの場合、アプリを閉じると、テストの結果、デスクトッププロセスが炎上します。あなたが尋ねる前に、私は電話しますUnhookWindowsHookEx
。また、常にCallNextHookEx
ハンドラー内から呼び出します。
限られたユーザーアカウントからテストアプリを実行していますが、これまでのところ、これが役割を果たすことを示すヒントは見つかりませんでした...(実際には驚きますが)
AFAICT、私は本ですべてをしました(明らかに私はしませんでしたが、今のところどこにいるのかわかりません)。
私はDelphi(2007)を使用していますが、それはそれほど重要ではないと思います。
編集:多分私は前にこれについて言及するべきでした:私はいくつかの例をダウンロードして試しました(残念ながらDelphiで利用できるものはそれほど多くありません-特にWH_SHELL
またはの場合はありませんWH_CBT
)。テストアプリのようにシステムをクラッシュさせることはありませんが、他のプロセスからのイベントをキャプチャすることはありません(ProcessExplorerでそれらが正常に読み込まれることを確認できますが)。したがって、システム構成に問題があるか、例が間違っているか、他のプロセスからイベントをキャプチャできないようです。誰かが私を啓発できますか?
EDIT2: OK、これが私のテストプロジェクトのソースです。
フックプロシージャを含むDLL:
library HookHelper;
uses
Windows;
{$R *.res}
type
THookCallback = procedure(ACode, AWParam, ALParam: Integer); stdcall;
var
WndHookCallback: THookCallback;
Hook: HHook;
function HookProc(ACode, AWParam, ALParam: Integer): Integer; stdcall;
begin
Result := CallNextHookEx(Hook, ACode, AWParam, ALParam);
if ACode < 0 then Exit;
try
if Assigned(WndHookCallback)
// and (ACode in [HSHELL_WINDOWCREATED, HSHELL_WINDOWDESTROYED]) then
and (ACode in [HCBT_CREATEWND, HCBT_DESTROYWND]) then
WndHookCallback(ACode, AWParam, ALParam);
except
// plop!
end;
end;
procedure InitHook(ACallback: THookCallback); register;
begin
// Hook := SetWindowsHookEx(WH_SHELL, @HookProc, HInstance, 0);
Hook := SetWindowsHookEx(WH_CBT, @HookProc, HInstance, 0);
if Hook = 0 then
begin
// ShowMessage(SysErrorMessage(GetLastError));
end
else
begin
WndHookCallback := ACallback;
end;
end;
procedure UninitHook; register;
begin
if Hook <> 0 then
UnhookWindowsHookEx(Hook);
WndHookCallback := nil;
end;
exports
InitHook,
UninitHook;
begin
end.
そして、フックを使用したアプリのメインフォーム:
unit MainFo;
interface
uses
Windows, SysUtils, Forms, Dialogs, Classes, Controls, Buttons, StdCtrls;
type
THookTest_Fo = class(TForm)
Hook_Btn: TSpeedButton;
Output_Lbx: TListBox;
Test_Btn: TButton;
procedure Hook_BtnClick(Sender: TObject);
procedure Test_BtnClick(Sender: TObject);
public
destructor Destroy; override;
end;
var
HookTest_Fo: THookTest_Fo;
implementation
{$R *.dfm}
type
THookCallback = procedure(ACode, AWParam, ALParam: Integer); stdcall;
procedure InitHook(const ACallback: THookCallback); register; external 'HookHelper.dll';
procedure UninitHook; register; external 'HookHelper.dll';
procedure HookCallback(ACode, AWParam, ALParam: Integer); stdcall;
begin
if Assigned(HookTest_Fo) then
case ACode of
// HSHELL_WINDOWCREATED:
HCBT_CREATEWND:
HookTest_Fo.Output_Lbx.Items.Add('created handle #' + IntToStr(AWParam));
// HSHELL_WINDOWDESTROYED:
HCBT_DESTROYWND:
HookTest_Fo.Output_Lbx.Items.Add('destroyed handle #' + IntToStr(AWParam));
else
HookTest_Fo.Output_Lbx.Items.Add(Format('code: %d, WParam: $%x, LParam: $%x', [ACode, AWParam, ALParam]));
end;
end;
procedure THookTest_Fo.Test_BtnClick(Sender: TObject);
begin
ShowMessage('Boo!');
end;
destructor THookTest_Fo.Destroy;
begin
UninitHook; // just to make sure
inherited;
end;
procedure THookTest_Fo.Hook_BtnClick(Sender: TObject);
begin
if Hook_Btn.Down then
InitHook(HookCallback)
else
UninitHook;
end;
end.