| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431 | {    This file is part of the Free Component Library    Implementation of TXMLConfig class    Copyright (c) 1999 - 2005 by Sebastian Guenther, [email protected]    See the file COPYING.FPC, included in this distribution,    for details about the copyright.    This program is distributed in the hope that it will be useful,    but WITHOUT ANY WARRANTY; without even the implied warranty of    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. **********************************************************************}{  TXMLConfig enables applications to use XML files for storing their  configuration data}{$MODE objfpc}{$H+}unit XMLCfg;interface{off $DEFINE MEM_CHECK}uses  {$IFDEF MEM_CHECK}MemCheck,{$ENDIF}  SysUtils, Classes, DOM, XMLRead, XMLWrite;resourcestring  SMissingPathName = 'A part of the pathname is invalid (missing)';  SEscapingNecessary = 'Invalid pathname, escaping must be enabled';  SWrongRootName = 'XML file has wrong root element name';type  EXMLConfigError = class(Exception);  {"APath" is the path and name of a value: A XML configuration file is   hierachical. "/" is the path delimiter, the part after the last "/"   is the name of the value. The path components will be mapped to XML   elements, the name will be an element attribute.}  TXMLConfig = class(TComponent)  private    FFilename: String;    FStartEmpty: Boolean;    FUseEscaping: Boolean;    FRootName: DOMString;    procedure SetFilename(const AFilename: String; ForceReload: Boolean);    procedure SetFilename(const AFilename: String);    procedure SetStartEmpty(AValue: Boolean);    procedure SetRootName(const AValue: DOMString);  protected    Doc: TXMLDocument;    FModified: Boolean;    procedure Loaded; override;    function FindNode(const APath: String; PathHasValue: boolean): TDomNode;    function Escape(const s: String): String;  public    constructor Create(AOwner: TComponent); override;    destructor Destroy; override;    procedure Clear;    procedure Flush;    // Writes the XML file    function  GetValue(const APath, ADefault: String): String;    function  GetValue(const APath: String; ADefault: Integer): Integer;    function  GetValue(const APath: String; ADefault: Boolean): Boolean;    procedure SetValue(const APath, AValue: String);    procedure SetDeleteValue(const APath, AValue, DefValue: String);    procedure SetValue(const APath: String; AValue: Integer);    procedure SetDeleteValue(const APath: String; AValue, DefValue: Integer);    procedure SetValue(const APath: String; AValue: Boolean);    procedure SetDeleteValue(const APath: String; AValue, DefValue: Boolean);    procedure DeletePath(const APath: string);    procedure DeleteValue(const APath: string);    property Modified: Boolean read FModified;  published    property Filename: String read FFilename write SetFilename;    property StartEmpty: Boolean read FStartEmpty write SetStartEmpty;    property UseEscaping: Boolean read FUseEscaping write FUseEscaping      default True;    property RootName: DOMString read FRootName write SetRootName;  end;// ===================================================================implementationconstructor TXMLConfig.Create(AOwner: TComponent);begin  inherited Create(AOwner);  FUseEscaping := True;  FRootName := 'CONFIG';  Doc := TXMLDocument.Create;  Doc.AppendChild(Doc.CreateElement(RootName));end;destructor TXMLConfig.Destroy;begin  if Assigned(Doc) then  begin    Flush;    Doc.Free;  end;  inherited Destroy;end;procedure TXMLConfig.Clear;begin  Doc.ReplaceChild(Doc.CreateElement(RootName), Doc.DocumentElement);end;procedure TXMLConfig.Flush;begin  if Modified then  begin    WriteXMLFile(Doc, Filename);    FModified := False;  end;end;function TXMLConfig.GetValue(const APath, ADefault: String): String;var  Node, Child, Attr: TDOMNode;  NodeName: String;  PathLen: integer;  StartPos, EndPos: integer;begin  Result := ADefault;  PathLen := Length(APath);  Node := Doc.DocumentElement;  StartPos := 1;  while True do  begin    EndPos := StartPos;    while (EndPos <= PathLen) and (APath[EndPos] <> '/') do      Inc(EndPos);    if EndPos > PathLen then      break;    SetLength(NodeName, EndPos - StartPos);    Move(APath[StartPos], NodeName[1], EndPos - StartPos);    StartPos := EndPos + 1;    Child := Node.FindNode(Escape(NodeName));    if not Assigned(Child) then      exit;    Node := Child;  end;  if StartPos > PathLen then    exit;  SetLength(NodeName, PathLen - StartPos + 1);  Move(APath[StartPos], NodeName[1], Length(NodeName));  Attr := Node.Attributes.GetNamedItem(Escape(NodeName));  if Assigned(Attr) then    Result := Attr.NodeValue;end;function TXMLConfig.GetValue(const APath: String; ADefault: Integer): Integer;begin  Result := StrToIntDef(GetValue(APath, IntToStr(ADefault)),ADefault);end;function TXMLConfig.GetValue(const APath: String; ADefault: Boolean): Boolean;var  s: String;begin  if ADefault then    s := 'True'  else    s := 'False';  s := GetValue(APath, s);  if AnsiCompareText(s, 'TRUE')=0 then    Result := True  else if AnsiCompareText(s, 'FALSE')=0 then    Result := False  else    Result := ADefault;end;procedure TXMLConfig.SetValue(const APath, AValue: String);var  Node, Child: TDOMNode;  NodeName: String;  PathLen: integer;  StartPos, EndPos: integer;begin  Node := Doc.DocumentElement;  PathLen := Length(APath);  StartPos:=1;  while True do  begin    EndPos := StartPos;    while (EndPos <= PathLen) and (APath[EndPos] <> '/') do      Inc(EndPos);    if EndPos > PathLen then      break;    SetLength(NodeName, EndPos - StartPos);    Move(APath[StartPos], NodeName[1], EndPos - StartPos);    StartPos := EndPos + 1;    NodeName := Escape(NodeName);    Child := Node.FindNode(NodeName);    if not Assigned(Child) then    begin      Child := Doc.CreateElement(NodeName);      Node.AppendChild(Child);    end;    Node := Child;  end;  if StartPos > PathLen then    exit;  SetLength(NodeName, PathLen - StartPos + 1);  Move(APath[StartPos], NodeName[1], Length(NodeName));  NodeName := Escape(NodeName);  if (not Assigned(TDOMElement(Node).GetAttributeNode(NodeName))) or    (TDOMElement(Node)[NodeName] <> AValue) then  begin    TDOMElement(Node)[NodeName] := AValue;    FModified := True;  end;end;procedure TXMLConfig.SetDeleteValue(const APath, AValue, DefValue: String);begin  if AValue = DefValue then    DeleteValue(APath)  else    SetValue(APath, AValue);end;procedure TXMLConfig.SetValue(const APath: String; AValue: Integer);begin  SetValue(APath, IntToStr(AValue));end;procedure TXMLConfig.SetDeleteValue(const APath: String; AValue,  DefValue: Integer);begin  if AValue = DefValue then    DeleteValue(APath)  else    SetValue(APath, AValue);end;procedure TXMLConfig.SetValue(const APath: String; AValue: Boolean);begin  if AValue then    SetValue(APath, 'True')  else    SetValue(APath, 'False');end;procedure TXMLConfig.SetDeleteValue(const APath: String; AValue,  DefValue: Boolean);begin  if AValue = DefValue then    DeleteValue(APath)  else    SetValue(APath,AValue);end;procedure TXMLConfig.DeletePath(const APath: string);var  Node: TDomNode;begin  Node := FindNode(APath, False);  if (Node = nil) or (Node.ParentNode = nil) then    exit;  Node.ParentNode.RemoveChild(Node);  FModified := True;end;procedure TXMLConfig.DeleteValue(const APath: string);var  Node: TDomNode;  StartPos: integer;  NodeName: string;begin  Node := FindNode(APath, True);  if not Assigned(Node) then    exit;  StartPos := Length(APath);  while (StartPos > 0) and (APath[StartPos] <> '/') do   Dec(StartPos);  NodeName := Escape(Copy(APath, StartPos+1, Length(APath) - StartPos));  if (not Assigned(TDOMElement(Node).GetAttributeNode(NodeName))) then    exit;  TDOMElement(Node).RemoveAttribute(NodeName);  FModified := True;end;procedure TXMLConfig.Loaded;begin  inherited Loaded;  if Length(Filename) > 0 then    SetFilename(Filename);              // Load the XML config fileend;function TXMLConfig.FindNode(const APath: String;  PathHasValue: boolean): TDomNode;var  NodePath: String;  StartPos, EndPos: integer;  PathLen: integer;begin  Result := Doc.DocumentElement;  PathLen := Length(APath);  StartPos := 1;  while Assigned(Result) do  begin    EndPos := StartPos;    while (EndPos <= PathLen) and (APath[EndPos] <> '/') do      Inc(EndPos);    if (EndPos > PathLen) and PathHasValue then      exit;    if EndPos = StartPos then      break;    SetLength(NodePath, EndPos - StartPos);    Move(APath[StartPos], NodePath[1], Length(NodePath));    Result := Result.FindNode(Escape(NodePath));    StartPos := EndPos + 1;    if StartPos > PathLen then      exit;  end;  Result := nil;end;function TXMLConfig.Escape(const s: String): String;const  AllowedChars = ['A'..'Z', 'a'..'z', '0'..'9', '.', '-', '_'];var  EscapingNecessary: Boolean;  i: Integer;begin  if Length(s) < 1 then    raise EXMLConfigError.Create(SMissingPathName);  if not (s[1] in ['A'..'Z', 'a'..'z', '_']) then    EscapingNecessary := True  else  begin    EscapingNecessary := False;    for i := 2 to Length(s) do      if not (s[i] in AllowedChars) then      begin        EscapingNecessary := True;	exit;      end;  end;  if EscapingNecessary then    if UseEscaping then    begin      Result := '_';      for i := 1 to Length(s) do        if s[i] in (AllowedChars - ['_']) then	  Result := Result + s[i]	else	  Result := Result + '_' + IntToHex(Ord(s[i]), 2);    end else      raise EXMLConfigError.Create(SEscapingNecessary)  else	// No escaping necessary    Result := s;end;procedure TXMLConfig.SetFilename(const AFilename: String; ForceReload: Boolean);begin  {$IFDEF MEM_CHECK}CheckHeapWrtMemCnt('TXMLConfig.SetFilename A '+AFilename);{$ENDIF}  if (not ForceReload) and (FFilename = AFilename) then    exit;  FFilename := AFilename;  if csLoading in ComponentState then    exit;  Flush;  FreeAndNil(Doc);  if FileExists(AFilename) and (not FStartEmpty) then    ReadXMLFile(Doc, AFilename);  if not Assigned(Doc) then    Doc := TXMLDocument.Create;  if not Assigned(Doc.DocumentElement) then    Doc.AppendChild(Doc.CreateElement(RootName))  else    if Doc.DocumentElement.NodeName <> RootName then      raise EXMLConfigError.Create('XML file has wrong root element name');  {$IFDEF MEM_CHECK}CheckHeapWrtMemCnt('TXMLConfig.SetFilename END');{$ENDIF}end;procedure TXMLConfig.SetFilename(const AFilename: String);begin  SetFilename(AFilename, False);end;procedure TXMLConfig.SetRootName(const AValue: DOMString);var  Cfg: TDOMElement;begin  if AValue <> RootName then  begin    FRootName := AValue;    Cfg := Doc.CreateElement(AValue);    while Assigned(Doc.DocumentElement.FirstChild) do      Cfg.AppendChild(Doc.DocumentElement.FirstChild);    Doc.ReplaceChild(Cfg, Doc.DocumentElement);    FModified := True;  end;end;procedure TXMLConfig.SetStartEmpty(AValue: Boolean);begin  if AValue <> StartEmpty then  begin    FStartEmpty := AValue;    if (not AValue) and not Modified then      SetFilename(Filename, True);  end;end;end.
 |