![]() |
Delphi-Version: 5
AlwaysOnTop mit mehreren Formularen
Ein freundliches Hallo an die Experten :). [Ggf. passt das Thema auch zu GUI-Design.]
Bislang setze ich Formulare OnTop mit der folgenden Funktion:
Delphi-Quellcode:
Das funzt auch wunderbar - jedenfalls in allen Anwendungen, die nur ein Formular haben bzw. bei denen nur eines OnTop sein soll.
procedure AlwaysOnTop(AOnTop: Boolean; const AFormHandle: THandle);
begin if AOnTop then SetWindowPos(AFormHandle, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOACTIVATE or SWP_NOMOVE or SWP_NOSIZE) else SetWindowPos(AFormHandle, HWND_NOTOPMOST, 0, 0, 0, 0, SWP_NOACTIVATE or SWP_NOMOVE or SWP_NOSIZE); end; Gestern fügte ich einem Projekt ein weiteres Formular hinzu, das wie das erste Formular OnTop sein soll. Jeden, der nun anmerkt, dass nicht beide OnTop sein können, kann ich beruhigen, denn die Formulare sind explizit nicht übereinander sondern immer nebeneinander auf dem Bildschirm. Es geht nur darum, dass beide Formulare über den Fenstern anderer Anwendungen liegen sollen. Nachfolgend ein simples Testprojekt, das das Verhalten ebenfalls zeigt. Form1 wird automatisch erzeugt, Form2 nicht. Unit1 mit Form1:
Delphi-Quellcode:
Unit2 mit Form2:
unit Unit1;
interface uses Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs, uGUIHelper, Unit2; type TForm1 = class(TForm) procedure FormCreate(Sender: TObject); procedure FormDestroy(Sender: TObject); procedure FormActivate(Sender: TObject); private Fform2: TForm2; public end; const bShowForm2: Boolean = True; var Form1: TForm1; implementation {$R *.DFM} procedure TForm1.FormCreate(Sender: TObject); begin if bShowForm2 then begin Fform2:= TForm2.Create(nil); Fform2.Show; end; end; procedure TForm1.FormDestroy(Sender: TObject); begin Fform2.Free; end; procedure TForm1.FormActivate(Sender: TObject); begin uGUIHelper.AlwaysOnTop(True, Self.Handle); end; end.
Delphi-Quellcode:
Problemstellung: Beim mehrfachen Umschalten zwischen Form2 und anderen Anwendungen passiert es in aller Regel beim zweiten Umschaltvorgang, dass mindestens Form2, manchmal auch beide Formulare das OnTop-Attribut verlieren. Schaltet man zwischen Form1 und anderen Anwendungen um, passiert dies nicht.
unit Unit2;
interface uses Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs; type TForm2 = class(TForm) procedure FormActivate(Sender: TObject); private public end; implementation uses uGUIHelper; {$R *.DFM} procedure TForm2.FormActivate(Sender: TObject); begin uGUIHelper.AlwaysOnTop(True, Self.Handle); end; end. Kann mir jemand erklären, warum das passiert? Ich schätze mal, die VCL macht da mehr, als ich in diesem Fall gebrauchen kann. Gibt es eine Möglichkeit, zu verhindern, dass die Formulare das Flag HWND_TOPMOST verlieren? Grüße Dalai |
AW: AlwaysOnTop mit mehreren Formularen
Das Property
Delphi-Quellcode:
macht eigentlich genau das und erhält diesen Status auch über ein Recreate des Handles hinaus aufrecht. Es sollte also ausreichen, im Objektinspektor bei beiden Forms das FormStyle-Property entsprechend zu setzen.
FormStyle = fsStayOnTop
|
AW: AlwaysOnTop mit mehreren Formularen
Über FormStyle hatte ich auch schon nachgedacht, aber ich meine mich zu erinnern, dass es damit irgendeinen Ärger gab. Dennoch habe ich es gerade im Testprojekt ausprobiert und stelle fest, dass das ähnlich unzuverlässig funktioniert, egal ob das Property per Code oder Objektinspektor gesetzt wird. Manchmal bleiben beide Forms OnTop, manchmal gar keine, manchmal nur die zweite. Offenbar hängt das auch von der Windows-Version ab.
Im Application.OnDeactivate die Funktion AlwaysOnTop für beide Formulare zu rufen funktioniert etwas besser, aber auch das ist noch weit von verlässlich entfernt... Da bin ich ja fast geneigt, den Inhalt der zweiten Form in ein eigenes simples Programm auszulagern. Andererseits bin ich der Meinung, dass das zum Funktionieren zu bringen sein muss. Grüße Dalai |
AW: AlwaysOnTop mit mehreren Formularen
Vielleicht ist auch noch wichtig, wie das Umschalten zwischen Programmen erfolgt. Getestet hab ich per Maus und Alt+Tab. Letzteres funktioniert etwas besser, aber die eigentliche Anwendung wird eher per Maus benutzt werden, und beim Umschalten damit bleiben die Fenster fast nie im Vordergrund.
Grüße Dalai |
AW: AlwaysOnTop mit mehreren Formularen
Zitat:
Beim Ändern wird die innere Form komplett neu generiert, anstatt nur die eine Option zu ändern. In Windows 10, da ist und bleibt zwar bei beiden Forms die Option aktiv (siehe Caption), aber nur der Erste, welcher diese Option setzt, ist wirklich ganz oben. Unabhängig vom ExStyle, gibt es ja auch nur eine Liste mit den Z-Positionen und da kann nur einer ganz oben sein, aber vielleicht wurde das früher im Windows mal anders behandelt. Eventuell gibt es auch noch einen Unterschied bei MultiMonitor-Systemen, wenn man jemanden glauben mag, falls man z.B. nach ![]() Hab hier jetzt nur einen Monitor aktiv (bzw. ist gespiegelt), aber könnte es nächste Woche mal ausprobieren.
Delphi-Quellcode:
uses Unit2;
implementation procedure TForm1.Button1Click(Sender: TObject); begin TForm2.Create(Self).Show; end; procedure TForm1.FormActivate(Sender: TObject); begin //SetWindowLong(Handle, GWL_EXSTYLE, GetWindowLong(Handle, GWL_EXSTYLE) or WS_EX_TOPMOST); // theoretisch, aber ändern via SetWindowLong, siehe https://docs.microsoft.com/de-de/windows/win32/winmsg/extended-window-styles SetWindowPos(Handle, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOACTIVATE or SWP_NOMOVE or SWP_NOSIZE); end; procedure TForm1.Timer1Timer(Sender: TObject); begin Caption := BoolToStr(GetWindowLong(Handle, GWL_EXSTYLE) and WS_EX_TOPMOST <> 0, True); end;
Delphi-Quellcode:
implementation
uses Unit1; procedure TForm2.FormActivate(Sender: TObject); begin SetWindowPos(Handle, {Form1.Handle}HWND_TOPMOST, 0, 0, 0, 0, SWP_NOACTIVATE or SWP_NOMOVE or SWP_NOSIZE); end; procedure TForm2.Timer1Timer(Sender: TObject); begin Caption := BoolToStr(GetWindowLong(Handle, GWL_EXSTYLE) and WS_EX_TOPMOST <> 0, True); end; |
AW: AlwaysOnTop mit mehreren Formularen
Zitat:
Delphi-Quellcode:
if FFormStyle <> Value then begin if ((Value = fsNormal) and (FFormStyle = fsStayOnTop)) or ((Value = fsStayOnTop) and (FFormStyle = fsNormal)) then begin FFormStyle := Value; if not (csDesigning in ComponentState) and HandleAllocated then SetWindowPos(Handle, HWND_STYLE[FFormStyle = fsStayOnTop], 0, 0, 0, 0, SWP_NOMOVE or SWP_NOSIZE or SWP_NOACTIVATE or SWP_NOOWNERZORDER); end else ... end; end; |
AW: AlwaysOnTop mit mehreren Formularen
Wenn du sagst, die Fenster verlieren ihre TOPMOST Eigenschaft, wie äußert sich das? Sind plötzlich andere Forms deiner eigenen Anwendung über diesen (eigentlich TOPMOST) Forms oder sind nur Fenster einer anderen Anwendung darüber?
|
AW: AlwaysOnTop mit mehreren Formularen
Windows 10 und Delphi 10.3, gestern ausprobiert: Nur das erste TopMost-Fesnter ist immer oben,
aber in den Fenstereigenschaften steht es dennoch bei Beiden drin. (Dachte vielleicht Windows schaltet es beim anderen Fenster ab, bzw. ignoriert das Setzen im Zweiten) Mir war so, als wenn da das Fenster immer so richtig schön kurz wegblinkte. Vielleicht wurde es ja inzwischen geändert. Aber ich geb mit Stolz zu, dass ich seit vielen Jahren mich erfolgreich von TopMost verabschiedet hab, da es mehr Arbeit machte, als Freude zu bereiten. |
AW: AlwaysOnTop mit mehreren Formularen
Grundsätzlich kann ja auch nur ein Fenster oben sein. TOPMOST heißt ja auch nur: das Fenster ist über den Fenstern, die nicht TOPMOST sind. Innerhalb von TOPMOST und NON-TOPMOST gibt es natürlich jeweils eine Reihenfolge.
Zitat:
|
AW: AlwaysOnTop mit mehreren Formularen
Es kommt drauf an, wie man es sieht/auslegt
TopMost = über ALLEM (da ginge nur Einer, so ala Highlander) TopMost = über allem, was nicht TopMost ist In einem Programm/Thread bekomm ich jetzt auch nur 1 Fenster, was immer oben bleibt, aber starte ich das Programm doppelt (pro Programm/Thread nur ein Fenster), dann geht es auch mehrfach. > das was den Fokus hat, jeweils ganz oben, aber alle immer über den normalen Fenstern |
AW: AlwaysOnTop mit mehreren Formularen
Zitat:
Zwischenzeitlich hab ich das zweite Form in eine eigene Anwendung überführt (weil ich für ein anderes Projekt fix eine Lösung brauchte), aber ich bin dennoch an einer Lösung innerhalb einer Anwendung interessiert. Grüße Dalai |
AW: AlwaysOnTop mit mehreren Formularen
siehe #5 ?
[edit] Hab's aber grad nochmal probiert ... ich glaub Delphi bzw. die VCL ist Schuld. Vorhin falsch geguckt, denn so ist es aktuell in Windows 10 + Delphi 10.3: * das zweite Fenster ist immer über dem Ersten * wären Beide gleich (StayOnTop oder nicht), müsste jeweils das Aktive oben sein * vermutlich irgendwas in Richtung PopupMode, aber das steht (standardmäßig) eigentlich auf pmNone :grueble: [edit2]
Delphi-Quellcode:
Keine Ahnung wer auf diese bescheuerte Idee gekommen ist auch bei NONE etwas zu machen.
procedure TCustomForm.CreateParams(var Params: TCreateParams);
... case LPopupMode of pmNone: begin if Application.MainFormOnTaskBar then begin // FCreatingMainForm is True when the MainForm is // being created, Self = Application.MainForm during CM_RECREATEWND. if FCreatingMainForm or (Self = Application.MainForm) then WndParent := 0 else if Assigned(Application.MainForm) and Application.MainForm.HandleAllocated then begin WndParent := Application.MainFormHandle; if WndParent = Application.MainForm.Handle then begin if Application.MainForm.PopupChildren.IndexOf(Self) < 0 then Application.MainForm.PopupChildren.Add(Self); <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<< FreeNotification(Application.MainForm); end; end else WndParent := Application.Handle; end else begin WndParent := Application.Handle; SetWindowLong(WndParent, GWL_EXSTYLE, GetWindowLong(WndParent, GWL_EXSTYLE) and not WS_EX_TOOLWINDOW); end; end; pmAuto: begin if FCreatingMainForm then WndParent := 0 // A main form can't be parented to another form else Lösung: Und statt OnActivate, ist sowieso CreateWnd besser. (auch wenn Beides geht)
Delphi-Quellcode:
[Edit3]
type
TForm3 = class(TForm) procedure FormCreate(Sender: TObject); protected procedure CreateWnd; override; end; var Form3: TForm3; implementation {$R *.dfm} procedure TForm3.CreateWnd; begin inherited; SetWindowPos(Handle, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOACTIVATE or SWP_NOMOVE or SWP_NOSIZE); end; procedure TForm3.FormCreate(Sender: TObject); begin PopupMode := TPopupMode(9); end; Oder Application.OnGetMainFormHandle benutzen und dort
Delphi-Quellcode:
zurückgeben.
HWND(-1)
|
AW: AlwaysOnTop mit mehreren Formularen
Zitat:
Zitat:
Man kann auch noch mit PopupParent experimentieren. Ist das nicht gesetzt, wird implizit das MainForm verwendet. Das hat aber nur Auswirkungen auf die eigenen Anwendung. |
AW: AlwaysOnTop mit mehreren Formularen
Ja, "normal" war es so, dass bei mehreren TopMost beide vor allen anderen Nicht-TopMost blieb
und bei mehreren TopMost dort jeweils das ganz obe, was man zuletzt angeklickt/fokusiert hat. Wenn man den "neuen" Mist im TCustomForm.CreateParams "repariert", dann ist es auch wieder so. Wenn die Entwickler das Verhalten ändern wollen, dann dürfen sie gern pmAuto als Default festlegen, aber niemals pmNone kaputt machen. |
AW: AlwaysOnTop mit mehreren Formularen
OK, nachdem ich wieder etwas Luft habe, hab mir das jetzt im Detail angeschaut. Leider musste ich an mehreren Stellen drehen, um einige Nebeneffekte zu unterbinden.
Zitat:
Zitat:
Dummerweise hat das zur Folge, dass die Form2 im Beispiel einen eigenen Button in der Taskleiste bekommt. Das kann ich gar nicht brauchen. Nach einigen Versuchen hab ich eine Lösung gefunden, aber ich frag lieber nach, ob das eine saubere Variante ist. Unit1:
Delphi-Quellcode:
Restliche Funktionen der Unit wie gehabt.
procedure TForm1.FormCreate(Sender: TObject);
begin if bShowForm2 then begin Fform2:= TForm2.Create(nil); Fform2.BorderStyle:= bsNone; Fform2.PopupParent:= Self; end; end; Unit2:
Delphi-Quellcode:
---
procedure TForm2.FormCreate(Sender: TObject);
begin Self.PopupMode:= TPopupMode(99); end; Eine andere Variante kam mir gerade aufgrund eines anderen Projektes in den Sinn. Unit1: wie im OP. [EDIT]Der Aufruf von TForm2.Create muss natürlich etwas anders aussehen, aber der Rest bleibt gleich.
Delphi-Quellcode:
[/EDIT]
procedure TForm1.FormCreate(Sender: TObject);
begin if bShowForm2 then begin Fform2:= TForm2.Create(nil, Self.Handle); Fform2.BorderStyle:= bsNone; end; end; Unit2:
Delphi-Quellcode:
Vorteil: Kein Button erscheint für Form2 in der Taskleiste und ein Setzen des PopupMode entfällt, was den Nebeneffekt hat, dass es sogar im Delphi 5 funktioniert ;).
type
TForm2 = class(TForm) procedure FormActivate(Sender: TObject); private FWndParent: HWND; protected procedure CreateParams(var Params: TCreateParams); override; public constructor Create(AOwner: TComponent; AWndParent: HWND); reintroduce; overload; end; implementation constructor TForm2.Create(AOwner: TComponent; AWndParent: HWND); begin Self.FWndParent:= AWndParent; inherited Create(AOwner); end; procedure TForm2.CreateParams(var Params: TCreateParams); begin inherited; Params.WndParent:= Self.FWndParent; end; procedure TForm2.FormActivate(Sender: TObject); begin uGUIHelper.AlwaysOnTop(True, Self.Handle); end; Meinungen zu den beiden Lösungsvarianten? Ist eine besser geeignet als die andere? Und wenn ja, warum? Grüße Dalai |
Alle Zeitangaben in WEZ +1. Es ist jetzt 08:08 Uhr. |
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024-2025 by Thomas Breitkreuz