This is a follow up to my earlier post: String to XmlNode Delphi (or how to add an xml fragment to TXMLDocument) It seemed appropriate to start a new question...
I am essentially adding well formed xml snippets to an existing xmldocument. The code suggested in the previous solution had been working great - until - I added [poPreserveWhiteSpace] to TXMLDocument.ParseOptions.
When I remove [poPreserveWhiteSpace] everything works fine, but whitespace is not preserved. It actually puts the closing tag on a new line.
Here is a code snippet ot the Target TXMLDocument.
StoredXMLObj := TXMLDocument.Create(self);
StoredXMLObj.Options := [doNodeAutoCreate, doNodeAutoIndent];
StoredXMLObj.ParseOptions := StoredXMLObj.ParseOptions + [poPreserveWhiteSpace];
StoredXMLObj.XML.Assign(StoredXML); //StoredXML is a TStringList with a complete XML Document
StoredXMLObj.Active := TRUE;
I have tried different combinations of the Options and ParseOptions above, but I can only get the code to work by removing [poPreserveWhiteSpace].
The code that triggers the exception is the second line of:
tmpNode := storedXMLObj.DocumentElement.ChildNodes[i]; // <Class> node
tmpNode.ChildNodes.Nodes[1].ChildNodes.Nodes[0].ChildNodes.Add(LoadXMLData(MissingElements[j]).DocumentElement); //TMPNode is an IXMLNode and MissingElements is a TStringList
I tried creating a reference to the return value of LoadXMLData(..), and setting those ParseOptions to match, before adding the xml snippet, but no luck there either.
Any thoughts?
Edit: Adding self contained sample code to demonstrate problem. Clarified Title. Here is some simplified code. Note that there will be an exception unless you comment out the line containing [poPreserveWhitespace]. **Edit2: Tweaking code to preserve whitespace as per Remy's suggestion. Still has problem when calling FormatXMLData.
procedure TForm2.BitBtn2Click(Sender: TObject);
var
FragmentXMLObj : TXMLDocument;
StoredXMLObj : TXMLDocument;
FragNode : IXMLNode; //THIS SHOULD BE IXMLNODE, RIGHT?
XMLStarting, XMLFragment, XMLMerged : TStringList;
i : integer;
begin
//StringLists to hold xml data
XMLStarting := TStringList.Create; //COMPLETE XML
XMLFragment := TStringList.Create; //XML FRAGMENT TO INSERT INTO COMPLETE XML
XMLMerged := TStringList.Create; //MERGE OF THE ABOVE TWO.
//STARTING XML
XMLStarting.Add('<?xml version="1.0" encoding="UTF-16" standalone="no"?>');
XMLStarting.Add('<Programs>');
XMLStarting.Add(' <Program_Group Batch_No="{12345678-1234-1234-1234-123456789ABC}" Description="FOO_824_1">');
XMLStarting.Add(' <Program Name="PROG_1">');
XMLStarting.Add(' <Class Name="CLASS_1">');
XMLStarting.Add(' <Property Name="DB" RttiType="tkString"> </Property>');
XMLStarting.Add(' <Property Name="SystemDate" RttiType="tkClass" ClassType="TXSDATE">12/30/1899</Property>');
XMLStarting.Add(' </Class>');
XMLStarting.Add(' </Program>');
XMLStarting.Add(' </Program_Group>');
XMLStarting.Add('</Programs>');
//XML DOCUMENT OBJECT
StoredXMLObj := TXMLDocument.create(self);
//PROBLEM LINE START
StoredXMLObj.ParseOptions := StoredXMLObj.ParseOptions + [poPreserveWhiteSpace];
//PROBLEM LINE END
StoredXMLObj.Options := [doNodeAutoCreate, doNodeAutoIndent];
StoredXMLObj.XML.Text := XMLStarting.Text;
StoredXMLObj.Active := TRUE;
//XML FRAGMENT WITH SPACES
XMLFragment.Add('<ParentNode>');
XMLFragment.Add('<Property Name="VRSN" RttiType="tkString"> </Property>');
XMLFragment.Add('<Property Name="ShowMetaData" RttiType="tkBoolean"> </Property>');
XMLFragment.Add('</ParentNode>');
//--OLD CODE THAT RAISES EXCEPTION--
//INSERTING XML FRAGMENT INTO STARTING XML
// FragNode := storedXMLObj.DocumentElement.ChildNodes[0];
// FragNode.ChildNodes.Nodes[0].ChildNodes.Nodes[0].ChildNodes.Add(LoadXMLData(XMLFragment.Text).DocumentElement.ChildNodes.Nodes[0]);
// FragNode.ChildNodes.Nodes[0].ChildNodes.Nodes[0].ChildNodes.Add(LoadXMLData(XMLFragment.Text).DocumentElement.ChildNodes.Nodes[1]);
//--OLD CODE THAT RAISES EXCEPTION--
FragNode := storedXMLObj.DocumentElement.ChildNodes[1];
FragmentXMLObj := TXMLDocument.Create(self);
FragmentXMLObj.ParseOptions := FragmentXMLObj.ParseOptions + [poPreserveWhiteSpace];
FragmentXMLObj.Options := [doNodeAutoCreate, doNodeAutoIndent];
FragmentXMLObj.LoadFromXML(XMLFragment.Text);
//FragNode.ChildNodes.Nodes[1].ChildNodes.Nodes[1].ChildNodes.Add(FragmentXMLObj.DocumentElement); //this also pulls in the parent tags, which I don't want.
for i := 0 to FragmentXMLObj.DocumentElement.ChildNodes.Count-1 do //easier to just pull in all the nodes (including whitespace, then formatxml to cleanup).
FragNode.ChildNodes.Nodes[1].ChildNodes.Nodes[1].ChildNodes.Add(FragmentXMLObj.DocumentElement.ChildNodes.Nodes[i]);
FragmentXMLObj.Free;
XMLMerged.Text := StoredXMLObj.XML.Text;
XMLMerged.Text := FormatXMLData(XMLMerged.Text); //UGH... FormatXMLData WIPES OUT WHITESPACE PROPERTY VALUES!! Doesn't seem to have any settings either...
XMLMerged.SaveToFile('c:\merged.xml');
XMLStarting.Free;
XMLFragment.Free;
XMLMerged.Free;
StoredXMLObj.Free;
end;
The Resulting Merged XML File... Whitespace property values got wiped out during formatting (and I do need to format the data, otw it is REALLY ugly).
<?xml version="1.0" encoding="UTF-16" standalone="no"?>
<Programs>
<Program_Group Batch_No="{12345678-1234-1234-1234-123456789ABC}" Description="FOO_824_1">
<Program Name="PROG_1">
<Class Name="CLASS_1">
<Property Name="DB" RttiType="tkString"/>
<Property Name="SystemDate" RttiType="tkClass" ClassType="TXSDATE">12/30/1899</Property>
<Property Name="VRSN" RttiType="tkString"/>
<Property Name="ShowMetaData" RttiType="tkBoolean"/>
</Class>
</Program>
</Program_Group>
</Programs>