![]() |
Frames
Hallo,
ich habe ein Frame das ich zur Laufzeit in ein Formular einbinde. Dies funktioniert auch. In dem Frame gibt es eine Query. Nun will ich einen neuen Datensatz einfügen bekomme aber die in dieser Procedere eine Zugriffsverletzung.
Delphi-Quellcode:
Irgendwie kann ich nicht auf die MitgliederQry zugreifen, aber warum nicht?
procedure TframeSpenden.qryMitgliederspendenNewRecord(DataSet: TDataSet);
begin qryMitgliederspenden.FieldByName('MITGLIEDERNR').AsInteger := frmMitglieder.MitgliederQry.FieldByName('MITGLIEDERNR').AsInteger; end; Die Unit Mitglieder habe ich in der Uses hinzugefügt. Was mache ich hier falsch. Danke für Eure Hilfe. |
AW: Frames
Ich halte die Frage für nicht lösbar basierend auf den angegebenen Informationen. Die ist höchstwahrscheinlich nur mit (
Delphi-Quellcode:
-?)Quelltext des Frames und der Form lösbar. FormCreate oder wo auch immer du den Frame erstellst, wäre auch interessant.
interface
In einem ersten Schritt könnte man eine Integervariable anlegen und schauen, ob es an der Quelle oder dem Ziel der Zuweisung liegt. |
AW: Frames
OK hier der Code vom Frame
Delphi-Quellcode:
Und hier das einbinden in der Form:
unit Spenden;
interface uses Winapi.Windows, Winapi.Messages, System.SysUtils, System.Variants, System.Classes, Vcl.Graphics, Vcl.Controls, Vcl.Forms, Vcl.Dialogs, cxGraphics, cxControls, cxLookAndFeels, cxLookAndFeelPainters, cxStyles, dxSkinsCore, dxSkinBlack, dxSkinBlue, dxSkinBlueprint, dxSkinCaramel, dxSkinCoffee, dxSkinDarkRoom, dxSkinDarkSide, dxSkinDevExpressDarkStyle, dxSkinDevExpressStyle, dxSkinFoggy, dxSkinGlassOceans, dxSkinHighContrast, dxSkiniMaginary, dxSkinLilian, dxSkinLiquidSky, dxSkinLondonLiquidSky, dxSkinMcSkin, dxSkinMetropolis, dxSkinMetropolisDark, dxSkinMoneyTwins, dxSkinOffice2007Black, dxSkinOffice2007Blue, dxSkinOffice2007Green, dxSkinOffice2007Pink, dxSkinOffice2007Silver, dxSkinOffice2010Black, dxSkinOffice2010Blue, dxSkinOffice2010Silver, dxSkinOffice2013DarkGray, dxSkinOffice2013LightGray, dxSkinOffice2013White, dxSkinOffice2016Colorful, dxSkinOffice2016Dark, dxSkinPumpkin, dxSkinSeven, dxSkinSevenClassic, dxSkinSharp, dxSkinSharpPlus, dxSkinSilver, dxSkinSpringTime, dxSkinStardust, dxSkinSummer2008, dxSkinTheAsphaltWorld, dxSkinsDefaultPainters, dxSkinValentine, dxSkinVisualStudio2013Blue, dxSkinVisualStudio2013Dark, dxSkinVisualStudio2013Light, dxSkinVS2010, dxSkinWhiteprint, dxSkinXmas2008Blue, cxCustomData, cxFilter, cxData, cxDataStorage, cxEdit, cxNavigator, cxDataControllerConditionalFormattingRulesManagerDialog, Data.DB, cxDBData, cxButtonEdit, cxGridCustomTableView, cxGridTableView, cxGridDBTableView, IBODataset, cxGridLevel, cxClasses, cxGridCustomView, cxGrid, cxCalendar, cxCurrencyEdit, cxCheckBox; type TframeSpenden = class(TFrame) cxGrSpendenView: TcxGridDBTableView; cxGrSpendenLevel: TcxGridLevel; cxGrSpenden: TcxGrid; qryMitgliederspenden: TIBOQuery; cxGrSpendenViewDatum: TcxGridDBColumn; cxGrSpendenViewSpendenfallnummer: TcxGridDBColumn; dsrcMitgliederspenden: TDataSource; cxGrSpendenViewSpendennummer: TcxGridDBColumn; cxGrSpendenViewSpendenfall: TcxGridDBColumn; cxGrSpendenViewBetrag: TcxGridDBColumn; cxGrSpendenViewBetraginWorten: TcxGridDBColumn; cxGrSpendenViewZuwendung: TcxGridDBColumn; cxGrSpendenViewQuittung: TcxGridDBColumn; procedure cxGrSpendenViewSpendenfallnummerPropertiesButtonClick( Sender: TObject; AButtonIndex: Integer); procedure qryMitgliederspendenNewRecord(DataSet: TDataSet); procedure qryMitgliederspendenAfterInsert(DataSet: TDataSet); private { Private-Deklarationen } public { Public-Deklarationen } end; implementation {$R *.dfm} uses Modul, Main, Mitglieder; procedure TframeSpenden.cxGrSpendenViewSpendenfallnummerPropertiesButtonClick( Sender: TObject; AButtonIndex: Integer); begin // Suchen einfügen end; procedure TframeSpenden.qryMitgliederspendenAfterInsert(DataSet: TDataSet); begin qryMitgliederspenden.FieldByName('DATUM').AsDateTime := Date; qryMitgliederspenden.FieldByName('ZUWENDUNGSART').AsString := 'Geldspende'; end; procedure TframeSpenden.qryMitgliederspendenNewRecord(DataSet: TDataSet); begin qryMitgliederspenden.FieldByName('MITGLIEDERNR').AsInteger := frmMitglieder.MitgliederQry.FieldByName('MITGLIEDERNR').AsInteger; end; end.
Delphi-Quellcode:
if dmMain.ibqryOptionen.FieldByName('SPENDEN').AsString = 'J' then
begin tshVerein.Caption := 'Spenden'; tshVerein.Visible := True; if not assigned(Spenden)then begin Spenden := TframeSpenden.Create(PanelVerein); Spenden.Parent := PanelVerein; end; Spenden.BringToFront; end; |
AW: Frames
Hi Walter Landwehr
Datenbankkomponenen und die dazugehörende Logik hat auf einem Frame auch rein gar nichts zu suchen. Eigentlich auch nicht mal auf einer Form, sondern in den meisten Fällen auf einem Datenmodul. Da befinden sich somit auch die gesamten SQL-Statements. DB-Komponenten können auf einem Formular platziert werden, wenn die Anwendung nicht wirklich viele dieser Dinger braucht. Von Formularen gibt es zur Laufzeit gerade mal dasjenige, das zur Designzeit bearbeitet wurde und somit auch nur eine Instanz der DB-Verbindung. DB-Komponenten auf Frames zu platzieren, ist hingegen eine sehr schlechte Idee. Vom Basisframe, den du zur Designzeit bearbeitest, siehst du zur Laufzeit gerade mal eine Instanz, die vom Basisfram,e geerbt hat - ob sie auch dessen DB-Verbindung erbt, weiss ich nicht, aber ich könnte mir gut vorstellen: Nein. Gruss Delbor PS:Was soll dieser Code ?:
Delphi-Quellcode:
Das Ereignis AfterInsert ist nicht dazu dda, einer Datenbank etwas hinzuzufügen, sondern um bestimmte Aufgaben ddurchzuführen, nachdem ein Isert stattgefunden hat. So liesse sich zB, eine offee Verbindung gleich wieder schliessen.
procedure TframeSpenden.qryMitgliederspendenAfterInsert(DataSet: TDataSet);
begin qryMitgliederspenden.FieldByName('DATUM').AsDateTime := Date; qryMitgliederspenden.FieldByName('ZUWENDUNGSART').AsString := 'Geldspende'; end; |
AW: Frames
OK Delbor,
das geht natürlich. Das mit dem Datenmodul ist etwas schwierigerer da es sich um MDIForm handelt. Also lass ich mal die Komponenten auf der Form. Vielen Dank trotzdem für Deinen Beitrag. |
AW: Frames
Zitat:
Ansonsten hast du natürlich recht. Connections, Query's, Tables usw. sollten seperat in einem Datenmodul erzeugt und gehandelt werden. @Walter Ein Datenmodul ist unabhängig von irgendwelchen Forms. Das was du im Designer siehst, ist lediglich ein Platzhalter, das du die entsprechenden Komponenten draufziehen kannst. Aber ein DM zur Laufzeit keine Visualisierung. Um auf die entsprechenden Komponenten zugreifen zu können. musst du nur die Unit inkludieren und dann via <Datenmodulname>.<Queryname> zugreifen, egal in welcher Form oder welchen Frame. Was mir noch auffällt ist, das du versuchst auf ein weiteres Formular (Mitglieder) zu zugreifen. Kann es evtl. sein, das es zu dieser Zeit noch garnicht existiert ? |
AW: Frames
Hi Ghostwalker
Zitat:
Ja, ![]() (Basis-)Frames werden vor der Mainform erstellt. Zumindest, wenn die Mainform oder eine ihr untergeordnete Form ein Frameinstanz enthält, da diese beim konstruieren der Form (Form.OnCreate läuft, zumindest bei den letzten Delphi-Versionen nach dem Constructor der Form ab) die Grundlage zur Erstellung der Frameinstanz(en) darstellen. Wie das abläuft, wenn zur Entwurfszeit keine Frameinstanzen auf der Form abgelegt sondern dynamisch zur Laufzeit erstellt werde, weiss ich nicht. Falls du mit 'nicht automatisch erzeugt' darauf anspielst, dass du in der Toolpalette nicht einfach auf 'Frames' klicken und ihn, wie alle andern Komponenten, in eine Form(...) einfügen kannst: einverstanden. Der Frame muss erst über <Datei>Neu>weiter>Frames> erzeugt und definiert werden. Und gerade das ist der Basisframe, den du zur Laufzeit nie zu Gesicht bekommst - was du siehst, sind immer Frameinstanzen. Es gibt auch noch die Möglichkeit, Frames in der Toolpalette unter einem eigenen Icon abzulegen. Dann dürfte das geschehn, was ich manuell mache: Ein früher mal erstellter Frame wird dem Projekt hinzugefügt, gleichzeitig aber wird eine Instanz dieses Frames auf dem von dir bestimmten Container erstellt. Gruss Delbor |
AW: Frames
Zitat:
Delphi-Quellcode:
oder ganz Sicher:
procedure TframeSpenden.qryMitgliederspendenNewRecord(DataSet: TDataSet);
begin if qryMitgliederspenden.Active then qryMitgliederspenden.FieldByName('MITGLIEDERNR').AsInteger := frmMitglieder.MitgliederQry.FieldByName('MITGLIEDERNR').AsInteger; end;
Delphi-Quellcode:
procedure TframeSpenden.qryMitgliederspendenNewRecord(DataSet: TDataSet);
begin if Assigned(frmMitglieder) then begin if qryMitgliederspenden.Active and frmMitglieder.MitgliederQry.Active then qryMitgliederspenden.FieldByName('MITGLIEDERNR').AsInteger := frmMitglieder.MitgliederQry.FieldByName('MITGLIEDERNR').AsInteger; end; end; |
AW: Frames
Zitat:
Auch wenn du einen früher erstellten Frame über die Toolpalette dem Projekt hinzufügst, wird noch keine konkrete Instanz erzeugt. Vielmehr wird eine Kopie erzeugt, die als Parent deinen Container hat. Erst wenn du das Programm compilierst und startest, werden die entsprechenden Instanzen erzeugt. Das man im Falle von DB-Komponenten die Daten der eingestellten Datenbank sieht (zu Designtime) liegt letztlich am Komponenteneditor und der IDE, hat aber nur wenig mit dem zu tun, was zur Laufzeit passiert. |
AW: Frames
Hi Ghostwalker
Ich sehe schon, wir sprechen vom gleichen und sind uns sehr einig. Das mit der Kopie zur Designzeit habe ich allerdings nicht gewusst. Entweder sagt die Onlinehilfe von Embarcadero nichts darüber aus und spricht immer nur von Frameinstanzen (Frame1, Frame2 etc.), oder ich habs schlicht überlesen. Das hier finde ich aber trotzdem zumindest alls irritierend: Zitat:
Die Eigenverantwortung des Programmieres beginnt dann, wenn er ein Objekt erstellt und diesem keinen Owner zuweist. So ein Objekt muss zwingend expilizit freigegeben werden. Gruss Delbor |
Alle Zeitangaben in WEZ +1. Es ist jetzt 22:26 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