5

我使用 Delphi XE2 构建了下面的代码。它创建 Form1,Form1 立即创建 Form2 的一个实例。当我按下 Form2 上的按钮时,会创建第二个 Form2。

现在,如果我将鼠标悬停在第二个最顶部的 Form2 上的按钮上,然后等待工具提示出现,在工具提示出现的那一刻,第一个Form2 就出现在前面,抢走了焦点。

仅当Application.MainFormOnTaskbar是时才会出现问题True。它还依赖于从 Form1 的FormCreate方法创建的第一个 Form2。如果我使用PostMessage()延迟创建第一个 Form2 直到应用程序完成初始化,问题就会消失。

我想了解为什么会这样。我已经了解到 Delphi 的 Application 对象处理了很多事情,包括提示显示,并且我知道 Delphi 可以在初始化期间重新创建一个窗口的句柄,但是我无法按照这个来完全解释上述行为(或确实上述两个事实是否相关)。

项目1.dpr

program Project1;

uses
  Vcl.Forms,
  Unit1 in 'Unit1.pas' {Form1},
  Unit2 in 'Unit2.pas' {Form2};

{$R *.res}

begin
  Application.Initialize;
  Application.MainFormOnTaskbar := True; // False makes problem go away
  Application.CreateForm(TForm1, Form1);
  Application.Run;
end.

单元1.pas

unit Unit1;
interface
uses
  Vcl.Forms, Unit2;

type
  TForm1 = class(TForm)
    procedure FormCreate(Sender: TObject);
  public
    procedure CreateForm2;
  end;

var
  Form1: TForm1;

implementation
{$R *.dfm}

procedure TForm1.FormCreate(Sender: TObject);
begin
  CreateForm2;
end;

procedure TForm1.CreateForm2;
var
  frm : TForm2;
begin
  frm := TForm2.Create(Application); // (Could pass Self - makes no difference)
  frm.Show;
end;

end.

单元2.pas

unit Unit2;
interface
uses
  Vcl.Forms, System.Classes, Vcl.Controls, Vcl.StdCtrls, WinApi.Windows;

type
  TForm2 = class(TForm)
    Button1: TButton; // This button has a hint
    procedure Button1Click(Sender: TObject);
  end;

var
  Form2: TForm2;

implementation
uses
  System.SysUtils, Unit1;

{$R *.dfm}

procedure TForm2.Button1Click(Sender: TObject);
begin
  Form1.CreateForm2;
end;

end.
4

1 回答 1

7

这里的关键问题是第一个实例TForm2被创建为应用程序窗口拥有的窗口,Application.Handle. 这里我指的是Windows 中 owner 的含义。在 VCL 语言中,这被称为弹出父级。

现在,当您创建第一个TForm2实例时,该Application.MainForm属性仍然是nil. 并且由于您没有明确指定PopupParent,因此中的代码TCustomForm.CreateParams将所有者设置为应用程序窗口。

您只是不希望您的窗口归隐藏的应用程序窗口所有。这就是为什么第一个TForm2实例有时会出现在所有其他窗口之后的原因,尤其是在您的主窗体之后。它只是用错误的所有者创建的。

拥有的表单Application.Handle显示在 中THintWindow.ActivateHint。发生这种情况是由于读取的行ParentWindow := Application.Handle。这之后是一个调用SetWindowPos(Handle, ...),导致错误拥有的表单出现在前面。据推测,该表格出现在最前面是因为它也归Application.Handle. 现在我对确切的机制没有一个明确的解释,但我觉得这不是很有趣,因为表单显然设置错误。

无论如何,根本问题是您创建了一个不正确拥有的窗口。因此,解决方案是确保正确拥有该窗口。通过分配PopupParent. 例如:

procedure TForm1.CreateForm2;
var
  frm : TForm2;
begin
  frm := TForm2.Create(Application); // (Could pass Self - makes no difference)
  frm.PopupParent := Self;
  frm.Show;
end;
于 2012-12-12T11:30:59.410 回答