![]() |
Single Instanz mit Extras, hier findet man ein Beispiel
Hallo Community!
Falls ihr auch schon mal auf der Suche nach einer Möglichkeit wart Eure Applikation in eine Single Instanz Anwendung zu verwandeln, ich habe auf ![]() Die Fakten: Es wird nicht der übliche Mutex Weg eingeschlagen sondern ein globales ClassName Unikat verwendet ( FindWindow() ) Es wird WM_COPYDATA für die Übertragung neuer Parameter benutzt Es besteht die Möglichkeit ein Notify Icon zu erzeugen Eine kleines Demo PopUp menu ist vorhanden Ich hoffe irgendwer kann es mal brauchen, für Vorschläge wie man es sonst noch machen könnte bin ich wie so oft sehr offen und dankbar! Im Archiv befindet sich ein Delphi 5 Kompilat zum rumspielen und der Sourcecode. Es sollte meines Erachtens auch mit Rio klappen wenn man den Namespace addiert, aber ob es da Wiederum einfachere Wege gibt dies alles zu realisieren, das kann ich an dieser Stelle nicht sagen, bitte an Fachmann/Frau wenden. Viel Spaß damit wünscht Euch KodeZwerg. |
AW: Single Instanz mit Extras, hier findet man ein Beispiel
Der "alten" DP fehlt eindeutig die "Danke"-Funktion der "neuen". Aber das soll ja noch kommen.
Also: Danke! |
AW: Single Instanz mit Extras, hier findet man ein Beispiel
Liste der Anhänge anzeigen (Anzahl: 1)
Zitat:
...ist eh nicht auf meinem Mist gewachsen, habs nur von StäckÖverflöw. Ich musste den obigen Download entfernen, mir ist da doch etwas aufgefallen was so noch nicht ganz rund war. Nun flutscht alles nach besten Wissen und Gewissen. Was ist anders? Nun ist die Möglichkeit das alte Fenster wieder herzustellen in den Hauptsource integriert worden. Gleichzeitig wird demonstriert wie man seine eigenen Parameter setzen und auswerten kann gezeigt. Warum? Weil es in der .DPR zu fehlern kam. Fehler welcher Art? Hat man App gestartet und als Notification Icon abgelegt, so wurde es zwar wieder hergestellt aber es fehlte ein Taskbar Eintrag. Sprich, es machte eine weitere arbeit mit dem Icon fast unmöglich. Originale Source teile sind noch vorhanden aber halt demarkiert. Viel Spaß damit, KodeZwerg. |
AW: Single Instanz mit Extras, hier findet man ein Beispiel
Zitat:
Wenn man so etwas Automatisiert dann kommt es nicht von Herzen. Die 5 Buchstaben sollte man doch wohl noch eintippen können. gruss |
AW: Single Instanz mit Extras, hier findet man ein Beispiel
So eine ähnliche Funktionalität benutze ich schon seit Jahre. Mit einem Unterschied. Ich frage schon in der DPR ab, ob alle Parameter gültig sind, damit nicht irgendetwas Komisches an die laufende Instanz weitergegeben wird.
|
AW: Single Instanz mit Extras, hier findet man ein Beispiel
Zitat:
Ich habe es nicht zufriedenstellend hinbekommen. Ein "ShowWindow()" ist nicht das selbe wie ein "Visible :=" User-Parameter sollen ja Wurst sein, das geht meinem Beispiel nichts an, die werden einfach nur an die Memo weitergeleitet. Beispiel: Ein Programm das als Parameter einen Pfad+Namen benötigt Interne switches kann man ja gestalten wie man möchte, okay, da fehlt eventuell noch ne Prüfung ob "-visible" mehr als einmal als Parameter vorkommt, aber das kann sich ja jeder so hinbiegen wie er es im Moment benötigt. Man kann sich die Parameter auch in eine Dynamische Liste speichern und verwerten, da gibt es viele Möglichkeiten. Die Demo dient nur als Beispiel / Grundgerüst. Herzliche Grüße, KodeZwerg. |
AW: Single Instanz mit Extras, hier findet man ein Beispiel
Nicht ganz.
Delphi-Quellcode:
Da wo die Kommentarzeile ist. Dort prüfe ich alle übergebenen Parameter auf Gültigkeit.
if Window=0 then begin
Application.Initialize; Application.MainFormOnTaskbar := True; Application.CreateForm(TMainForm, MainForm); Application.Run; end else begin ////////////////////////////////////////////////////////////////////////// FillChar(CopyDataStruct, Sizeof(CopyDataStruct), 0); for i := 1 to ParamCount do begin Arg := ParamStr(i); CopyDataStruct.cbData := (Length(Arg)+1)*SizeOf(Char); CopyDataStruct.lpData := PChar(Arg); SendMessage(Window, WM_COPYDATA, 0, NativeInt(@CopyDataStruct)); end; SetForegroundWindow(Window); end; Wenn mein Programm den Parameter close unterstützt dann prüfe ich hier, ob close in der Übergabe drin ist. Wenn nicht, wird dieser fehlerhafte Parameter garnicht erst ans Hauptprogramm weitergegeben. |
AW: Single Instanz mit Extras, hier findet man ein Beispiel
@DieDolly: Dein Code versagt wenn die Applikation als Notification Icon dargestellt wird.
Viele Grüße, KodeZwerg. |
AW: Single Instanz mit Extras, hier findet man ein Beispiel
Das ist nicht mein Code. Der ist von stackoverflow.
Ich verstehe das mit deinem Icon nicht. |
AW: Single Instanz mit Extras, hier findet man ein Beispiel
Zitat:
Delphi-Quellcode:
Hier die Auszüge aus dem Hauptsource:
program OneApp;
uses Windows, Messages, Forms, ShellApi, uMain in 'uMain.pas' {frmMain}; {$R *.res} procedure Main; var i: Integer; Arg: String; Window: HWND; CopyDataStruct: TCopyDataStruct; NotifyIconData: TNotifyIconData; Begin // try find our window handle by custom classname, kinda bulletproof Window := FindWindow( SWindowClassName, nil ); // no old instance found = create a new one if Window = 0 then begin Application.Initialize; Application.Title := 'OneApp'; // Application.MainFormOnTaskbar := True; // unsupported in Delphi 5 // Application.ShowMainForm := False; // removed, was used while testing Application.CreateForm( TfrmMain, frmMain ); Application.Run; end else // old instance found! copy any commandline(s) to it begin // do we need to copy data? if ( ParamCount > 0 ) then begin FillChar( CopyDataStruct, SizeOf( CopyDataStruct ), 0 ); for i := 1 to ParamCount do begin // ich habe schon verstanden was Du mir sagen wolltest, Du prüfst hier an dieser Stelle nach irgendwas Arg := ParamStr( i ); CopyDataStruct.cbData := ( Length( Arg ) +1 ) * SizeOf( Char ); CopyDataStruct.lpData := PChar( Arg ); SendMessage( Window, WM_COPYDATA, 0, LPARAM( @CopyDataStruct ) ); end; end; // in meinem Beispiel wiederum soll der User den "-visible" switch erst gar nicht kennen // da wir uns hier in dem "der Prozess existiert" Block befinden // addiere ich einfach mein "machs Sichtbar" und verarbeite es im Hauptsource // setup argument to bring window back Arg := '-visible'; CopyDataStruct.cbData := ( Length( Arg ) +1 ) * SizeOf( Char ); CopyDataStruct.lpData := PChar( Arg ); SendMessage( Window, WM_COPYDATA, 0, LPARAM( @CopyDataStruct ) ); // Alles was nötig war um ein Fenster aus egal was für einem Status Wiederherzustellen steht hier demarkiert da sich nun der Hauptsource darum kümmert (* Switched to above handling method // Bring old Window back to Foreground in any case // an dieser Stellte fehlte Deinem Beispiel der ShowWindow() Befehl // damit wird ein unsichtbares Fenster wieder sichtbar ShowWindow( Window, SW_SHOW ); SetForegroundWindow( Window ); // try find notification icon by Application.Title and remove it Window := FindWindow( nil, PChar( Application.Title ) ); if Window <> 0 then begin NotifyIconData.cbSize := SizeOf( TNotifyIconData ); NotifyIconData.Wnd := Window; NotifyIconData.uID := 0; Shell_NotifyIcon( NIM_DELETE, @NotifyIconData ); end; *) end; End; Begin Main; End.
Delphi-Quellcode:
// ist nicht wirklich schön das ich ne Vcl Aufrufe, kann man auch als seperate methode ablegen und hier aufrufen
// this invoke our commandline copy action procedure TfrmMain.ProcessArgument( const Arg: String ); begin // filter out fixed parameters and do specific actions if Arg = '-visible' then mnuShowClick( Self ) else // for demonstration purposes just throw given argument to Memo1 Memo1.Lines.Add( Arg ); end; // hier ist der "magische" Unterschied // ich sage der Vcl Visible := True anstelle des ShowWindow() damit ich auch ein Taskbar Icon habe // bring window back from notify status procedure TfrmMain.mnuShowClick(Sender: TObject); begin // show window // Show; Visible := True; // if at all; remove old icon if Shell_NotifyIcon( NIM_DELETE, @NotifyIconData ) then Memo1.Lines.Add( 'Shell_NotifyIcon NIM_DELETE: ' + 'Success!' ) else begin Memo1.Lines.Add( 'Shell_NotifyIcon NIM_DELETE: ' + 'Failed!' ); Memo1.Lines.Add( 'Error: ' + SysErrorMessage( GetLastError ) ); end; end; |
Alle Zeitangaben in WEZ +1. Es ist jetzt 18:30 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