mirror of
https://github.com/Kirill/simplexml.git
synced 2025-02-13 16:32:15 +02:00
4480 lines
119 KiB
ObjectPascal
4480 lines
119 KiB
ObjectPascal
{************************************************************
|
|
SimpleXML - Библиотека для синтаксического разбора текстов XML
|
|
и преобразования в иерархию XML-объектов.
|
|
И наоборот: можно сформировать иерархию XML-объектов, и
|
|
уже из нее получить текст XML.
|
|
Достойная замена для MSXML. При использовании Ansi-строк
|
|
работает быстрее и кушает меньше памяти.
|
|
|
|
(с) Авторские права 2002,2003 Михаил Власов.
|
|
Библиотека бесплатная и может быть использована по любому назначению.
|
|
Разрешается внесение любых изменений и использование измененных
|
|
библиотек без ограничений.
|
|
Единственное требование: Данный текст должен присутствовать
|
|
без изменений во всех модификациях библиотеки.
|
|
|
|
Все пожелания приветствую по адресу misha@integro.ru
|
|
Так же рекомендую посетить мою страничку: http://mv.rb.ru
|
|
Там Вы всегда найдете самую последнюю версию библиотеки.
|
|
Желаю приятного программирования, Михаил Власов.
|
|
|
|
Текущая версия: 1.0.1
|
|
*************************************************************}
|
|
unit SimpleXML;
|
|
|
|
interface
|
|
|
|
uses
|
|
Windows, SysUtils, Classes;
|
|
|
|
const
|
|
BinXmlSignatureSize = Length('< binary-xml >');
|
|
BinXmlSignature: String = '< binary-xml >';
|
|
|
|
BINXML_USE_WIDE_CHARS = 1;
|
|
BINXML_COMPRESSED = 2;
|
|
|
|
XSTR_NULL = '{{null}}';
|
|
|
|
NODE_INVALID = $00000000;
|
|
NODE_ELEMENT = $00000001;
|
|
NODE_ATTRIBUTE = $00000002;
|
|
NODE_TEXT = $00000003;
|
|
NODE_CDATA_SECTION = $00000004;
|
|
NODE_ENTITY_REFERENCE = $00000005;
|
|
NODE_ENTITY = $00000006;
|
|
NODE_PROCESSING_INSTRUCTION = $00000007;
|
|
NODE_COMMENT = $00000008;
|
|
NODE_DOCUMENT = $00000009;
|
|
NODE_DOCUMENT_TYPE = $0000000A;
|
|
NODE_DOCUMENT_FRAGMENT = $0000000B;
|
|
NODE_NOTATION = $0000000C;
|
|
|
|
type
|
|
// TXmlString - тип строковых переменных, используемых в SimpleXML.
|
|
// Может быть String или WideString.
|
|
|
|
{ $DEFINE XML_WIDE_CHARS}
|
|
|
|
{$IFDEF XML_WIDE_CHARS}
|
|
PXmlChar = PWideChar;
|
|
TXmlChar = WideChar;
|
|
TXmlString = WideString;
|
|
{$ELSE}
|
|
PXmlChar = PChar;
|
|
TXmlChar = Char;
|
|
TXmlString = String;
|
|
{$ENDIF}
|
|
|
|
IXmlDocument = interface;
|
|
IXmlElement = interface;
|
|
IXmlText = interface;
|
|
IXmlCDATASection = interface;
|
|
IXmlComment = interface;
|
|
IXmlProcessingInstruction = interface;
|
|
|
|
|
|
// IXmlBase - базовый интерфейс для всех интерфейсов SimpleXML.
|
|
IXmlBase = interface
|
|
// GetObject - возвращает ссылку на объект, реализующий интерфейс.
|
|
function GetObject: TObject;
|
|
end;
|
|
|
|
// IXmlNameTable - таблица имен. Каждому имени сопоставляется некий
|
|
// уникальный числовой идентификатор. Используется для хранения
|
|
// названий тэгов и атрибутов.
|
|
IXmlNameTable = interface(IXmlBase)
|
|
// GetID - возвращает числовой идентификатор указанной строки.
|
|
function GetID(const aName: TXmlString): Integer;
|
|
// GetID - возвращает строку, соответствующую указанному числовому
|
|
// идентификатору.
|
|
function GetName(anID: Integer): TXmlString;
|
|
end;
|
|
|
|
IXmlNode = interface;
|
|
|
|
// IXmlNodeList - список узлов. Список организован в виде массива.
|
|
// Доступ к элементам списка по индексу
|
|
IXmlNodeList = interface(IXmlBase)
|
|
// Get_Count - количество узлов в списке
|
|
function Get_Count: Integer;
|
|
// Get_Item - получить узел по индексу
|
|
function Get_Item(anIndex: Integer): IXmlNode;
|
|
// Get_XML - возвращает представление элементов списка в формате XML
|
|
function Get_XML: TXmlString;
|
|
|
|
property Count: Integer read Get_Count;
|
|
property Item[anIndex: Integer]: IXmlNode read Get_Item; default;
|
|
property XML: TXmlString read Get_XML;
|
|
end;
|
|
|
|
// IXmlNode - узел XML-дерева
|
|
IXmlNode = interface(IXmlBase)
|
|
// Get_NameTable - таблица имен, используемая данным узлом
|
|
function Get_NameTable: IXmlNameTable;
|
|
// Get_NodeName - возвращает название узла. Интерпретация названия узла
|
|
// зависит от его типа
|
|
function Get_NodeName: TXmlString;
|
|
// Get_NodeNameID - возвращает код названия узла
|
|
function Get_NodeNameID: Integer;
|
|
// Get_NodeType - возвращает тип узла
|
|
function Get_NodeType: Integer;
|
|
// Get_Text - возвращает текст узла
|
|
function Get_Text: TXmlString;
|
|
// Set_Text - изменяет текст узла
|
|
procedure Set_Text(const aValue: TXmlString);
|
|
// Get_DataType - возаращает тип данных узла в терминах вариантов
|
|
function Get_DataType: Integer;
|
|
// Get_TypedValue - возвращает
|
|
function Get_TypedValue: Variant;
|
|
// Set_TypedValue - изменяет текст узла на типизированное значение
|
|
procedure Set_TypedValue(const aValue: Variant);
|
|
// Get_XML - возвращает представление узла и всех вложенных узлов
|
|
// в формате XML.
|
|
function Get_XML: TXmlString;
|
|
|
|
// CloneNode - создает точную копию данного узла
|
|
// Если задан признак aDeep, то создастся копия
|
|
// всей ветви иерархии от данного узла.
|
|
function CloneNode(aDeep: Boolean = True): IXmlNode;
|
|
|
|
// Get_ParentNode - возвращает родительский узел
|
|
function Get_ParentNode: IXmlNode;
|
|
// Get_OwnerDocument - возвращает XML-документ,
|
|
// в котором расположен данный узел
|
|
function Get_OwnerDocument: IXmlDocument;
|
|
|
|
// Get_ChildNodes - возвращает список дочерних узлов
|
|
function Get_ChildNodes: IXmlNodeList;
|
|
// AppendChild - добавляет указанный узел в конец списка дочерних узлов
|
|
procedure AppendChild(const aChild: IXmlNode);
|
|
// InsertBefore - добавляет указанный узел в указанное место списка дочерних узлов
|
|
procedure InsertBefore(const aChild, aBefore: IXmlNode);
|
|
// ReplaceChild - заменяет указанный узел другим узлом
|
|
procedure ReplaceChild(const aNewChild, anOldChild: IXmlNode);
|
|
// RemoveChild - удаляет указанный узел из списка дочерних узлов
|
|
procedure RemoveChild(const aChild: IXmlNode);
|
|
|
|
// AppendElement - создает элемент и добавляет его в конец списка
|
|
// в конец списка дочерних объектов
|
|
function AppendElement(aNameID: Integer): IXmlElement; overload;
|
|
function AppendElement(const aName: TxmlString): IXmlElement; overload;
|
|
|
|
// AppendText - создает текстовый узел и добавляет его
|
|
// в конец списка дочерних объектов
|
|
function AppendText(const aData: TXmlString): IXmlText;
|
|
|
|
// AppendCDATA - создает секцию CDATA и добавляет ее
|
|
// в конец списка дочерних объектов
|
|
function AppendCDATA(const aData: TXmlString): IXmlCDATASection;
|
|
|
|
// AppendComment - создает комментарий и добавляет его
|
|
// в конец списка дочерних объектов
|
|
function AppendComment(const aData: TXmlString): IXmlComment;
|
|
|
|
// AppendProcessingInstruction - создает инструкцию и добавляет её
|
|
// в конец списка дочерних объектов
|
|
function AppendProcessingInstruction(aTargetID: Integer;
|
|
const aData: TXmlString): IXmlProcessingInstruction; overload;
|
|
function AppendProcessingInstruction(const aTarget: TXmlString;
|
|
const aData: TXmlString): IXmlProcessingInstruction; overload;
|
|
|
|
// GetChildText - возвращает значение дочернего узла
|
|
// SetChildText - добавляет или изменяет значение дочернего узла
|
|
function GetChildText(const aName: TXmlString; const aDefault: TXmlString = ''): TXmlString; overload;
|
|
function GetChildText(aNameID: Integer; const aDefault: TXmlString = ''): TXmlString; overload;
|
|
procedure SetChildText(const aName, aValue: TXmlString); overload;
|
|
procedure SetChildText(aNameID: Integer; const aValue: TXmlString); overload;
|
|
|
|
// NeedChild - возвращает дочерний узел с указанным именем.
|
|
// Если узел не найден, то генерируется исключение
|
|
function NeedChild(aNameID: Integer): IXmlNode; overload;
|
|
function NeedChild(const aName: TXmlString): IXmlNode; overload;
|
|
|
|
// EnsureChild - возвращает дочерний узел с указанным именем.
|
|
// Если узел не найден, то он будет создан
|
|
function EnsureChild(aNameID: Integer): IXmlNode; overload;
|
|
function EnsureChild(const aName: TXmlString): IXmlNode; overload;
|
|
|
|
// RemoveAllChilds - удаляет все дочерние узлы
|
|
procedure RemoveAllChilds;
|
|
|
|
// SelectNodes - производит выборку узлов, удовлетворяющих
|
|
// указанным критериям
|
|
function SelectNodes(const anExpression: TXmlString): IXmlNodeList;
|
|
// SelectSingleNode - производит поиск первого узла, удовлетворяющего
|
|
// указанным критериям
|
|
function SelectSingleNode(const anExpression: TXmlString): IXmlNode;
|
|
// FindElement - производит поиск первого узла, удовлетворяющего
|
|
// указанным критериям
|
|
function FindElement(const anElementName, anAttrName: String; const anAttrValue: Variant): IXmlElement;
|
|
|
|
// Get_AttrCount - возвращает количество атрибутов
|
|
function Get_AttrCount: Integer;
|
|
// Get_AttrNameID - возвращает код названия атрибута
|
|
function Get_AttrNameID(anIndex: Integer): Integer;
|
|
// Get_AttrName - возвращает название атрибута
|
|
function Get_AttrName(anIndex: Integer): TXmlString;
|
|
// RemoveAttr - удаляет атрибут
|
|
procedure RemoveAttr(const aName: TXmlString); overload;
|
|
procedure RemoveAttr(aNameID: Integer); overload;
|
|
// RemoveAllAttrs - удаляет все атрибуты
|
|
procedure RemoveAllAttrs;
|
|
|
|
// AttrExists - проверяет, задан ли указанный атрибут.
|
|
function AttrExists(aNameID: Integer): Boolean; overload;
|
|
function AttrExists(const aName: TXmlString): Boolean; overload;
|
|
|
|
// GetAttrType - возаращает тип данных атрибута в терминах вариантов
|
|
function GetAttrType(aNameID: Integer): Integer; overload;
|
|
function GetAttrType(const aName: TXmlString): Integer; overload;
|
|
|
|
// GetAttrType - возвращает тип атрибута
|
|
// Result
|
|
// GetVarAttr - возвращает типизированное значение указанного атрибута.
|
|
// Если атрибут не задан, то возвращается значение по умолчанию
|
|
// SetAttr - изменяет или добавляет указанный атрибут
|
|
function GetVarAttr(aNameID: Integer; const aDefault: Variant): Variant; overload;
|
|
function GetVarAttr(const aName: TXmlString; const aDefault: Variant): Variant; overload;
|
|
procedure SetVarAttr(aNameID: Integer; const aValue: Variant); overload;
|
|
procedure SetVarAttr(const aName: TXmlString; aValue: Variant); overload;
|
|
|
|
// NeedAttr - возвращает строковое значение указанного атрибута.
|
|
// Если атрибут не задан, то генерируется исключение
|
|
function NeedAttr(aNameID: Integer): TXmlString; overload;
|
|
function NeedAttr(const aName: TXmlString): TXmlString; overload;
|
|
|
|
// GetAttr - возвращает строковое значение указанного атрибута.
|
|
// Если атрибут не задан, то возвращается значение по умолчанию
|
|
// SetAttr - изменяет или добавляет указанный атрибут
|
|
function GetAttr(aNameID: Integer; const aDefault: TXmlString = ''): TXmlString; overload;
|
|
function GetAttr(const aName: TXmlString; const aDefault: TXmlString = ''): TXmlString; overload;
|
|
procedure SetAttr(aNameID: Integer; const aValue: TXmlString); overload;
|
|
procedure SetAttr(const aName, aValue: TXmlString); overload;
|
|
|
|
// GetBoolAttr - возвращает целочисленное значение указанного атрибута
|
|
// SetBoolAttr - изменяет или добавляет указанный атрибут целочисленным
|
|
// значением
|
|
function GetBoolAttr(aNameID: Integer; aDefault: Boolean = False): Boolean; overload;
|
|
function GetBoolAttr(const aName: TXmlString; aDefault: Boolean = False): Boolean; overload;
|
|
procedure SetBoolAttr(aNameID: Integer; aValue: Boolean = False); overload;
|
|
procedure SetBoolAttr(const aName: TXmlString; aValue: Boolean); overload;
|
|
|
|
// GetIntAttr - возвращает целочисленное значение указанного атрибута
|
|
// SetIntAttr - изменяет или добавляет указанный атрибут целочисленным
|
|
// значением
|
|
function GetIntAttr(aNameID: Integer; aDefault: Integer = 0): Integer; overload;
|
|
function GetIntAttr(const aName: TXmlString; aDefault: Integer = 0): Integer; overload;
|
|
procedure SetIntAttr(aNameID: Integer; aValue: Integer); overload;
|
|
procedure SetIntAttr(const aName: TXmlString; aValue: Integer); overload;
|
|
|
|
// GetDateTimeAttr - возвращает целочисленное значение указанного атрибута
|
|
// SetDateTimeAttr - изменяет или добавляет указанный атрибут целочисленным
|
|
// значением
|
|
function GetDateTimeAttr(aNameID: Integer; aDefault: TDateTime = 0): TDateTime; overload;
|
|
function GetDateTimeAttr(const aName: TXmlString; aDefault: TDateTime = 0): TDateTime; overload;
|
|
procedure SetDateTimeAttr(aNameID: Integer; aValue: TDateTime); overload;
|
|
procedure SetDateTimeAttr(const aName: TXmlString; aValue: TDateTime); overload;
|
|
|
|
// GetFloatAttr - возвращает значение указанного атрибута в виде
|
|
// вещественного числа
|
|
// SetFloatAttr - изменяет или добавляет указанный атрибут вещественным
|
|
// значением
|
|
function GetFloatAttr(aNameID: Integer; aDefault: Double = 0): Double; overload;
|
|
function GetFloatAttr(const aName: TXmlString; aDefault: Double = 0): Double; overload;
|
|
procedure SetFloatAttr(aNameID: Integer; aValue: Double); overload;
|
|
procedure SetFloatAttr(const aName: TXmlString; aValue: Double); overload;
|
|
|
|
// GetHexAttr - получение значения указанного атрибута в целочисленном виде.
|
|
// Строковое значение атрибута преобразуется в целое число. Исходная
|
|
// строка должна быть задана в шестнадцатиричном виде без префиксов
|
|
// ("$", "0x" и пр.) Если преобразование не может быть выполнено,
|
|
// генерируется исключение.
|
|
// Если атрибут не задан, возвращается значение параметра aDefault.
|
|
// SetHexAttr - изменение значения указанного атрибута на строковое
|
|
// представление целого числа в шестнадцатиричном виде без префиксов
|
|
// ("$", "0x" и пр.) Если преобразование не может быть выполнено,
|
|
// генерируется исключение.
|
|
// Если атрибут не был задан, до он будет добавлен.
|
|
// Если был задан, то будет изменен.
|
|
function GetHexAttr(const aName: TXmlString; aDefault: Integer = 0): Integer; overload;
|
|
function GetHexAttr(aNameID: Integer; aDefault: Integer = 0): Integer; overload;
|
|
procedure SetHexAttr(const aName: TXmlString; aValue: Integer; aDigits: Integer = 8); overload;
|
|
procedure SetHexAttr(aNameID: Integer; aValue: Integer; aDigits: Integer = 8); overload;
|
|
|
|
// GetEnumAttr - ищет значение атрибута в указанном списке строк и
|
|
// возвращает индекс найденной строки. Если атрибут задан но не найден
|
|
// в списке, то генерируется исключение.
|
|
// Если атрибут не задан, возвращается значение параметра aDefault.
|
|
function GetEnumAttr(const aName: TXmlString;
|
|
const aValues: array of TXmlString; aDefault: Integer = 0): Integer; overload;
|
|
function GetEnumAttr(aNameID: Integer;
|
|
const aValues: array of TXmlString; aDefault: Integer = 0): Integer; overload;
|
|
|
|
function NeedEnumAttr(const aName: TXmlString;
|
|
const aValues: array of TXmlString): Integer; overload;
|
|
function NeedEnumAttr(aNameID: Integer;
|
|
const aValues: array of TXmlString): Integer; overload;
|
|
|
|
function Get_Values(const aName: String): Variant;
|
|
procedure Set_Values(const aName: String; const aValue: Variant);
|
|
|
|
function AsElement: IXmlElement;
|
|
function AsText: IXmlText;
|
|
function AsCDATASection: IXmlCDATASection;
|
|
function AsComment: IXmlComment;
|
|
function AsProcessingInstruction: IXmlProcessingInstruction;
|
|
|
|
property NodeName: TXmlString read Get_NodeName;
|
|
property NodeNameID: Integer read Get_NodeNameID;
|
|
property NodeType: Integer read Get_NodeType;
|
|
property ParentNode: IXmlNode read Get_ParentNode;
|
|
property OwnerDocument: IXmlDocument read Get_OwnerDocument;
|
|
property NameTable: IXmlNameTable read Get_NameTable;
|
|
property ChildNodes: IXmlNodeList read Get_ChildNodes;
|
|
property AttrCount: Integer read Get_AttrCount;
|
|
property AttrNames[anIndex: Integer]: TXmlString read Get_AttrName;
|
|
property AttrNameIDs[anIndex: Integer]: Integer read Get_AttrNameID;
|
|
property Text: TXmlString read Get_Text write Set_Text;
|
|
property DataType: Integer read Get_DataType;
|
|
property TypedValue: Variant read Get_TypedValue write Set_TypedValue;
|
|
property XML: TXmlString read Get_XML;
|
|
property Values[const aName: String]: Variant read Get_Values write Set_Values; default;
|
|
end;
|
|
|
|
IXmlElement = interface(IXmlNode)
|
|
// ReplaceTextByCDATASection - удаляет все текстовые элементы и добавляет
|
|
// одну секцию CDATA, содержащую указанный текст
|
|
procedure ReplaceTextByCDATASection(const aText: TXmlString);
|
|
|
|
// ReplaceTextByBynaryData - удаляет все текстовые элементы и добавляет
|
|
// один текстовый элемент, содержащий указанные двоичные данные
|
|
// в формате "base64".
|
|
// Если параметр aMaxLineLength не равен нулю, то производится разбивка
|
|
// полученой строки на строки длиной aMaxLineLength.
|
|
// Строки разделяются парой символов #13#10 (CR,LF).
|
|
// После последней строки указанные символы не вставляются.
|
|
procedure ReplaceTextByBynaryData(const aData; aSize: Integer;
|
|
aMaxLineLength: Integer);
|
|
|
|
// GetTextAsBynaryData - cобирает все текстовые элементы в одну строку и
|
|
// производит преобразование из формата "base64" в двоичные данные.
|
|
// При преобразовании игнорируются все пробельные символы (с кодом <= ' '),
|
|
// содержащиеся в исходной строке.
|
|
function GetTextAsBynaryData: TXmlString;
|
|
|
|
end;
|
|
|
|
IXmlCharacterData = interface(IXmlNode)
|
|
end;
|
|
|
|
IXmlText = interface(IXmlCharacterData)
|
|
end;
|
|
|
|
IXmlCDATASection = interface(IXmlCharacterData)
|
|
end;
|
|
|
|
IXmlComment = interface(IXmlCharacterData)
|
|
end;
|
|
|
|
IXmlProcessingInstruction = interface(IXmlNode)
|
|
end;
|
|
|
|
IXmlDocument = interface(IXmlNode)
|
|
function Get_DocumentElement: IXmlElement;
|
|
function Get_BinaryXML: String;
|
|
function Get_PreserveWhiteSpace: Boolean;
|
|
procedure Set_PreserveWhiteSpace(aValue: Boolean);
|
|
|
|
function NewDocument(const aVersion, anEncoding: TXmlString;
|
|
aRootElementNameID: Integer): IXmlElement; overload;
|
|
function NewDocument(const aVersion, anEncoding,
|
|
aRootElementName: TXmlString): IXmlElement; overload;
|
|
|
|
function CreateElement(aNameID: Integer): IXmlElement; overload;
|
|
function CreateElement(const aName: TXmlString): IXmlElement; overload;
|
|
function CreateText(const aData: TXmlString): IXmlText;
|
|
function CreateCDATASection(const aData: TXmlString): IXmlCDATASection;
|
|
function CreateComment(const aData: TXmlString): IXmlComment;
|
|
function CreateProcessingInstruction(const aTarget,
|
|
aData: TXmlString): IXmlProcessingInstruction; overload;
|
|
function CreateProcessingInstruction(aTargetID: Integer;
|
|
const aData: TXmlString): IXmlProcessingInstruction; overload;
|
|
|
|
procedure LoadXML(const aXML: TXmlString);
|
|
procedure LoadBinaryXML(const aXML: String);
|
|
|
|
procedure Load(aStream: TStream); overload;
|
|
procedure Load(const aFileName: TXmlString); overload;
|
|
|
|
procedure LoadResource(aType, aName: PChar);
|
|
|
|
procedure Save(aStream: TStream); overload;
|
|
procedure Save(const aFileName: TXmlString); overload;
|
|
|
|
procedure SaveBinary(aStream: TStream; anOptions: LongWord = 0); overload;
|
|
procedure SaveBinary(const aFileName: TXmlString; anOptions: LongWord = 0); overload;
|
|
|
|
property PreserveWhiteSpace: Boolean read Get_PreserveWhiteSpace write Set_PreserveWhiteSpace;
|
|
property DocumentElement: IXmlElement read Get_DocumentElement;
|
|
property BinaryXML: String read Get_BinaryXML;
|
|
end;
|
|
|
|
function CreateNameTable(aHashTableSize: Integer = 4096): IXmlNameTable;
|
|
function CreateXmlDocument(
|
|
const aRootElementName: String = '';
|
|
const aVersion: String = '1.0';
|
|
const anEncoding: String = ''; // SimpleXmlDefaultEncoding
|
|
const aNames: IXmlNameTable = nil): IXmlDocument;
|
|
|
|
function CreateXmlElement(const aName: TXmlString; const aNameTable: IXmlNameTable = nil): IXmlElement;
|
|
function LoadXmlDocumentFromXML(const aXML: TXmlString): IXmlDocument;
|
|
function LoadXmlDocumentFromBinaryXML(const aXML: String): IXmlDocument;
|
|
|
|
function LoadXmlDocument(aStream: TStream): IXmlDocument; overload;
|
|
function LoadXmlDocument(const aFileName: TXmlString): IXmlDocument; overload;
|
|
function LoadXmlDocument(aResType, aResName: PChar): IXmlDocument; overload;
|
|
|
|
|
|
var
|
|
DefaultNameTable: IXmlNameTable = nil;
|
|
DefaultPreserveWhiteSpace: Boolean = False;
|
|
DefaultEncoding: String = 'windows-1251';
|
|
DefaultIndentText: String = ^I;
|
|
|
|
resourcestring
|
|
SSimpleXmlError1 = 'Ошибка получения элемента списка: индекс выходит за пределы';
|
|
SSimpleXmlError2 = 'Не завершено определение элемента';
|
|
SSimpleXmlError3 = 'Некорретный символ в имени элемента';
|
|
SSimpleXmlError4 = 'Ошибка чтения двоичного XML: некорректный тип узла';
|
|
SSimpleXmlError5 = 'Ошибка записи двоичного XML: некорректный тип узла';
|
|
SSimpleXmlError6 = 'Неверное значение атрибута "%s" элемента "%s".'^M^J +
|
|
'Допустимые значения:'^M^J +
|
|
'%s';
|
|
SSimpleXmlError7 = 'Не найден атрибут "%s"';
|
|
SSimpleXmlError8 = 'Не задан атрибут "%s"';
|
|
SSimpleXmlError9 = 'Данная возможность не поддерживается SimpleXML';
|
|
SSimpleXmlError10 = 'Ошибка: не найден дочерний элемент "%s".';
|
|
SSimpleXmlError11 = 'Имя должно начинаться с буквы или "_"';
|
|
SSimpleXmlError12 = 'Ожидается число';
|
|
SSimpleXmlError13 = 'Ожидается шестнадцатеричное число';
|
|
SSimpleXmlError14 = 'Ожидается "#" или имя упрамляющего символа';
|
|
SSimpleXmlError15 = 'Некорректное имя управляющего символа';
|
|
SSimpleXmlError16 = 'Ожидается "%c"';
|
|
SSimpleXmlError17 = 'Ожидается "%s"';
|
|
SSimpleXmlError18 = 'Символ "<" не может использоваться в значениях атрибутов';
|
|
SimpleXmlError19 = 'Ожидается "%s"';
|
|
SSimpleXmlError20 = 'Ожидается значение атрибута';
|
|
SSimpleXmlError21 = 'Ожидается строковая константа';
|
|
SimpleXmlError22 = 'Ожидается "%s"';
|
|
SSimpleXmlError23 = 'Ошибка чтения данных.';
|
|
SSimpleXmlError24 = 'Ошибка чтения значения: некорректный тип.';
|
|
SSimpleXmlError25 = 'Ошибка записи значения: некорректный тип.';
|
|
|
|
function XSTRToFloat(s: TXmlString): Double;
|
|
function FloatToXSTR(v: Double): TXmlString;
|
|
function DateTimeToXSTR(v: TDateTime): TXmlString;
|
|
function VarToXSTR(const v: TVarData): TXmlString;
|
|
|
|
function TextToXML(const aText: TXmlString): TXmlString;
|
|
function BinToBase64(const aBin; aSize, aMaxLineLength: Integer): String;
|
|
function Base64ToBin(const aBase64: String): String;
|
|
function IsXmlDataString(const aData: String): Boolean;
|
|
function XmlIsInBinaryFormat(const aData: String): Boolean;
|
|
procedure PrepareToSaveXml(var anElem: IXmlElement; const aChildName: String);
|
|
function PrepareToLoadXml(var anElem: IXmlElement; const aChildName: String): Boolean;
|
|
|
|
implementation
|
|
|
|
uses
|
|
Variants, DateUtils;
|
|
|
|
function TextToXML(const aText: TXmlString): TXmlString;
|
|
var
|
|
i, j: Integer;
|
|
begin
|
|
j := 0;
|
|
for i := 1 to Length(aText) do
|
|
case aText[i] of
|
|
'<', '>': Inc(j, 4);
|
|
'&': Inc(j, 5);
|
|
'"': Inc(j, 6);
|
|
else
|
|
Inc(j);
|
|
end;
|
|
if j = Length(aText) then
|
|
Result := aText
|
|
else begin
|
|
SetLength(Result, j);
|
|
j := 1;
|
|
for i := 1 to Length(aText) do
|
|
case aText[i] of
|
|
'<': begin Move(PChar('<')^, Result[j], 4); Inc(j, 4) end;
|
|
'>': begin Move(PChar('>')^, Result[j], 4); Inc(j, 4) end;
|
|
'&': begin Move(PChar('&')^, Result[j], 5); Inc(j, 5) end;
|
|
'"': begin Move(PChar('"')^, Result[j], 6); Inc(j, 6) end;
|
|
else begin Result[j] := aText[i]; Inc(j) end;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
function XSTRToFloat(s: TXmlString): Double;
|
|
var
|
|
aPos: Integer;
|
|
begin
|
|
if '.' = DecimalSeparator then
|
|
aPos := Pos(',', s)
|
|
else if ',' = DecimalSeparator then
|
|
aPos := Pos('.', s)
|
|
else begin
|
|
aPos := Pos(',', s);
|
|
if aPos = 0 then
|
|
aPos := Pos('.', s);
|
|
end;
|
|
|
|
if aPos <> 0 then
|
|
s[aPos] := TXmlChar(DecimalSeparator);
|
|
Result := StrToFloat(s);
|
|
end;
|
|
|
|
function FloatToXSTR(v: Double): TXmlString;
|
|
var
|
|
aPos: Integer;
|
|
begin
|
|
Result := FloatToStr(v);
|
|
aPos := Pos(DecimalSeparator, Result);
|
|
if aPos <> 0 then
|
|
Result[aPos] := '.';
|
|
end;
|
|
|
|
function XSTRToDateTime(const s: String): TDateTime;
|
|
var
|
|
aPos: Integer;
|
|
|
|
function FetchTo(aStop: Char): Integer;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
i := aPos;
|
|
while (i <= Length(s)) and (s[i] in ['0'..'9']) do
|
|
Inc(i);
|
|
if i > aPos then
|
|
Result := StrToInt(Copy(s, aPos, i - aPos))
|
|
else
|
|
Result := 0;
|
|
if (i <= Length(s)) and (s[i] = aStop) then
|
|
aPos := i + 1
|
|
else
|
|
aPos := Length(s) + 1;
|
|
end;
|
|
|
|
var
|
|
y, m, d, h, n, ss: Integer;
|
|
begin
|
|
aPos := 1;
|
|
y := FetchTo('-'); m := FetchTo('-'); d := FetchTo('T');
|
|
h := FetchTo('-'); n := FetchTo('-'); ss := FetchTo('-');
|
|
Result := EncodeDateTime(y, m, d, h, n, ss, 0);
|
|
end;
|
|
|
|
function DateTimeToXSTR(v: TDateTime): TXmlString;
|
|
var
|
|
y, m, d, h, n, s, ms: Word;
|
|
begin
|
|
DecodeDateTime(v, y, m, d, h, n, s, ms);
|
|
Result := Format('%.4d-%.2d-%.2dT%.2d-%.2d-%.2d', [y, m, d, h, n, s])
|
|
end;
|
|
|
|
function VarToXSTR(const v: TVarData): TXmlString;
|
|
const
|
|
BoolStr: array[Boolean] of TXmlString = ('0', '1');
|
|
var
|
|
p: Pointer;
|
|
begin
|
|
case v.VType of
|
|
varNull: Result := XSTR_NULL;
|
|
varSmallint: Result := IntToStr(v.VSmallInt);
|
|
varInteger: Result := IntToStr(v.VInteger);
|
|
varSingle: Result := FloatToXSTR(v.VSingle);
|
|
varDouble: Result := FloatToXSTR(v.VDouble);
|
|
varCurrency: Result := FloatToXSTR(v.VCurrency);
|
|
varDate: Result := DateTimeToXSTR(v.VDate);
|
|
varOleStr: Result := v.VOleStr;
|
|
varBoolean: Result := BoolStr[v.VBoolean = True];
|
|
varShortInt: Result := IntToStr(v.VShortInt);
|
|
varByte: Result := IntToStr(v.VByte);
|
|
varWord: Result := IntToStr(v.VWord);
|
|
varLongWord: Result := IntToStr(v.VLongWord);
|
|
varInt64: Result := IntToStr(v.VInt64);
|
|
varString: Result := String(v.VString);
|
|
varArray + varByte:
|
|
begin
|
|
p := VarArrayLock(Variant(v));
|
|
try
|
|
Result := BinToBase64(p^, VarArrayHighBound(Variant(v), 1) - VarArrayLowBound(Variant(v), 1) + 1, 0);
|
|
finally
|
|
VarArrayUnlock(Variant(v))
|
|
end
|
|
end;
|
|
else
|
|
Result := Variant(v)
|
|
end;
|
|
end;
|
|
|
|
procedure PrepareToSaveXml(var anElem: IXmlElement; const aChildName: String);
|
|
begin
|
|
if aChildName <> '' then
|
|
anElem := anElem.AppendElement(aChildName);
|
|
end;
|
|
|
|
function PrepareToLoadXml(var anElem: IXmlElement; const aChildName: String): Boolean;
|
|
begin
|
|
if (aChildName <> '') and Assigned(anElem) then
|
|
anElem := anElem.selectSingleNode(aChildName).AsElement;
|
|
Result := Assigned(anElem);
|
|
end;
|
|
|
|
function LoadXMLResource(aModule: HMODULE; aName, aType: PChar; const aXMLDoc: IXmlDocument): boolean;
|
|
var
|
|
aRSRC: HRSRC;
|
|
aGlobal: HGLOBAL;
|
|
aSize: DWORD;
|
|
aPointer: Pointer;
|
|
|
|
aStream: TStringStream;
|
|
begin
|
|
Result := false;
|
|
|
|
aRSRC := FindResource(aModule, aName, aType);
|
|
if aRSRC <> 0 then begin
|
|
aGlobal := LoadResource(aModule, aRSRC);
|
|
aSize := SizeofResource(aModule, aRSRC);
|
|
if (aGlobal <> 0) and (aSize <> 0) then begin
|
|
aPointer := LockResource(aGlobal);
|
|
if Assigned(aPointer) then begin
|
|
aStream := TStringStream.Create('');
|
|
try
|
|
aStream.WriteBuffer(aPointer^, aSize);
|
|
aXMLDoc.LoadXML(aStream.DataString);
|
|
Result := true;
|
|
finally
|
|
aStream.Free;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
function IsXmlDataString(const aData: String): Boolean;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
Result := Copy(aData, 1, BinXmlSignatureSize) = BinXmlSignature;
|
|
if not Result then begin
|
|
i := 1;
|
|
while (i <= Length(aData)) and (aData[i] in [#10, #13, #9, ' ']) do
|
|
Inc(i);
|
|
Result := Copy(aData, i, Length('<?xml ')) = '<?xml ';
|
|
end;
|
|
end;
|
|
|
|
function XmlIsInBinaryFormat(const aData: String): Boolean;
|
|
begin
|
|
Result := Copy(aData, 1, BinXmlSignatureSize) = BinXmlSignature
|
|
end;
|
|
|
|
var
|
|
Base64Map: array [0..63] of Char = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';
|
|
|
|
type
|
|
PChars = ^TChars;
|
|
TChars = packed record a, b, c, d: Char end;
|
|
POctet = ^TOctet;
|
|
TOctet = packed record a, b, c: Byte; end;
|
|
|
|
procedure OctetToChars(po: POctet; aCount: Integer; pc: PChars);
|
|
var
|
|
o: Integer;
|
|
begin
|
|
if aCount = 1 then begin
|
|
o := po.a shl 16;
|
|
LongWord(pc^) := $3D3D3D3D;
|
|
pc.a := Base64Map[(o shr 18) and $3F];
|
|
pc.b := Base64Map[(o shr 12) and $3F];
|
|
end
|
|
else if aCount = 2 then begin
|
|
o := po.a shl 16 or po.b shl 8;
|
|
LongWord(pc^) := $3D3D3D3D;
|
|
pc.a := Base64Map[(o shr 18) and $3F];
|
|
pc.b := Base64Map[(o shr 12) and $3F];
|
|
pc.c := Base64Map[(o shr 6) and $3F];
|
|
end
|
|
else if aCount > 2 then begin
|
|
o := po.a shl 16 or po.b shl 8 or po.c;
|
|
LongWord(pc^) := $3D3D3D3D;
|
|
pc.a := Base64Map[(o shr 18) and $3F];
|
|
pc.b := Base64Map[(o shr 12) and $3F];
|
|
pc.c := Base64Map[(o shr 6) and $3F];
|
|
pc.d := Base64Map[o and $3F];
|
|
end;
|
|
end;
|
|
|
|
function BinToBase64(const aBin; aSize, aMaxLineLength: Integer): String;
|
|
var
|
|
o: POctet;
|
|
c: PChars;
|
|
aCount: Integer;
|
|
i: Integer;
|
|
begin
|
|
o := @aBin;
|
|
aCount := aSize;
|
|
SetLength(Result, ((aCount + 2) div 3)*4);
|
|
c := PChars(Result);
|
|
while aCount > 0 do begin
|
|
OctetToChars(o, aCount, c);
|
|
Inc(o);
|
|
Inc(c);
|
|
Dec(aCount, 3);
|
|
end;
|
|
if aMaxLineLength > 0 then begin
|
|
i := aMaxLineLength;
|
|
while i <= Length(Result) do begin
|
|
Insert(#13#10, Result, i);
|
|
Inc(i, 2 + aMaxLineLength);
|
|
end
|
|
end;
|
|
end;
|
|
|
|
function CharTo6Bit(c: Char): Byte;
|
|
begin
|
|
if (c >= 'A') and (c <= 'Z') then
|
|
Result := Ord(c) - Ord('A')
|
|
else if (c >= 'a') and (c <= 'z') then
|
|
Result := Ord(c) - Ord('a') + 26
|
|
else if (c >= '0') and (c <= '9') then
|
|
Result := Ord(c) - Ord('0') + 52
|
|
else if c = '+' then
|
|
Result := 62
|
|
else if c = '/' then
|
|
Result := 63
|
|
else
|
|
Result := 0
|
|
end;
|
|
|
|
procedure CharsToOctet(c: PChars; o: POctet);
|
|
var
|
|
i: Integer;
|
|
begin
|
|
if c.c = '=' then begin // 1 byte
|
|
i := CharTo6Bit(c.a) shl 18 or CharTo6Bit(c.b) shl 12;
|
|
o.a := (i shr 16) and $FF;
|
|
end
|
|
else if c.d = '=' then begin // 2 bytes
|
|
i := CharTo6Bit(c.a) shl 18 or CharTo6Bit(c.b) shl 12 or CharTo6Bit(c.c) shl 6;
|
|
o.a := (i shr 16) and $FF;
|
|
o.b := (i shr 8) and $FF;
|
|
end
|
|
else begin // 3 bytes
|
|
i := CharTo6Bit(c.a) shl 18 or CharTo6Bit(c.b) shl 12 or CharTo6Bit(c.c) shl 6 or CharTo6Bit(c.d);
|
|
o.a := (i shr 16) and $FF;
|
|
o.b := (i shr 8) and $FF;
|
|
o.c := i and $FF;
|
|
end;
|
|
end;
|
|
|
|
function Base64ToBin(const aBase64: String): String;
|
|
var
|
|
o: POctet;
|
|
c: PChars;
|
|
aCount: Integer;
|
|
s: String;
|
|
i, j: Integer;
|
|
begin
|
|
s := aBase64;
|
|
i := 1;
|
|
while i <= Length(s) do begin
|
|
while (i <= Length(s)) and (s[i] > ' ') do
|
|
Inc(i);
|
|
if i <= Length(s) then begin
|
|
j := i;
|
|
while (j <= Length(s)) and (s[j] <= ' ') do
|
|
Inc(j);
|
|
Delete(s, i, j - i);
|
|
end;
|
|
end;
|
|
|
|
if Length(s) < 4 then
|
|
Result := ''
|
|
else begin
|
|
aCount := ((Length(s) + 3) div 4)*3;
|
|
if aCount > 0 then begin
|
|
if s[Length(s) - 1] = '=' then
|
|
Dec(aCount, 2)
|
|
else if s[Length(s)] = '=' then
|
|
Dec(aCount);
|
|
SetLength(Result, aCount);
|
|
FillChar(Result[1], aCount, '*');
|
|
c := @s[1];
|
|
o := @Result[1];
|
|
while aCount > 0 do begin
|
|
CharsToOctet(c, o);
|
|
Inc(o);
|
|
Inc(c);
|
|
Dec(aCount, 3);
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
|
|
type
|
|
TBinXmlReader = class
|
|
private
|
|
FOptions: LongWord;
|
|
public
|
|
procedure Read(var aBuf; aSize: Integer); virtual; abstract;
|
|
|
|
function ReadLongint: Longint;
|
|
function ReadAnsiString: String;
|
|
function ReadWideString: WideString;
|
|
function ReadXmlString: TXmlString;
|
|
procedure ReadVariant(var v: TVarData);
|
|
end;
|
|
|
|
TStmXmlReader = class(TBinXmlReader)
|
|
private
|
|
FStream: TStream;
|
|
FOptions: LongWord;
|
|
FBufStart,
|
|
FBufEnd,
|
|
FBufPtr: PChar;
|
|
FBufSize,
|
|
FRestSize: Integer;
|
|
public
|
|
constructor Create(aStream: TStream; aBufSize: Integer);
|
|
destructor Destroy; override;
|
|
|
|
procedure Read(var aBuf; aSize: Integer); override;
|
|
end;
|
|
|
|
TStrXmlReader = class(TBinXmlReader)
|
|
private
|
|
FString: String;
|
|
FOptions: LongWord;
|
|
FPtr: PChar;
|
|
FRestSize: Integer;
|
|
public
|
|
constructor Create(const aStr: String);
|
|
|
|
procedure Read(var aBuf; aSize: Integer); override;
|
|
end;
|
|
|
|
TBinXmlWriter = class
|
|
private
|
|
FOptions: LongWord;
|
|
public
|
|
procedure Write(const aBuf; aSize: Integer); virtual; abstract;
|
|
|
|
procedure WriteLongint(aValue: Longint);
|
|
procedure WriteAnsiString(const aValue: String);
|
|
procedure WriteWideString(const aValue: WideString);
|
|
procedure WriteXmlString(const aValue: TXmlString);
|
|
procedure WriteVariant(const v: TVarData);
|
|
end;
|
|
|
|
TStmXmlWriter = class(TBinXmlWriter)
|
|
private
|
|
FStream: TStream;
|
|
FBufStart,
|
|
FBufEnd,
|
|
FBufPtr: PChar;
|
|
FBufSize: Integer;
|
|
public
|
|
constructor Create(aStream: TStream; anOptions: LongWord; aBufSize: Integer);
|
|
destructor Destroy; override;
|
|
|
|
procedure Write(const aBuf; aSize: Integer); override;
|
|
end;
|
|
|
|
TStrXmlWriter = class(TBinXmlWriter)
|
|
private
|
|
FData: String;
|
|
FBufStart,
|
|
FBufEnd,
|
|
FBufPtr: PChar;
|
|
FBufSize: Integer;
|
|
procedure FlushBuf;
|
|
public
|
|
constructor Create(anOptions: LongWord; aBufSize: Integer);
|
|
destructor Destroy; override;
|
|
|
|
procedure Write(const aBuf; aSize: Integer); override;
|
|
end;
|
|
|
|
TXmlBase = class(TInterfacedObject, IXmlBase)
|
|
protected
|
|
// реализация интерфейса IXmlBase
|
|
function GetObject: TObject;
|
|
public
|
|
end;
|
|
|
|
PNameIndexArray = ^TNameIndexArray;
|
|
TNameIndexArray = array of Longint;
|
|
TXmlNameTable = class(TXmlBase, IXmlNameTable)
|
|
private
|
|
FNames: array of TXmlString;
|
|
FHashTable: array of TNameIndexArray;
|
|
|
|
FXmlTextNameID: Integer;
|
|
FXmlCDATASectionNameID: Integer;
|
|
FXmlCommentNameID: Integer;
|
|
FXmlDocumentNameID: Integer;
|
|
FXmlID: Integer;
|
|
protected
|
|
function GetID(const aName: TXmlString): Integer;
|
|
function GetName(anID: Integer): TXmlString;
|
|
public
|
|
constructor Create(aHashTableSize: Integer);
|
|
|
|
procedure LoadBinXml(aReader: TBinXmlReader);
|
|
procedure SaveBinXml(aWriter: TBinXmlWriter);
|
|
end;
|
|
|
|
{ TXmlBase }
|
|
|
|
function TXmlBase.GetObject: TObject;
|
|
begin
|
|
Result := Self;
|
|
end;
|
|
|
|
{ TXmlNameTable }
|
|
|
|
constructor TXmlNameTable.Create(aHashTableSize: Integer);
|
|
begin
|
|
inherited Create;
|
|
SetLength(FHashTable, aHashTableSize);
|
|
FXmlTextNameID := GetID('#text');
|
|
FXmlCDATASectionNameID := GetID('#cdata-section');
|
|
FXmlCommentNameID := GetID('#comment');
|
|
FXmlDocumentNameID := GetID('#document');
|
|
FXmlID := GetID('xml');
|
|
end;
|
|
|
|
procedure TXmlNameTable.LoadBinXml(aReader: TBinXmlReader);
|
|
var
|
|
aCount: LongInt;
|
|
anIndex, i: Integer;
|
|
begin
|
|
// Считать массив имен
|
|
aCount := aReader.ReadLongint;
|
|
SetLength(FNames, aCount);
|
|
for i := 0 to aCount - 1 do
|
|
FNames[i] := aReader.ReadXmlString;
|
|
|
|
// Считать хэш-таблицу
|
|
SetLength(FHashTable, aReader.ReadLongint);
|
|
for i := 0 to Length(FHashTable) - 1 do
|
|
SetLength(FHashTable[i], 0);
|
|
aCount := aReader.ReadLongint;
|
|
for i := 0 to aCount - 1 do begin
|
|
anIndex := aReader.ReadLongInt;
|
|
SetLength(FHashTable[anIndex], aReader.ReadLongInt);
|
|
aReader.Read(FHashTable[anIndex][0], Length(FHashTable[anIndex])*sizeof(Longint));
|
|
end;
|
|
end;
|
|
|
|
procedure TXmlNameTable.SaveBinXml(aWriter: TBinXmlWriter);
|
|
var
|
|
aCount: LongInt;
|
|
i: Integer;
|
|
begin
|
|
// Записать массив имен
|
|
aCount := Length(FNames);
|
|
aWriter.WriteLongint(aCount);
|
|
for i := 0 to aCount - 1 do
|
|
aWriter.WriteXmlString(FNames[i]);
|
|
|
|
// Записать хэш-таблицу
|
|
aWriter.WriteLongint(Length(FHashTable));
|
|
aCount := 0;
|
|
for i := 0 to Length(FHashTable) - 1 do
|
|
if Length(FHashTable[i]) > 0 then
|
|
Inc(aCount);
|
|
aWriter.WriteLongint(aCount);
|
|
for i := 0 to Length(FHashTable) - 1 do begin
|
|
aCount := Length(FHashTable[i]);
|
|
if aCount > 0 then begin
|
|
aWriter.WriteLongint(i);
|
|
aWriter.WriteLongint(aCount);
|
|
aWriter.Write(FHashTable[i][0], aCount*sizeof(Longint));
|
|
end
|
|
end;
|
|
end;
|
|
|
|
function TXmlNameTable.GetID(const aName: TXmlString): Integer;
|
|
|
|
function NameHashKey(const aName: TXmlString): UINT;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
Result := 0;
|
|
for i := 1 to Length(aName) do
|
|
Result := UINT((int64(Result) shl 5) + Result + Ord(aName[i]));
|
|
end;
|
|
|
|
var
|
|
i: Integer;
|
|
aNameIndexes: PNameIndexArray;
|
|
begin
|
|
if aName = '' then
|
|
Result := -1
|
|
else begin
|
|
aNameIndexes := @FHashTable[NameHashKey(aName) mod UINT(Length(FHashTable))];
|
|
for i := 0 to Length(aNameIndexes^) - 1 do begin
|
|
Result := aNameIndexes^[i];
|
|
if FNames[Result] = aName then
|
|
Exit
|
|
end;
|
|
Result := Length(FNames);
|
|
SetLength(FNames, Result + 1);
|
|
FNames[Result] := aName;
|
|
|
|
SetLength(aNameIndexes^, Length(aNameIndexes^) + 1);
|
|
aNameIndexes^[Length(aNameIndexes^) - 1] := Result;
|
|
end;
|
|
end;
|
|
|
|
function TXmlNameTable.GetName(anID: Integer): TXmlString;
|
|
begin
|
|
if anID < 0 then
|
|
Result := ''
|
|
else
|
|
Result := FNames[anID]
|
|
end;
|
|
|
|
function CreateNameTable(aHashTableSize: Integer): IXmlNameTable;
|
|
begin
|
|
Result := TXmlNameTable.Create(aHashTableSize)
|
|
end;
|
|
|
|
type
|
|
TXmlNode = class;
|
|
TXmlToken = class
|
|
private
|
|
FValueBuf: TXmlString;
|
|
FValueStart,
|
|
FValuePtr,
|
|
FValueEnd: PXmlChar;
|
|
public
|
|
constructor Create;
|
|
procedure Clear;
|
|
procedure AppendChar(aChar: TXmlChar);
|
|
procedure AppendText(aText: PXmlChar; aCount: Integer);
|
|
function Length: Integer;
|
|
|
|
property ValueStart: PXmlChar read FValueStart;
|
|
end;
|
|
|
|
TXmlSource = class
|
|
private
|
|
FTokenStack: array of TXmlToken;
|
|
FTokenStackTop: Integer;
|
|
FToken: TXmlToken;
|
|
function ExpectQuotedText(aQuote: TXmlChar): TXmlString;
|
|
public
|
|
CurChar: TXmlChar;
|
|
constructor Create;
|
|
destructor Destroy; override;
|
|
|
|
function EOF: Boolean; virtual; abstract;
|
|
function Next: Boolean; virtual; abstract;
|
|
|
|
procedure SkipBlanks;
|
|
function ExpectXmlName: TXmlString;
|
|
function ExpectXmlEntity: TXmlChar;
|
|
procedure ExpectChar(aChar: TXmlChar);
|
|
procedure ExpectText(aText: PXmlChar);
|
|
function ExpectDecimalInteger: Integer;
|
|
function ExpectHexInteger: Integer;
|
|
function ParseTo(aText: PXmlChar): TXmlString;
|
|
procedure ParseAttrs(aNode: TXmlNode);
|
|
|
|
procedure NewToken;
|
|
procedure AppendTokenChar(aChar: TXmlChar);
|
|
procedure AppendTokenText(aText: PXmlChar; aCount: Integer);
|
|
function AcceptToken: TXmlString;
|
|
procedure DropToken;
|
|
end;
|
|
|
|
TXmlStrSource = class(TXmlSource)
|
|
private
|
|
FSource: TXmlString;
|
|
FSourcePtr,
|
|
FSourceEnd: PXmlChar;
|
|
public
|
|
constructor Create(const aSource: TXmlString);
|
|
function EOF: Boolean; override;
|
|
function Next: Boolean; override;
|
|
end;
|
|
|
|
TXmlStmSource = class(TXmlSource)
|
|
private
|
|
FStream: TStream;
|
|
FBufStart,
|
|
FBufPtr,
|
|
FBufEnd: PChar;
|
|
FBufSize: Integer;
|
|
FSize: Integer;
|
|
public
|
|
constructor Create(aStream: TStream; aBufSize: Integer);
|
|
function EOF: Boolean; override;
|
|
function Next: Boolean; override;
|
|
destructor Destroy; override;
|
|
end;
|
|
|
|
TXmlNodeList = class(TXmlBase, IXmlNodeList)
|
|
private
|
|
FOwnerNode: TXmlNode;
|
|
|
|
FItems: array of TXmlNode;
|
|
FCount: Integer;
|
|
procedure Grow;
|
|
protected
|
|
function Get_Count: Integer;
|
|
function Get_Item(anIndex: Integer): IXmlNode;
|
|
function Get_XML: TXmlString;
|
|
public
|
|
constructor Create(anOwnerNode: TXmlNode);
|
|
destructor Destroy; override;
|
|
|
|
function IndexOf(aNode: TXmlNode): Integer;
|
|
procedure ParseXML(aXML: TXmlSource; aNames: TXmlNameTable; aPreserveWhiteSpace: Boolean);
|
|
|
|
procedure LoadBinXml(aReader: TBinXmlReader; aCount: Integer; aNames: TXmlNameTable);
|
|
procedure SaveBinXml(aWriter: TBinXmlWriter);
|
|
|
|
procedure Insert(aNode: TXmlNode; anIndex: Integer);
|
|
function Remove(aNode: TXmlNode): Integer;
|
|
procedure Delete(anIndex: Integer);
|
|
procedure Replace(anIndex: Integer; aNode: TXmlNode);
|
|
procedure Clear;
|
|
end;
|
|
|
|
PXmlAttrData = ^TXmlAttrData;
|
|
TXmlAttrData = record
|
|
NameID: Integer;
|
|
Value: Variant;
|
|
end;
|
|
|
|
TXmlDocument = class;
|
|
TXmlNode = class(TXmlBase, IXmlNode)
|
|
private
|
|
FParentNode: TXmlNode;
|
|
// FNames - таблица имен. Задается извне
|
|
FNames: TXmlNameTable;
|
|
// Количество атрибутов в массиве FAttrs
|
|
FAttrCount: Integer;
|
|
// Массив атрибутов
|
|
FAttrs: array of TXmlAttrData;
|
|
// Список дочерних узлов
|
|
FChilds: TXmlNodeList;
|
|
function GetChilds: TXmlNodeList; virtual;
|
|
function FindFirstChild(aNameID: Integer): TXmlNode;
|
|
function GetAttrsXML: TXmlString;
|
|
function FindAttrData(aNameID: Integer): PXmlAttrData;
|
|
function GetOwnerDocument: TXmlDocument;
|
|
procedure SetNameTable(aValue: TXmlNameTable; aMap: TList);
|
|
procedure SetNodeNameID(aValue: Integer); virtual;
|
|
function DoCloneNode(aDeep: Boolean): IXmlNode; virtual; abstract;
|
|
|
|
protected
|
|
// IXmlNode
|
|
function Get_NameTable: IXmlNameTable;
|
|
function Get_NodeName: TXmlString;
|
|
|
|
function Get_NodeNameID: Integer; virtual; abstract;
|
|
function Get_NodeType: Integer; virtual; abstract;
|
|
function Get_Text: TXmlString; virtual; abstract;
|
|
procedure Set_Text(const aValue: TXmlString); virtual; abstract;
|
|
function CloneNode(aDeep: Boolean): IXmlNode;
|
|
|
|
procedure LoadBinXml(aReader: TBinXmlReader);
|
|
procedure SaveBinXml(aWriter: TBinXmlWriter);
|
|
|
|
function Get_DataType: Integer; virtual;
|
|
function Get_TypedValue: Variant; virtual;
|
|
procedure Set_TypedValue(const aValue: Variant); virtual;
|
|
|
|
function Get_XML: TXmlString; virtual; abstract;
|
|
|
|
function Get_OwnerDocument: IXmlDocument; virtual;
|
|
function Get_ParentNode: IXmlNode;
|
|
|
|
function Get_ChildNodes: IXmlNodeList; virtual;
|
|
procedure AppendChild(const aChild: IXmlNode);
|
|
|
|
function AppendElement(aNameID: Integer): IXmlElement; overload;
|
|
function AppendElement(const aName: TxmlString): IXmlElement; overload;
|
|
function AppendText(const aData: TXmlString): IXmlText;
|
|
function AppendCDATA(const aData: TXmlString): IXmlCDATASection;
|
|
function AppendComment(const aData: TXmlString): IXmlComment;
|
|
function AppendProcessingInstruction(aTargetID: Integer;
|
|
const aData: TXmlString): IXmlProcessingInstruction; overload;
|
|
function AppendProcessingInstruction(const aTarget: TXmlString;
|
|
const aData: TXmlString): IXmlProcessingInstruction; overload;
|
|
|
|
procedure InsertBefore(const aChild, aBefore: IXmlNode);
|
|
procedure ReplaceChild(const aNewChild, anOldChild: IXmlNode);
|
|
procedure RemoveChild(const aChild: IXmlNode);
|
|
function GetChildText(const aName: TXmlString; const aDefault: TXmlString = ''): TXmlString; overload;
|
|
function GetChildText(aNameID: Integer; const aDefault: TXmlString = ''): TXmlString; overload;
|
|
procedure SetChildText(const aName, aValue: TXmlString); overload;
|
|
procedure SetChildText(aNameID: Integer; const aValue: TXmlString); overload;
|
|
|
|
function NeedChild(aNameID: Integer): IXmlNode; overload;
|
|
function NeedChild(const aName: TXmlString): IXmlNode; overload;
|
|
function EnsureChild(aNameID: Integer): IXmlNode; overload;
|
|
function EnsureChild(const aName: TXmlString): IXmlNode; overload;
|
|
|
|
procedure RemoveAllChilds;
|
|
|
|
function SelectNodes(const anExpression: TXmlString): IXmlNodeList;
|
|
function SelectSingleNode(const anExpression: TXmlString): IXmlNode;
|
|
function FindElement(const anElementName, anAttrName: String; const anAttrValue: Variant): IXmlElement;
|
|
|
|
function Get_AttrCount: Integer;
|
|
function Get_AttrNameID(anIndex: Integer): Integer;
|
|
function Get_AttrName(anIndex: Integer): TXmlString;
|
|
procedure RemoveAttr(const aName: TXmlString); overload;
|
|
procedure RemoveAttr(aNameID: Integer); overload;
|
|
procedure RemoveAllAttrs;
|
|
|
|
function AttrExists(aNameID: Integer): Boolean; overload;
|
|
function AttrExists(const aName: TXmlString): Boolean; overload;
|
|
|
|
function GetAttrType(aNameID: Integer): Integer; overload;
|
|
function GetAttrType(const aName: TXmlString): Integer; overload;
|
|
|
|
function GetVarAttr(aNameID: Integer; const aDefault: Variant): Variant; overload;
|
|
function GetVarAttr(const aName: TXmlString; const aDefault: Variant): Variant; overload;
|
|
procedure SetVarAttr(aNameID: Integer; const aValue: Variant); overload;
|
|
procedure SetVarAttr(const aName: TXmlString; aValue: Variant); overload;
|
|
|
|
function NeedAttr(aNameID: Integer): TXmlString; overload;
|
|
function NeedAttr(const aName: TXmlString): TXmlString; overload;
|
|
|
|
function GetAttr(aNameID: Integer; const aDefault: TXmlString = ''): TXmlString; overload;
|
|
function GetAttr(const aName: TXmlString; const aDefault: TXmlString = ''): TXmlString; overload;
|
|
procedure SetAttr(aNameID: Integer; const aValue: TXmlString); overload;
|
|
procedure SetAttr(const aName, aValue: TXmlString); overload;
|
|
|
|
function GetBoolAttr(aNameID: Integer; aDefault: Boolean = False): Boolean; overload;
|
|
function GetBoolAttr(const aName: TXmlString; aDefault: Boolean = False): Boolean; overload;
|
|
procedure SetBoolAttr(aNameID: Integer; aValue: Boolean = False); overload;
|
|
procedure SetBoolAttr(const aName: TXmlString; aValue: Boolean); overload;
|
|
|
|
function GetIntAttr(aNameID: Integer; aDefault: Integer = 0): Integer; overload;
|
|
function GetIntAttr(const aName: TXmlString; aDefault: Integer = 0): Integer; overload;
|
|
procedure SetIntAttr(aNameID: Integer; aValue: Integer); overload;
|
|
procedure SetIntAttr(const aName: TXmlString; aValue: Integer); overload;
|
|
|
|
function GetDateTimeAttr(aNameID: Integer; aDefault: TDateTime = 0): TDateTime; overload;
|
|
function GetDateTimeAttr(const aName: TXmlString; aDefault: TDateTime = 0): TDateTime; overload;
|
|
procedure SetDateTimeAttr(aNameID: Integer; aValue: TDateTime); overload;
|
|
procedure SetDateTimeAttr(const aName: TXmlString; aValue: TDateTime); overload;
|
|
|
|
function GetFloatAttr(aNameID: Integer; aDefault: Double = 0): Double; overload;
|
|
function GetFloatAttr(const aName: TXmlString; aDefault: Double = 0): Double; overload;
|
|
procedure SetFloatAttr(aNameID: Integer; aValue: Double); overload;
|
|
procedure SetFloatAttr(const aName: TXmlString; aValue: Double); overload;
|
|
|
|
function GetHexAttr(const aName: TXmlString; aDefault: Integer = 0): Integer; overload;
|
|
function GetHexAttr(aNameID: Integer; aDefault: Integer = 0): Integer; overload;
|
|
procedure SetHexAttr(const aName: TXmlString; aValue: Integer; aDigits: Integer = 8); overload;
|
|
procedure SetHexAttr(aNameID: Integer; aValue: Integer; aDigits: Integer = 8); overload;
|
|
|
|
function GetEnumAttr(const aName: TXmlString;
|
|
const aValues: array of TXmlString; aDefault: Integer = 0): Integer; overload;
|
|
function GetEnumAttr(aNameID: Integer;
|
|
const aValues: array of TXmlString; aDefault: Integer = 0): Integer; overload;
|
|
function NeedEnumAttr(const aName: TXmlString;
|
|
const aValues: array of TXmlString): Integer; overload;
|
|
function NeedEnumAttr(aNameID: Integer;
|
|
const aValues: array of TXmlString): Integer; overload;
|
|
|
|
|
|
function Get_Values(const aName: String): Variant;
|
|
procedure Set_Values(const aName: String; const aValue: Variant);
|
|
|
|
function AsElement: IXmlElement; virtual;
|
|
function AsText: IXmlText; virtual;
|
|
function AsCDATASection: IXmlCDATASection; virtual;
|
|
function AsComment: IXmlComment; virtual;
|
|
function AsProcessingInstruction: IXmlProcessingInstruction; virtual;
|
|
|
|
public
|
|
constructor Create(aNames: TXmlNameTable);
|
|
destructor Destroy; override;
|
|
end;
|
|
|
|
TXmlElement = class(TXmlNode, IXmlElement)
|
|
private
|
|
FNameID: Integer;
|
|
FData: Variant;
|
|
procedure RemoveTextNodes;
|
|
procedure SetNodeNameID(aValue: Integer); override;
|
|
function DoCloneNode(aDeep: Boolean): IXmlNode; override;
|
|
protected
|
|
function GetChilds: TXmlNodeList; override;
|
|
|
|
function Get_NodeNameID: Integer; override;
|
|
function Get_NodeType: Integer; override;
|
|
function Get_Text: TXmlString; override;
|
|
procedure Set_Text(const aValue: TXmlString); override;
|
|
function Get_DataType: Integer; override;
|
|
function Get_TypedValue: Variant; override;
|
|
procedure Set_TypedValue(const aValue: Variant); override;
|
|
function Get_XML: TXmlString; override;
|
|
function AsElement: IXmlElement; override;
|
|
function Get_ChildNodes: IXmlNodeList; override;
|
|
|
|
// IXmlElement
|
|
procedure ReplaceTextByCDATASection(const aText: TXmlString);
|
|
procedure ReplaceTextByBynaryData(const aData; aSize: Integer;
|
|
aMaxLineLength: Integer);
|
|
function GetTextAsBynaryData: TXmlString;
|
|
public
|
|
constructor Create(aNames: TXmlNameTable; aNameID: Integer);
|
|
end;
|
|
|
|
TXmlCharacterData = class(TXmlNode, IXmlCharacterData)
|
|
private
|
|
FData: TXmlString;
|
|
protected
|
|
function Get_Text: TXmlString; override;
|
|
procedure Set_Text(const aValue: TXmlString); override;
|
|
public
|
|
constructor Create(aNames: TXmlNameTable; const aData: TXmlString);
|
|
end;
|
|
|
|
TXmlText = class(TXmlNode, IXmlText)
|
|
private
|
|
FData: Variant;
|
|
function DoCloneNode(aDeep: Boolean): IXmlNode; override;
|
|
protected
|
|
function Get_NodeNameID: Integer; override;
|
|
function Get_NodeType: Integer; override;
|
|
function Get_Text: TXmlString; override;
|
|
procedure Set_Text(const aValue: TXmlString); override;
|
|
function Get_DataType: Integer; override;
|
|
function Get_TypedValue: Variant; override;
|
|
procedure Set_TypedValue(const aValue: Variant); override;
|
|
function Get_XML: TXmlString; override;
|
|
function AsText: IXmlText; override;
|
|
public
|
|
constructor Create(aNames: TXmlNameTable; const aData: Variant);
|
|
end;
|
|
|
|
TXmlCDATASection = class(TXmlCharacterData, IXmlCDATASection)
|
|
protected
|
|
function Get_NodeNameID: Integer; override;
|
|
function Get_NodeType: Integer; override;
|
|
function Get_XML: TXmlString; override;
|
|
function AsCDATASection: IXmlCDATASection; override;
|
|
function DoCloneNode(aDeep: Boolean): IXmlNode; override;
|
|
public
|
|
end;
|
|
|
|
TXmlComment = class(TXmlCharacterData, IXmlComment)
|
|
protected
|
|
function Get_NodeNameID: Integer; override;
|
|
function Get_NodeType: Integer; override;
|
|
function Get_XML: TXmlString; override;
|
|
function AsComment: IXmlComment; override;
|
|
function DoCloneNode(aDeep: Boolean): IXmlNode; override;
|
|
public
|
|
end;
|
|
|
|
TXmlProcessingInstruction = class(TXmlNode, IXmlProcessingInstruction)
|
|
private
|
|
FTargetID: Integer;
|
|
FData: String;
|
|
procedure SetNodeNameID(aValue: Integer); override;
|
|
function DoCloneNode(aDeep: Boolean): IXmlNode; override;
|
|
protected
|
|
function Get_NodeNameID: Integer; override;
|
|
function Get_NodeType: Integer; override;
|
|
function Get_Text: TXmlString; override;
|
|
procedure Set_Text(const aText: TXmlString); override;
|
|
function Get_XML: TXmlString; override;
|
|
function AsProcessingInstruction: IXmlProcessingInstruction; override;
|
|
|
|
public
|
|
constructor Create(aNames: TXmlNameTable; aTargetID: Integer;
|
|
const aData: TXmlString);
|
|
end;
|
|
|
|
TXmlDocument = class(TXmlNode, IXmlDocument)
|
|
private
|
|
FPreserveWhiteSpace: Boolean;
|
|
|
|
function DoCloneNode(aDeep: Boolean): IXmlNode; override;
|
|
protected
|
|
function Get_NodeNameID: Integer; override;
|
|
function Get_NodeType: Integer; override;
|
|
function Get_Text: TXmlString; override;
|
|
procedure Set_Text(const aText: TXmlString); override;
|
|
function Get_XML: TXmlString; override;
|
|
function Get_PreserveWhiteSpace: Boolean;
|
|
procedure Set_PreserveWhiteSpace(aValue: Boolean);
|
|
|
|
function NewDocument(const aVersion, anEncoding: TXmlString;
|
|
aRootElementNameID: Integer): IXmlElement; overload;
|
|
function NewDocument(const aVersion, anEncoding,
|
|
aRootElementName: TXmlString): IXmlElement; overload;
|
|
|
|
function CreateElement(aNameID: Integer): IXmlElement; overload;
|
|
function CreateElement(const aName: TXmlString): IXmlElement; overload;
|
|
function CreateText(const aData: TXmlString): IXmlText;
|
|
function CreateCDATASection(const aData: TXmlString): IXmlCDATASection;
|
|
function CreateComment(const aData: TXmlString): IXmlComment;
|
|
function Get_DocumentElement: IXmlElement;
|
|
function CreateProcessingInstruction(const aTarget,
|
|
aData: TXmlString): IXmlProcessingInstruction; overload;
|
|
function CreateProcessingInstruction(aTargetID: Integer;
|
|
const aData: TXmlString): IXmlProcessingInstruction; overload;
|
|
procedure LoadXML(const aXML: TXmlString);
|
|
|
|
procedure Load(aStream: TStream); overload;
|
|
procedure Load(const aFileName: TXmlString); overload;
|
|
|
|
procedure LoadResource(aType, aName: PChar);
|
|
|
|
procedure Save(aStream: TStream); overload;
|
|
procedure Save(const aFileName: TXmlString); overload;
|
|
|
|
procedure SaveBinary(aStream: TStream; anOptions: LongWord); overload;
|
|
procedure SaveBinary(const aFileName: TXmlString; anOptions: LongWord); overload;
|
|
|
|
function Get_BinaryXML: String;
|
|
procedure LoadBinaryXML(const aXML: String);
|
|
public
|
|
constructor Create(aNames: TXmlNameTable);
|
|
end;
|
|
|
|
{ TXmlNodeList }
|
|
|
|
procedure TXmlNodeList.Clear;
|
|
var
|
|
i: Integer;
|
|
aNode: TXmlNode;
|
|
begin
|
|
for i := 0 to FCount - 1 do begin
|
|
aNode := FItems[i];
|
|
if Assigned(FOwnerNode) then
|
|
aNode.FParentNode := nil;
|
|
aNode._Release;
|
|
end;
|
|
FCount := 0;
|
|
end;
|
|
|
|
procedure TXmlNodeList.Delete(anIndex: Integer);
|
|
var
|
|
aNode: TXmlNode;
|
|
begin
|
|
aNode := FItems[anIndex];
|
|
Dec(FCount);
|
|
if anIndex < FCount then
|
|
Move(FItems[anIndex + 1], FItems[anIndex],
|
|
(FCount - anIndex)*SizeOf(TXmlNode));
|
|
if Assigned(aNode) then begin
|
|
if Assigned(FOwnerNode) then
|
|
aNode.FParentNode := nil;
|
|
aNode._Release;
|
|
end;
|
|
end;
|
|
|
|
constructor TXmlNodeList.Create(anOwnerNode: TXmlNode);
|
|
begin
|
|
inherited Create;
|
|
FOwnerNode := anOwnerNode;
|
|
end;
|
|
|
|
destructor TXmlNodeList.Destroy;
|
|
begin
|
|
Clear;
|
|
inherited;
|
|
end;
|
|
|
|
function TXmlNodeList.Get_Item(anIndex: Integer): IXmlNode;
|
|
begin
|
|
if (anIndex < 0) or (anIndex >= FCount) then
|
|
raise Exception.Create(SSimpleXmlError1);
|
|
Result := FItems[anIndex]
|
|
end;
|
|
|
|
function TXmlNodeList.Get_Count: Integer;
|
|
begin
|
|
Result := FCount
|
|
end;
|
|
|
|
function TXmlNodeList.IndexOf(aNode: TXmlNode): Integer;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
for i := 0 to FCount - 1 do
|
|
if FItems[i] = aNode then begin
|
|
Result := i;
|
|
Exit
|
|
end;
|
|
Result := -1;
|
|
end;
|
|
|
|
procedure TXmlNodeList.Grow;
|
|
var
|
|
aDelta: Integer;
|
|
begin
|
|
if Length(FItems) > 64 then
|
|
aDelta := Length(FItems) div 4
|
|
else
|
|
if Length(FItems) > 8 then
|
|
aDelta := 16
|
|
else
|
|
aDelta := 4;
|
|
SetLength(FItems, Length(FItems) + aDelta);
|
|
end;
|
|
|
|
procedure TXmlNodeList.Insert(aNode: TXmlNode; anIndex: Integer);
|
|
begin
|
|
if anIndex = -1 then
|
|
anIndex := FCount;
|
|
if FCount = Length(FItems) then
|
|
Grow;
|
|
if anIndex < FCount then
|
|
Move(FItems[anIndex], FItems[anIndex + 1],
|
|
(FCount - anIndex)*SizeOf(TXmlNode));
|
|
FItems[anIndex] := aNode;
|
|
Inc(FCount);
|
|
if aNode <> nil then begin
|
|
aNode._AddRef;
|
|
if Assigned(FOwnerNode) then begin
|
|
aNode.FParentNode := FOwnerNode;
|
|
aNode.SetNameTable(FOwnerNode.FNames, nil);
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
function TXmlNodeList.Remove(aNode: TXmlNode): Integer;
|
|
begin
|
|
Result := IndexOf(aNode);
|
|
if Result <> -1 then
|
|
Delete(Result);
|
|
end;
|
|
|
|
procedure TXmlNodeList.Replace(anIndex: Integer; aNode: TXmlNode);
|
|
var
|
|
anOldNode: TXmlNode;
|
|
begin
|
|
anOldNode := FItems[anIndex];
|
|
if aNode <> anOldNode then begin
|
|
if Assigned(anOldNode) then begin
|
|
if Assigned(FOwnerNode) then
|
|
anOldNode.FParentNode := nil;
|
|
anOldNode._Release;
|
|
end;
|
|
FItems[anIndex] := aNode;
|
|
if Assigned(aNode) then begin
|
|
aNode._AddRef;
|
|
if Assigned(FOwnerNode) then begin
|
|
aNode.FParentNode := FOwnerNode;
|
|
aNode.SetNameTable(FOwnerNode.FNames, nil);
|
|
end
|
|
end
|
|
end;
|
|
end;
|
|
|
|
function TXmlNodeList.Get_XML: TXmlString;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
Result := '';
|
|
for i := 0 to FCount - 1 do
|
|
Result := Result + FItems[i].Get_XML;
|
|
end;
|
|
|
|
procedure TXmlNodeList.ParseXML(aXML: TXmlSource; aNames: TXmlNameTable; aPreserveWhiteSpace: Boolean);
|
|
|
|
// на входе: символ текста
|
|
// на выходе: символ разметки '<'
|
|
procedure ParseText;
|
|
var
|
|
aText: String;
|
|
begin
|
|
aXml.NewToken;
|
|
while not aXML.EOF and (aXML.CurChar <> '<') do
|
|
if aXML.CurChar = '&' then
|
|
aXml.AppendTokenChar(aXml.ExpectXmlEntity)
|
|
else begin
|
|
aXml.AppendTokenChar(aXML.CurChar);
|
|
aXML.Next;
|
|
end;
|
|
aText := aXml.AcceptToken;
|
|
if aPreserveWhiteSpace or (Trim(aText) <> '') then
|
|
Insert(TXmlText.Create(aNames, aText), -1);
|
|
end;
|
|
|
|
// CurChar - '?'
|
|
procedure ParseProcessingInstruction;
|
|
var
|
|
aTarget: TXmlString;
|
|
aNode: TXmlProcessingInstruction;
|
|
begin
|
|
aXML.Next;
|
|
aTarget := aXML.ExpectXmlName;
|
|
aNode := TXmlProcessingInstruction.Create(aNames, aNames.GetID(aTarget), '');
|
|
Insert(aNode, -1);
|
|
if aNode.FTargetID = aNames.FXmlID then begin
|
|
aXml.ParseAttrs(aNode);
|
|
aXml.ExpectText('?>');
|
|
end
|
|
else
|
|
aNode.FData := aXml.ParseTo('?>');
|
|
end;
|
|
|
|
// на входе: первый '--'
|
|
// на выходе: символ после '-->'
|
|
procedure ParseComment;
|
|
begin
|
|
aXml.ExpectText('--');
|
|
Insert(TXmlComment.Create(aNames, aXml.ParseTo('-->')), -1);
|
|
end;
|
|
|
|
// на входе: '[CDATA['
|
|
// на выходе: символ после ']]>'
|
|
procedure ParseCDATA;
|
|
begin
|
|
aXml.ExpectText('[CDATA[');
|
|
Insert(TXmlCDATASection.Create(aNames, aXml.ParseTo(']]>')), -1);
|
|
end;
|
|
|
|
|
|
// на входе: 'DOCTYPE'
|
|
// на выходе: символ после '>'
|
|
procedure ParseDOCTYPE;
|
|
begin
|
|
aXml.ExpectText('DOCTYPE');
|
|
aXml.ParseTo('>');
|
|
end;
|
|
|
|
// на входе: 'имя-элемента'
|
|
// на выходе: символ после '>'
|
|
procedure ParseElement;
|
|
var
|
|
aNameID: Integer;
|
|
aNode: TXmlElement;
|
|
begin
|
|
aNameID := aNames.GetID(aXml.ExpectXmlName);
|
|
if aXml.EOF then
|
|
raise Exception.Create(SSimpleXMLError2);
|
|
if not ((aXml.CurChar <= ' ') or (aXml.CurChar = '/') or (aXml.CurChar = '>')) then
|
|
raise Exception.Create(SSimpleXMLError3);
|
|
aNode := TXmlElement.Create(aNames, aNameID);
|
|
Insert(aNode, -1);
|
|
aXml.ParseAttrs(aNode);
|
|
if aXml.CurChar = '/' then
|
|
aXml.ExpectText('/>')
|
|
else begin
|
|
aXml.ExpectChar('>');
|
|
aNode.GetChilds.ParseXML(aXml, aNames, aPreserveWhiteSpace);
|
|
aXml.ExpectChar('/');
|
|
aXml.ExpectText(PXmlChar(aNames.GetName(aNameID)));
|
|
aXml.SkipBlanks;
|
|
aXml.ExpectChar('>');
|
|
end;
|
|
end;
|
|
|
|
begin
|
|
while not aXML.EOF do begin
|
|
ParseText;
|
|
if aXML.CurChar = '<' then // символ разметки
|
|
if aXML.Next then
|
|
if aXML.CurChar = '/' then // закрывающий тэг элемента
|
|
Exit
|
|
else if aXML.CurChar = '?' then // инструкция
|
|
ParseProcessingInstruction
|
|
else if aXML.CurChar = '!' then begin
|
|
if aXML.Next then
|
|
if aXML.CurChar = '-' then // коментарий
|
|
ParseComment
|
|
else if aXML.CurChar = '[' then // секция CDATA
|
|
ParseCDATA
|
|
else
|
|
ParseDOCTYPE
|
|
end
|
|
else // открывающий тэг элемента
|
|
ParseElement
|
|
end;
|
|
end;
|
|
|
|
procedure TXmlNodeList.LoadBinXml(aReader: TBinXmlReader;
|
|
aCount: Integer; aNames: TXmlNameTable);
|
|
var
|
|
i: Integer;
|
|
aNodeType: Byte;
|
|
aNode: TXmlNode;
|
|
aNameID: LongInt;
|
|
begin
|
|
Clear;
|
|
SetLength(FItems, aCount);
|
|
for i := 0 to aCount - 1 do begin
|
|
aReader.Read(aNodeType, sizeof(aNodeType));
|
|
case aNodeType of
|
|
NODE_ELEMENT:
|
|
begin
|
|
aNameID := aReader.ReadLongint;
|
|
aNode := TXmlElement.Create(aNames, aNameID);
|
|
Insert(aNode, -1);
|
|
aReader.ReadVariant(TVarData(TXmlElement(aNode).FData));
|
|
aNode.LoadBinXml(aReader);
|
|
end;
|
|
NODE_TEXT:
|
|
begin
|
|
aNode := TXmlText.Create(aNames, Unassigned);
|
|
Insert(aNode, -1);
|
|
aReader.ReadVariant(TVarData(TXmlText(aNode).FData));
|
|
end;
|
|
NODE_CDATA_SECTION:
|
|
Insert(TXmlCDATASection.Create(aNames, aReader.ReadXmlString), -1);
|
|
NODE_PROCESSING_INSTRUCTION:
|
|
begin
|
|
aNameID := aReader.ReadLongint;
|
|
aNode := TXmlProcessingInstruction.Create(aNames, aNameID,
|
|
aReader.ReadXmlString);
|
|
Insert(aNode, -1);
|
|
aNode.LoadBinXml(aReader);
|
|
end;
|
|
NODE_COMMENT:
|
|
Insert(TXmlComment.Create(aNames, aReader.ReadXmlString), -1);
|
|
else
|
|
raise Exception.Create(SSimpleXMLError4);
|
|
end
|
|
end;
|
|
end;
|
|
|
|
procedure TXmlNodeList.SaveBinXml(aWriter: TBinXmlWriter);
|
|
const
|
|
EmptyVar: TVarData = (VType:varEmpty);
|
|
var
|
|
aCount: LongInt;
|
|
i: Integer;
|
|
aNodeType: Byte;
|
|
aNode: TXmlNode;
|
|
begin
|
|
aCount := FCount;
|
|
for i := 0 to aCount - 1 do begin
|
|
aNode := FItems[i];
|
|
aNodeType := aNode.Get_NodeType;
|
|
aWriter.Write(aNodeType, sizeof(aNodeType));
|
|
case aNodeType of
|
|
NODE_ELEMENT:
|
|
with TXmlElement(aNode) do begin
|
|
aWriter.WriteLongint(FNameID);
|
|
if Assigned(FChilds) and (FChilds.FCount > 0) or VarIsEmpty(FData) then
|
|
aWriter.WriteVariant(EmptyVar)
|
|
else
|
|
aWriter.WriteVariant(TVarData(FData));
|
|
SaveBinXml(aWriter);
|
|
end;
|
|
NODE_TEXT:
|
|
aWriter.WriteVariant(TVarData(TXmlText(aNode).FData));
|
|
NODE_CDATA_SECTION:
|
|
aWriter.WriteXmlString(TXmlCDATASection(aNode).FData);
|
|
NODE_PROCESSING_INSTRUCTION:
|
|
begin
|
|
aWriter.WriteLongint(TXmlProcessingInstruction(aNode).FTargetID);
|
|
aWriter.WriteXmlString(TXmlProcessingInstruction(aNode).FData);
|
|
aNode.SaveBinXml(aWriter);
|
|
end;
|
|
NODE_COMMENT:
|
|
aWriter.WriteXmlString(TXmlComment(aNode).FData);
|
|
else
|
|
raise Exception.Create(SSimpleXmlError5);
|
|
end
|
|
end;
|
|
end;
|
|
|
|
{ TXmlNode }
|
|
|
|
constructor TXmlNode.Create(aNames: TXmlNameTable);
|
|
begin
|
|
inherited Create;
|
|
FNames := aNames;
|
|
FNames._AddRef;
|
|
end;
|
|
|
|
destructor TXmlNode.Destroy;
|
|
begin
|
|
if Assigned(FChilds) then
|
|
FChilds._Release;
|
|
FNames._Release;
|
|
inherited;
|
|
end;
|
|
|
|
function TXmlNode.GetChilds: TXmlNodeList;
|
|
begin
|
|
if not Assigned(FChilds) then begin
|
|
FChilds := TXmlNodeList.Create(Self);
|
|
FChilds._AddRef;
|
|
end;
|
|
Result := FChilds;
|
|
end;
|
|
|
|
procedure TXmlNode.AppendChild(const aChild: IXmlNode);
|
|
begin
|
|
GetChilds.Insert(aChild.GetObject as TXmlNode, -1);
|
|
end;
|
|
|
|
function TXmlNode.Get_AttrCount: Integer;
|
|
begin
|
|
Result := FAttrCount;
|
|
end;
|
|
|
|
function TXmlNode.Get_AttrName(anIndex: Integer): TXmlString;
|
|
begin
|
|
Result := FNames.GetName(FAttrs[anIndex].NameID);
|
|
end;
|
|
|
|
function TXmlNode.Get_AttrNameID(anIndex: Integer): Integer;
|
|
begin
|
|
Result := FAttrs[anIndex].NameID;
|
|
end;
|
|
|
|
function TXmlNode.Get_ChildNodes: IXmlNodeList;
|
|
begin
|
|
Result := GetChilds
|
|
end;
|
|
|
|
function TXmlNode.Get_NameTable: IXmlNameTable;
|
|
begin
|
|
Result := FNames
|
|
end;
|
|
|
|
function TXmlNode.GetAttr(const aName, aDefault: TXmlString): TXmlString;
|
|
begin
|
|
Result := GetAttr(FNames.GetID(aName), aDefault)
|
|
end;
|
|
|
|
function TXmlNode.GetAttr(aNameID: Integer;
|
|
const aDefault: TXmlString): TXmlString;
|
|
var
|
|
aData: PXmlAttrData;
|
|
begin
|
|
aData := FindAttrData(aNameID);
|
|
if Assigned(aData) then
|
|
Result := aData.Value
|
|
else
|
|
Result := aDefault
|
|
end;
|
|
|
|
function TXmlNode.GetBoolAttr(aNameID: Integer;
|
|
aDefault: Boolean): Boolean;
|
|
var
|
|
aData: PXmlAttrData;
|
|
begin
|
|
aData := FindAttrData(aNameID);
|
|
if Assigned(aData) then
|
|
Result := aData.Value
|
|
else
|
|
Result := aDefault
|
|
end;
|
|
|
|
function TXmlNode.GetBoolAttr(const aName: TXmlString;
|
|
aDefault: Boolean): Boolean;
|
|
begin
|
|
Result := GetBoolAttr(FNames.GetID(aName), aDefault)
|
|
end;
|
|
|
|
function TXmlNode.FindFirstChild(aNameID: Integer): TXmlNode;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
if Assigned(FChilds) then
|
|
for i := 0 to FChilds.FCount - 1 do begin
|
|
Result := FChilds.FItems[i];
|
|
if Result.Get_NodeNameID = aNameID then
|
|
Exit
|
|
end;
|
|
Result := nil
|
|
end;
|
|
|
|
function TXmlNode.GetChildText(aNameID: Integer;
|
|
const aDefault: TXmlString): TXmlString;
|
|
var
|
|
aChild: TXmlNode;
|
|
begin
|
|
aChild := FindFirstChild(aNameID);
|
|
if Assigned(aChild) then
|
|
Result := aChild.Get_Text
|
|
else
|
|
Result := aDefault
|
|
end;
|
|
|
|
function TXmlNode.GetChildText(const aName: TXmlString;
|
|
const aDefault: TXmlString): TXmlString;
|
|
begin
|
|
Result := GetChildText(FNames.GetID(aName), aDefault);
|
|
end;
|
|
|
|
function TXmlNode.GetEnumAttr(const aName: TXmlString;
|
|
const aValues: array of TXmlString; aDefault: Integer): Integer;
|
|
begin
|
|
Result := GetEnumAttr(FNames.GetID(aName), aValues, aDefault);
|
|
end;
|
|
|
|
function EnumAttrValue(aNode: TXmlNode; anAttrData: PXmlAttrData;
|
|
const aValues: array of TXmlString): Integer;
|
|
var
|
|
anAttrValue: TXmlString;
|
|
s: String;
|
|
i: Integer;
|
|
begin
|
|
anAttrValue := anAttrData.Value;
|
|
for Result := 0 to Length(aValues) - 1 do
|
|
if AnsiCompareText(anAttrValue, aValues[Result]) = 0 then
|
|
Exit;
|
|
if Length(aValues) = 0 then
|
|
s := ''
|
|
else begin
|
|
s := aValues[0];
|
|
for i := 1 to Length(aValues) - 1 do
|
|
s := s + ^M^J + aValues[i];
|
|
end;
|
|
raise Exception.CreateFmt(SSimpleXmlError6,
|
|
[aNode.FNames.GetName(anAttrData.NameID), aNode.Get_NodeName, s]);
|
|
end;
|
|
|
|
function TXmlNode.GetEnumAttr(aNameID: Integer;
|
|
const aValues: array of TXmlString; aDefault: Integer): Integer;
|
|
var
|
|
anAttrData: PXmlAttrData;
|
|
begin
|
|
anAttrData := FindAttrData(aNameID);
|
|
if Assigned(anAttrData) then
|
|
Result := EnumAttrValue(Self, anAttrData, aValues)
|
|
else
|
|
Result := aDefault;
|
|
end;
|
|
|
|
function TXmlNode.NeedEnumAttr(const aName: TXmlString;
|
|
const aValues: array of TXmlString): Integer;
|
|
begin
|
|
Result := NeedEnumAttr(FNames.GetID(aName), aValues)
|
|
end;
|
|
|
|
function TXmlNode.NeedEnumAttr(aNameID: Integer;
|
|
const aValues: array of TXmlString): Integer;
|
|
var
|
|
anAttrData: PXmlAttrData;
|
|
begin
|
|
anAttrData := FindAttrData(aNameID);
|
|
if Assigned(anAttrData) then
|
|
Result := EnumAttrValue(Self, anAttrData, aValues)
|
|
else
|
|
raise Exception.CreateFmt(SSimpleXMLError7, [FNames.GetName(aNameID)]);
|
|
end;
|
|
|
|
function TXmlNode.GetFloatAttr(const aName: TXmlString;
|
|
aDefault: Double): Double;
|
|
begin
|
|
Result := GetFloatAttr(FNames.GetID(aName), aDefault);
|
|
end;
|
|
|
|
function TXmlNode.GetFloatAttr(aNameID: Integer;
|
|
aDefault: Double): Double;
|
|
var
|
|
aData: PXmlAttrData;
|
|
begin
|
|
aData := FindAttrData(aNameID);
|
|
if Assigned(aData) then
|
|
if VarIsNumeric(aData.Value) then
|
|
Result := aData.Value
|
|
else
|
|
Result := XSTRToFloat(aData.Value)
|
|
else
|
|
Result := aDefault
|
|
end;
|
|
|
|
function TXmlNode.GetHexAttr(aNameID, aDefault: Integer): Integer;
|
|
var
|
|
anAttr: PXmlAttrData;
|
|
begin
|
|
anAttr := FindAttrData(aNameID);
|
|
if Assigned(anAttr) then
|
|
Result := StrToInt('$' + anAttr.Value)
|
|
else
|
|
Result := aDefault;
|
|
end;
|
|
|
|
function TXmlNode.GetHexAttr(const aName: TXmlString;
|
|
aDefault: Integer): Integer;
|
|
begin
|
|
Result := GetHexAttr(FNames.GetID(aName), aDefault)
|
|
end;
|
|
|
|
function TXmlNode.GetIntAttr(aNameID, aDefault: Integer): Integer;
|
|
var
|
|
anAttr: PXmlAttrData;
|
|
begin
|
|
anAttr := FindAttrData(aNameID);
|
|
if Assigned(anAttr) then
|
|
Result := anAttr.Value
|
|
else
|
|
Result := aDefault;
|
|
end;
|
|
|
|
function TXmlNode.GetIntAttr(const aName: TXmlString;
|
|
aDefault: Integer): Integer;
|
|
begin
|
|
Result := GetIntAttr(FNames.GetID(aName), aDefault)
|
|
end;
|
|
|
|
function TXmlNode.NeedAttr(aNameID: Integer): TXmlString;
|
|
var
|
|
anAttr: PXmlAttrData;
|
|
begin
|
|
anAttr := FindAttrData(aNameID);
|
|
if not Assigned(anAttr) then
|
|
raise Exception.CreateFmt(SSimpleXmlError8, [FNames.GetName(aNameID)]);
|
|
Result := anAttr.Value
|
|
end;
|
|
|
|
function TXmlNode.NeedAttr(const aName: TXmlString): TXmlString;
|
|
begin
|
|
Result := NeedAttr(FNames.GetID(aName))
|
|
end;
|
|
|
|
function TXmlNode.GetVarAttr(aNameID: Integer;
|
|
const aDefault: Variant): Variant;
|
|
var
|
|
anAttr: PXmlAttrData;
|
|
begin
|
|
anAttr := FindAttrData(aNameID);
|
|
if Assigned(anAttr) then
|
|
Result := anAttr.Value
|
|
else
|
|
Result := aDefault;
|
|
end;
|
|
|
|
function TXmlNode.GetVarAttr(const aName: TXmlString;
|
|
const aDefault: Variant): Variant;
|
|
begin
|
|
Result := GetVarAttr(FNames.GetID(aName), aDefault)
|
|
end;
|
|
|
|
function TXmlNode.Get_NodeName: TXmlString;
|
|
begin
|
|
Result := FNames.GetName(Get_NodeNameID);
|
|
end;
|
|
|
|
function TXmlNode.GetOwnerDocument: TXmlDocument;
|
|
var
|
|
aResult: TXmlNode;
|
|
begin
|
|
aResult := Self;
|
|
repeat
|
|
if aResult is TXmlDocument then
|
|
break
|
|
else
|
|
aResult := aResult.FParentNode;
|
|
until not Assigned(aResult);
|
|
Result := TXmlDocument(aResult)
|
|
end;
|
|
|
|
function TXmlNode.Get_OwnerDocument: IXmlDocument;
|
|
var
|
|
aDoc: TXmlDocument;
|
|
begin
|
|
aDoc := GetOwnerDocument;
|
|
if Assigned(aDoc) then
|
|
Result := aDoc
|
|
else
|
|
Result := nil;
|
|
end;
|
|
|
|
function TXmlNode.Get_ParentNode: IXmlNode;
|
|
begin
|
|
Result := FParentNode
|
|
end;
|
|
|
|
function TXmlNode.Get_TypedValue: Variant;
|
|
begin
|
|
Result := Get_Text
|
|
end;
|
|
|
|
procedure TXmlNode.InsertBefore(const aChild, aBefore: IXmlNode);
|
|
var
|
|
i: Integer;
|
|
aChilds: TXmlNodeList;
|
|
begin
|
|
aChilds := GetChilds;
|
|
if Assigned(aBefore) then
|
|
i := aChilds.IndexOf(aBefore.GetObject as TXmlNode)
|
|
else
|
|
i := aChilds.FCount;
|
|
GetChilds.Insert(aChild.GetObject as TXmlNode, i)
|
|
end;
|
|
|
|
procedure TXmlNode.RemoveAllAttrs;
|
|
begin
|
|
FAttrCount := 0;
|
|
end;
|
|
|
|
procedure TXmlNode.RemoveAllChilds;
|
|
begin
|
|
if Assigned(FChilds) then
|
|
FChilds.Clear
|
|
end;
|
|
|
|
procedure TXmlNode.RemoveAttr(const aName: TXmlString);
|
|
begin
|
|
RemoveAttr(FNames.GetID(aName));
|
|
end;
|
|
|
|
procedure TXmlNode.RemoveAttr(aNameID: Integer);
|
|
var
|
|
a1, a2: PXmlAttrData;
|
|
i: Integer;
|
|
begin
|
|
a1 := @FAttrs[0];
|
|
i := 0;
|
|
while (i < FAttrCount) and (a1.NameID <> aNameID) do begin
|
|
Inc(a1);
|
|
Inc(i)
|
|
end;
|
|
if i < FAttrCount then begin
|
|
a2 := a1;
|
|
Inc(a2);
|
|
while i < FAttrCount - 1 do begin
|
|
a1^ := a2^;
|
|
Inc(a1);
|
|
Inc(a2);
|
|
Inc(i)
|
|
end;
|
|
VarClear(a1.Value);
|
|
Dec(FAttrCount);
|
|
end;
|
|
end;
|
|
|
|
procedure TXmlNode.RemoveChild(const aChild: IXmlNode);
|
|
begin
|
|
GetChilds.Remove(aChild.GetObject as TXmlNode)
|
|
end;
|
|
|
|
procedure TXmlNode.ReplaceChild(const aNewChild, anOldChild: IXmlNode);
|
|
var
|
|
i: Integer;
|
|
aChilds: TXmlNodeList;
|
|
begin
|
|
aChilds := GetChilds;
|
|
i := aChilds.IndexOf(anOldChild.GetObject as TXmlNode);
|
|
if i <> -1 then
|
|
aChilds.Replace(i, aNewChild.GetObject as TXmlNode)
|
|
end;
|
|
|
|
function NameCanBeginWith(aChar: TXmlChar): Boolean;
|
|
begin
|
|
{$IFDEF XML_WIDE_CHARS}
|
|
Result := (aChar = '_') or IsCharAlphaW(aChar)
|
|
{$ELSE}
|
|
Result := (aChar = '_') or IsCharAlpha(aChar)
|
|
{$ENDIF}
|
|
end;
|
|
|
|
function NameCanContain(aChar: TXmlChar): Boolean;
|
|
begin
|
|
{$IFDEF XML_WIDE_CHARS}
|
|
Result := (aChar = '_') or (aChar = '-') or (aChar = ':') or (aChar = '.') or
|
|
IsCharAlphaNumericW(aChar)
|
|
{$ELSE}
|
|
Result := (aChar in ['_', '-', ':', '.']) or IsCharAlphaNumeric(aChar)
|
|
{$ENDIF}
|
|
end;
|
|
|
|
function IsName(const s: TXmlString): Boolean;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
if s = '' then
|
|
Result := False
|
|
else if not NameCanBeginWith(s[1]) then
|
|
Result := False
|
|
else begin
|
|
for i := 2 to Length(s) do
|
|
if not NameCanContain(s[i]) then begin
|
|
Result := False;
|
|
Exit
|
|
end;
|
|
Result := True;
|
|
end;
|
|
end;
|
|
|
|
const
|
|
ntComment = -2;
|
|
ntNode = -3;
|
|
ntProcessingInstruction = -4;
|
|
ntText = -5;
|
|
|
|
type
|
|
TAxis = (axAncestor, axAncestorOrSelf, axAttribute, axChild,
|
|
axDescendant, axDescendantOrSelf, axFollowing, axFollowingSibling,
|
|
axParent, axPreceding, axPrecedingSibling, axSelf);
|
|
|
|
TPredicate = class
|
|
function Check(aNode: TXmlNode): Boolean; virtual; abstract;
|
|
end;
|
|
|
|
TLocationStep = class
|
|
Next: TLocationStep;
|
|
Axis: TAxis;
|
|
NodeTest: Integer;
|
|
Predicates: TList;
|
|
end;
|
|
|
|
|
|
|
|
function TXmlNode.SelectNodes(
|
|
const anExpression: TXmlString): IXmlNodeList;
|
|
var
|
|
aNodes: TXmlNodeList;
|
|
aChilds: TXmlNodeList;
|
|
aChild: TXmlNode;
|
|
aNameID: Integer;
|
|
i: Integer;
|
|
{
|
|
aPath: TXmlPath;
|
|
}
|
|
begin
|
|
if IsName(anExpression) then begin
|
|
aNodes := TXmlNodeList.Create(nil);
|
|
Result := aNodes;
|
|
aChilds := GetChilds;
|
|
aNameID := FNames.GetID(anExpression);
|
|
for i := 0 to aChilds.FCount - 1 do begin
|
|
aChild := aChilds.FItems[i];
|
|
if (aChild.Get_NodeType = NODE_ELEMENT) and (aChild.Get_NodeNameID = aNameID) then
|
|
aNodes.Insert(aChild, aNodes.FCount);
|
|
end;
|
|
end
|
|
else begin
|
|
raise
|
|
Exception.Create(SSimpleXmlError9);
|
|
{
|
|
aPath := TXmlPath.Create;
|
|
try
|
|
aPath.Init(anExpression);
|
|
Result := aPath.SelectNodes(Self);
|
|
finally
|
|
aPath.Free
|
|
end
|
|
}
|
|
end;
|
|
end;
|
|
|
|
function TXmlNode.SelectSingleNode(
|
|
const anExpression: TXmlString): IXmlNode;
|
|
var
|
|
aChilds: TXmlNodeList;
|
|
aChild: TXmlNode;
|
|
aNameID: Integer;
|
|
i: Integer;
|
|
begin
|
|
if IsName(anExpression) then begin
|
|
aChilds := GetChilds;
|
|
aNameID := FNames.GetID(anExpression);
|
|
for i := 0 to aChilds.FCount - 1 do begin
|
|
aChild := aChilds.FItems[i];
|
|
if (aChild.Get_NodeType = NODE_ELEMENT) and (aChild.Get_NodeNameID = aNameID) then begin
|
|
Result := aChild;
|
|
Exit
|
|
end
|
|
end;
|
|
Result := nil;
|
|
end
|
|
else begin
|
|
raise
|
|
Exception.Create(SSimpleXmlError9)
|
|
end
|
|
end;
|
|
|
|
function TXmlNode.FindElement(const anElementName, anAttrName: String;
|
|
const anAttrValue: Variant): IXmlElement;
|
|
var
|
|
aChild: TXmlNode;
|
|
aNameID, anAttrNameID: Integer;
|
|
i: Integer;
|
|
pa: PXmlAttrData;
|
|
begin
|
|
if Assigned(FChilds) then begin
|
|
aNameID := FNames.GetID(anElementName);
|
|
anAttrNameID := FNames.GetID(anAttrName);
|
|
|
|
for i := 0 to FChilds.FCount - 1 do begin
|
|
aChild := FChilds.FItems[i];
|
|
if (aChild.Get_NodeType = NODE_ELEMENT) and (aChild.Get_NodeNameID = aNameID) then begin
|
|
pa := aChild.FindAttrData(anAttrNameID);
|
|
try
|
|
if Assigned(pa) and VarSameValue(pa.Value, anAttrValue) then begin
|
|
Result := aChild.AsElement;
|
|
Exit
|
|
end
|
|
except
|
|
// Исключительная ситуация может возникнуть в том случае,
|
|
// если произойдет сбой в функции VarSameValue.
|
|
// Иными словами - если значения нельзя сравнивать.
|
|
end;
|
|
end
|
|
end;
|
|
end;
|
|
Result := nil;
|
|
end;
|
|
|
|
procedure TXmlNode.Set_TypedValue(const aValue: Variant);
|
|
begin
|
|
Set_Text(aValue)
|
|
end;
|
|
|
|
procedure TXmlNode.SetAttr(const aName, aValue: TXmlString);
|
|
begin
|
|
SetVarAttr(FNames.GetID(aName), aValue)
|
|
end;
|
|
|
|
procedure TXmlNode.SetAttr(aNameID: Integer; const aValue: TXmlString);
|
|
begin
|
|
SetVarAttr(aNameID, aValue)
|
|
end;
|
|
|
|
procedure TXmlNode.SetBoolAttr(aNameID: Integer; aValue: Boolean);
|
|
begin
|
|
SetVarAttr(aNameID, aValue)
|
|
end;
|
|
|
|
procedure TXmlNode.SetBoolAttr(const aName: TXmlString; aValue: Boolean);
|
|
begin
|
|
SetVarAttr(FNames.GetID(aName), aValue)
|
|
end;
|
|
|
|
procedure TXmlNode.SetChildText(const aName: TXmlString;
|
|
const aValue: TXmlString);
|
|
begin
|
|
SetChildText(FNames.GetID(aName), aValue)
|
|
end;
|
|
|
|
procedure TXmlNode.SetChildText(aNameID: Integer; const aValue: TXmlString);
|
|
var
|
|
aChild: TXmlNode;
|
|
begin
|
|
aChild := FindFirstChild(aNameID);
|
|
if not Assigned(aChild) then begin
|
|
aChild := TXmlElement.Create(FNames, aNameID);
|
|
with GetChilds do
|
|
Insert(aChild, FCount);
|
|
end;
|
|
aChild.Set_Text(aValue)
|
|
end;
|
|
|
|
procedure TXmlNode.SetFloatAttr(aNameID: Integer; aValue: Double);
|
|
begin
|
|
SetVarAttr(aNameID, aValue)
|
|
end;
|
|
|
|
procedure TXmlNode.SetFloatAttr(const aName: TXmlString; aValue: Double);
|
|
begin
|
|
SetVarAttr(FNames.GetID(aName), aValue);
|
|
end;
|
|
|
|
procedure TXmlNode.SetHexAttr(const aName: TXmlString; aValue,
|
|
aDigits: Integer);
|
|
begin
|
|
SetVarAttr(FNames.GetID(aName), IntToHex(aValue, aDigits))
|
|
end;
|
|
|
|
procedure TXmlNode.SetHexAttr(aNameID, aValue, aDigits: Integer);
|
|
begin
|
|
SetVarAttr(aNameID, IntToHex(aValue, aDigits))
|
|
end;
|
|
|
|
procedure TXmlNode.SetIntAttr(aNameID, aValue: Integer);
|
|
begin
|
|
SetVarAttr(aNameID, aValue)
|
|
end;
|
|
|
|
procedure TXmlNode.SetIntAttr(const aName: TXmlString; aValue: Integer);
|
|
begin
|
|
SetVarAttr(FNames.GetID(aName), aValue)
|
|
end;
|
|
|
|
procedure TXmlNode.SetVarAttr(const aName: TXmlString; aValue: Variant);
|
|
begin
|
|
SetVarAttr(FNames.GetID(aName), aValue)
|
|
end;
|
|
|
|
procedure TXmlNode.SetVarAttr(aNameID: Integer; const aValue: Variant);
|
|
var
|
|
anAttr: PXmlAttrData;
|
|
var
|
|
aDelta: Integer;
|
|
begin
|
|
anAttr := FindAttrData(aNameID);
|
|
if not Assigned(anAttr) then begin
|
|
if FAttrCount = Length(FAttrs) then begin
|
|
if FAttrCount > 64 then
|
|
aDelta := FAttrCount div 4
|
|
else if FAttrCount > 8 then
|
|
aDelta := 16
|
|
else
|
|
aDelta := 4;
|
|
SetLength(FAttrs, FAttrCount + aDelta);
|
|
end;
|
|
anAttr := @FAttrs[FAttrCount];
|
|
anAttr.NameID := aNameID;
|
|
Inc(FAttrCount);
|
|
end;
|
|
anAttr.Value := aValue
|
|
end;
|
|
|
|
function TXmlNode.FindAttrData(aNameID: Integer): PXmlAttrData;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
Result := @FAttrs[0];
|
|
for i := 0 to FAttrCount - 1 do
|
|
if Result.NameID = aNameID then
|
|
Exit
|
|
else
|
|
Inc(Result);
|
|
Result := nil;
|
|
end;
|
|
|
|
function TXmlNode.AsElement: IXmlElement;
|
|
begin
|
|
Result := nil
|
|
end;
|
|
|
|
function TXmlNode.AsCDATASection: IXmlCDATASection;
|
|
begin
|
|
Result := nil
|
|
end;
|
|
|
|
function TXmlNode.AsComment: IXmlComment;
|
|
begin
|
|
Result := nil
|
|
end;
|
|
|
|
function TXmlNode.AsText: IXmlText;
|
|
begin
|
|
Result := nil
|
|
end;
|
|
|
|
function TXmlNode.AsProcessingInstruction: IXmlProcessingInstruction;
|
|
begin
|
|
Result := nil
|
|
end;
|
|
|
|
function TXmlNode.AppendCDATA(const aData: TXmlString): IXmlCDATASection;
|
|
var
|
|
aChild: TXmlCDATASection;
|
|
begin
|
|
aChild := TXmlCDATASection.Create(FNames, aData);
|
|
GetChilds.Insert(aChild, -1);
|
|
Result := aChild
|
|
end;
|
|
|
|
function TXmlNode.AppendComment(const aData: TXmlString): IXmlComment;
|
|
var
|
|
aChild: TXmlComment;
|
|
begin
|
|
aChild := TXmlComment.Create(FNames, aData);
|
|
GetChilds.Insert(aChild, -1);
|
|
Result := aChild
|
|
end;
|
|
|
|
function TXmlNode.AppendElement(const aName: TxmlString): IXmlElement;
|
|
var
|
|
aChild: TXmlElement;
|
|
begin
|
|
aChild := TXmlElement.Create(FNames, FNames.GetID(aName));
|
|
GetChilds.Insert(aChild, -1);
|
|
Result := aChild
|
|
end;
|
|
|
|
function TXmlNode.AppendElement(aNameID: Integer): IXmlElement;
|
|
var
|
|
aChild: TXmlElement;
|
|
begin
|
|
aChild := TXmlElement.Create(FNames, aNameID);
|
|
GetChilds.Insert(aChild, -1);
|
|
Result := aChild
|
|
end;
|
|
|
|
function TXmlNode.AppendProcessingInstruction(const aTarget,
|
|
aData: TXmlString): IXmlProcessingInstruction;
|
|
var
|
|
aChild: TXmlProcessingInstruction;
|
|
begin
|
|
aChild := TXmlProcessingInstruction.Create(FNames, FNames.GetID(aTarget), aData);
|
|
GetChilds.Insert(aChild, -1);
|
|
Result := aChild
|
|
end;
|
|
|
|
function TXmlNode.AppendProcessingInstruction(aTargetID: Integer;
|
|
const aData: TXmlString): IXmlProcessingInstruction;
|
|
var
|
|
aChild: TXmlProcessingInstruction;
|
|
begin
|
|
aChild := TXmlProcessingInstruction.Create(FNames, aTargetID, aData);
|
|
GetChilds.Insert(aChild, -1);
|
|
Result := aChild
|
|
end;
|
|
|
|
function TXmlNode.AppendText(const aData: TXmlString): IXmlText;
|
|
var
|
|
aChild: TXmlText;
|
|
begin
|
|
aChild := TXmlText.Create(FNames, aData);
|
|
GetChilds.Insert(aChild, -1);
|
|
Result := aChild
|
|
end;
|
|
|
|
function TXmlNode.GetAttrsXML: TXmlString;
|
|
var
|
|
a: PXmlAttrData;
|
|
i: Integer;
|
|
begin
|
|
Result := '';
|
|
if FAttrCount > 0 then begin
|
|
a := @FAttrs[0];
|
|
for i := 0 to FAttrCount - 1 do begin
|
|
Result := Result + ' ' + FNames.GetName(a.NameID) + '="' + TextToXML(VarToXSTR(TVarData(a.Value))) + '"';
|
|
Inc(a);
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
procedure TXmlNode.LoadBinXml(aReader: TBinXmlReader);
|
|
var
|
|
aCount: LongInt;
|
|
a: PXmlAttrData;
|
|
i: Integer;
|
|
begin
|
|
// Считать атрибуты
|
|
RemoveAllAttrs;
|
|
aCount := aReader.ReadLongint;
|
|
SetLength(FAttrs, aCount);
|
|
FAttrCount := aCount;
|
|
a := @FAttrs[0];
|
|
for i := 0 to aCount - 1 do begin
|
|
a.NameID := aReader.ReadLongint;
|
|
aReader.ReadVariant(TVarData(a.Value));
|
|
Inc(a);
|
|
end;
|
|
|
|
// Считать дочерние узлы
|
|
aCount := aReader.ReadLongint;
|
|
if aCount > 0 then
|
|
GetChilds.LoadBinXml(aReader, aCount, FNames);
|
|
end;
|
|
|
|
procedure TXmlNode.SaveBinXml(aWriter: TBinXmlWriter);
|
|
var
|
|
aCount: LongInt;
|
|
a: PXmlAttrData;
|
|
i: Integer;
|
|
begin
|
|
// Считать атрибуты
|
|
aCount := FAttrCount;
|
|
aWriter.WriteLongint(aCount);
|
|
a := @FAttrs[0];
|
|
for i := 0 to aCount - 1 do begin
|
|
aWriter.WriteLongint(a.NameID);
|
|
aWriter.WriteVariant(TVarData(a.Value));
|
|
Inc(a);
|
|
end;
|
|
|
|
// Записать дочерние узлы
|
|
if Assigned(FChilds) then begin
|
|
aWriter.WriteLongint(FChilds.FCount);
|
|
FChilds.SaveBinXml(aWriter);
|
|
end
|
|
else
|
|
aWriter.WriteLongint(0);
|
|
end;
|
|
|
|
function TXmlNode.Get_DataType: Integer;
|
|
begin
|
|
{$IFDEF XML_WIDE_CHARS}
|
|
Result := varOleStr
|
|
{$ELSE}
|
|
Result := varString
|
|
{$ENDIF}
|
|
end;
|
|
|
|
function TXmlNode.AttrExists(aNameID: Integer): Boolean;
|
|
begin
|
|
Result := FindAttrData(aNameID) <> nil
|
|
end;
|
|
|
|
function TXmlNode.AttrExists(const aName: TXmlString): Boolean;
|
|
begin
|
|
Result := FindAttrData(FNames.GetID(aName)) <> nil
|
|
end;
|
|
|
|
function TXmlNode.GetAttrType(aNameID: Integer): Integer;
|
|
var
|
|
a: PXmlAttrData;
|
|
begin
|
|
a := FindAttrData(aNameID);
|
|
if Assigned(a) then
|
|
Result := TVarData(a.Value).VType
|
|
else
|
|
{$IFDEF XML_WIDE_CHARS}
|
|
Result := varOleStr
|
|
{$ELSE}
|
|
Result := varString
|
|
{$ENDIF}
|
|
end;
|
|
|
|
function TXmlNode.GetAttrType(const aName: TXmlString): Integer;
|
|
begin
|
|
Result := GetAttrType(FNames.GetID(aName));
|
|
end;
|
|
|
|
function TXmlNode.Get_Values(const aName: String): Variant;
|
|
var
|
|
aChild: IXmlNode;
|
|
begin
|
|
if aName = '' then
|
|
Result := Get_TypedValue
|
|
else if aName[1] = '@' then
|
|
Result := GetVarAttr(Copy(aName, 2, Length(aName) - 1), '')
|
|
else begin
|
|
aChild := SelectSingleNode(aName);
|
|
if Assigned(aChild) then
|
|
Result := aChild.TypedValue
|
|
else
|
|
Result := ''
|
|
end
|
|
end;
|
|
|
|
procedure TXmlNode.Set_Values(const aName: String; const aValue: Variant);
|
|
var
|
|
aChild: IXmlNode;
|
|
begin
|
|
if aName = '' then
|
|
Set_TypedValue(aValue)
|
|
else if aName[1] = '@' then
|
|
SetVarAttr(Copy(aName, 2, Length(aName) - 1), aValue)
|
|
else begin
|
|
aChild := SelectSingleNode(aName);
|
|
if not Assigned(aChild) then
|
|
aChild := AppendElement(aName);
|
|
aChild.TypedValue := aValue;
|
|
end
|
|
end;
|
|
|
|
function TXmlNode.GetDateTimeAttr(aNameID: Integer;
|
|
aDefault: TDateTime): TDateTime;
|
|
var
|
|
anAttr: PXmlAttrData;
|
|
begin
|
|
anAttr := FindAttrData(aNameID);
|
|
if Assigned(anAttr) then begin
|
|
if (VarType(anAttr.Value) = varString) or (VarType(anAttr.Value) = varOleStr) then
|
|
Result := XSTRToDateTime(anAttr.Value)
|
|
else
|
|
Result := VarAsType(anAttr.Value, varDate)
|
|
end
|
|
else
|
|
Result := aDefault;
|
|
end;
|
|
|
|
function TXmlNode.GetDateTimeAttr(const aName: TXmlString;
|
|
aDefault: TDateTime): TDateTime;
|
|
begin
|
|
Result := GetDateTimeAttr(FNames.GetID(aName), aDefault)
|
|
end;
|
|
|
|
procedure TXmlNode.SetDateTimeAttr(aNameID: Integer; aValue: TDateTime);
|
|
begin
|
|
SetVarAttr(aNameID, VarAsType(aValue, varDate))
|
|
end;
|
|
|
|
procedure TXmlNode.SetDateTimeAttr(const aName: TXmlString;
|
|
aValue: TDateTime);
|
|
begin
|
|
SetVarAttr(aName, VarAsType(aValue, varDate))
|
|
end;
|
|
|
|
function TXmlNode.EnsureChild(aNameID: Integer): IXmlNode;
|
|
var
|
|
aChild: TXmlNode;
|
|
begin
|
|
aChild := FindFirstChild(aNameID);
|
|
if Assigned(aChild) then
|
|
Result := aChild
|
|
else
|
|
Result := AppendElement(aNameID)
|
|
end;
|
|
|
|
function TXmlNode.EnsureChild(const aName: TXmlString): IXmlNode;
|
|
begin
|
|
Result := EnsureChild(FNames.GetID(aName))
|
|
end;
|
|
|
|
function TXmlNode.NeedChild(aNameID: Integer): IXmlNode;
|
|
var
|
|
aChild: TXmlNode;
|
|
begin
|
|
aChild := FindFirstChild(aNameID);
|
|
if not Assigned(aChild) then
|
|
raise Exception.CreateFmt(SSimpleXmlError10, [FNames.GetName(aNameID)]);
|
|
Result := aChild
|
|
end;
|
|
|
|
function TXmlNode.NeedChild(const aName: TXmlString): IXmlNode;
|
|
begin
|
|
Result := NeedChild(FNames.GetID(aName));
|
|
end;
|
|
|
|
procedure TXmlNode.SetNameTable(aValue: TXmlNameTable; aMap: TList);
|
|
var
|
|
i: Integer;
|
|
aNewMap: Boolean;
|
|
begin
|
|
if aValue <> FNames then begin
|
|
aNewMap := not Assigned(aMap);
|
|
if aNewMap then begin
|
|
aMap := TList.Create;
|
|
for i := 0 to Length(FNames.FNames) do
|
|
aMap.Add(Pointer(aValue.GetID(FNames.FNames[i])));
|
|
end;
|
|
try
|
|
SetNodeNameID(Integer(aMap[Get_NodeNameID]));
|
|
for i := 0 to Length(FAttrs) do
|
|
with FAttrs[i] do
|
|
NameID := Integer(aMap[NameID]);
|
|
if Assigned(FChilds) then
|
|
for i := 0 to FChilds.FCount - 1 do
|
|
FChilds.FItems[i].SetNameTable(aValue, aMap);
|
|
finally
|
|
if aNewMap then
|
|
aMap.Free
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
procedure TXmlNode.SetNodeNameID(aValue: Integer);
|
|
begin
|
|
|
|
end;
|
|
|
|
|
|
function TXmlNode.CloneNode(aDeep: Boolean): IXmlNode;
|
|
begin
|
|
Result := DoCloneNode(aDeep)
|
|
end;
|
|
|
|
{ TXmlElement }
|
|
|
|
constructor TXmlElement.Create(aNames: TXmlNameTable; aNameID: Integer);
|
|
begin
|
|
inherited Create(aNames);
|
|
FNameID := aNameID;
|
|
end;
|
|
|
|
function TXmlElement.Get_NodeNameID: Integer;
|
|
begin
|
|
Result := FNameID
|
|
end;
|
|
|
|
function TXmlElement.Get_NodeType: Integer;
|
|
begin
|
|
Result := NODE_ELEMENT
|
|
end;
|
|
|
|
function TXmlElement.GetChilds: TXmlNodeList;
|
|
begin
|
|
Result := inherited GetChilds;
|
|
if not TVarData(FData).VType in [varEmpty, varNull] then begin
|
|
AppendChild(TXmlText.Create(FNames, FData));
|
|
VarClear(FData);
|
|
end;
|
|
end;
|
|
|
|
function TXmlElement.Get_Text: TXmlString;
|
|
var
|
|
aChilds: TXmlNodeList;
|
|
aChild: TXmlNode;
|
|
aChildText: TXmlString;
|
|
i: Integer;
|
|
begin
|
|
Result := '';
|
|
if Assigned(FChilds) and (FChilds.FCount > 0) then begin
|
|
aChilds := FChilds;
|
|
for i := 0 to aChilds.FCount - 1 do begin
|
|
aChild := aChilds.FItems[i];
|
|
if aChild.Get_NodeType in [NODE_ELEMENT, NODE_TEXT, NODE_CDATA_SECTION] then begin
|
|
aChildText := aChild.Get_Text;
|
|
if aChildText <> '' then
|
|
if Result = '' then
|
|
Result := aChildText
|
|
else
|
|
Result := Result + ' ' + aChildText
|
|
end
|
|
end;
|
|
end
|
|
else if VarIsEmpty(FData) then
|
|
Result := ''
|
|
else
|
|
Result := VarToXSTR(TVarData(FData))
|
|
end;
|
|
|
|
function TXmlElement.GetTextAsBynaryData: TXmlString;
|
|
begin
|
|
Result := Base64ToBin(Get_Text);
|
|
end;
|
|
|
|
procedure TXmlElement.ReplaceTextByBynaryData(const aData; aSize: Integer;
|
|
aMaxLineLength: Integer);
|
|
begin
|
|
RemoveTextNodes;
|
|
GetChilds.Insert(TXmlText.Create(FNames, BinToBase64(aData, aSize, aMaxLineLength)), -1);
|
|
end;
|
|
|
|
procedure TXmlElement.RemoveTextNodes;
|
|
var
|
|
i: Integer;
|
|
aNode: TXmlNode;
|
|
begin
|
|
if Assigned(FChilds) then
|
|
for i := FChilds.FCount - 1 downto 0 do begin
|
|
aNode := FChilds.FItems[i];
|
|
if aNode.Get_NodeType in [NODE_TEXT, NODE_CDATA_SECTION] then
|
|
FChilds.Remove(aNode);
|
|
end;
|
|
end;
|
|
|
|
procedure TXmlElement.ReplaceTextByCDATASection(const aText: TXmlString);
|
|
|
|
procedure AddCDATASection(const aText: TXmlString);
|
|
var
|
|
i: Integer;
|
|
aChilds: TXmlNodeList;
|
|
begin
|
|
i := Pos(']]>', aText);
|
|
aChilds := GetChilds;
|
|
if i = 0 then
|
|
aChilds.Insert(TXmlCDATASection.Create(FNames, aText), aChilds.FCount)
|
|
else begin
|
|
aChilds.Insert(TXmlCDATASection.Create(FNames, Copy(aText, 1, i)), aChilds.FCount);
|
|
AddCDATASection(Copy(aText, i + 1, Length(aText) - i - 1));
|
|
end;
|
|
end;
|
|
|
|
begin
|
|
RemoveTextNodes;
|
|
AddCDATASection(aText);
|
|
end;
|
|
|
|
procedure TXmlElement.Set_Text(const aValue: TXmlString);
|
|
begin
|
|
if Assigned(FChilds) then
|
|
FChilds.Clear;
|
|
FData := aValue;
|
|
end;
|
|
|
|
function TXmlElement.AsElement: IXmlElement;
|
|
begin
|
|
Result := Self
|
|
end;
|
|
|
|
var
|
|
FGetXMLIntend: Integer = 0;
|
|
|
|
function GetIndentStr: String;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
SetLength(Result, FGetXMLIntend*Length(DefaultIndentText));
|
|
for i := 0 to FGetXMLIntend - 1 do
|
|
Move(DefaultIndentText[1], Result[i*Length(DefaultIndentText) + 1], Length(DefaultIndentText));
|
|
end;
|
|
|
|
function HasCRLF(const s: String): Boolean;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
for i := 1 to Length(s) do
|
|
if (s[i] = ^M) or (s[i] = ^J) then begin
|
|
Result := True;
|
|
Exit
|
|
end;
|
|
Result := False;
|
|
end;
|
|
|
|
function EndWithCRLF(const s: String): Boolean;
|
|
begin
|
|
Result :=
|
|
(Length(s) > 1) and
|
|
(s[Length(s) - 1] = ^M) and
|
|
(s[Length(s)] = ^J);
|
|
end;
|
|
|
|
function TXmlElement.Get_XML: TXmlString;
|
|
var
|
|
aChildsXML: TXmlString;
|
|
aTag: TXmlString;
|
|
aDoc: TXmlDocument;
|
|
aPreserveWhiteSpace: Boolean;
|
|
begin
|
|
aDoc := GetOwnerDocument;
|
|
if Assigned(aDoc) then
|
|
aPreserveWhiteSpace := aDoc.Get_PreserveWhiteSpace
|
|
else
|
|
aPreserveWhiteSpace := DefaultPreserveWhiteSpace;
|
|
if aPreserveWhiteSpace then begin
|
|
if Assigned(FChilds) and (FChilds.FCount > 0) then
|
|
aChildsXML := FChilds.Get_XML
|
|
else if VarIsEmpty(FData) then
|
|
aChildsXML := ''
|
|
else
|
|
aChildsXML := TextToXML(VarToXSTR(TVarData(FData)));
|
|
|
|
aTag := FNames.GetName(FNameID);
|
|
Result := '<' + aTag + GetAttrsXML;
|
|
if aChildsXML = '' then
|
|
Result := Result + '/>'
|
|
else
|
|
Result := Result + '>' + aChildsXML + '</' + aTag + '>'
|
|
end
|
|
else begin
|
|
if Assigned(FChilds) and (FChilds.FCount > 0) then begin
|
|
Inc(FGetXMLIntend);
|
|
try
|
|
aChildsXML := FChilds.Get_XML
|
|
finally
|
|
Dec(FGetXMLIntend)
|
|
end
|
|
end
|
|
else if VarIsEmpty(FData) then
|
|
aChildsXML := ''
|
|
else
|
|
aChildsXML := TextToXML(VarToXSTR(TVarData(FData)));
|
|
aTag := FNames.GetName(FNameID);
|
|
Result := ^M^J+GetIndentStr + '<' + aTag + GetAttrsXML;
|
|
if aChildsXML = '' then
|
|
Result := Result + '/>'
|
|
else if HasCRLF(aChildsXML) then
|
|
if EndWithCRLF(aChildsXML) then
|
|
Result := Result + '>' + aChildsXML + GetIndentStr + '</' + aTag + '>'
|
|
else
|
|
Result := Result + '>' + aChildsXML + ^M^J + GetIndentStr + '</' + aTag + '>'
|
|
else
|
|
Result := Result + '>' + aChildsXML + '</' + aTag + '>';
|
|
end;
|
|
end;
|
|
|
|
function TXmlElement.Get_TypedValue: Variant;
|
|
begin
|
|
if Assigned(FChilds) and (FChilds.FCount > 0) then
|
|
Result := Get_Text
|
|
else
|
|
Result := FData
|
|
end;
|
|
|
|
procedure TXmlElement.Set_TypedValue(const aValue: Variant);
|
|
begin
|
|
if Assigned(FChilds) then
|
|
FChilds.Clear;
|
|
FData := aValue;
|
|
end;
|
|
|
|
function TXmlElement.Get_DataType: Integer;
|
|
begin
|
|
if (Assigned(FChilds) and (FChilds.FCount > 0)) or VarIsEmpty(FData) then
|
|
{$IFDEF XML_WIDE_CHARS}
|
|
Result := varOleStr
|
|
{$ELSE}
|
|
Result := varString
|
|
{$ENDIF}
|
|
else
|
|
Result := TVarData(FData).VType;
|
|
end;
|
|
|
|
function TXmlElement.Get_ChildNodes: IXmlNodeList;
|
|
begin
|
|
Result := inherited Get_ChildNodes;
|
|
end;
|
|
|
|
procedure TXmlElement.SetNodeNameID(aValue: Integer);
|
|
begin
|
|
FNameID := aValue
|
|
end;
|
|
|
|
function TXmlElement.DoCloneNode(aDeep: Boolean): IXmlNode;
|
|
var
|
|
aClone: TXmlElement;
|
|
i: Integer;
|
|
begin
|
|
aClone := TXmlElement.Create(FNames, FNameID);
|
|
Result := aClone;
|
|
SetLength(aClone.FAttrs, FAttrCount);
|
|
aClone.FAttrCount := FAttrCount;
|
|
for i := 0 to FAttrCount - 1 do
|
|
aClone.FAttrs[i] := FAttrs[i];
|
|
if aDeep and Assigned(FChilds) and (FChilds.FCount > 0) then
|
|
for i := 0 to FChilds.FCount - 1 do
|
|
aClone.AppendChild(FChilds.FItems[i].CloneNode(True));
|
|
end;
|
|
|
|
{ TXmlCharacterData }
|
|
|
|
constructor TXmlCharacterData.Create(aNames: TXmlNameTable;
|
|
const aData: TXmlString);
|
|
begin
|
|
inherited Create(aNames);
|
|
FData := aData;
|
|
end;
|
|
|
|
function TXmlCharacterData.Get_Text: TXmlString;
|
|
var
|
|
aDoc: TXmlDocument;
|
|
aPreserveWhiteSpace: Boolean;
|
|
begin
|
|
aDoc := GetOwnerDocument;
|
|
if Assigned(aDoc) then
|
|
aPreserveWhiteSpace := aDoc.Get_PreserveWhiteSpace
|
|
else
|
|
aPreserveWhiteSpace := DefaultPreserveWhiteSpace;
|
|
if aPreserveWhiteSpace then
|
|
Result := FData
|
|
else
|
|
Result := Trim(FData);
|
|
end;
|
|
|
|
procedure TXmlCharacterData.Set_Text(const aValue: TXmlString);
|
|
begin
|
|
FData := aValue
|
|
end;
|
|
|
|
{ TXmlText }
|
|
|
|
function TXmlText.AsText: IXmlText;
|
|
begin
|
|
Result := Self;
|
|
end;
|
|
|
|
constructor TXmlText.Create(aNames: TXmlNameTable; const aData: Variant);
|
|
begin
|
|
inherited Create(aNames);
|
|
FData := aData;
|
|
end;
|
|
|
|
function TXmlText.DoCloneNode(aDeep: Boolean): IXmlNode;
|
|
begin
|
|
Result := TXmlText.Create(FNames, FData);
|
|
end;
|
|
|
|
function TXmlText.Get_DataType: Integer;
|
|
begin
|
|
Result := TVarData(FData).VType
|
|
end;
|
|
|
|
function TXmlText.Get_NodeNameID: Integer;
|
|
begin
|
|
Result := FNames.FXmlTextNameID
|
|
end;
|
|
|
|
function TXmlText.Get_NodeType: Integer;
|
|
begin
|
|
Result := NODE_TEXT
|
|
end;
|
|
|
|
function TXmlText.Get_Text: TXmlString;
|
|
begin
|
|
Result := VarToXSTR(TVarData(FData))
|
|
end;
|
|
|
|
function TXmlText.Get_TypedValue: Variant;
|
|
begin
|
|
Result := FData
|
|
end;
|
|
|
|
function TXmlText.Get_XML: TXmlString;
|
|
begin
|
|
Result := TextToXML(VarToXSTR(TVarData(FData)));
|
|
end;
|
|
|
|
procedure TXmlText.Set_Text(const aValue: TXmlString);
|
|
begin
|
|
FData := aValue
|
|
end;
|
|
|
|
procedure TXmlText.Set_TypedValue(const aValue: Variant);
|
|
begin
|
|
FData := aValue
|
|
end;
|
|
|
|
{ TXmlCDATASection }
|
|
|
|
function TXmlCDATASection.AsCDATASection: IXmlCDATASection;
|
|
begin
|
|
Result := Self
|
|
end;
|
|
|
|
function TXmlCDATASection.DoCloneNode(aDeep: Boolean): IXmlNode;
|
|
begin
|
|
Result := TXmlCDATASection.Create(FNames, FData);
|
|
end;
|
|
|
|
function TXmlCDATASection.Get_NodeNameID: Integer;
|
|
begin
|
|
Result := FNames.FXmlCDATASectionNameID
|
|
end;
|
|
|
|
function TXmlCDATASection.Get_NodeType: Integer;
|
|
begin
|
|
Result := NODE_CDATA_SECTION
|
|
end;
|
|
|
|
function GenCDATAXML(const aValue: TXmlString): TXmlString;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
i := Pos(']]>', aValue);
|
|
if i = 0 then
|
|
Result := '<![CDATA[' + aValue + ']]>'
|
|
else
|
|
Result := '<![CDATA[' + Copy(aValue, 1, i) + ']]>' + GenCDATAXML(Copy(aValue, i + 1, Length(aValue) - i - 1));
|
|
end;
|
|
|
|
function TXmlCDATASection.Get_XML: TXmlString;
|
|
begin
|
|
Result := GenCDATAXML(FData);
|
|
end;
|
|
|
|
{ TXmlComment }
|
|
|
|
function TXmlComment.AsComment: IXmlComment;
|
|
begin
|
|
Result := Self
|
|
end;
|
|
|
|
function TXmlComment.DoCloneNode(aDeep: Boolean): IXmlNode;
|
|
begin
|
|
Result := TXmlComment.Create(FNames, FData);
|
|
end;
|
|
|
|
function TXmlComment.Get_NodeNameID: Integer;
|
|
begin
|
|
Result := FNames.FXmlCommentNameID
|
|
end;
|
|
|
|
function TXmlComment.Get_NodeType: Integer;
|
|
begin
|
|
Result := NODE_COMMENT
|
|
end;
|
|
|
|
function TXmlComment.Get_XML: TXmlString;
|
|
begin
|
|
Result := '<!--' + FData + '-->'
|
|
end;
|
|
|
|
{ TXmlDocument }
|
|
|
|
constructor TXmlDocument.Create(aNames: TXmlNameTable);
|
|
begin
|
|
inherited Create(aNames);
|
|
FPreserveWhiteSpace := DefaultPreserveWhiteSpace;
|
|
end;
|
|
|
|
function TXmlDocument.CreateCDATASection(
|
|
const aData: TXmlString): IXmlCDATASection;
|
|
begin
|
|
Result := TXmlCDATASection.Create(FNames, aData)
|
|
end;
|
|
|
|
function TXmlDocument.CreateComment(const aData: TXmlString): IXmlComment;
|
|
begin
|
|
Result := TXmlComment.Create(FNames, aData)
|
|
end;
|
|
|
|
function TXmlDocument.CreateElement(aNameID: Integer): IXmlElement;
|
|
begin
|
|
Result := TXmlElement.Create(FNames, aNameID)
|
|
end;
|
|
|
|
function TXmlDocument.CreateElement(const aName: TXmlString): IXmlElement;
|
|
begin
|
|
Result := TXmlElement.Create(FNames, FNames.GetID(aName));
|
|
end;
|
|
|
|
function TXmlDocument.CreateProcessingInstruction(const aTarget,
|
|
aData: TXmlString): IXmlProcessingInstruction;
|
|
begin
|
|
Result := TXmlProcessingInstruction.Create(FNames, FNames.GetID(aTarget), aData)
|
|
end;
|
|
|
|
function TXmlDocument.CreateProcessingInstruction(aTargetID: Integer;
|
|
const aData: TXmlString): IXmlProcessingInstruction;
|
|
begin
|
|
Result := TXmlProcessingInstruction.Create(FNames, aTargetID, aData)
|
|
end;
|
|
|
|
function TXmlDocument.CreateText(const aData: TXmlString): IXmlText;
|
|
begin
|
|
Result := TXmlText.Create(FNames, aData)
|
|
end;
|
|
|
|
function TXmlDocument.DoCloneNode(aDeep: Boolean): IXmlNode;
|
|
var
|
|
aClone: TXmlDocument;
|
|
i: Integer;
|
|
begin
|
|
aClone := TXmlDocument.Create(FNames);
|
|
Result := aClone;
|
|
if aDeep and Assigned(FChilds) and (FChilds.FCount > 0) then
|
|
for i := 0 to FChilds.FCount - 1 do
|
|
aClone.AppendChild(FChilds.FItems[i].CloneNode(True));
|
|
end;
|
|
|
|
function TXmlDocument.Get_BinaryXML: String;
|
|
var
|
|
aWriter: TStrXmlWriter;
|
|
begin
|
|
aWriter := TStrXmlWriter.Create(0, $10000);
|
|
try
|
|
FNames.SaveBinXml(aWriter);
|
|
SaveBinXml(aWriter);
|
|
aWriter.FlushBuf;
|
|
Result := aWriter.FData;
|
|
finally
|
|
aWriter.Free
|
|
end
|
|
end;
|
|
|
|
function TXmlDocument.Get_DocumentElement: IXmlElement;
|
|
var
|
|
aChilds: TXmlNodeList;
|
|
aChild: TXmlNode;
|
|
i: Integer;
|
|
begin
|
|
aChilds := GetChilds;
|
|
for i := 0 to aChilds.FCount - 1 do begin
|
|
aChild := aChilds.FItems[i];
|
|
if aChild.Get_NodeType = NODE_ELEMENT then begin
|
|
Result := aChild.AsElement;
|
|
Exit
|
|
end
|
|
end;
|
|
Result := nil;
|
|
end;
|
|
|
|
function TXmlDocument.Get_NodeNameID: Integer;
|
|
begin
|
|
Result := FNames.FXmlDocumentNameID
|
|
end;
|
|
|
|
function TXmlDocument.Get_NodeType: Integer;
|
|
begin
|
|
Result := NODE_DOCUMENT
|
|
end;
|
|
|
|
function TXmlDocument.Get_PreserveWhiteSpace: Boolean;
|
|
begin
|
|
Result := FPreserveWhiteSpace;
|
|
end;
|
|
|
|
function TXmlDocument.Get_Text: TXmlString;
|
|
var
|
|
aChilds: TXmlNodeList;
|
|
aChild: TXmlNode;
|
|
aChildText: TXmlString;
|
|
i: Integer;
|
|
begin
|
|
Result := '';
|
|
aChilds := GetChilds;
|
|
for i := 0 to aChilds.FCount - 1 do begin
|
|
aChild := aChilds.FItems[i];
|
|
if aChild.Get_NodeType in [NODE_ELEMENT, NODE_TEXT, NODE_CDATA_SECTION] then begin
|
|
aChildText := aChild.Get_Text;
|
|
if aChildText <> '' then
|
|
if Result = '' then
|
|
Result := aChildText
|
|
else
|
|
Result := Result + ' ' + aChildText
|
|
end
|
|
end;
|
|
end;
|
|
|
|
function TXmlDocument.Get_XML: TXmlString;
|
|
begin
|
|
Result := GetChilds.Get_XML
|
|
end;
|
|
|
|
procedure TXmlDocument.Load(aStream: TStream);
|
|
var
|
|
aXml: TXmlStmSource;
|
|
aBinarySign: String;
|
|
aReader: TBinXmlReader;
|
|
begin
|
|
RemoveAllChilds;
|
|
RemoveAllAttrs;
|
|
if aStream.Size > BinXmlSignatureSize then begin
|
|
SetLength(aBinarySign, BinXmlSignatureSize);
|
|
aStream.ReadBuffer(aBinarySign[1], BinXmlSignatureSize);
|
|
if aBinarySign = BinXmlSignature then begin
|
|
FNames._Release;
|
|
FNames := TXmlNameTable.Create(4096);
|
|
FNames._AddRef;
|
|
aReader := TStmXmlReader.Create(aStream, $10000);
|
|
try
|
|
FNames.LoadBinXml(aReader);
|
|
LoadBinXml(aReader);
|
|
finally
|
|
aReader.Free
|
|
end;
|
|
Exit;
|
|
end;
|
|
aStream.Position := aStream.Position - BinXmlSignatureSize;
|
|
end;
|
|
aXml := TXmlStmSource.Create(aStream, 1 shl 16);
|
|
try
|
|
GetChilds.ParseXML(aXml, FNames, FPreserveWhiteSpace);
|
|
finally
|
|
aXml.Free
|
|
end
|
|
end;
|
|
|
|
procedure TXmlDocument.Load(const aFileName: TXmlString);
|
|
var
|
|
aFile: TFileStream;
|
|
begin
|
|
aFile := TFileStream.Create(aFileName, fmShareDenyNone, fmShareDenyWrite);
|
|
try
|
|
Load(aFile);
|
|
finally
|
|
aFile.Free
|
|
end
|
|
end;
|
|
|
|
procedure TXmlDocument.LoadBinaryXML(const aXML: String);
|
|
var
|
|
aReader: TStrXmlReader;
|
|
begin
|
|
RemoveAllChilds;
|
|
RemoveAllAttrs;
|
|
aReader := TStrXmlReader.Create(aXML);
|
|
try
|
|
FNames._Release;
|
|
FNames := TXmlNameTable.Create(4096);
|
|
FNames._AddRef;
|
|
FNames.LoadBinXml(aReader);
|
|
LoadBinXml(aReader);
|
|
finally
|
|
aReader.Free
|
|
end
|
|
end;
|
|
|
|
procedure TXmlDocument.LoadResource(aType, aName: PChar);
|
|
var
|
|
aRSRC: HRSRC;
|
|
aGlobal: HGLOBAL;
|
|
aSize: DWORD;
|
|
aPointer: Pointer;
|
|
aStream: TStringStream;
|
|
begin
|
|
aRSRC := FindResource(HInstance, aName, aType);
|
|
if aRSRC <> 0 then begin
|
|
aGlobal := Windows.LoadResource(HInstance, aRSRC);
|
|
aSize := SizeofResource(HInstance, aRSRC);
|
|
if (aGlobal <> 0) and (aSize <> 0) then begin
|
|
aPointer := LockResource(aGlobal);
|
|
if Assigned(aPointer) then begin
|
|
aStream := TStringStream.Create('');
|
|
try
|
|
aStream.WriteBuffer(aPointer^, aSize);
|
|
LoadXML(aStream.DataString);
|
|
finally
|
|
aStream.Free;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
procedure TXmlDocument.LoadXML(const aXML: TXmlString);
|
|
var
|
|
aSource: TXmlStrSource;
|
|
begin
|
|
if XmlIsInBinaryFormat(aXML) then
|
|
LoadBinaryXML(aXML)
|
|
else begin
|
|
RemoveAllChilds;
|
|
RemoveAllAttrs;
|
|
aSource := TXmlStrSource.Create(aXML);
|
|
try
|
|
GetChilds.ParseXML(aSource, FNames, FPreserveWhiteSpace);
|
|
finally
|
|
aSource.Free
|
|
end
|
|
end
|
|
end;
|
|
|
|
function TXmlDocument.NewDocument(const aVersion, anEncoding,
|
|
aRootElementName: TXmlString): IXmlElement;
|
|
begin
|
|
Result := NewDocument(aVersion, anEncoding, FNames.GetID(aRootElementName));
|
|
end;
|
|
|
|
function TXmlDocument.NewDocument(const aVersion, anEncoding: TXmlString;
|
|
aRootElementNameID: Integer): IXmlElement;
|
|
var
|
|
aChilds: TXmlNodeList;
|
|
aRoot: TXmlElement;
|
|
e: String;
|
|
begin
|
|
aChilds := GetChilds;
|
|
aChilds.Clear;
|
|
if anEncoding = '' then
|
|
e := DefaultEncoding
|
|
else
|
|
e := anEncoding;
|
|
aChilds.Insert(TXmlProcessingInstruction.Create(FNames, FNames.FXmlID,
|
|
'version="' + aVersion + '" encoding="' + e + '"'), 0);
|
|
aRoot := TXmlElement.Create(FNames, aRootElementNameID);
|
|
aChilds.Insert(aRoot, 1);
|
|
Result := aRoot;
|
|
end;
|
|
|
|
procedure TXmlDocument.Save(aStream: TStream);
|
|
var
|
|
aXml: TXmlString;
|
|
begin
|
|
aXml := Get_XML;
|
|
aStream.WriteBuffer(aXml[1], sizeof(TXmlChar)*Length(aXml));
|
|
end;
|
|
|
|
procedure TXmlDocument.Save(const aFileName: TXmlString);
|
|
var
|
|
aFile: TFileStream;
|
|
begin
|
|
aFile := TFileStream.Create(aFileName, fmCreate, fmShareDenyWrite);
|
|
try
|
|
Save(aFile);
|
|
finally
|
|
aFile.Free
|
|
end
|
|
end;
|
|
|
|
procedure TXmlDocument.SaveBinary(aStream: TStream; anOptions: LongWord);
|
|
var
|
|
aWriter: TBinXmlWriter;
|
|
begin
|
|
aWriter := TStmXmlWriter.Create(aStream, anOptions, 65536);
|
|
try
|
|
FNames.SaveBinXml(aWriter);
|
|
SaveBinXml(aWriter);
|
|
finally
|
|
aWriter.Free
|
|
end
|
|
end;
|
|
|
|
procedure TXmlDocument.SaveBinary(const aFileName: TXmlString; anOptions: LongWord);
|
|
var
|
|
aFile: TFileStream;
|
|
begin
|
|
aFile := TFileStream.Create(aFileName, fmCreate, fmShareDenyWrite);
|
|
try
|
|
SaveBinary(aFile, anOptions);
|
|
finally
|
|
aFile.Free
|
|
end
|
|
end;
|
|
|
|
procedure TXmlDocument.Set_PreserveWhiteSpace(aValue: Boolean);
|
|
begin
|
|
FPreserveWhiteSpace := aValue;
|
|
end;
|
|
|
|
procedure TXmlDocument.Set_Text(const aText: TXmlString);
|
|
var
|
|
aChilds: TXmlNodeList;
|
|
begin
|
|
aChilds := GetChilds;
|
|
aChilds.Clear;
|
|
aChilds.Insert(TXmlText.Create(FNames, aText), 0);
|
|
end;
|
|
|
|
{ TXmlProcessingInstruction }
|
|
|
|
function TXmlProcessingInstruction.AsProcessingInstruction: IXmlProcessingInstruction;
|
|
begin
|
|
Result := Self
|
|
end;
|
|
|
|
constructor TXmlProcessingInstruction.Create(aNames: TXmlNameTable;
|
|
aTargetID: Integer; const aData: TXmlString);
|
|
begin
|
|
inherited Create(aNames);
|
|
FTargetID := aTargetID;
|
|
FData := aData;
|
|
end;
|
|
|
|
function TXmlProcessingInstruction.DoCloneNode(aDeep: Boolean): IXmlNode;
|
|
begin
|
|
Result := TXmlProcessingInstruction.Create(FNames, FTargetID, FData);
|
|
end;
|
|
|
|
function TXmlProcessingInstruction.Get_NodeNameID: Integer;
|
|
begin
|
|
Result := FTargetID
|
|
end;
|
|
|
|
function TXmlProcessingInstruction.Get_NodeType: Integer;
|
|
begin
|
|
Result := NODE_PROCESSING_INSTRUCTION
|
|
end;
|
|
|
|
function TXmlProcessingInstruction.Get_Text: TXmlString;
|
|
begin
|
|
Result := FData;
|
|
end;
|
|
|
|
function TXmlProcessingInstruction.Get_XML: TXmlString;
|
|
begin
|
|
if FData = '' then
|
|
Result := '<?' + FNames.GetName(FTargetID) + GetAttrsXML + '?>'
|
|
else
|
|
Result := '<?' + FNames.GetName(FTargetID) + ' ' + FData + '?>'
|
|
end;
|
|
|
|
procedure TXmlProcessingInstruction.SetNodeNameID(aValue: Integer);
|
|
begin
|
|
FTargetID := aValue
|
|
end;
|
|
|
|
procedure TXmlProcessingInstruction.Set_Text(const aText: TXmlString);
|
|
begin
|
|
FData := aText
|
|
end;
|
|
|
|
{ TXmlStrSource }
|
|
|
|
constructor TXmlStrSource.Create(const aSource: TXmlString);
|
|
begin
|
|
inherited Create;
|
|
FSource := aSource;
|
|
FSourcePtr := PXmlChar(FSource);
|
|
FSourceEnd := FSourcePtr + Length(FSource);
|
|
if FSourcePtr = FSourceEnd then
|
|
CurChar := #0
|
|
else
|
|
CurChar := FSourcePtr^;
|
|
end;
|
|
|
|
function TXmlStrSource.EOF: Boolean;
|
|
begin
|
|
Result := FSourcePtr = FSourceEnd
|
|
end;
|
|
|
|
function TXmlStrSource.Next: Boolean;
|
|
begin
|
|
if FSourcePtr < FSourceEnd then
|
|
Inc(FSourcePtr);
|
|
if FSourcePtr = FSourceEnd then begin
|
|
Result := False;
|
|
CurChar := #0;
|
|
end
|
|
else begin
|
|
Result := True;
|
|
CurChar := FSourcePtr^;
|
|
end;
|
|
end;
|
|
|
|
{ TXmlSource }
|
|
|
|
procedure TXmlSource.NewToken;
|
|
begin
|
|
Inc(FTokenStackTop);
|
|
if FTokenStackTop < Length(FTokenStack) then begin
|
|
FToken := FTokenStack[FTokenStackTop];
|
|
FToken.Clear
|
|
end
|
|
else begin
|
|
SetLength(FTokenStack, FTokenStackTop + 1);
|
|
FToken := TXmlToken.Create;
|
|
FTokenStack[FTokenStackTop] := FToken;
|
|
end
|
|
end;
|
|
|
|
function TXmlSource.AcceptToken: TXmlString;
|
|
begin
|
|
SetLength(Result, FToken.FValuePtr - FToken.ValueStart);
|
|
if Length(Result) > 0 then
|
|
Move(FToken.ValueStart^, Result[1], Length(Result)*sizeof(TXmlChar));
|
|
DropToken;
|
|
end;
|
|
|
|
procedure TXmlSource.SkipBlanks;
|
|
begin
|
|
while not EOF and (CurChar <= ' ') do
|
|
Next;
|
|
end;
|
|
|
|
// на входе - первый символ имени
|
|
// на выходе - первый символ, который не является допустимым для имен
|
|
function TXmlSource.ExpectXmlName: TXmlString;
|
|
begin
|
|
if not NameCanBeginWith(CurChar) then
|
|
raise Exception.Create(SSimpleXmlError11);
|
|
NewToken;
|
|
AppendTokenChar(CurChar);
|
|
while Next and NameCanContain(CurChar) do
|
|
AppendTokenChar(CurChar);
|
|
Result := AcceptToken;
|
|
end;
|
|
|
|
// на входе - первый символ числа
|
|
// на выходе - первый символ, который не является допустимым для чисел
|
|
function TXmlSource.ExpectDecimalInteger: Integer;
|
|
var
|
|
s: TXmlString;
|
|
e: Integer;
|
|
begin
|
|
NewToken;
|
|
while (CurChar >= '0') and (CurChar <= '9') do begin
|
|
AppendTokenChar(CurChar);
|
|
Next;
|
|
end;
|
|
s := AcceptToken;
|
|
if Length(s) = 0 then
|
|
raise Exception.Create(SSimpleXmlError12);
|
|
Val(s, Result, e);
|
|
end;
|
|
|
|
// на входе - первый символ числа
|
|
// на выходе - первый символ, который не является допустимым для
|
|
// щестнадцатиричных чисел
|
|
function TXmlSource.ExpectHexInteger: Integer;
|
|
var
|
|
s: TXmlString;
|
|
e: Integer;
|
|
begin
|
|
NewToken;
|
|
{$IFDEF XML_WIDE_CHARS}
|
|
while (CurChar >= '0') and (CurChar <= '9') or
|
|
(CurChar >= 'A') and (CurChar <= 'F') or
|
|
(CurChar >= 'a') and (CurChar <= 'f') do begin
|
|
{$ELSE}
|
|
while CurChar in ['0'..'9', 'A'..'F', 'a'..'f'] do begin
|
|
{$ENDIF}
|
|
AppendTokenChar(CurChar);
|
|
Next;
|
|
end;
|
|
s := '$';
|
|
s := s + AcceptToken;
|
|
if Length(s) = 1 then
|
|
raise Exception.Create(SSimpleXmlError13);
|
|
Val(s, Result, e);
|
|
end;
|
|
|
|
// на входе: "&"
|
|
// на выходе: следующий за ";"
|
|
function TXmlSource.ExpectXmlEntity: TXmlChar;
|
|
var
|
|
s: TXmlString;
|
|
begin
|
|
if not Next then
|
|
raise Exception.Create(SSimpleXmlError14);
|
|
if CurChar = '#' then begin
|
|
if not Next then
|
|
raise Exception.Create(SSimpleXmlError12);
|
|
if CurChar = 'x' then begin
|
|
Next;
|
|
Result := TXmlChar(ExpectHexInteger);
|
|
end
|
|
else
|
|
Result := TXmlChar(ExpectDecimalInteger);
|
|
ExpectChar(';');
|
|
end
|
|
else begin
|
|
s := ExpectXmlName;
|
|
ExpectChar(';');
|
|
if s = 'amp' then
|
|
Result := '&'
|
|
else if s = 'quot' then
|
|
Result := '"'
|
|
else if s = 'lt' then
|
|
Result := '<'
|
|
else if s = 'gt' then
|
|
Result := '>'
|
|
else if s = 'apos' then
|
|
Result := ''''
|
|
else
|
|
raise Exception.Create(SSimpleXmlError15);
|
|
end
|
|
end;
|
|
|
|
procedure TXmlSource.ExpectChar(aChar: TXmlChar);
|
|
begin
|
|
if EOF or (CurChar <> aChar) then
|
|
raise Exception.CreateFmt(SSimpleXmlError16, [aChar]);
|
|
Next;
|
|
end;
|
|
|
|
procedure TXmlSource.ExpectText(aText: PXmlChar);
|
|
begin
|
|
while aText^ <> #0 do begin
|
|
if (CurChar <> aText^) or EOF then
|
|
raise Exception.CreateFmt(SSimpleXmlError17, [aText]);
|
|
Inc(aText);
|
|
Next;
|
|
end;
|
|
end;
|
|
|
|
// на входе: открывающая кавычка
|
|
// на выходе: символ, следующий за закрывающей кавычкой
|
|
function TXmlSource.ExpectQuotedText(aQuote: TXmlChar): TXmlString;
|
|
begin
|
|
NewToken;
|
|
Next;
|
|
while not EOF and (CurChar <> aQuote) do begin
|
|
if CurChar = '&' then
|
|
AppendTokenChar(ExpectXmlEntity)
|
|
else if CurChar = '<' then
|
|
raise Exception.Create(SSimpleXmlError18)
|
|
else begin
|
|
AppendTokenChar(CurChar);
|
|
Next;
|
|
end
|
|
end;
|
|
if EOF then
|
|
raise Exception.CreateFmt(SimpleXmlError19, [aQuote]);
|
|
Next;
|
|
Result := AcceptToken;
|
|
end;
|
|
|
|
procedure TXmlSource.ParseAttrs(aNode: TXmlNode);
|
|
var
|
|
aName: TXmlString;
|
|
aValue: TXmlString;
|
|
begin
|
|
SkipBlanks;
|
|
while not EOF and NameCanBeginWith(CurChar) do begin
|
|
aName := ExpectXmlName;
|
|
SkipBlanks;
|
|
ExpectChar('=');
|
|
SkipBlanks;
|
|
if EOF then
|
|
raise Exception.Create(SSimpleXmlError20);
|
|
if (CurChar = '''') or (CurChar = '"') then
|
|
aValue := ExpectQuotedText(CurChar)
|
|
else
|
|
raise Exception.Create(SSimpleXmlError21);
|
|
aNode.SetAttr(aName, aValue);
|
|
SkipBlanks;
|
|
end;
|
|
end;
|
|
|
|
function StrEquals(p1, p2: PXmlChar; aLen: Integer): Boolean;
|
|
begin
|
|
{$IFDEF XML_WIDE_CHARS}
|
|
while aLen > 0 do
|
|
if p1^ <> p2^ then begin
|
|
Result := False;
|
|
Exit
|
|
end
|
|
else if (p1^ = #0) or (p2^ = #0) then begin
|
|
Result := p1^ = p2^;
|
|
Exit
|
|
end
|
|
else begin
|
|
Inc(p1);
|
|
Inc(p2);
|
|
Dec(aLen);
|
|
end;
|
|
Result := True;
|
|
{$ELSE}
|
|
Result := StrLComp(p1, p2, aLen) = 0
|
|
{$ENDIF}
|
|
end;
|
|
|
|
// на входе: первый символ текста
|
|
// на выходе: символ, следующий за последним символом ограничителя
|
|
function TXmlSource.ParseTo(aText: PXmlChar): TXmlString;
|
|
var
|
|
aCheck: PXmlChar;
|
|
p: PXmlChar;
|
|
begin
|
|
NewToken;
|
|
aCheck := aText;
|
|
while not EOF do begin
|
|
if CurChar = aCheck^ then begin
|
|
Inc(aCheck);
|
|
Next;
|
|
if aCheck^ = #0 then begin
|
|
Result := AcceptToken;
|
|
Exit;
|
|
end;
|
|
end
|
|
else if aCheck = aText then begin
|
|
AppendTokenChar(CurChar);
|
|
Next;
|
|
end
|
|
else begin
|
|
p := aText + 1;
|
|
while (p < aCheck) and not StrEquals(p, aText, aCheck - p) do
|
|
Inc(p);
|
|
AppendTokenText(aText, p - aText);
|
|
if p < aCheck then
|
|
aCheck := p
|
|
else
|
|
aCheck := aText;
|
|
end;
|
|
end;
|
|
raise Exception.CreateFmt(SimpleXmlError22, [aText]);
|
|
end;
|
|
|
|
procedure TXmlSource.AppendTokenChar(aChar: TXmlChar);
|
|
begin
|
|
FToken.AppendChar(aChar);
|
|
end;
|
|
|
|
procedure TXmlSource.AppendTokenText(aText: PXmlChar; aCount: Integer);
|
|
begin
|
|
FToken.AppendText(aText, aCount)
|
|
end;
|
|
|
|
procedure TXmlSource.DropToken;
|
|
begin
|
|
Dec(FTokenStackTop);
|
|
if FTokenStackTop >= 0 then
|
|
FToken := FTokenStack[FTokenStackTop]
|
|
else
|
|
FToken := nil
|
|
end;
|
|
|
|
constructor TXmlSource.Create;
|
|
begin
|
|
inherited Create;
|
|
FTokenStackTop := -1;
|
|
end;
|
|
|
|
destructor TXmlSource.Destroy;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
for i := 0 to Length(FTokenStack) - 1 do
|
|
FTokenStack[i].Free;
|
|
inherited;
|
|
end;
|
|
|
|
{ TXmlToken }
|
|
|
|
procedure TXmlToken.AppendChar(aChar: TXmlChar);
|
|
var
|
|
aSaveLength: Integer;
|
|
begin
|
|
if FValuePtr >= FValueEnd then begin
|
|
aSaveLength := FValuePtr - FValueStart;
|
|
SetLength(FValueBuf, aSaveLength + 1);
|
|
FValueStart := PXmlChar(FValueBuf);
|
|
FValuePtr := FValueStart + aSaveLength;
|
|
FValueEnd := FValueStart + System.Length(FValueBuf);
|
|
end;
|
|
FValuePtr^ := aChar;
|
|
Inc(FValuePtr);
|
|
end;
|
|
|
|
procedure TXmlToken.AppendText(aText: PXmlChar; aCount: Integer);
|
|
var
|
|
aSaveLength: Integer;
|
|
begin
|
|
if (FValuePtr + aCount) > FValueEnd then begin
|
|
aSaveLength := FValuePtr - FValueStart;
|
|
SetLength(FValueBuf, aSaveLength + aCount);
|
|
FValueStart := PXmlChar(FValueBuf);
|
|
FValuePtr := FValueStart + aSaveLength;
|
|
FValueEnd := FValueStart + System.Length(FValueBuf);
|
|
end;
|
|
Move(aText^, FValuePtr^, aCount*sizeof(TXmlChar));
|
|
Inc(FValuePtr, aCount);
|
|
end;
|
|
|
|
procedure TXmlToken.Clear;
|
|
begin
|
|
FValuePtr := FValueStart;
|
|
end;
|
|
|
|
constructor TXmlToken.Create;
|
|
begin
|
|
inherited Create;
|
|
SetLength(FValueBuf, 32);
|
|
FValueStart := PXmlChar(FValueBuf);
|
|
FValuePtr := FValueStart;
|
|
FValueEnd := FValueStart + 32;
|
|
end;
|
|
|
|
function TXmlToken.Length: Integer;
|
|
begin
|
|
Result := FValuePtr - FValueStart;
|
|
end;
|
|
|
|
{$IFDEF XML_WIDE_CHARS}
|
|
function AnsiToUnicode(c: AnsiChar): WideChar;
|
|
begin
|
|
MultiByteToWideChar(CP_ACP, 0, @c, 1, @Result, 1);
|
|
end;
|
|
{$ENDIF}
|
|
|
|
{ TXmlStmSource }
|
|
|
|
constructor TXmlStmSource.Create(aStream: TStream; aBufSize: Integer);
|
|
var
|
|
aSize: Integer;
|
|
begin
|
|
inherited Create;
|
|
FStream := aStream;
|
|
FBufSize := aBufSize;
|
|
FBufStart := AllocMem(aBufSize);
|
|
FBufPtr := FBufStart;
|
|
FBufEnd := FBufStart;
|
|
FSize := aStream.Size;
|
|
if FSize = 0 then
|
|
CurChar := #0
|
|
else begin
|
|
if FSize < FBufSize then
|
|
aSize := FSize
|
|
else
|
|
aSize := FBufSize;
|
|
FStream.ReadBuffer(FBufStart^, aSize);
|
|
FBufEnd := FBufStart + aSize;
|
|
FBufPtr := FBufStart;
|
|
Dec(FSize, aSize);
|
|
{$IFDEF XML_WIDE_CHARS}
|
|
CurChar := AnsiToUnicode(FBufPtr^);
|
|
{$ELSE}
|
|
CurChar := FBufPtr^;
|
|
{$ENDIF}
|
|
end
|
|
end;
|
|
|
|
destructor TXmlStmSource.Destroy;
|
|
begin
|
|
FreeMem(FBufStart);
|
|
inherited;
|
|
end;
|
|
|
|
function TXmlStmSource.EOF: Boolean;
|
|
begin
|
|
Result := (FBufPtr = FBufEnd) and (FSize = 0)
|
|
end;
|
|
|
|
function TXmlStmSource.Next: Boolean;
|
|
var
|
|
aSize: Integer;
|
|
begin
|
|
if FBufPtr < FBufEnd then
|
|
Inc(FBufPtr);
|
|
if FBufPtr = FBufEnd then
|
|
if FSize = 0 then begin
|
|
Result := False;
|
|
CurChar := #0;
|
|
end
|
|
else begin
|
|
if FSize < FBufSize then
|
|
aSize := FSize
|
|
else
|
|
aSize := FBufSize;
|
|
FStream.ReadBuffer(FBufStart^, aSize);
|
|
FBufEnd := FBufStart + aSize;
|
|
FBufPtr := FBufStart;
|
|
Dec(FSize, aSize);
|
|
Result := True;
|
|
{$IFDEF XML_WIDE_CHARS}
|
|
CurChar := AnsiToUnicode(FBufPtr^);
|
|
{$ELSE}
|
|
CurChar := FBufPtr^;
|
|
{$ENDIF}
|
|
end
|
|
else begin
|
|
Result := True;
|
|
{$IFDEF XML_WIDE_CHARS}
|
|
CurChar := AnsiToUnicode(FBufPtr^);
|
|
{$ELSE}
|
|
CurChar := FBufPtr^;
|
|
{$ENDIF}
|
|
end;
|
|
end;
|
|
|
|
{ TStmXmlReader }
|
|
|
|
constructor TStmXmlReader.Create(aStream: TStream; aBufSize: Integer);
|
|
begin
|
|
inherited Create;
|
|
FStream := aStream;
|
|
FRestSize := aStream.Size - aStream.Position;
|
|
FBufSize := aBufSize;
|
|
FBufStart := AllocMem(aBufSize);
|
|
FBufEnd := FBufStart;
|
|
FBufPtr := FBufEnd;
|
|
Read(FOptions, sizeof(FOptions));
|
|
end;
|
|
|
|
destructor TStmXmlReader.Destroy;
|
|
begin
|
|
FreeMem(FBufStart);
|
|
inherited;
|
|
end;
|
|
|
|
procedure TStmXmlReader.Read(var aBuf; aSize: Integer);
|
|
var
|
|
aBufRest: Integer;
|
|
aDst: PChar;
|
|
aBufSize: Integer;
|
|
begin
|
|
if aSize > FRestSize then
|
|
raise Exception.Create(SSimpleXmlError23);
|
|
|
|
aBufRest := FBufEnd - FBufPtr;
|
|
if aSize <= aBufRest then begin
|
|
Move(FBufPtr^, aBuf, aSize);
|
|
Inc(FBufPtr, aSize);
|
|
Dec(FRestSize, aSize);
|
|
end
|
|
else begin
|
|
aDst := @aBuf;
|
|
Move(FBufPtr^, aDst^, aBufRest);
|
|
Inc(aDst, aBufRest);
|
|
FStream.ReadBuffer(aDst^, aSize - aBufRest);
|
|
Dec(FRestSize, aSize);
|
|
|
|
if FRestSize < FBufSize then
|
|
aBufSize := FRestSize
|
|
else
|
|
aBufSize := FBufSize;
|
|
FBufPtr := FBufStart;
|
|
FBufEnd := FBufStart + aBufSize;
|
|
if aBufSize > 0 then
|
|
FStream.ReadBuffer(FBufStart^, aBufSize);
|
|
end;
|
|
end;
|
|
|
|
{ TStrXmlReader }
|
|
|
|
constructor TStrXmlReader.Create(const aStr: String);
|
|
var
|
|
aSig: array [1..BinXmlSignatureSize] of Char;
|
|
begin
|
|
inherited Create;
|
|
FString := aStr;
|
|
FRestSize := Length(aStr);
|
|
FPtr := PChar(FString);
|
|
Read(aSig, BinXmlSignatureSize);
|
|
Read(FOptions, sizeof(FOptions));
|
|
end;
|
|
|
|
procedure TStrXmlReader.Read(var aBuf; aSize: Integer);
|
|
begin
|
|
if aSize > FRestSize then
|
|
raise Exception.Create(SSimpleXmlError23);
|
|
Move(FPtr^, aBuf, aSize);
|
|
Inc(FPtr, aSize);
|
|
Dec(FRestSize, aSize);
|
|
end;
|
|
|
|
{ TBinXmlReader }
|
|
|
|
function TBinXmlReader.ReadAnsiString: String;
|
|
var
|
|
aLength: LongInt;
|
|
begin
|
|
aLength := ReadLongint;
|
|
if aLength = 0 then
|
|
Result := ''
|
|
else begin
|
|
SetLength(Result, aLength);
|
|
Read(Result[1], aLength);
|
|
end
|
|
end;
|
|
|
|
function TBinXmlReader.ReadLongint: Longint;
|
|
var
|
|
b: Byte;
|
|
begin
|
|
Result := 0;
|
|
Read(Result, 1);
|
|
if Result >= $80 then
|
|
if Result = $FF then
|
|
Read(Result, sizeof(Result))
|
|
else begin
|
|
Read(b, 1);
|
|
Result := (Result and $7F) shl 8 or b;
|
|
end
|
|
end;
|
|
|
|
procedure TBinXmlReader.ReadVariant(var v: TVarData);
|
|
var
|
|
aDataType: Word;
|
|
aSize: Longint;
|
|
p: Pointer;
|
|
begin
|
|
VarClear(Variant(v));
|
|
aDataType := ReadLongint;
|
|
case aDataType of
|
|
varEmpty: ;
|
|
varNull: ;
|
|
varSmallint:
|
|
Read(v.VSmallint, sizeof(SmallInt));
|
|
varInteger:
|
|
Read(v.VInteger, sizeof(Integer));
|
|
varSingle:
|
|
Read(v.VSingle, sizeof(Single));
|
|
varDouble:
|
|
Read(v.VDouble, sizeof(Double));
|
|
varCurrency:
|
|
Read(v.VCurrency, sizeof(Currency));
|
|
varDate:
|
|
Read(v.VDate, sizeof(TDateTime));
|
|
varOleStr:
|
|
Variant(v) := ReadWideString;
|
|
varBoolean:
|
|
Read(v.VBoolean, sizeof(WordBool));
|
|
varShortInt:
|
|
Read(v.VShortInt, sizeof(ShortInt));
|
|
varByte:
|
|
Read(v.VByte, sizeof(Byte));
|
|
varWord:
|
|
Read(v.VWord, sizeof(Word));
|
|
varLongWord:
|
|
Read(v.VLongWord, sizeof(LongWord));
|
|
varInt64:
|
|
Read(v.VInt64, sizeof(Int64));
|
|
varString:
|
|
Variant(v) := ReadAnsiString;
|
|
varArray + varByte:
|
|
begin
|
|
aSize := ReadLongint;
|
|
Variant(v) := VarArrayCreate([0, aSize - 1], varByte);
|
|
p := VarArrayLock(Variant(v));
|
|
try
|
|
Read(p^, aSize);
|
|
finally
|
|
VarArrayUnlock(Variant(v))
|
|
end
|
|
end;
|
|
else
|
|
raise Exception.Create(SSimpleXmlError24);
|
|
end;
|
|
v.VType := aDataType;
|
|
end;
|
|
|
|
function TBinXmlReader.ReadWideString: WideString;
|
|
var
|
|
aLength: LongInt;
|
|
begin
|
|
aLength := ReadLongint;
|
|
if aLength = 0 then
|
|
Result := ''
|
|
else begin
|
|
SetLength(Result, aLength);
|
|
Read(Result[1], aLength*sizeof(WideChar));
|
|
end
|
|
end;
|
|
|
|
function TBinXmlReader.ReadXmlString: TXmlString;
|
|
begin
|
|
if (FOptions and BINXML_USE_WIDE_CHARS) <> 0 then
|
|
Result := ReadWideString
|
|
else
|
|
Result := ReadAnsiString
|
|
end;
|
|
|
|
{ TStmXmlWriter }
|
|
|
|
constructor TStmXmlWriter.Create(aStream: TStream; anOptions: LongWord;
|
|
aBufSize: Integer);
|
|
begin
|
|
inherited Create;
|
|
FStream := aStream;
|
|
FOptions := anOptions;
|
|
FBufSize := aBufSize;
|
|
FBufStart := AllocMem(aBufSize);
|
|
FBufEnd := FBufStart + aBufSize;
|
|
FBufPtr := FBufStart;
|
|
Write(BinXmlSignature[1], BinXmlSignatureSize);
|
|
Write(FOptions, sizeof(FOptions));
|
|
end;
|
|
|
|
destructor TStmXmlWriter.Destroy;
|
|
begin
|
|
if FBufPtr > FBufStart then
|
|
FStream.WriteBuffer(FBufStart^, FBufPtr - FBufStart);
|
|
FreeMem(FBufStart);
|
|
inherited;
|
|
end;
|
|
|
|
procedure TStmXmlWriter.Write(const aBuf; aSize: Integer);
|
|
var
|
|
aBufRest: Integer;
|
|
begin
|
|
aBufRest := FBufEnd - FBufPtr;
|
|
if aSize <= aBufRest then begin
|
|
Move(aBuf, FBufPtr^, aSize);
|
|
Inc(FBufPtr, aSize);
|
|
end
|
|
else begin
|
|
if FBufPtr > FBufStart then begin
|
|
FStream.WriteBuffer(FBufStart^, FBufPtr - FBufStart);
|
|
FBufPtr := FBufStart;
|
|
end;
|
|
FStream.WriteBuffer(aBuf, aSize);
|
|
end
|
|
end;
|
|
|
|
{ TStrXmlWriter }
|
|
|
|
constructor TStrXmlWriter.Create(anOptions: LongWord; aBufSize: Integer);
|
|
begin
|
|
inherited Create;
|
|
FData := '';
|
|
FOptions := anOptions;
|
|
FBufSize := aBufSize;
|
|
FBufStart := AllocMem(aBufSize);
|
|
FBufEnd := FBufStart + aBufSize;
|
|
FBufPtr := FBufStart;
|
|
Write(BinXmlSignature[1], BinXmlSignatureSize);
|
|
Write(FOptions, sizeof(FOptions));
|
|
end;
|
|
|
|
destructor TStrXmlWriter.Destroy;
|
|
begin
|
|
FreeMem(FBufStart);
|
|
inherited;
|
|
end;
|
|
|
|
procedure TStrXmlWriter.FlushBuf;
|
|
var
|
|
aPrevSize: Integer;
|
|
aSize: Integer;
|
|
begin
|
|
aPrevSize := Length(FData);
|
|
aSize := FBufPtr - FBufStart;
|
|
SetLength(FData, aPrevSize + aSize);
|
|
Move(FBufStart^, FData[aPrevSize + 1], aSize);
|
|
FBufPtr := FBufStart;
|
|
end;
|
|
|
|
procedure TStrXmlWriter.Write(const aBuf; aSize: Integer);
|
|
var
|
|
aBufRest: Integer;
|
|
aPrevSize: Integer;
|
|
aBufSize: Integer;
|
|
begin
|
|
aBufRest := FBufEnd - FBufPtr;
|
|
if aSize <= aBufRest then begin
|
|
Move(aBuf, FBufPtr^, aSize);
|
|
Inc(FBufPtr, aSize);
|
|
end
|
|
else begin
|
|
aPrevSize := Length(FData);
|
|
aBufSize := FBufPtr - FBufStart;
|
|
SetLength(FData, aPrevSize + aBufSize + aSize);
|
|
if aBufSize > 0 then
|
|
Move(FBufStart^, FData[aPrevSize + 1], aBufSize);
|
|
Move(aBuf, FData[aPrevSize + aBufSize + 1], aSize);
|
|
FBufPtr := FBufStart;
|
|
end
|
|
end;
|
|
|
|
{ TBinXmlWriter }
|
|
|
|
procedure TBinXmlWriter.WriteAnsiString(const aValue: String);
|
|
var
|
|
aLength: LongInt;
|
|
begin
|
|
aLength := Length(aValue);
|
|
WriteLongint(aLength);
|
|
if aLength > 0 then
|
|
Write(aValue[1], aLength);
|
|
end;
|
|
|
|
procedure TBinXmlWriter.WriteLongint(aValue: Longint);
|
|
var
|
|
b: array [0..1] of Byte;
|
|
begin
|
|
if aValue < 0 then begin
|
|
b[0] := $FF;
|
|
Write(b[0], 1);
|
|
Write(aValue, SizeOf(aValue));
|
|
end
|
|
else if aValue < $80 then
|
|
Write(aValue, 1)
|
|
else if aValue <= $7FFF then begin
|
|
b[0] := (aValue shr 8) or $80;
|
|
b[1] := aValue and $FF;
|
|
Write(b, 2);
|
|
end
|
|
else begin
|
|
b[0] := $FF;
|
|
Write(b[0], 1);
|
|
Write(aValue, SizeOf(aValue));
|
|
end;
|
|
end;
|
|
|
|
procedure TBinXmlWriter.WriteVariant(const v: TVarData);
|
|
var
|
|
aSize: Integer;
|
|
p: Pointer;
|
|
begin
|
|
WriteLongint(v.VType);
|
|
case v.VType of
|
|
varEmpty: ;
|
|
varNull: ;
|
|
varSmallint:
|
|
Write(v.VSmallint, sizeof(SmallInt));
|
|
varInteger:
|
|
Write(v.VInteger, sizeof(Integer));
|
|
varSingle:
|
|
Write(v.VSingle, sizeof(Single));
|
|
varDouble:
|
|
Write(v.VDouble, sizeof(Double));
|
|
varCurrency:
|
|
Write(v.VCurrency, sizeof(Currency));
|
|
varDate:
|
|
Write(v.VDate, sizeof(TDateTime));
|
|
varOleStr:
|
|
WriteWideString(Variant(v));
|
|
varBoolean:
|
|
Write(v.VBoolean, sizeof(WordBool));
|
|
varShortInt:
|
|
Write(v.VShortInt, sizeof(ShortInt));
|
|
varByte:
|
|
Write(v.VByte, sizeof(Byte));
|
|
varWord:
|
|
Write(v.VWord, sizeof(Word));
|
|
varLongWord:
|
|
Write(v.VLongWord, sizeof(LongWord));
|
|
varInt64:
|
|
Write(v.VInt64, sizeof(Int64));
|
|
varString:
|
|
WriteAnsiString(Variant(v));
|
|
varArray + varByte:
|
|
begin
|
|
aSize := VarArrayHighBound(Variant(v), 1) - VarArrayLowBound(Variant(v), 1) + 1;
|
|
WriteLongint(aSize);
|
|
p := VarArrayLock(Variant(v));
|
|
try
|
|
Write(p^, aSize);
|
|
finally
|
|
VarArrayUnlock(Variant(v))
|
|
end
|
|
end;
|
|
else
|
|
raise Exception.Create(SSimpleXmlError25);
|
|
end;
|
|
end;
|
|
|
|
procedure TBinXmlWriter.WriteWideString(const aValue: WideString);
|
|
var
|
|
aLength: LongInt;
|
|
begin
|
|
aLength := Length(aValue);
|
|
WriteLongint(aLength);
|
|
if aLength > 0 then
|
|
Write(aValue[1], aLength*sizeof(WideChar));
|
|
end;
|
|
|
|
|
|
procedure TBinXmlWriter.WriteXmlString(const aValue: TXmlString);
|
|
begin
|
|
if (FOptions and BINXML_USE_WIDE_CHARS) <> 0 then
|
|
WriteWideString(aValue)
|
|
else
|
|
WriteAnsiString(aValue)
|
|
end;
|
|
|
|
var
|
|
DefaultNameTableImpl: TXmlNameTable = nil;
|
|
|
|
function CreateXmlElement(const aName: TXmlString; const aNameTable: IXmlNameTable): IXmlElement;
|
|
var
|
|
aNameTableImpl: TXmlNameTable;
|
|
begin
|
|
if Assigned(aNameTable) then
|
|
aNameTableImpl := aNameTable.GetObject as TXmlNameTable
|
|
else
|
|
aNameTableImpl := DefaultNameTableImpl;
|
|
Result := TXmlElement.Create(aNameTableImpl, aNameTableImpl.GetID(aName));
|
|
end;
|
|
|
|
function CreateXmlDocument(
|
|
const aRootElementName: String;
|
|
const aVersion: String;
|
|
const anEncoding: String;
|
|
const aNames: IXmlNameTable): IXmlDocument;
|
|
var
|
|
aNameTable: TXmlNameTable;
|
|
s: String;
|
|
begin
|
|
if Assigned(aNames) then
|
|
aNameTable := aNames.GetObject as TXmlNameTable
|
|
else
|
|
aNameTable := DefaultNameTableImpl;
|
|
if anEncoding = '' then
|
|
s := DefaultEncoding
|
|
else
|
|
s := anEncoding;
|
|
Result := TXmlDocument.Create(aNameTable);
|
|
if aRootElementName <> '' then
|
|
Result.NewDocument(aVersion, anEncoding, aRootElementName);
|
|
end;
|
|
|
|
function LoadXmlDocumentFromXML(const aXML: TXmlString): IXmlDocument;
|
|
begin
|
|
Result := TXmlDocument.Create(DefaultNameTableImpl);
|
|
Result.LoadXML(aXML);
|
|
end;
|
|
|
|
function LoadXmlDocumentFromBinaryXML(const aXML: String): IXmlDocument;
|
|
begin
|
|
Result := TXmlDocument.Create(DefaultNameTableImpl);
|
|
Result.LoadBinaryXML(aXML);
|
|
end;
|
|
|
|
function LoadXmlDocument(aStream: TStream): IXmlDocument;
|
|
begin
|
|
Result := TXmlDocument.Create(DefaultNameTableImpl);
|
|
Result.Load(aStream);
|
|
end;
|
|
|
|
function LoadXmlDocument(const aFileName: TXmlString): IXmlDocument; overload;
|
|
begin
|
|
Result := TXmlDocument.Create(DefaultNameTableImpl);
|
|
Result.Load(aFileName);
|
|
end;
|
|
|
|
function LoadXmlDocument(aResType, aResName: PChar): IXmlDocument; overload;
|
|
begin
|
|
Result := TXmlDocument.Create(DefaultNameTableImpl);
|
|
Result.LoadResource(aResType, aResName);
|
|
end;
|
|
|
|
initialization
|
|
DefaultNameTableImpl := TXmlNameTable.Create(4096);
|
|
DefaultNameTable := DefaultNameTableImpl;
|
|
finalization
|
|
DefaultNameTable := nil;
|
|
DefaultNameTableImpl := nil;
|
|
end.
|