检测聚焦控制变化的灵活方法

Wod*_*dzu 6 delphi components focus delphi-2009

我需要编写一个组件,它将在其他组件中注册,并检测其中一个注册组件是否获得焦点.

例如,对于我的组件,TFocusObserver我正在注册三个对象.

FocusObserver.Register(MyMemo);
FocusObserver.Register(MyButton);
FocusObserver.Register(MyEdit);
Run Code Online (Sandbox Code Playgroud)

现在,如果其中一个组件获得焦点,那么FocusObserver就会启动一些通知事件.

我正在寻找如何检测焦点变化,并发现这TScreen.OnActiveControlChange正是我需要的.所以我的组件可以连接到这个事件.问题是不止一个TFocusObserver可能存在,或者在将来可能想要使用的其他somoene中存在OnActiveControlChange.

这是我将从多播事件中受益的时间 - 它将立即解决我的问题.

我在想如何解决这个问题,目前我有两个想法:

  1. 以某种方式扩展TScreen所以它将为我提供一个更多的事件.
  2. 引入一个中间对象,该对象将挂钩OnActiveControlChange并为其他对象公开一个多播事件.

在简要介绍一下这些来源之后,我不清楚如何通过使用第一个想法来解决它,而第二个想法的缺点是有人可以简单地指定另一种方法,OnActiveControlChange并且一切都会崩溃.

会对一些建议表示感谢.

bal*_*azs 9

如果你的focusObserver类可以是TWinControl的后代,那么你可以这样做:

TFocusObserver = class( TWinControl )

  procedure CMFocusChanged(var Message: TCMFocusChanged); message CM_FOCUSCHANGED;
end;
Run Code Online (Sandbox Code Playgroud)

procedure TFocusObserver.CMFocusChanged(var Message: TCMFocusChanged);
var
  LControl: TWinControl;

begin
      LControl := TWinControl(Message.Sender);

      if LControl <> nil then
      begin
        form1.Caption := lControl.Name;
      end;
end;
Run Code Online (Sandbox Code Playgroud)

这里的主要想法是观看CM_FOCUSCHANGED.

第二种方法:

注册控件时,请更换它WindowProc.这是一个小代码片段:

TRegisteredComp = class
  private
    fControl: TControl;
    fowndproc: TWndMethod;
    procedure HookWndProc(var Message: TMessage);
  public
    constructor Create( c: TControl );
    destructor Destroy; override;
  end;

  TFocusObserver = class
  private
    l: TList;
   public
    constructor Create;
    destructor Destroy; override;
    procedure reg( c: TControl );

  end;
Run Code Online (Sandbox Code Playgroud)

并在实施中:

constructor TFocusObserver.Create;
begin
  l := TList.Create;
end;

destructor TFocusObserver.Destroy;
var i: integer;
begin
  for i := 0 to l.Count - 1 do
    TRegisteredComp(l[i]).Free;
  l.Free;
  inherited;
end;

procedure TFocusObserver.reg( c: TControl );
var
  rc: TRegisteredComp;
begin
  rc := TRegisteredComp.Create( c );
  l.Add( rc );
end;

constructor TRegisteredComp.Create(c: TControl);
begin
  fControl := c;
  fowndproc := c.WindowProc;
  c.WindowProc := HookWndProc;
end;

destructor TRegisteredComp.Destroy;
begin
  fControl.WindowProc := fowndproc;
  inherited;
end;

procedure TRegisteredComp.HookWndProc(var Message: TMessage);
begin
  if ( Message.Msg = CM_FOCUSCHANGED ) and
    ( TControl(Message.LParam) = fControl ) then
    form1.ListBox1.Items.Add( 'focused: ' + fControl.Name );

  fowndproc( Message );
end;
Run Code Online (Sandbox Code Playgroud)

而不仅仅是注册你想要观看的控件,例如:

procedure TForm1.FormCreate(Sender: TObject);
var
  i: Integer;
begin
  fo := TFocusObserver.Create;
  for i := 0 to ControlCount - 1 do
    fo.reg( Controls[i] );
end;
Run Code Online (Sandbox Code Playgroud)

听起来怎么样?