123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421 |
- /*
- * Copyright (c) 2005, The haXe Project Contributors
- * All rights reserved.
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions are met:
- *
- * - Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * - Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- *
- * THIS SOFTWARE IS PROVIDED BY THE HAXE PROJECT CONTRIBUTORS "AS IS" AND ANY
- * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
- * DISCLAIMED. IN NO EVENT SHALL THE HAXE PROJECT CONTRIBUTORS BE LIABLE FOR
- * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
- * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
- * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
- * DAMAGE.
- */
- package js;
- import Xml;
- class JsXml__ {
- static var enode = ~/^<([a-zA-Z0-9:_-]+)/;
- static var ecdata = ~/^<!\[CDATA\[/i;
- static var edoctype = ~/^<!DOCTYPE/i;
- static var eend = ~/^<\/([a-zA-Z0-9:_-]+)>/;
- static var epcdata = ~/^[^<]+/;
- static var ecomment = ~/^<!--/;
- static var eprolog = ~/^<\?[^\?]+\?>/;
- static var eattribute = ~/^\s*([a-zA-Z0-9:_-]+)\s*=\s*(["'])([^\2]*?)\2/; //"
- static var eclose = ~/^[ \r\n\t]*(>|(\/>))/;
- static var ecdata_end = ~/\]\]>/;
- static var edoctype_elt = ~/[\[|\]>]/;
- static var ecomment_end = ~/-->/;
- public var nodeType(default,null) : XmlType;
- public var nodeName(getNodeName,setNodeName) : String;
- public var nodeValue(getNodeValue,setNodeValue) : String;
- public var parent(getParent,null) : JsXml__;
- var _nodeName : String;
- var _nodeValue : String;
- var _attributes : Hash<String>;
- var _children : Array<JsXml__>;
- var _parent : JsXml__;
- public static function parse( str : String ) : Xml {
- var rules = [enode,epcdata,eend,ecdata,edoctype,ecomment,eprolog];
- var nrules = rules.length;
- var current = Xml.createDocument();
- var stack = new List();
- while( str.length > 0 ) {
- var i = 0;
- while( i < nrules ) {
- var r = rules[i];
- if( r.match(str) ) {
- switch( i ) {
- case 0: // Node
- var x = Xml.createElement(r.matched(1));
- current.addChild(x);
- str = r.matchedRight();
- while( eattribute.match(str) ) {
- x.set(eattribute.matched(1),eattribute.matched(3));
- str = eattribute.matchedRight();
- }
- if( !eclose.match(str) ) {
- i = nrules;
- break;
- }
- if( eclose.matched(1) == ">" ) {
- stack.push(current);
- current = x;
- }
- str = eclose.matchedRight();
- case 1: // PCData
- var x = Xml.createPCData(r.matched(0));
- current.addChild(x);
- str = r.matchedRight();
- case 2: // End Node
- untyped if( current._children != null && current._children.length == 0 ) {
- var e = Xml.createPCData("");
- current.addChild(e);
- }
- untyped if( r.matched(1) != current._nodeName || stack.isEmpty() ) {
- i = nrules;
- break;
- }
- current = stack.pop();
- str = r.matchedRight();
- case 3: // CData
- str = r.matchedRight();
- if( !ecdata_end.match(str) )
- throw "End of CDATA section not found";
- var x = Xml.createCData(ecdata_end.matchedLeft());
- current.addChild(x);
- str = ecdata_end.matchedRight();
- case 4: // DocType
- var pos = 0;
- var count = 0;
- var old = str;
- while( true ) {
- if( !edoctype_elt.match(str) )
- throw "End of DOCTYPE section not found";
- var p = edoctype_elt.matchedPos();
- pos += p.pos + p.len;
- str = edoctype_elt.matchedRight();
- switch( edoctype_elt.matched(0) ) {
- case "[": count++;
- case "]": count--; if( count < 0 ) throw "Invalid ] found in DOCTYPE declaration";
- default:
- if( count == 0 )
- break;
- }
- }
- var x = Xml.createDocType(old.substr(0,pos));
- current.addChild(x);
- case 5: // Comment
- if( !ecomment_end.match(str) )
- throw "Unclosed Comment";
- var p = ecomment_end.matchedPos();
- var x = Xml.createComment(str.substr(0,p.pos+p.len));
- current.addChild(x);
- str = ecomment_end.matchedRight();
- case 6: // Prolog
- var x = Xml.createProlog(r.matched(0));
- current.addChild(x);
- str = r.matchedRight();
- }
- break;
- }
- i += 1;
- }
- if( i == nrules ) {
- if( str.length > 10 )
- throw ("Xml parse error : Unexpected "+str.substr(0,10)+"...");
- else
- throw ("Xml parse error : Unexpected "+str);
- }
- }
- untyped return current;
- }
- private function new(){
- }
- static function createElement( name : String ) : JsXml__ {
- var r = new JsXml__();
- r.nodeType = Xml.Element;
- r._children = new Array();
- r._attributes = new Hash();
- r.setNodeName( name );
- return r;
- }
- static function createPCData( data : String ) : JsXml__ {
- var r = new JsXml__();
- r.nodeType = Xml.PCData;
- r.setNodeValue( data );
- return r;
- }
- static function createCData( data : String ) : JsXml__ {
- var r = new JsXml__();
- r.nodeType = Xml.CData;
- r.setNodeValue( data );
- return r;
- }
- static function createComment( data : String ) : JsXml__ {
- var r = new JsXml__();
- r.nodeType = Xml.Comment;
- r.setNodeValue( data );
- return r;
- }
- static function createDocType( data : String ) : JsXml__ {
- var r = new JsXml__();
- r.nodeType = Xml.DocType;
- r.setNodeValue( data );
- return r;
- }
- static function createProlog( data : String ) : JsXml__ {
- var r = new JsXml__();
- r.nodeType = Xml.Prolog;
- r.setNodeValue( data );
- return r;
- }
- static function createDocument() : JsXml__ {
- var r = new JsXml__();
- r.nodeType = Xml.Document;
- r._children = new Array();
- return r;
- }
- private function getNodeName() : String {
- if( nodeType != Xml.Element )
- throw "bad nodeType";
- return _nodeName;
- }
- private function setNodeName( n : String ) : String {
- if( nodeType != Xml.Element )
- throw "bad nodeType";
- return _nodeName = n;
- }
- private function getNodeValue() : String {
- if( nodeType == Xml.Element || nodeType == Xml.Document )
- throw "bad nodeType";
- return _nodeValue;
- }
- private function setNodeValue( v : String ) : String {
- if( nodeType == Xml.Element || nodeType == Xml.Document )
- throw "bad nodeType";
- return _nodeValue = v;
- }
- private function getParent() {
- return _parent;
- }
- public function get( att : String ) : String {
- if( nodeType != Xml.Element )
- throw "bad nodeType";
- return _attributes.get( att );
- }
- public function set( att : String, value : String ) : Void {
- if( nodeType != Xml.Element )
- throw "bad nodeType";
- _attributes.set( att, value );
- }
- public function remove( att : String ) : Void{
- if( nodeType != Xml.Element )
- throw "bad nodeType";
- _attributes.remove( att );
- }
- public function exists( att : String ) : Bool {
- if( nodeType != Xml.Element )
- throw "bad nodeType";
- return _attributes.exists( att );
- }
- public function attributes() : Iterator<String> {
- if( nodeType != Xml.Element )
- throw "bad nodeType";
- return _attributes.keys();
- }
- public function iterator() : Iterator<JsXml__> {
- if( _children == null ) throw "bad nodetype";
- return untyped {
- cur: 0,
- x: this._children,
- hasNext : function(){
- return this.cur < this.x.length;
- },
- next : function(){
- return this.x[this.cur++];
- }
- }
- }
- public function elements(){
- if( _children == null ) throw "bad nodetype";
- return untyped {
- cur: 0,
- x: this._children,
- hasNext : function() {
- var k = this.cur;
- var l = this.x.length;
- while( k < l ) {
- if( this.x[k].nodeType == Xml.Element )
- break;
- k += 1;
- }
- this.cur = k;
- return k < l;
- },
- next : function() {
- var k = this.cur;
- var l = this.x.length;
- while( k < l ) {
- var n = this.x[k];
- k += 1;
- if( n.nodeType == Xml.Element ) {
- this.cur = k;
- return n;
- }
- }
- return null;
- }
- }
- }
- public function elementsNamed( name : String ) {
- if( _children == null ) throw "bad nodetype";
- return untyped {
- cur: 0,
- x: this._children,
- hasNext : function() {
- var k = this.cur;
- var l = this.x.length;
- while( k < l ) {
- var n = this.x[k];
- if( n.nodeType == Xml.Element && n._nodeName == name )
- break;
- k++;
- }
- this.cur = k;
- return k < l;
- },
- next : function() {
- var k = this.cur;
- var l = this.x.length;
- while( k < l ) {
- var n = this.x[k];
- k++;
- if( n.nodeType == Xml.Element && n._nodeName == name ) {
- this.cur = k;
- return n;
- }
- }
- return null;
- }
- }
- }
- public function firstChild() : JsXml__ {
- if( _children == null ) throw "bad nodetype";
- return _children[0];
- }
- public function firstElement() : JsXml__ {
- if( _children == null ) throw "bad nodetype";
- var cur = 0;
- var l = _children.length;
- while( cur < l ) {
- var n = _children[cur];
- if( n.nodeType == Xml.Element )
- return n;
- cur++;
- }
- return null;
- }
- public function addChild( x : JsXml__ ) : Void {
- if( _children == null ) throw "bad nodetype";
- if( x._parent != null ) x._parent._children.remove(x);
- x._parent = this;
- _children.push( x );
- }
- public function removeChild( x : JsXml__ ) : Bool {
- if( _children == null ) throw "bad nodetype";
- var b = _children.remove( x );
- if( b )
- x._parent = null;
- return b;
- }
- public function insertChild( x : JsXml__, 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 );
- }
- public function toString() {
- if( nodeType == Xml.PCData )
- return _nodeValue;
- if( nodeType == Xml.CData )
- return "<![CDATA["+_nodeValue+"]]>";
- if( nodeType == Xml.Comment || nodeType == Xml.DocType || nodeType == Xml.Prolog )
- return _nodeValue;
- var s = new StringBuf();
- if( nodeType == Xml.Element ) {
- s.add("<");
- s.add(_nodeName);
- for( k in _attributes.keys() ){
- s.add(" ");
- s.add(k);
- s.add("=\"");
- s.add(_attributes.get(k));
- s.add("\"");
- }
- if( _children.length == 0 ) {
- s.add("/>");
- return s.toString();
- }
- s.add(">");
- }
- for( x in iterator() )
- s.add(x.toString());
- if( nodeType == Xml.Element ) {
- s.add("</");
- s.add(_nodeName);
- s.add(">");
- }
- return s.toString();
- }
- }
|