![]() |
Wann wird fremde TForm ausgeblendet
Moin Moin,
wie "elegant" mitbekommen, wann eine fremde TForm freigegeben ausgeblendet wird? egal wie (VCL oder NonVCL) Freigeben ist einfach. * einfach ein WinWintrol dranhängen ... mein HWND verschwindet, wenn der Parent verschwindet * oder ein TComponent mit Owner dranhängen ... wird Parent freigegeben, dann ich auch Aber leider dauert es "ewig", vom Ausblenden, bis zur Freigabe, drum wollte ich nun versuchen, mich ans "Hide" zu hängen. * TApplicationEvents.OnMessage auf WM_DESTROY bzw. WM_QUIT ... unschön, weil dann ja bei ALLEM mein "Hook" drin * Form.OnHide oder OnClose überschreiben * WndProc der Form hooken (SetWindowLong+GWL_WNDPROC -> CM_VISIBLECHANGED) * VMT des DoHide überschreiben * * gefallem mit diese Hooks eigentlich nicht wirklich, da ich dort ja prinzipiell ja die Komponente selbst "verändern" würde (ungünstig, wenn ich einen Fehler machen würde) WM_SHOWWINDOW bei meinen WinControl ausprobiert, aber es wird nie aufgerufen ... somit leider nicht möglich dort das SW_PARENTCLOSING prüfen zu wollen. |
AW: Wann wird fremde TForm ausgeblendet
Ich bin mir nicht sicher, ob ich die Frage richtig verstehe. Aktuell wäre mein Vorschlag
![]() |
AW: Wann wird fremde TForm ausgeblendet
Es geht also um eine "fremde" TForm im gleichen Delphi-Programm?
Willst Du etwas, das immer funktioniert, also für beliebige TForms? Oder reicht etwas, das für eine bestimmte TForm funktioniert? TForm.OnHide wäre da das einfachste, sofern dieses Event nicht belegt ist. Wenn doch, aber der Wert sich nicht ändert, kann man den originalen Event speichern und ihn aus dem eigenen aufrufen, vor, oder nach dem eigenen Code. Edit: "VCL oder non-VCL" bedeutet VCL oder FMX ? Oder sprechen wir hier doch nicht von einem Delphi-Programm? |
AW: Wann wird fremde TForm ausgeblendet
Jupp, eine fremde Form im Programm.
Es ist ja nicht "meine" Form, und somit weiß ich nicht, ob nicht jetzt/irgendwann schon was am OnHide hängt. (Ja, eine Kopie des Wertes und dann das Aufrufen ... Problem ist dann, wenn irgendwann noch wer Anderes auch sowas macht) Ich will einfach nur mitbekommen, wann sie ausgelendet wird, also eigentlich das Programm "anfängt" sich zu beenden. Aber das so früh wie möglich. (über die "normalen" Wege der OTA, sind zwischen dem Verschwinden der MainForm aus der Taskleiste und meinem Taskbareintrag 30-60 Sekunden, wo man nix sieht, bis es wirklich weg ist) z.B. die Hauptform vom Delphi, in einem Designtimepackage. :duck: Zitat:
z.B. das WM_SHOWWINDOW+SW_PARENTCLOSING hatte ich mit einem TPanel probiert (hätte aber auch eine TForm sein können) oder eben per CreateWindowEx. Das Projekt macht jetzt nicht viel,
Delphi-Quellcode:
:freak: 37 KB als Release (2-3 MB als Debug)
unit Package1Unit;
interface uses Winapi.Windows, Winapi.Messages, System.SysUtils, System.Classes, System.Win.ComObj, Winapi.ShlObj, Winapi.Dwmapi, Vcl.ExtCtrls, Vcl.Forms; //procedure Register; implementation type TStartDummy = class private class procedure AssignToStart; class procedure WaitTimer(Sender: TObject); end; TDestroyDummy = class(TPanel) // TPanel wegen WMShowWindow (eigentlich TComponent) public procedure BeforeDestruction; override; private procedure WMShowWindow(var Msg: TWMShowWindow); message WM_SHOWWINDOW; end; class procedure TStartDummy.AssignToStart; var Taskbar: ITaskbarList3; begin for var i := Screen.CustomFormCount - 1 downto 0 do if (Screen.CustomForms[i].ClassType = TForm) and (Screen.CustomForms[i].Caption = 'SplashScreen') then begin // in Taskleiste anteigen ShowWindow(Screen.CustomForms[i].Handle, SW_HIDE); SetWindowLong(Screen.CustomForms[i].Handle, GWL_EXSTYLE, GetWindowLong(Screen.CustomForms[i].Handle, GWL_EXSTYLE) or WS_EX_TOOLWINDOW or WS_EX_APPWINDOW); ShowWindow(Screen.CustomForms[i].Handle, SW_SHOW); // Texte und Arbeitsanzeige Screen.CustomForms[i].Caption := 'Delphi wird gestartet'; Taskbar := CreateComObject(CLSID_TaskbarList) as ITaskbarList3; {CheckOSError}(Taskbar.SetProgressState(Screen.CustomForms[i].Handle, TBPF_INDETERMINATE)); //{CheckOSError}(Taskbar.SetThumbnailTooltip(Screen.CustomForms[i].Handle, 'Delphi wird gestartet')); end; //TDestroyDummy.Create(Application.MainForm); // an Hauptfenster hängen with TDestroyDummy.Create(Application.MainForm) do begin // an Hauptfenster hängen Parent := Application.MainForm; BevelOuter := TPanelBevel.bvNone; Width := 1; Height := 1; TabStop := False; Visible := True; end; end; class procedure TStartDummy.WaitTimer(Sender: TObject); var B: Boolean; begin try B := Assigned(Application) and Assigned(Application.MainForm); for var i := Screen.CustomFormCount - 1 downto 0 do if (Screen.CustomForms[i].ClassType = TForm) and (Screen.CustomForms[i].Caption = 'SplashScreen') then B := True; if B then begin TStartDummy.AssignToStart; Sender.Free; end; except end; end; procedure TDestroyDummy.BeforeDestruction; var WndClass: TWndClass; TaskWnd: HWND; Taskbar: ITaskbarList3; Param: BOOL; begin try FillChar(WndClass, SizeOf(WndClass), 0); WndClass.lpfnWndProc := @DefWindowProc; WndClass.hInstance := HInstance; WndClass.hIcon := 0; WndClass.lpszClassName := 'TaskbarDestroyDummy'; {Win32Check}(RegisterClassW(WndClass) {<> 0}); TaskWnd := CreateWindowEx(WS_EX_TOOLWINDOW or WS_EX_APPWINDOW, WndClass.lpszClassName, PChar('Delphi wird beended'), WS_VISIBLE {or WS_POPUP or WS_CAPTION or WS_CLIPSIBLINGS}, 100, 100, 0, 0, 0, 0, HInstance, nil); {Win32Check(TaskWnd <> 0);} Taskbar := CreateComObject(CLSID_TaskbarList) as ITaskbarList3; {CheckOSError}(Taskbar.SetProgressState(TaskWnd, TBPF_INDETERMINATE)); {CheckOSError}(Taskbar.SetThumbnailTooltip(TaskWnd, 'Delphi wird beendet')); Param := True; {CheckOSError}(DwmSetWindowAttribute(TaskWnd, DWMWA_FORCE_ICONIC_REPRESENTATION, @Param, SizeOf(BOOL))); {CheckOSError}(DwmSetWindowAttribute(TaskWnd, DWMWA_HAS_ICONIC_BITMAP, @Param, SizeOf(BOOL))); inherited; except end; end; procedure TDestroyDummy.WMShowWindow(var Msg: TWMShowWindow); begin MessageBox(0, PChar(IntToStr(Msg.Status)), nil, 0); end; //procedure Register; // wenn in HKEY_CURRENT_USER\Software\Embarcadero\BDS\22.0\Known IDE Packages //begin // TStartDummy.AssignToStart; //end; initialization with TTimer.Create(nil) do begin // wenn in HKEY_CURRENT_USER\Software\Embarcadero\BDS\22.0\Known Packages OnTimer := TStartDummy.WaitTimer; Interval := 1; Enabled := True; end; finalization try if Assigned(Application) and Assigned(Application.MainForm) then for var C in Application.MainForm do if C is TDestroyDummy then begin C.Free; Break; end; except end; end. "register" kommt erst eine Weile nach dem erscheinen des SplashScreen (Known IDE Packages früher oder Known Packages bissl bis viel später) "initialization" im Known IDE Packages leider schon vor dem SplashScreen (drum der böse Timer) Und dann das die Registrierung von "Known Packages" ins "Known IDE Packages" verschieben, bzw. garnicht das Package "installieren" und direkt den Eintrag dort rein.
Delphi-Quellcode:
C:\Users\Public\Documents\Embarcadero\Studio\22.0\Bpl\Package1.bpl
Achtung, ist es im "Known IDE Packages" registriert, dann lässt sich das Projekt im Delphi nicht mehr kompilieren. (vorher also entfernen oder umbennen ...
Delphi-Quellcode:
anhängen)
_
Der Code ist für Delphi 11. Ab Delphi 12 ist es nochmal extrem einfacher, da dort die VCL alles bietet, von Form.ShowInTaskBar über TTaskbar (Form.TaskbarHandler).
Delphi-Quellcode:
// Der SplashScreen ist eine manuell zusammengebaute TForm.
SplashScreen.Caption := 'Delphi startet'; SplashScreen.ShowInTaskBar := True; TaskBar := TTaskBar.Create(SplashScreen); // SplashScreen.TaskbarHandler TaskBar.ProgressState := TTaskBarProgressState.Indeterminate; TaskBar.ToolTip := 'Delphi startet'; Echt mal, ich verstehe einfach nicht, warum sich Emba seit Jahrzehnten dagegen wehrt, beim Starten das Delphi in der Taskleiste anzuzeigen, also SetWindowLong+WS_EX_APPWINDOW, bzw. aktuell TForm.ShowInTaskBar. :wall: (wenigsten ist der nicht StayOnTop) |
AW: Wann wird fremde TForm ausgeblendet
Ach, es geht um die Delphi-IDE. Warum hast Du das nicht gleich geschrieben?
Zum Hooken von Events in der IDE hätte ich noch folgendes beizutragen: ![]() Allerdings bin ich wohl der einzige, der sich an diesen "Standard" hält (immerhin 2 Plugins), aber es kann nicht schaden, das zu tun. |
AW: Wann wird fremde TForm ausgeblendet
Hey, an einen Hook im Screen hatte ich auch schon gedacht. (change des ActiveControl/ActiveForm)
Leider hat TList kein OnChange, sondern nur ein virtuellen Notify. Ein einfaches überschreiben der VMT würde alle TList hooken, aber immernoch enorm weniger, als ein ApplicationEvents.OnMessage. Aber wenn, dann hätte ich wohl den TVirtualMethodInterceptor bemüht. (der kopiert ja die ClassInfo und hookt dort dann nur für diese Instanz) An TScreen.FForms bzw. FCustomForms wäre ich über einen bösen Cast mit kopierter Klasse bin ich über die RTTI rangekommen. Beim Hook des Notify war ich mir noch unsicher.
Delphi-Quellcode:
Da hier in der OTA/NTA nichts möglich ist
var R := TRttiContext.Create.GetType(Vcl.Forms.TScreen);
var L := R.GetField('FForms').GetValue(Screen).AsObject as TList; R := TRttiContext.Create.GetType(TList); R.GetMethod('Notify').VirtualIndex und auch die VCL direkt nichts bietet, suche ich halt einfach nur einen "schönen" Weg, möglichst mit nativen Mitteln des Delphi (VCL/GDI), um rauszubekommen, wann diese TForm hidden wird. Mir fällt grad ein, anstatt an die Screen->TList könnte ich auch direkt die VMT der TAppBuilder (DoHide) :stupid: (auch wenn Delphi bei vielen eigenen Klassen die erweiterte RTTI deaktiviert hat ... die VMT der ClassInfo bleibt) |
AW: Wann wird fremde TForm ausgeblendet
Liste der Anhänge anzeigen (Anzahl: 2)
Wow, die haben in D12 ja echt mal was verändert.
Der SplashScreen ist jetzt eine eigene Form-Klasse. ![]() ![]() Wollte erst die 12, aber dann fiel mir ein, dass man bei der 11 länger etwas von sieht. Jetzt nur noch das ShowInTaskBar auf True und vielleicht noch ein TTaskbar drauf. :stupid: Muß ich wohl mal einen neuen FeatureRequest erstellen. Selbst ein halbwegs frisches D12, noch ohne Unmassen an Erweiteurngen und Fremdkomponenten, braucht hier beim Beenden schon 10-15 Sekunden, bis es wirklich weg ist, nachdem IDE-Fenster und Taskbar-Button verschwunden sind. OK, es war "erstmal" einfacher, als gedacht. OnShow wird "aktuell" von Emba nicht verwendet, also hab ich mich dort angehängt. :stupid: (hab da eine billige Assigned-Prüfung mit Warnmeldung, falls die das ändern) Also, für D11.3 und D12.1 hab ich's nun erstmal am Laufen.
|
AW: Wann wird fremde TForm ausgeblendet
From the video, i think i do understand the question, so here a suggest with different approach :
Replace BorlndMM.dll with one that load (just an import of one function might be enough) another DLL (yours) in this case BorlndMM.dll still serve as memory manager, but will invoke your dll before even the IDE start, don't forget to make your injected dll is not using share memory. |
AW: Wann wird fremde TForm ausgeblendet
Normal fällt einfach nicht auf, wenn Delphi startet und noch schlimmer, wenn es beim Starten im Hintergrund hängen bleibt.
Sowie, dass Delphi eigentlich noch offen ist, wenn es zu lange zum Beenden braucht. Hier ebenfalls nett, wenn es durch einen Fehler ewig unbemerkt hängen bleibt. ![]() ![]() Was auch nicht gleich auffällt, wenn man Delphi ausversehn mehrfach startet. Oder wenn es hängt, man denkt man hatte vielleicht vergessen zu klicken, bzw. nicht richtig geklickt, und startet es dann nochmals. (wenn man den Splashscreen nicht sieht, weil ein anderes Programm vorne liegt) Die Splashscreens überdecken sich und in der Taskleiste taucht normal ja erst das Hauptfenster auf. ![]() Komisch sieht es aus, wenn das Delphi im Hintergrund weiter/schneller ist und dann hinter/unter dem sichtbaren Splashscreen noch ein anderer Teil hervorragt. (bei genügend Erweiterungen, wenn sich das Fenster vergrößert) |
AW: Wann wird fremde TForm ausgeblendet
Zitat:
Wenn man TApplicationEvents kennt, dann fragt man sich, warum es sowas nicht auch für Screen gibt, bzw. warum die TScreen-Events nicht auch im TApplicationEvents mit enthalten sind. :freak: Zitat:
OK, via RTTI-Invoke könnte man da bestimmt so einiges machen, aber "schön" ist das auch nicht. Zitat:
Auch bei WinAPI hab ich aufgegeben auf biegen und brechen Altes unterstüzen zu wollen (9x ist tot , XP ist tot ähhh 7 ist tot die ersten 10 sind tot) |
AW: Wann wird fremde TForm ausgeblendet
|
Alle Zeitangaben in WEZ +1. Es ist jetzt 16:22 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