Moin,
ich bin gerade dabei eine Software von
BDE auf Firebird mit Hilfe von ZEOS umzustellen. In der Anwendung wurden einige TTables verwendet (gekapselt in davon abgeleiteten Klassen). Diese habe ich dann umgeschrieben, sodass sie von TZTable abgeleitet sind und bisher einige Dinge reingepackt, die unterschiedlich waren.
Ich will also nur durch Veränderung dieser Komponente auf ZEOS umstellen, die Anwendung selbst soll möglichst nicht angefasst werden, da dies viel Aufwand beudeten würde.
Zu den Features, die ich einbauen musste gehört die automatische Erstellung des UpdateSQL-Objekts, damit das ZTable auch Daten ändern kann. Das funktioniert auch schon (habs in OnAfterOpen reingepackt und die
SQL-Statements erstellen lassen).
Und nun zum Problem: In der Anwendung wird ein neuer Datensatz eingefügt:
Delphi-Quellcode:
table.Insert;
table.FieldByName('AdrTyp').Value := 3;
table.FieldByName('AdrArt').Value := 1;
table.FieldByName('Nummer').Value := NeueNummer;
table.Post;
Jetzt gibt es ein Feld Name1 das in der Datenbank als NotNull erstellt wurde. Aufgrund meiner UpdateSQL-Scripte wird jetzt natürlich folgender
SQL-Code erzeugt:
INSERT INTO tabelle (AdrTyp,AdrArt,Nummer,Name1) VALUES (3,1,99999,NULL);
da ja alle Felder darin vorkommen. Das läuft aber natürlich auf einen Fehler, weil Name1 nicht NULL sein darf.
Ich muss also vor dem Insert (bzw OnBeforePost) die
SQL-Skripte erneut schreiben und zwar vor allem das Insert-Statement und dort nur die Felder eintragen, die geändert wurden.
Hat jemand eine Ahnung, wie ich das machen kann?
Mit folgendem Code hab ichs mal probiert:
Delphi-Quellcode:
for i:=0
to Fields.Count-1
do begin
//***Update-Object konfigurieren***
if Fields[i].FieldKind<>fkData
then //Abbruch, wenn kein physisches DB-Feld
Continue;
//Key-Felder
if (pfInKey
IN Fields[i].ProviderFlags)
then begin
if keysql<>'
'
then
keysql:=keysql+'
and ';
keysql:=keysql+Fields[i].FieldName+'
= :OLD_'+Fields[i].FieldName;
end;
//Alle Felder für alternative WHERE-Bedingung
if altkeysql<>'
'
then
altkeysql:=altkeysql+'
and ';
altkeysql:=altkeysql+Fields[i].FieldName+'
= :OLD_'+Fields[i].FieldName;
//Felder und Werte für UPDATE-Statement
if modsql<>'
'
then
modsql:=modsql+'
, ';
modsql:=modsql+Fields[i].FieldName+'
= :'+Fields[i].FieldName;
if (Fields[i].OldValue<>Fields[i].NewValue)
or (
not Fields[i].IsNull)
then begin //kritischer Punkt
//Werte für Insert-Statement
if inssql<>'
'
then
inssql:=inssql+'
, ';
inssql:=inssql+'
:'+Fields[i].FieldName;
//Felder für Insert-Statement
if insfields<>'
'
then
insfields:=insfields+'
, ';
insfields:=insfields+Fields[i].FieldName;
end;
end;
if keysql='
'
then keysql:=altkeysql;
UpdateObject.DeleteSQL.Text:='
DELETE FROM '+TableName+'
WHERE '+keysql+'
;';
UpdateObject.ModifySQL.Text:='
UPDATE '+TableName+'
SET '+modsql+'
WHERE '+keysql+'
;';
UpdateObject.InsertSQL.Text:='
INSERT INTO '+TableName+'
('+insfields+'
) VALUES ('+inssql+'
);';
Nur sind in diesem Fall OldValue und NewValue gleich ...