| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500 | /* * Copyright (C)2005-2019 Haxe Foundation * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. */package cpp;enum abstract XmlType(Int) {   /**      Represents an XML element type.   **/   var Element = 0;   /**      Represents XML parsed character data type.   **/   var PCData = 1;   /**      Represents XML character data type.   **/   var CData = 2;   /**      Represents an XML comment type.   **/   var Comment = 3;   /**      Represents an XML doctype element type.   **/   var DocType = 4;   /**       Represents an XML processing instruction type.   **/   var ProcessingInstruction = 5;   /**      Represents an XML document type.   **/   var Document = 6;}class NativeXmlState{   var cur : Xml;   public function new(x:Xml)   {      x._children = new Array<Xml>();      cur = x;   }   @:keep   public function xml(name:String, att:Dynamic<String>)   {      var x = new Xml();      x._parent = cur;      x.nodeType = Xml.Element;      x._nodeName = name;      x._attributes = att;      x._children = new Array<Xml>();      cur.addChild(x);      cur = x;   }   @:keep   public function cdata(text:String)   {      var x = new Xml();      x._parent = cur;      x.nodeType = Xml.CData;      x._nodeValue = text;      cur.addChild(x);   }   @:keep   public function pcdata(text:String)   {      var x = new Xml();      x._parent = cur;      x.nodeType = Xml.PCData;      x._nodeValue = text;      cur.addChild(x);   }   @:keep   public function comment(text:String)   {      var x = new Xml();      x._parent = cur;      if( text.length>1 && StringTools.fastCodeAt(text,0) == 63 )      {         x.nodeType = Xml.ProcessingInstruction;         text = text.substr(1, text.length - 2);      }      else      {         x.nodeType = Xml.Comment;      }      x._nodeValue = text;      cur.addChild(x);   }   @:keep   public function doctype(text:String)   {      var x = new Xml();      x._parent = cur;      x.nodeType = Xml.DocType;      x._nodeValue = text.substr(1);      cur.addChild(x);   }   @:keep   public function done()   {     cur = cur._parent;   }}private class NativeXmlIterator{   var cur = 0;   var children:Array<Xml>;   public function new(inChildren:Array<Xml>)   {      children = inChildren;      cur = 0;   }   public function hasNext() : Bool   {      var k = cur;      var l = children.length;      while( k < l )      {         if (children[k].nodeType == Xml.Element)            break;         k += 1;      }      cur = k;      return k < l;   }   public function next():Xml   {      var k = cur;      var l = children.length;      while( k < l )      {         var n = children[k];         k += 1;         if( n.nodeType == Xml.Element )         {            cur = k;            return n;         }      }      return null;   }}private class NativeXmlNamedIterator{   var cur = 0;   var children:Array<Xml>;   var name:String;   public function new(inChildren:Array<Xml>, inName:String)   {      children = inChildren;      name = inName;      cur = 0;   }   public function hasNext() : Bool   {      var k = cur;      var l = children.length;      while( k < l )      {         var n = children[k];         if( n.nodeType == Xml.Element && n._nodeName == name )            break;         k++;     }     cur = k;     return k < l;   }   public function next():Xml   {      var k = cur;      var l = children.length;      while( k < l )      {         var n = children[k];         k++;         if( n.nodeType == Xml.Element && n._nodeName == name ) {            cur = k;            return n;         }      }      return null;   }}@:cppInclude("./NativeXmlImport.cpp")@:allow(cpp.NativeXmlState) @:allow(cpp.NativeXmlIterator) @:allow(cpp.NativeXmlNamedIterator)class Xml {   static inline var Element = XmlType.Element;   static inline var PCData = XmlType.PCData;   static inline var CData = XmlType.CData;   static inline var Comment = XmlType.Comment;   static inline var DocType = XmlType.DocType;   static inline var ProcessingInstruction = XmlType.ProcessingInstruction;   static inline var Document = XmlType.Document;   private var _nodeName : String;   private var _nodeValue : String;   private var _attributes : Dynamic<String>;   private var _children : Array<Xml>;   private var _parent : Xml;   function new() : Void {   }   @:native("parse_xml")   extern static function parse_xml(str:String, state:NativeXmlState);   public static function parse( str : String ) : Xml   {      var x = new Xml();      var state = new NativeXmlState(x);      parse_xml(str,state);      x.nodeType = Xml.Document;      return x;   }   public static function createElement( name : String ) : Xml {      var r = new Xml();      r.nodeType = Xml.Element;      r._nodeName = name;      r._attributes = null;      r._children = new Array();      return r;   }   public static function createPCData( data : String ) : Xml {      var r = new Xml();      r.nodeType = Xml.PCData;      r._nodeValue = data;      return r;   }   public static function createCData( data : String ) : Xml {      var r = new Xml();      r.nodeType = Xml.CData;      r._nodeValue = data;      return r;   }   public static function createComment( data : String ) : Xml {      var r = new Xml();      r.nodeType = Xml.Comment;      r._nodeValue = data;      return r;   }   public static function createDocType( data : String ) : Xml {      var r = new Xml();      r.nodeType = Xml.DocType;      r._nodeValue = data;      return r;   }   public static function createProcessingInstruction( data : String ) : Xml {      var r = new Xml();      r.nodeType = Xml.ProcessingInstruction;      r._nodeValue = data;      return r;   }   public static function createDocument() : Xml {      var r = new Xml();      r.nodeType = Xml.Document;      r._children = new Array();      return r;   }   public var nodeType(default,null) : XmlType;   public var nodeName(get,set) : String;   public var nodeValue(get,set) : String;   private function get_nodeName() : String {      if( nodeType != Xml.Element )         throw "bad nodeType";      return _nodeName;   }   private function set_nodeName( n : String ) : String {      if( nodeType != Xml.Element )         throw "bad nodeType";      return _nodeName = n;   }   private function get_nodeValue() : String {      if( nodeType == Xml.Element || nodeType == Xml.Document )         throw "bad nodeType";      return _nodeValue;   }   private function set_nodeValue( v : String ) : String {      if( nodeType == Xml.Element || nodeType == Xml.Document )         throw "bad nodeType";      return _nodeValue = v;   }   public var parent(get,null) : Xml;   private function get_parent() : Xml {      return _parent;   }   public function get( att : String ) : String {      if( nodeType != Xml.Element )         throw "bad nodeType";      return Reflect.field( _attributes, att );   }   public function set( att : String, value : String ) : Void {      if( nodeType != Xml.Element )         throw "bad nodeType";      if (_attributes==null)         _attributes = {};      Reflect.setField (_attributes, att, value );      return;   }   public function remove( att : String ) : Void{      if( nodeType != Xml.Element )         throw "bad nodeType";      Reflect.deleteField( _attributes, att );      return;   }   public function exists( att : String ) : Bool {      if( nodeType != Xml.Element )         throw "bad nodeType";      return Reflect.hasField( _attributes, att );   }   public function attributes() : Iterator<String> {      if( nodeType != Xml.Element )         throw "bad nodeType";      return Reflect.fields( _attributes ).iterator();   }   public function iterator() : Iterator<Xml> {      if( _children == null )         throw "bad nodetype";      return untyped _children.iterator();   }   public function elements(): Iterator<Xml>   {      if( _children == null )         throw "bad nodetype";      return new NativeXmlIterator(_children);   }   public function elementsNamed( name : String ) : Iterator<Xml>   {      if( _children == null )         throw "bad nodetype";      return new NativeXmlNamedIterator(_children,name);   }   public function firstChild() : Xml {      if( _children == null )         throw "bad nodetype";      return _children[0];   }   public function firstElement() : Xml {      if( _children == null )         throw "bad nodetype";      for( cur in 0..._children.length ) {         var n:Xml = _children[cur];         if( n.nodeType == Xml.Element )            return n;      }      return null;   }   public function addChild( x : Xml ) : Void {      if( _children == null )         throw "bad nodetype";      if( x._parent != null ) x._parent._children.remove(x);      x._parent = this;      _children.push( x );      return;   }   public function removeChild( x : Xml ) : Bool {      if( _children == null )         throw "bad nodetype";      var b = _children.remove( x );      if( b ) x._parent = null;      return b;   }   public function insertChild( x : Xml, pos : Int ) : Void {      if( _children == null )         throw "bad nodetype";      if( x._parent != null ) x._parent._children.remove(x);      x._parent = this;      _children.insert( pos, x );      return;   }   public function toString() : String {      var s = new StringBuf();      toStringRec(s);      return s.toString();   }   private function toStringRec(s: StringBuf) : Void {      switch( nodeType ) {      case Xml.Document:         for( x in _children )            x.toStringRec(s);      case Xml.Element:         s.addChar("<".code);         s.add(_nodeName);         for( k in Reflect.fields(_attributes) ) {            s.addChar(" ".code);            s.add(k);            s.addChar("=".code);            s.addChar("\"".code);            s.add(Reflect.field(_attributes,k));            s.addChar("\"".code);         }         if( _children.length == 0 ) {            s.addChar("/".code);            s.addChar(">".code);            return;         }         s.addChar(">".code);         for( x in _children )            x.toStringRec(s);         s.addChar("<".code);         s.addChar("/".code);         s.add(_nodeName);         s.addChar(">".code);      case Xml.PCData:         s.add(StringTools.htmlEscape(_nodeValue));      case Xml.CData:         s.add("<![CDATA[");         s.add(_nodeValue);         s.add("]]>");      case Xml.Comment:         s.add("<!--");         s.add(_nodeValue);         s.add("-->");      case Xml.DocType:         s.add("<!DOCTYPE ");         s.add(_nodeValue);         s.add(">");      case Xml.ProcessingInstruction:         s.add("<?");         s.add(_nodeValue);         s.add("?>");      }   }}
 |