在我的TComponent中,有一个点需要侦听键事件并在组件中拦截ESC键并处理它,使用/“吃”击键,这样所有者表单就不会在那个阶段处理它。就像在TDragObject中一样,当您开始执行拖动并通过按ESC取消它。
问题是TDragObject的AllocateHWnd是由它的所有者表单CN_KEYDOWN通知的。但没人通知我的组件。
我需要用自己的WindowProc替换表单的吗?如果是的话,那么如何正确地“按书”来做呢?
仅仅是100%的澄清:
TMyComponent = class(TComponent)我做了一个小小的测试,它似乎奏效了:
TMyComponent = class(TComponent)
private
FOldWindowProc: TWndMethod;
FParentForm: TCustomForm;
procedure FormWindowProc(var Message: TMessage);
public
constructor Create(AOwner: TComponent); override;
destructor Destroy; override;
end;
...
constructor TMyComponent.Create(AOwner: TComponent);
begin
if not (AOwner is TWinControl) then
raise Exception.Create('TMyComponent.Create: Owner must be a TWinControl');
inherited Create(AOwner);
// hook parent form
FParentForm := GetParentForm(TWinControl(Owner));
if Assigned(FParentForm) then
begin
FOldWindowProc := FParentForm.WindowProc;
FParentForm.WindowProc := FormWindowProc;
end;
end;
destructor TMyComponent.Destroy;
begin
// unhook parent form
if Assigned(FParentForm) then
FParentForm.WindowProc := FOldWindowProc;
inherited;
end;
procedure TMyComponent.FormWindowProc(var Message: TMessage);
begin
FOldWindowProc(Message);
if Message.Msg = CM_CHILDKEY then // CM_CHILDKEY -> CM_DIALOGKEY -> CM_DIALOGCHAR
begin
OutputDebugString('CM_CHILDKEY');
if Message.WParam = VK_ESCAPE then
begin
Beep;
// do my stuff...
Message.Result := 1; // consume keystroke
end;
end;
end; 我想知道这是否是正确/唯一的办法。
发布于 2013-01-16 01:42:53
一种方法可能是在组件内部创建一个TApplicationEvents对象,然后使用它的OnMessage事件来查看主线程消息队列中的消息,例如击键,然后在其他OnMessage处理它们之前。
https://stackoverflow.com/questions/14347351
复制相似问题