如何将鼠标事件重定向到另一个控件?

时间:2012-12-13 20:26:46

标签: delphi delphi-7

我的情况是TImageTPanel部分涵盖了它,并且他们共享同一个父母:

------------------
|  Image1        |
|  ------------  |
|  |  Panel1  |  |
|  ------------  |
|                |
------------------

Panel1正在接收鼠标按下/移动/向上事件并对其进行处理(Image1也是如此),但在某些情况下,我想将鼠标按下“重定向”到Image1,就像模拟单击Image1而不是Panel1一样

这是我做的:

procedure TForm1.Image1MouseDown(Sender: TObject; Button: TMouseButton;
  Shift: TShiftState; X, Y: Integer);
begin
  if (ssLeft in Shift) then
    Beep;
end;

procedure TForm1.Image1MouseMove(Sender: TObject; Shift: TShiftState; 
  X, Y: Integer);
begin
  //...
end;

procedure TForm1.Image1MouseUp(Sender: TObject; Button: TMouseButton;
  Shift: TShiftState; X, Y: Integer);
begin
  ShowMessage('boo!');
end;

procedure TForm1.Panel1MouseDown(Sender: TObject; Button: TMouseButton;
  Shift: TShiftState; X, Y: Integer);
var
  P: TPoint;
begin
  if FRedirectToImage then begin
    ReleaseCapture; // do I need to send a WM_LBUTTONUP as well to the panel?        
    GetCursorPos(P);
    P := ScreenToClient(P);
    Image1.Perform(WM_LBUTTONDOWN, MK_LBUTTON, Longint(PointToSmallPoint(P)));
    Exit;
  end;

  // Normal handling
  if (ssLeft in Shift) then begin
    // ...
  end;
end;

它按预期工作但我不确定这是正确的方式。
我的问题是,我做得对吗?有更好的或推荐的方式吗?


更新(1): 按建议处理WM_NCHITTEST是一个有效答案,我也考虑过了。即使将Panel1.Enabled设置为False,也会将鼠标消息路由到基础Image1控件。

但是(!)考虑这种情况,我单击Panel上的x位置,仍然需要将消息路由到Image1:

------------------
|  Image1        |
|          --------------
|          |  Panel1  x |
|          --------------
|                |
------------------

我的方法有效,但WM_NCHITTEST不适用于所描述的方案。如果我的方法有效,我仍然没有得到答案。 (或者我可以用上面的场景问另一个问题?)

3 个答案:

答案 0 :(得分:7)

处理发送到面板的wm_NCHitTest条消息并返回htTransparent。操作系统会将鼠标消息发送到下一个控件,而无需程序中的任何进一步处理。 (从操作系统的角度来看,“下一个控件向下”是面板和图像的父控件; VCL负责将鼠标消息路由回图像控件,就像所有TGraphicControl个后代一样,因为它们不是真正的窗口控件。)

这样的事情:

procedure TParentForm.PanelWindowProc(var Msg: TMessage);
begin
  FPrevPanelWindowProc(Msg);
  if (Msg.Message = wm_NCHitTest) and FRedirectToImage then
    Msg.Result := htTransparent;
end;

将该方法分配给面板的WindowProc方法。将属性的先前值存储在表单的字段中。

var
  FPrevPanelWindowProc: TWndMethod;

FPrevPanelWindowProc := Panel.WindowProc;
Panel.WindowProc := Self.PanelWindowProc;

答案 1 :(得分:5)

您可以派生面板类来处理WM_NCHITTEST消息,以便返回HTTRANSPARENT您希望面板下方控件接收鼠标消息的区域。 E.g:

procedure TMyPanel.WMNCHitTest(var Message: TWMNCHitTest);
var
  Pt: TPoint;
begin
  Pt := ScreenToClient(SmallPointToPoint(Message.Pos));
  if (Pt.X < 80) and (Pt.Y < 60) then // devise your logic here...
    Message.Result := HTTRANSPARENT
  else
    inherited;
end;

显然这只是一个测试,您可以在组件中发布一个字段,以便解析该控件所在的位置等。

答案 2 :(得分:5)

