![]() |
AW: TPopupMenu Popup überschreiben / DoPopup wird nie aufgerufen
Nur so 'ne Idee: Du könntest auch in der WndProc auf WM_INITMENUPOPUP (Sent when a drop-down menu or submenu is about to become active.) reagieren und das aktuelle PopupMenu anhand des Handles suchen. Dann sparst du dir die Ableitung von TPopupMenu ganz:
Delphi-Quellcode:
case message.msg of WM_INITMENUPOPUP: begin for I := 0 to Count - 1 do begin if TPopupMenu(Items[I]).Handle = TWMInitMenuPopup(Message).MenuPopup then begin { sollte nie erreicht werden, wenn ein SubMenu geöffnet wird } PopupMenu := TPopupMenu(Items[I]); Break; end; end; end; WM_ENTERMENULOOP: begin MenuLoop := True; // Menu sichtbar end; WM_EXITMENULOOP: begin MenuLoop := False; // Menu nicht mehr sichtbar PopupMenu := nil; end; end; |
AW: TPopupMenu Popup überschreiben / DoPopup wird nie aufgerufen
Das ist natürlich die elegantere Lösung.
Delphi-Quellcode:
Warum genau denn bis Count-1? Geht man hier einfach auf verdacht bis in einen sehr hohen Bereich?
for I := 0 to Count - 1 do
|
AW: TPopupMenu Popup überschreiben / DoPopup wird nie aufgerufen
Zitat:
Delphi-Quellcode:
die gesamte Liste. Da wir in der Message lediglich das Handle des Menüs bekommen, suchen wir die dazu passende Instanz in der For-Schleife. Sobald diese gefunden ist, bricht das Break die Schleife ab.
for I := 0 to Count - 1 do
|
AW: TPopupMenu Popup überschreiben / DoPopup wird nie aufgerufen
Danke für die Erklärung!
|
AW: TPopupMenu Popup überschreiben / DoPopup wird nie aufgerufen
Es gibt leider ein neues komisches Problem :(
Wenn man das Popupmenu wie folgt sucht sollte es eigentlich der Variablen zugewiesen werden
Delphi-Quellcode:
Wenn ich in einer anderen Unit nun etwas teste, bekomme ich 'X' nie zu sehen, obwohl die Unit korrekt in den uses steht. Stattdessen sehen ich "nil":
WM_INITMENUPOPUP: // PopupMenu anhand des Handles finden
begin for i := 0 to Count - 1 do begin if TPopupMenu(Items[i]).Handle = TWMInitMenuPopup(Message).MenuPopup then begin PopupMenu := TPopupMenu(Items[i]); Break; end; end; end;
Delphi-Quellcode:
Selbst mein alter Code mit procedure DoPopup(Sender: TObject); override; funktioniert jetzt nicht mehr und DoPopup wird nie aufgerufen. Das ist mehr als seltsam. Mal funktioniert das, mal nicht, obwohl ich doch nur wissen möchte, welches das aktuell geöffnete PopupMenu ist ...
// Steht im OnPopup meines PopupMenus
if TPopupListEx(PopupList).PopupMenu = nil then ShowMessage('nil') else if TPopupListEx(PopupList).PopupMenu = FrmMain.PopupMenu2 then showmessage('X'); Noch seltamer: jetzt bekomme ich nicht mal mehr diese Showmessage angezeigt
Delphi-Quellcode:
procedure TForm1.PopupMenu2Popup(Sender: TObject);
begin showmessage('Popup'); ... end; |
AW: TPopupMenu Popup überschreiben / DoPopup wird nie aufgerufen
Prüf doch mal die Reihenfolge der Aufrufe im Debugger. Eventuell kommt die WM_INITMENUPOPUP Message ja erst nach dem DoPopup.
|
AW: TPopupMenu Popup überschreiben / DoPopup wird nie aufgerufen
Ich werde mir das gleich mal genau angucken.
Ein Problem habe ich jedenfalls schon gefunden. Ich verwende ein ListView. Beim Selektieren eines Items wird Prozedur X aufgerufen. Diese prüft gewisse Dinge und passt das dementsprechend eine Toolbar an. Nun aber das Problem. das Popup-Event meines PopupMenus ruft diese Prozedur auch auf. In der Reihenfolge kommt aber zuerst ListView > OnSelectItem. Das heißt hier wird Prozedur X zweimal aufgerufen denn OnSelectItem. In Prozedur X prüfe ich außerdem, welches PopupMenu gerade geöffnet ist, um unnötigen Code nicht erst auszuführen. Durch dieses Reihenfolgen-Problem wird X aber zweimal aufgerufen. Wenn ich in OnSelectItem also herausfinden könnte, ob es einen Rechtsklick der Maus gab bzw ob der Windows-Kontextmenu-Knopf (VK_APPS) gedrückt wurde, könnte ich das verhindern. Aber wie kommt man in OnSelectItem an diese Infos? OnMouseDown ist schon zu spät. Das wird erst nach OnSelectItem ausgeführt. Ich hatte gerade ganz grob das hier im Kopf:
Delphi-Quellcode:
Was mir auch bei einem anderen Problem helfen würde, wäre wenn ich das MenuClose von PopupMenu überschreiben könnte. Aber das darf man scheinbar nicht.
if (GetAsyncKeyState(VK_RBUTTON) = 0) and (GetAsyncKeyState(VK_APPS) = 0) then
|
AW: TPopupMenu Popup überschreiben / DoPopup wird nie aufgerufen
Keine Ahnung, ob ich das richtig verstanden habe, aber kannst du der Methode X nicht einen zusätzlichen Parameter mitgeben, der das Popupmenü angibt? Im OnSelectItem übergibst du einfach nil und im OnPopup dann (Sender as TPopupMenu). Kannst du dir gesamte Mimik mit WndProc und geerbten Klassen dann nicht einfach sparen?
|
AW: TPopupMenu Popup überschreiben / DoPopup wird nie aufgerufen
Ich denke das würde funktionieren.
Ich dachte nur, dass ich diese Unit mit WndProc usw. später gut verwenden könnte, wenn ich mal wissen muss welches PopupMenu denn offen ist. Ich habe jetzt mal all den PopupMenu WndProc-Kram gelöscht und mache es mit dem Parameter. Sollte ich irgendwann in Zukunft meine WndProc-Sache brauchen, kann ich sie ja noch immer nutzen. Jetzt habe ich nur noch das Problem mit GetKeyState. DAMIT das sauber bleibt mache ich dafür ein eigenes Thema auf. |
Alle Zeitangaben in WEZ +1. Es ist jetzt 05:23 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