【问题标题】:How can my TComponent intercept the ESC key and handle it?我的 TComponent 如何拦截 ESC 键并进行处理?
【发布时间】:2013-01-15 21:33:33
【问题描述】:

在我的TComponent 中,有一点我想监听键事件并拦截ESC 键并在我的组件中处理它,消耗/“吃”击键,例如所有者表单不会在那个阶段处理它。就像在TDragObject 中一样,当您开始拖动并按ESC 取消它时。

问题是TDragObjectAllocateHWnd,由其所有者表单通知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; 

我想知道这是否是正确/唯一的方法。

【问题讨论】:

  • 拖动涉及一个新的模态循环。这不是你的选择。如果没有主人的合作,很难看出如何干净利落地做到这一点。
  • @DavidHeffernan,OP 说“就像在 TDragObject 中一样”,这只是一个例子,我假设 OP 只想要 ESC 键,仅此而已。 ESC 是一个对话框键。我现在没有 1 分钟的时间来查找代码/API/windows 消息。
  • @Cosmin 拖动操作的模态循环拥有并抽取队列。因此可以掌握按键。但是表单上的组件没有那么奢侈。你打算如何进入应用的消息循环?
  • 如果 " 有一个点我想听关键事件" 仅表示特定的持续时间,您不必在整个生命周期内更换窗口过程的组件。顺便说一句,还有其他方法,例如安装消息挂钩 (SetWindowsHookEx),或拥有自己的消息循环...我想讨论 正确 方法需要了解确切的场景。
  • 检查是否收到 WM_CANCELMODE 消息。这是 Windows 告诉您 Esc 键(或其他取消捕获/拖动的事件)必须结束的常用方式。

标签: delphi delphi-7


【解决方案1】:

一种方法可能是在组件内部创建一个TApplicationEvents 对象,然后使用它的OnMessage 事件在VCL 的其余部分处理它们之前从主线程消息队列中查看消息,例如击键.

【讨论】:

  • 组件真的可以做到这一点吗?如果应用程序分配 OnMessage 怎么办?这不会是冲突吗?
  • TApplicationEvents 被设计为一个多播类。多个实例接收相同的事件。 IOW,当单个消息到达队列时,所有分配的TApplicationEvents.OnMessage 处理程序都会得到它,如果每个人都使用TApplicationEvents,那就是。如果有人直接使用TApplication.OnMessage,那么TApplicationEvents.OnMessage 可能会中断,反之亦然,是的。这不是一个完美的系统,但总比没有好。
  • OnMessage 来自多个 TApplicationEvents 组件的事件将按创建这些组件时的相反顺序触发。
  • 你认为这比像我一样挂钩 WindowProc 更好吗?
  • 另一种方法是使用SetWindowsHookEx() 创建线程特定的键盘钩子或getmessage/wndproc 钩子。这比尝试挂钩 VCL 事件更加孤立。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2012-12-26
  • 1970-01-01
相关资源
最近更新 更多