如果您想要重定向鼠标事件的控件不在其整个客户区 inside 中,那么应该将这些事件重定向到的控件(正如您在问题中所示)更新),然后WM_NCHITTEST消息可能会发送到另一个控件。然后唯一的方法仍然是使用IMHO,重定向所有鼠标消息。

正如@David在评论中提到的那样,您可以通过为OnMessageTApplication事件编写事件处理程序,以全局方式执行此消息重定向。或者使用TApplicationEvents对象。

在以下示例中,您可以定义要重定向的消息范围,以及指定该重定向的源和目标控件列表。对于重定向使用OnMessage对象的TApplication事件,但由于您的目标是TGraphicControl后代,因此您不仅可以更改传入邮件的收件人,还可以必须通过Perform方法自己吃这条消息并在目标控件上执行消息。

以下是显示如何将所有鼠标消息从Panel1重定向到Image1的代码。如果需要,您可以获得整个测试项目from here

unit Unit1;

interface

uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, StdCtrls, ExtCtrls;

type
  TMsgRange = record
    MsgFrom: UINT;
    MsgTo: UINT;
  end;
  TRedirect = record
    Source: HWND;
    Target: TControl;
  end;

type
  TForm1 = class(TForm)
    Memo1: TMemo;
    Panel1: TPanel;
    Image1: TImage;
    procedure FormCreate(Sender: TObject);
    procedure Image1MouseDown(Sender: TObject; Button: TMouseButton;
      Shift: TShiftState; X, Y: Integer);
    procedure Panel1MouseDown(Sender: TObject; Button: TMouseButton;
      Shift: TShiftState; X, Y: Integer);
    procedure Image1MouseUp(Sender: TObject; Button: TMouseButton;
      Shift: TShiftState; X, Y: Integer);
    procedure Panel1MouseUp(Sender: TObject; Button: TMouseButton;
      Shift: TShiftState; X, Y: Integer);
  private
    FRedirectList: array of TRedirect;
    FRedirectEnabled: Boolean;
    FRedirectMsgRange: TMsgRange;
    procedure ApplicationMessage(var AMessage: TMsg; var Handled: Boolean);
  public
    { Public declarations }
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}

procedure TForm1.ApplicationMessage(var AMessage: TMsg; var Handled: Boolean);
var
  I: Integer;
begin
  if FRedirectEnabled and (AMessage.message >= FRedirectMsgRange.MsgFrom) and
    (AMessage.message <= FRedirectMsgRange.MsgTo) then
  begin
    for I := 0 to High(FRedirectList) do
      if (AMessage.hwnd = FRedirectList[I].Source) and
        Assigned(FRedirectList[I].Target) then
      begin
        Handled := True;
        FRedirectList[I].Target.Perform(AMessage.message,
          AMessage.wParam, AMessage.lParam);
        Break;
      end;
  end;
end;

procedure TForm1.FormCreate(Sender: TObject);
begin
  FRedirectEnabled := True;
  FRedirectMsgRange.MsgFrom := WM_MOUSEFIRST;
  FRedirectMsgRange.MsgTo := WM_MOUSELAST;
  SetLength(FRedirectList, 1);
  FRedirectList[0].Source := Panel1.Handle;
  FRedirectList[0].Target := Image1;
  Application.OnMessage := ApplicationMessage;
end;

procedure TForm1.Image1MouseDown(Sender: TObject; Button: TMouseButton;
  Shift: TShiftState; X, Y: Integer);
begin
  Memo1.Lines.Add('Image1MouseDown')
end;

procedure TForm1.Image1MouseUp(Sender: TObject; Button: TMouseButton;
  Shift: TShiftState; X, Y: Integer);
begin
  Memo1.Lines.Add('Image1MouseUp')
end;

procedure TForm1.Panel1MouseDown(Sender: TObject; Button: TMouseButton;
  Shift: TShiftState; X, Y: Integer);
begin
  Memo1.Lines.Add('Panel1MouseDown')
end;

procedure TForm1.Panel1MouseUp(Sender: TObject; Button: TMouseButton;
  Shift: TShiftState; X, Y: Integer);
begin
  Memo1.Lines.Add('Panel1MouseUp')
end;

end.