![]() |
TInterfacedObject Subclass
Seit ich meine Listbox auf Interface umgestellt habe kann ich die Listbox nicht mehr subclassen.
Gibt es eine andere möglichkeit TObject auf ein Interface umzulegen? Neu..
Delphi-Quellcode:
vorher..
ISkinListBox = interface
['{38EF3B4F-86A1-45D0-A7F3-4E45E125979D}'] function GetHandle: hWnd; property Handle: hWnd read GetHandle; procedure SetFont(nPointSize: Integer; FontName: PAnsiChar; AktForecolor: COLORREF; InAktForecolor: COLORREF; Shadow: Boolean; SOffset: Integer; ShadowColor: COLORREF); end; TSkinListBox = class(TInterfacedObject, ISkinListBox)
Delphi-Quellcode:
TSkinListBox = class(TObject)
Delphi-Quellcode:
procedure TSkinListBox.SubClass(WinHandle: HWND);
var Method: TMethod; begin Method.Code := @TSkinListBox.ListBoxProc; Method.Data := Self; FEnumProcInst := MakeProcInstance(Method); PrevWndProc := SetWindowLong(WinHandle, GWL_WNDPROC, integer(@WndProc)); PrevWndProcLB := SetWindowLong(Handle, GWL_WNDPROC, integer(FEnumProcInst)); end; Vorher einwandfrei Funktioniert. gruss |
AW: TInterfacedObject Subclass
Zitat:
|
AW: TInterfacedObject Subclass
Zitat:
Es kracht auf jedenfall in der Winproc. Kann aber ohne probleme compilieren. Die Listbox befindet sich in einer DLL und wird von außen aufgerufen.
Delphi-Quellcode:
InstrumentList := CTRL_ListBoxCreate(hMain, PAnsiChar(SKAERO_FOLDER + 'Sound.png'),
145, 62, 610, 268, ID_INSTRUMENTLIST, True, 18, SKAERO_INACTIVECAPTION); SKAERO_SetAnchorMode(InstrumentList.Handle, ANCHOR_RIGHT); SKAERO_SetZorder(InstrumentList.Handle, ANCHOR_RIGHT); InstrumentList.SetFont(SKAERO_CAPTIONFONTHEIGHT, PAnsiChar(SKAERO_TEXTFONT), SKAERO_ACTIVECAPTION, SKAERO_INACTIVECAPTION, TRUE, 2, 0); Rückgabe der Winproc
Delphi-Quellcode:
Result := CallWindowProc(Pointer(PrevWndProcLB), WinHandle, Msg, wP, lP);
gruss |
AW: TInterfacedObject Subclass
Alle Variablen auf ISkinListBox umgestellt?
Ist sichergestellt, das es nirgends im Programm eine Variable gibt, die direkt auf das Objekt "TSkinListBox" verweist? Existiert mindestens eine Interfacevariable, die das verwendete Objekt TSkinListBox referenziert, so lange wie das geskinnte Objekt existiert? |
AW: TInterfacedObject Subclass
Warum nicht so?
Delphi-Quellcode:
TSkinListBox = class(TListBox, ISkinListBox);
In der Klasse TComponent sind die Methoden _AddRef, _Release und QueryInterface schon implementiert. Deshalb kann man Komponenten und Controls als Basisklasse verwenden und damit weitere Interfaces implementieren. |
AW: TInterfacedObject Subclass
Zitat:
Also die ich definiert habe im ISkinListBox Interface selbst ja .. Zumindest meldet der Compiler da keine Fehler. Zitat:
Delphi-Quellcode:
TMPlayList: TSkinListBox;
Zitat:
Delphi-Quellcode:
InstrumentList.Handle
Welches beim erstellen der ListBox zurückgegeben wird. Ich habe aber bemerkt das die ListBox kurz nach dem erstellen wieder zerstört wird
Delphi-Quellcode:
Obwohl ich diese selber nicht beende.
destructor TSkinListBox.Destroy;
begin UnSubClass(FHOwner); inherited Destroy; end; Beim beenden. error: to many consecutive exceptions. Aber welche meldet er nicht. Denke das hat damit zu tun das die ListBox schon zerstört wurde. er springt dann in
Delphi-Quellcode:
sagt mir aber ehrlich gesagt nicht viel!
001D3D32 8B08 mov ecx,[eax]
001D3D34 FF51FC call dword ptr [ecx-$04] 001D3D37 C3 ret TObject.InitInstance: 001D3D38 53 push ebx gruss |
AW: TInterfacedObject Subclass
Zitat:
gruss |
AW: TInterfacedObject Subclass
Zitat:
Aber jeder soll seine eigenen Erfahrungen machen... |
AW: TInterfacedObject Subclass
Zitat:
Um ein Controll vernünftig zu Überzeichen kommt man da nicht drumherum. Und ob es eine Zeitverschwendung ist ? Für mich nicht! Und Erfahrungen habe ich gute gemacht. gruss |
AW: TInterfacedObject Subclass
Auch wenn es eine blöde frage zu sein scheint.
aber wie komme ich an die WinProc der Hauptanwendung? Denke das ich das problem erkannt habe. ich muss die Winproc der Anwendung subclassen nicht meine eigene. :(
Delphi-Quellcode:
PrevWndProc := SetWindowLong(WinHandle, GWL_WNDPROC, integer(@WndProc));
anstelle von
Delphi-Quellcode:
PrevWndProc := SetWindowLong(WinHandle, GWL_WNDPROC, integer(@TSkinListBox.ListBoxProc));
Aber ich habe die Vermutung ... Gar nicht! gruss |
AW: TInterfacedObject Subclass
Zitat:
|
AW: TInterfacedObject Subclass
Zitat:
Delphi-Quellcode:
type
ISkinListBox = interface ['{38EF3B4F-86A1-45D0-A7F3-4E45E125979D}'] function GetHandle: hWnd; property Handle: hWnd read GetHandle; procedure SetFont(nPointSize: Integer; FontName: PAnsiChar; AktForecolor: COLORREF; InAktForecolor: COLORREF; Shadow: Boolean; SOffset: Integer; ShadowColor: COLORREF); end; TSkinListBox = class(TInterfacedObject, ISkinListBox) private LStyle: DWORD; FHOwner: HWND; FHandle: HWND; FEnumProcInst: Pointer; ImgIcon: Cardinal; ImgIconH: Cardinal; ImgIconW: Cardinal; function MakeProcInstance(M: TMethod): Pointer; procedure SetCTLFont(hCtL: HWND; Font: hFont); procedure SubClass(WinHandle: HWND); procedure UnSubClass(WinHandle: HWND); function GetHandle: hWnd; public property Handle: HWND Read FHandle; procedure SetFont(nPointSize: Integer; FontName: PAnsiChar; AktForecolor: COLORREF; InAktForecolor: COLORREF; Shadow: Boolean; SOffset: Integer; ShadowColor: COLORREF); function ListBoxProc(WinHandle: HWND; Msg: UINT; wP: WParam; lP: LParam): LRESULT; stdcall; procedure DrawItem(WinHandle: HWND; Dc: Hdc; Index: Integer;Rect: TRect; Selected: Bool); procedure InitTrackbar; procedure ListSetTopIndex(hList: HWND; nTopIndex: Integer); function ListGetTopIndex(hList: HWND): Integer; function ListGetSel(hList: HWND; nSelected: Integer): Bool; function GetItemHeight(hList: HWND; ItemHeigh: Integer): Integer; function ListCount(hList: HWND): Integer; procedure ListDeleteAll(hList: HWND); procedure ListDelete(hList: HWND; nIndex: Integer); procedure ListSelectPlus(hList: HWND; nSelected: Integer); function ListFindString(hList: HWND; Tmp: string): Integer; function ListGetCurSel(hList: HWND): Integer; function ListAdd(hList: HWND; Tmp: string): Integer; function ListGetText(hList: HWND; Item: Integer): PAnsiChar; constructor Create(hOwner: HWND; FullpathImageName: string; x, y, xW, yH, ListID: integer; Visible: Boolean; ItemHeight: Integer; BackColor: COLORREF); destructor Destroy; override; end; type LBTYPE = Record AktForecolor : COLORREF; InAktForecolor : COLORREF; Backcolor : COLORREF; Shadow : Boolean; ShadowColor : COLORREF; ShadowOffset : Integer; ForeColorSelected : COLORREF; BackColorSelected : COLORREF; PointSize : Integer; DrawStyle : Integer; BorderStyle : Integer; Icon : string; ItemHeight : Integer; Handle : HWND; Left : Integer; Top : Integer; Width : Integer; Height : Integer; Font : HFONT; end;
Delphi-Quellcode:
constructor TSkinListBox.Create(hOwner: HWND; FullpathImageName: string; x, y, xW, yH,
ListID: integer; Visible: Boolean; ItemHeight: Integer; BackColor: COLORREF); begin with SkinEngine do begin // LBS_NOTIFY übergeben ohne wird kein Event // auf LBN_DBLCLK ausgelößt if Visible = True then begin LStyle := LBS_HASSTRINGS or LBS_OWNERDRAWFIXED or LBS_NOINTEGRALHEIGHT or LBS_NOTIFY or WS_CHILD or WS_VISIBLE; end else LStyle := LBS_HASSTRINGS or LBS_OWNERDRAWFIXED or LBS_NOINTEGRALHEIGHT or LBS_NOTIFY or WS_CHILD; // Propertys der Listbox festlegen ListBoxType.Backcolor := BackColor; ListBoxType.ForeColorSelected := GetSysColor(COLOR_HIGHLIGHTTEXT); ListBoxType.BackColorSelected := GetSysColor(COLOR_HIGHLIGHT); ListBoxType.BorderStyle := EDGE_RAISED; ListBoxType.Left := x; ListBoxType.Top := y; ListBoxType.Width := xW; ListBoxType.Height := yH; ListBoxType.Icon := FullpathImageName; ListBoxType.ShadowColor := (RGB(0,0,0)); ListBoxType.Shadow:= FALSE; ListBoxType.ShadowOffset := 3; ListBoxType.ItemHeight := ItemHeight; // ListBox erstellen FHandle := CreateWindowEx(WS_EX_TRANSPARENT, SKLISTBOX, nil, LStyle, ListBoxType.Left, ListBoxType.Top, ListBoxType.Width, ListBoxType.Height, hOwner, ListID, skInstance, nil); if FHandle <> 0 then begin ListBoxType.Handle := FHandle; SendMessage(Handle, LB_SETITEMHEIGHT, 0, ListBoxType.ItemHeight); ListBoxType.DrawStyle := CreateSolidBrush(ListBoxType.Backcolor); FHOwner := hOwner; SubClass(FHOwner); end; end; end; gruss |
AW: TInterfacedObject Subclass
Und wo wird hier TSkinListBox.Create aufgerufen? Welche Variable hält die erzeugte Instanz? Was für einen Typ hat diese Variable?
Was da im Create passiert, ist wahrscheinlich völlig irrelevant. Die Tatsache, daß du das Window-Handle abspeicherst, hat mit der Lebensdauer der Instanz nichts zu tun. Ich hätte jetzt hier so etwas erwartet:
Delphi-Quellcode:
var
SkinListBox: ISkinListBox; // Wichtig! Hier muss eine Interface-Variable stehen. IInterface ginge auch, nicht aber TSkinListBox oder TObject. ... SkinListBox := TSkinListBox.Create(...); |
AW: TInterfacedObject Subclass
Zitat:
Delphi-Quellcode:
Aber in meiner Anwendung nicht in der TSkinList Classe.
InstrumentList: ISkinListBox;
Wird doch von außerhalb aufgerufen...
Delphi-Quellcode:
In der DLL..
InstrumentList := CTRL_ListBoxCreate(hMain, PAnsiChar(SKAERO_FOLDER + 'Sound.png'),
145, 62, 610, 268, ID_INSTRUMENTLIST, True, 18, SKAERO_INACTIVECAPTION); SKAERO_SetAnchorMode(InstrumentList.Handle, ANCHOR_RIGHT); SKAERO_SetZorder(InstrumentList.Handle, ANCHOR_RIGHT); InstrumentList.SetFont(SKAERO_CAPTIONFONTHEIGHT, PAnsiChar(SKAERO_TEXTFONT), SKAERO_ACTIVECAPTION, SKAERO_INACTIVECAPTION, TRUE, 2, 0);
Delphi-Quellcode:
Zurückgegeben wird ein handle mit dem ich in der Anwendung
function CTRL_ListBoxCreate(hOwner: HWND; FullpathImageName: string; x, y, xW, yH,
ListID: integer; Visible: Boolean; ItemHeight: Integer; BackColor: COLORREF): ISkinListBox; stdcall; begin result := TSkinListBox.Create(hOwner, FullpathImageName, x, y, xW, yH, ListID, Visible, ItemHeight, BackColor); end; in verbindung mit der übergebenen ID der ListBox arbeite. Zitat:
Du kennst doch die Zusammenhänge nicht wie ich mit den Daten im weiteren verlauf arbeite. Alles was deklariert wurde wird auch im späteren verlauf verwendet. Und funktioniert in meiner anderen Anwendung "OHNE" Dll. gruss |
AW: TInterfacedObject Subclass
Zitat:
Kannst du nicht einen Breakpoint in das Destroy setzen und schauen, wodurch es ausgelöst wurde? BTW, es macht zwar bislang hier keinen Unterschied, aber man sollte doch immer
Delphi-Quellcode:
aufrufen. Man weiß ja nie, was in zukünftigen Delphi-Versionen in
inherited Create
Delphi-Quellcode:
so alles noch eingebaut wird.
TObject.Create
|
AW: TInterfacedObject Subclass
Habs schon versucht.. mit Breakpoint
Das problem ist das der Compiler nur den ASM Bildschirm öffnet und da kann ich nur lesen Zitat:
Zitat:
gruss |
AW: TInterfacedObject Subclass
InitInstance ist auch in Assembler, da wird man nicht viel mehr sehen können. Aber vielleicht kannst du im Aufrufstack etwas erkennen?
In der DLL wird nach dem Create dreimal auf InstrumentList zugegriffen. Kommt es überhaupt dazu oder knallt es schon beim Create? Ich weiß, daß man in einer DLL nicht immer alles das machen kann, was in einer Exe problemlos möglich ist. Ich habe aber noch keine Ahnung, was das hier sein könnte. Zumindest würde man mit diesem Wissen die Suche in eine andere Richtung bringen. |
AW: TInterfacedObject Subclass
Zitat:
Lasse ich alles andere weg dann habe ich aber keinen Zugriff mehr auf meine Default WinProc in der Anwendung selbst. Das merkt man weil ich keine Messagen mehr verarbeiten kann zum beispiel wenn ich auf einen Button Klicke. Die Anwendung läßt sich dann nicht mehr schließen. Ich glaube das es an der Winproc selbst liegt bzw.. wie diese gesubclassed wird. Mein Fehler ist das ich wie vorher schon mal erwähnt nicht die WinProc der Anwendung sondern die der ListBox subclasse. Es werden meineserachtens keine Messagen an die Hauptanwendung geschickt bzw.. dort verarbeitet. Zitat:
Das sie funktioniert siehst du am Bild.. Aber nur in der Anwenung ohne DLL gruss |
AW: TInterfacedObject Subclass
GetWindowLong?
|
AW: TInterfacedObject Subclass
Zitat:
Ja damit komme ich an die Winproc (Hätte mir auch selbst einfallen können ;) ) Aber funktionieren tut es trotzdem nicht. Habe jetzt das Interface zur ListBox entfernt also zurück auf TObject und siehe da jetzt funktioniert es wieder. Warum ich die ListBox nicht mit dem Interface initialisiert bekomme ist mir noch nicht ganz klar. Wäre aber die Ideale lösung. Muss trotzdem noch einges ändern damit die Listbox von außen besser bedienbar ist. Siehe Bild! Wie du sehen kannst kopiere ich den Hintergrund ab der position 0, 0, 610, 268 mit dem Handle meines Frames in die Listbox So emuliere ich quasi eine Transparente ListBox. Das es jetzt so komisch aussieht ist absicht ;) Normalerweise muss ich das Handle der Anwendung selbst verwenden und die Position auf der meines Frames setzen. 145, 62, 610, 268 Habe es jetzt mal absichtlich so gemacht damit du sehen kannst das es wieder funktioniert. gruss |
Alle Zeitangaben in WEZ +1. Es ist jetzt 00:19 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