//------------------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation. All rights reserved.
//
// Microsoft
//------------------------------------------------------------------------------
namespace System.Xml
{
using System;
using System.Text;
using System.IO;
using System.Diagnostics;
using System.Collections;
using System.Collections.Generic;
using System.Xml.Schema;
using System.Globalization;
internal class XmlNodeReaderNavigator {
XmlNode curNode;
XmlNode elemNode;
XmlNode logNode;
int attrIndex;
int logAttrIndex;
//presave these 2 variables since they shouldn't change.
XmlNameTable nameTable;
XmlDocument doc;
int nAttrInd; //used to identify virtual attributes of DocumentType node and XmlDeclaration node
const String strPublicID = "PUBLIC";
const String strSystemID = "SYSTEM";
const String strVersion = "version";
const String strStandalone = "standalone";
const String strEncoding = "encoding";
//caching variables for perf reasons
int nDeclarationAttrCount;
int nDocTypeAttrCount;
//variables for roll back the moves
int nLogLevel;
int nLogAttrInd;
bool bLogOnAttrVal;
bool bCreatedOnAttribute;
internal struct VirtualAttribute {
internal String name;
internal String value;
internal VirtualAttribute(String name, String value) {
this.name = name;
this.value = value;
}
};
internal VirtualAttribute [] decNodeAttributes = {
new VirtualAttribute( null, null ),
new VirtualAttribute( null, null ),
new VirtualAttribute( null, null )
};
internal VirtualAttribute [] docTypeNodeAttributes = {
new VirtualAttribute( null, null ),
new VirtualAttribute( null, null )
};
bool bOnAttrVal;
public XmlNodeReaderNavigator( XmlNode node ) {
curNode = node;
logNode = node;
XmlNodeType nt = curNode.NodeType;
if ( nt == XmlNodeType.Attribute ) {
elemNode = null;
attrIndex = -1;
bCreatedOnAttribute = true;
}
else {
elemNode = node;
attrIndex = -1;
bCreatedOnAttribute = false;
}
//presave this for pref reason since it shouldn't change.
if ( nt == XmlNodeType.Document )
this.doc = (XmlDocument)curNode;
else
this.doc = node.OwnerDocument;
this.nameTable = doc.NameTable;
this.nAttrInd = -1;
//initialize the caching variables
this.nDeclarationAttrCount = -1;
this.nDocTypeAttrCount = -1;
this.bOnAttrVal = false;
this.bLogOnAttrVal = false;
}
public XmlNodeType NodeType {
get {
XmlNodeType nt = curNode.NodeType;
if ( nAttrInd != -1 ) {
Debug.Assert( nt == XmlNodeType.XmlDeclaration || nt == XmlNodeType.DocumentType );
if ( this.bOnAttrVal )
return XmlNodeType.Text;
else
return XmlNodeType.Attribute;
}
return nt;
}
}
public String NamespaceURI {
get { return curNode.NamespaceURI; }
}
public String Name {
get {
if ( nAttrInd != -1 ) {
Debug.Assert( curNode.NodeType == XmlNodeType.XmlDeclaration || curNode.NodeType == XmlNodeType.DocumentType );
if ( this.bOnAttrVal )
return String.Empty; //Text node's name is String.Empty
else {
Debug.Assert( nAttrInd >= 0 && nAttrInd < AttributeCount );
if ( curNode.NodeType == XmlNodeType.XmlDeclaration )
return decNodeAttributes[nAttrInd].name;
else
return docTypeNodeAttributes[nAttrInd].name;
}
}
if ( IsLocalNameEmpty ( curNode.NodeType ) )
return String.Empty;
return curNode.Name;
}
}
public String LocalName {
get {
if ( nAttrInd != -1 )
//for the nodes in this case, their LocalName should be the same as their name
return Name;
if ( IsLocalNameEmpty( curNode.NodeType ))
return String.Empty;
return curNode.LocalName;
}
}
internal bool IsOnAttrVal {
get {
return this.bOnAttrVal;
}
}
internal XmlNode OwnerElementNode {
get {
if( this.bCreatedOnAttribute )
return null;
return this.elemNode;
}
}
internal bool CreatedOnAttribute {
get {
return this.bCreatedOnAttribute;
}
}
private bool IsLocalNameEmpty ( XmlNodeType nt) {
switch ( nt ) {
case XmlNodeType.None :
case XmlNodeType.Text :
case XmlNodeType.CDATA :
case XmlNodeType.Comment :
case XmlNodeType.Document :
case XmlNodeType.DocumentFragment :
case XmlNodeType.Whitespace :
case XmlNodeType.SignificantWhitespace :
case XmlNodeType.EndElement :
case XmlNodeType.EndEntity :
return true;
case XmlNodeType.Element :
case XmlNodeType.Attribute :
case XmlNodeType.EntityReference :
case XmlNodeType.Entity :
case XmlNodeType.ProcessingInstruction :
case XmlNodeType.DocumentType :
case XmlNodeType.Notation :
case XmlNodeType.XmlDeclaration :
return false;
default :
return true;
}
}
public String Prefix {
get { return curNode.Prefix; }
}
public bool HasValue {
//In DOM, DocumentType node and XmlDeclaration node doesn't value
//In XPathNavigator, XmlDeclaration node's value is its InnerText; DocumentType doesn't have value
//In XmlReader, DocumentType node's value is its InternalSubset which is never null ( at least String.Empty )
get {
if ( nAttrInd != -1 ) {
//Pointing at the one of virtual attributes of Declaration or DocumentType nodes
Debug.Assert( curNode.NodeType == XmlNodeType.XmlDeclaration || curNode.NodeType == XmlNodeType.DocumentType );
Debug.Assert( nAttrInd >= 0 && nAttrInd < AttributeCount );
return true;
}
if ( curNode.Value != null || curNode.NodeType == XmlNodeType.DocumentType )
return true;
return false;
}
}
public String Value {
//See comments in HasValue
get {
String retValue = null;
XmlNodeType nt = curNode.NodeType;
if ( nAttrInd != -1 ) {
//Pointing at the one of virtual attributes of Declaration or DocumentType nodes
Debug.Assert( nt == XmlNodeType.XmlDeclaration || nt == XmlNodeType.DocumentType );
Debug.Assert( nAttrInd >= 0 && nAttrInd < AttributeCount );
if ( curNode.NodeType == XmlNodeType.XmlDeclaration )
return decNodeAttributes[nAttrInd].value;
else
return docTypeNodeAttributes[nAttrInd].value;
}
if ( nt == XmlNodeType.DocumentType )
retValue = ((XmlDocumentType)curNode).InternalSubset; //in this case nav.Value will be null
else if ( nt == XmlNodeType.XmlDeclaration ) {
StringBuilder strb = new StringBuilder(String.Empty);
if ( nDeclarationAttrCount == -1 )
InitDecAttr();
for ( int i = 0; i < nDeclarationAttrCount; i++ ) {
strb.Append(decNodeAttributes[i].name + "=\"" +decNodeAttributes[i].value + "\"");
if( i != ( nDeclarationAttrCount-1 ) )
strb.Append( " " );
}
retValue = strb.ToString();
} else
retValue = curNode.Value;
return ( retValue == null )? String.Empty : retValue;
}
}
public String BaseURI {
get { return curNode.BaseURI; }
}
public XmlSpace XmlSpace {
get { return curNode.XmlSpace; }
}
public String XmlLang {
get { return curNode.XmlLang; }
}
public bool IsEmptyElement {
get {
if (curNode.NodeType == XmlNodeType.Element) {
return((XmlElement)curNode).IsEmpty;
}
return false;
}
}
public bool IsDefault {
get {
if (curNode.NodeType == XmlNodeType.Attribute) {
return !((XmlAttribute)curNode).Specified;
}
return false;
}
}
public IXmlSchemaInfo SchemaInfo {
get {
return curNode.SchemaInfo;
}
}
public XmlNameTable NameTable {
get { return nameTable; }
}
public int AttributeCount {
get {
if( this.bCreatedOnAttribute )
return 0;
XmlNodeType nt = curNode.NodeType;
if ( nt == XmlNodeType.Element )
return ((XmlElement)curNode).Attributes.Count;
else if ( nt == XmlNodeType.Attribute
|| ( this.bOnAttrVal && nt != XmlNodeType.XmlDeclaration && nt != XmlNodeType.DocumentType ) )
return elemNode.Attributes.Count;
else if ( nt == XmlNodeType.XmlDeclaration ) {
if ( nDeclarationAttrCount != -1 )
return nDeclarationAttrCount;
InitDecAttr();
return nDeclarationAttrCount;
} else if ( nt == XmlNodeType.DocumentType ) {
if ( nDocTypeAttrCount != -1 )
return nDocTypeAttrCount;
InitDocTypeAttr();
return nDocTypeAttrCount;
}
return 0;
}
}
private void CheckIndexCondition(int attributeIndex) {
if (attributeIndex < 0 || attributeIndex >= AttributeCount) {
throw new ArgumentOutOfRangeException( "attributeIndex" );
}
}
//8 functions below are the helper functions to deal with virtual attributes of XmlDeclaration nodes and DocumentType nodes.
private void InitDecAttr() {
int i = 0;
String strTemp = doc.Version;
if ( strTemp != null && strTemp.Length != 0 ) {
decNodeAttributes[i].name = strVersion;
decNodeAttributes[i].value = strTemp;
i++;
}
strTemp = doc.Encoding;
if ( strTemp != null && strTemp.Length != 0 ) {
decNodeAttributes[i].name = strEncoding;
decNodeAttributes[i].value = strTemp;
i++;
}
strTemp = doc.Standalone;
if ( strTemp != null && strTemp.Length != 0 ) {
decNodeAttributes[i].name = strStandalone;
decNodeAttributes[i].value = strTemp;
i++;
}
nDeclarationAttrCount = i;
}
public String GetDeclarationAttr( XmlDeclaration decl, String name ) {
//PreCondition: curNode is pointing at Declaration node or one of its virtual attributes
if ( name == strVersion )
return decl.Version;
if ( name == strEncoding )
return decl.Encoding;
if ( name == strStandalone )
return decl.Standalone;
return null;
}
public String GetDeclarationAttr( int i ) {
if ( nDeclarationAttrCount == -1 )
InitDecAttr();
return decNodeAttributes[i].value;
}
public int GetDecAttrInd( String name ) {
if ( nDeclarationAttrCount == -1 )
InitDecAttr();
for ( int i = 0 ; i < nDeclarationAttrCount; i++ ) {
if ( decNodeAttributes[i].name == name )
return i;
}
return -1;
}
private void InitDocTypeAttr() {
int i = 0;
XmlDocumentType docType = doc.DocumentType;
if ( docType == null ) {
nDocTypeAttrCount = 0;
return;
}
String strTemp = docType.PublicId;
if ( strTemp != null ) {
docTypeNodeAttributes[i].name = strPublicID;
docTypeNodeAttributes[i].value = strTemp;
i++;
}
strTemp = docType.SystemId;
if ( strTemp != null ) {
docTypeNodeAttributes[i].name = strSystemID;
docTypeNodeAttributes[i].value = strTemp;
i++;
}
nDocTypeAttrCount = i;
}
public String GetDocumentTypeAttr ( XmlDocumentType docType, String name ) {
//PreCondition: nav is pointing at DocumentType node or one of its virtual attributes
if ( name == strPublicID )
return docType.PublicId;
if ( name == strSystemID )
return docType.SystemId;
return null;
}
public String GetDocumentTypeAttr( int i ) {
if ( nDocTypeAttrCount == -1 )
InitDocTypeAttr();
return docTypeNodeAttributes[i].value;
}
public int GetDocTypeAttrInd( String name ) {
if ( nDocTypeAttrCount == -1 )
InitDocTypeAttr();
for ( int i = 0 ; i < nDocTypeAttrCount; i++ ) {
if ( docTypeNodeAttributes[i].name == name )
return i;
}
return -1;
}
private String GetAttributeFromElement( XmlElement elem, String name ) {
XmlAttribute attr = elem.GetAttributeNode( name );
if ( attr != null )
return attr.Value;
return null;
}
public String GetAttribute( String name ) {
if( this.bCreatedOnAttribute )
return null;
switch ( curNode.NodeType ) {
case XmlNodeType.Element:
return GetAttributeFromElement((XmlElement)curNode, name);
case XmlNodeType.Attribute :
return GetAttributeFromElement((XmlElement)elemNode, name);
case XmlNodeType.XmlDeclaration:
return GetDeclarationAttr( (XmlDeclaration)curNode, name );
case XmlNodeType.DocumentType:
return GetDocumentTypeAttr( (XmlDocumentType)curNode, name );
}
return null;
}
private String GetAttributeFromElement( XmlElement elem, String name, String ns ) {
XmlAttribute attr = elem.GetAttributeNode( name, ns );
if ( attr != null )
return attr.Value;
return null;
}
public String GetAttribute( String name, String ns ) {
if( this.bCreatedOnAttribute )
return null;
switch ( curNode.NodeType ) {
case XmlNodeType.Element:
return GetAttributeFromElement((XmlElement)curNode, name, ns);
case XmlNodeType.Attribute :
return GetAttributeFromElement((XmlElement)elemNode, name, ns);
case XmlNodeType.XmlDeclaration:
return (ns.Length == 0) ? GetDeclarationAttr( (XmlDeclaration)curNode, name ) : null;
case XmlNodeType.DocumentType:
return (ns.Length == 0) ? GetDocumentTypeAttr( (XmlDocumentType)curNode, name ) : null;
}
return null;
}
public String GetAttribute( int attributeIndex ) {
if( this.bCreatedOnAttribute )
return null;
switch ( curNode.NodeType ) {
case XmlNodeType.Element:
CheckIndexCondition( attributeIndex );
return ((XmlElement)curNode).Attributes[attributeIndex].Value;
case XmlNodeType.Attribute :
CheckIndexCondition( attributeIndex );
return ((XmlElement)elemNode).Attributes[attributeIndex].Value;
case XmlNodeType.XmlDeclaration: {
CheckIndexCondition( attributeIndex );
return GetDeclarationAttr( attributeIndex );
}
case XmlNodeType.DocumentType: {
CheckIndexCondition( attributeIndex );
return GetDocumentTypeAttr( attributeIndex );
}
}
throw new ArgumentOutOfRangeException( "attributeIndex" ); //for other senario, AttributeCount is 0, i has to be out of range
}
public void LogMove( int level ) {
logNode = curNode;
nLogLevel = level;
nLogAttrInd = nAttrInd;
logAttrIndex = attrIndex;
this.bLogOnAttrVal = this.bOnAttrVal;
}
//The function has to be used in pair with ResetMove when the operation fails after LogMove() is
// called because it relies on the values of nOrigLevel, logNav and nOrigAttrInd to be acurate.
public void RollBackMove( ref int level ) {
curNode = logNode;
level = nLogLevel;
nAttrInd = nLogAttrInd;
attrIndex = logAttrIndex;
this.bOnAttrVal = this.bLogOnAttrVal;
}
private bool IsOnDeclOrDocType {
get {
XmlNodeType nt = curNode.NodeType;
return ( nt == XmlNodeType.XmlDeclaration || nt == XmlNodeType.DocumentType );
}
}
public void ResetToAttribute( ref int level ) {
//the current cursor is pointing at one of the attribute children -- this could be caused by
// the calls to ReadAttributeValue(..)
if( this.bCreatedOnAttribute )
return;
if ( this.bOnAttrVal ) {
if ( IsOnDeclOrDocType ) {
level-=2;
} else {
while ( curNode.NodeType != XmlNodeType.Attribute && ( ( curNode = curNode.ParentNode ) != null ) )
level-- ;
}
this.bOnAttrVal = false;
}
}
public void ResetMove( ref int level, ref XmlNodeType nt ) {
LogMove( level );
if( this.bCreatedOnAttribute )
return;
if ( nAttrInd != -1 ) {
Debug.Assert( IsOnDeclOrDocType );
if ( this.bOnAttrVal ) {
level--;
this.bOnAttrVal = false;
}
nLogAttrInd = nAttrInd;
level--;
nAttrInd = -1;
nt = curNode.NodeType;
return;
}
if ( this.bOnAttrVal && curNode.NodeType != XmlNodeType.Attribute )
ResetToAttribute( ref level );
if ( curNode.NodeType == XmlNodeType.Attribute ) {
curNode = ((XmlAttribute)curNode).OwnerElement;
attrIndex = -1;
level--;
nt = XmlNodeType.Element;
}
if ( curNode.NodeType == XmlNodeType.Element )
elemNode = curNode;
}
public bool MoveToAttribute( string name ) {
return MoveToAttribute( name, string.Empty );
}
private bool MoveToAttributeFromElement( XmlElement elem, String name, String ns ) {
XmlAttribute attr = null;
if( ns.Length == 0 )
attr = elem.GetAttributeNode( name );
else
attr = elem.GetAttributeNode( name, ns );
if ( attr != null ) {
this.bOnAttrVal = false;
elemNode = elem;
curNode = attr;
attrIndex = elem.Attributes.FindNodeOffsetNS(attr);
if (attrIndex != -1) {
return true;
}
}
return false;
}
public bool MoveToAttribute( string name, string namespaceURI ) {
if( this.bCreatedOnAttribute )
return false;
XmlNodeType nt = curNode.NodeType;
if ( nt == XmlNodeType.Element )
return MoveToAttributeFromElement((XmlElement)curNode, name, namespaceURI );
else if ( nt == XmlNodeType.Attribute )
return MoveToAttributeFromElement((XmlElement)elemNode, name, namespaceURI );
else if ( nt == XmlNodeType.XmlDeclaration && namespaceURI.Length == 0 ) {
if ( ( nAttrInd = GetDecAttrInd( name ) ) != -1 ) {
this.bOnAttrVal = false;
return true;
}
} else if ( nt == XmlNodeType.DocumentType && namespaceURI.Length == 0 ) {
if ( ( nAttrInd = GetDocTypeAttrInd( name ) ) != -1 ) {
this.bOnAttrVal = false;
return true;
}
}
return false;
}
public void MoveToAttribute( int attributeIndex ) {
if( this.bCreatedOnAttribute )
return;
XmlAttribute attr = null;
switch ( curNode.NodeType ) {
case XmlNodeType.Element:
CheckIndexCondition( attributeIndex );
attr = ((XmlElement)curNode).Attributes[attributeIndex];
if ( attr != null ) {
elemNode = curNode;
curNode = (XmlNode) attr;
attrIndex = attributeIndex;
}
break;
case XmlNodeType.Attribute:
CheckIndexCondition( attributeIndex );
attr = ((XmlElement)elemNode).Attributes[attributeIndex];
if ( attr != null ) {
curNode = (XmlNode) attr;
attrIndex = attributeIndex;
}
break;
case XmlNodeType.XmlDeclaration :
case XmlNodeType.DocumentType :
CheckIndexCondition( attributeIndex );
nAttrInd = attributeIndex;
break;
}
}
public bool MoveToNextAttribute( ref int level ) {
if( this.bCreatedOnAttribute )
return false;
XmlNodeType nt = curNode.NodeType;
if ( nt == XmlNodeType.Attribute ) {
if( attrIndex >= ( elemNode.Attributes.Count-1 ) )
return false;
else {
curNode = elemNode.Attributes[++attrIndex];
return true;
}
} else if ( nt == XmlNodeType.Element ) {
if ( curNode.Attributes.Count > 0 ) {
level++;
elemNode = curNode;
curNode = curNode.Attributes[0];
attrIndex = 0;
return true;
}
} else if ( nt == XmlNodeType.XmlDeclaration ) {
if ( nDeclarationAttrCount == -1 )
InitDecAttr();
nAttrInd++;
if ( nAttrInd < nDeclarationAttrCount ) {
if ( nAttrInd == 0 ) level++;
this.bOnAttrVal = false;
return true;
}
nAttrInd--;
} else if ( nt == XmlNodeType.DocumentType ) {
if ( nDocTypeAttrCount == -1 )
InitDocTypeAttr();
nAttrInd++;
if ( nAttrInd < nDocTypeAttrCount ) {
if ( nAttrInd == 0 ) level++;
this.bOnAttrVal = false;
return true;
}
nAttrInd--;
}
return false;
}
public bool MoveToParent() {
XmlNode parent = curNode.ParentNode;
if ( parent != null ) {
curNode = parent;
if( !bOnAttrVal )
attrIndex = 0;
return true;
}
return false;
}
public bool MoveToFirstChild() {
XmlNode firstChild = curNode.FirstChild;
if ( firstChild != null ) {
curNode = firstChild;
if( !bOnAttrVal )
attrIndex = -1;
return true;
}
return false;
}
private bool MoveToNextSibling( XmlNode node ) {
XmlNode nextSibling = node.NextSibling;
if ( nextSibling != null ) {
curNode = nextSibling;
if( !bOnAttrVal )
attrIndex = -1;
return true;
}
return false;
}
public bool MoveToNext() {
if ( curNode.NodeType != XmlNodeType.Attribute )
return MoveToNextSibling( curNode );
else
return MoveToNextSibling( elemNode );
}
public bool MoveToElement() {
if( this.bCreatedOnAttribute )
return false;
switch ( curNode.NodeType ) {
case XmlNodeType.Attribute :
if ( elemNode != null ) {
curNode = elemNode;
attrIndex = -1;
return true;
}
break;
case XmlNodeType.XmlDeclaration :
case XmlNodeType.DocumentType : {
if ( nAttrInd != -1 ) {
nAttrInd = -1;
return true;
}
break;
}
}
return false;
}
public String LookupNamespace(string prefix) {
if( this.bCreatedOnAttribute )
return null;
if ( prefix == "xmlns" ) {
return nameTable.Add( XmlReservedNs.NsXmlNs );
}
if ( prefix == "xml" ) {
return nameTable.Add( XmlReservedNs.NsXml );
}
// construct the name of the xmlns attribute
string attrName;
if ( prefix == null )
prefix = string.Empty;
if ( prefix.Length == 0 )
attrName = "xmlns";
else
attrName = "xmlns:" + prefix;
// walk up the XmlNode parent chain, looking for the xmlns attribute
XmlNode node = curNode;
while ( node != null ) {
if ( node.NodeType == XmlNodeType.Element ) {
XmlElement elem = (XmlElement)node;
if ( elem.HasAttributes ) {
XmlAttribute attr = elem.GetAttributeNode( attrName );
if ( attr != null ) {
return attr.Value;
}
}
}
else if ( node.NodeType == XmlNodeType.Attribute ) {
node = ((XmlAttribute)node).OwnerElement;
continue;
}
node = node.ParentNode;
}
if ( prefix.Length == 0 ) {
return string.Empty;
}
return null;
}
internal string DefaultLookupNamespace( string prefix ) {
if( !this.bCreatedOnAttribute ) {
if ( prefix == "xmlns" ) {
return nameTable.Add( XmlReservedNs.NsXmlNs );
}
if ( prefix == "xml" ) {
return nameTable.Add( XmlReservedNs.NsXml );
}
if ( prefix == string.Empty ) {
return nameTable.Add( string.Empty );
}
}
return null;
}
internal String LookupPrefix(string namespaceName) {
if( this.bCreatedOnAttribute || namespaceName == null ) {
return null;
}
if ( namespaceName == XmlReservedNs.NsXmlNs ) {
return nameTable.Add( "xmlns" );
}
if ( namespaceName == XmlReservedNs.NsXml ) {
return nameTable.Add( "xml" );
}
if ( namespaceName == string.Empty ) {
return string.Empty;
}
// walk up the XmlNode parent chain, looking for the xmlns attribute with namespaceName value
XmlNode node = curNode;
while ( node != null ) {
if ( node.NodeType == XmlNodeType.Element ) {
XmlElement elem = (XmlElement)node;
if ( elem.HasAttributes ) {
XmlAttributeCollection attrs = elem.Attributes;
for ( int i = 0; i < attrs.Count; i++ ) {
XmlAttribute a = attrs[i];
if ( a.Value == namespaceName ) {
if ( a.Prefix.Length == 0 && a.LocalName == "xmlns" ) {
if ( LookupNamespace( string.Empty ) == namespaceName ) {
return string.Empty;
}
}
else if ( a.Prefix == "xmlns" ) {
string pref = a.LocalName;
if ( LookupNamespace( pref ) == namespaceName ) {
return nameTable.Add( pref );
}
}
}
}
}
}
else if ( node.NodeType == XmlNodeType.Attribute ) {
node = ((XmlAttribute)node).OwnerElement;
continue;
}
node = node.ParentNode;
}
return null;
}
internal IDictionary GetNamespacesInScope( XmlNamespaceScope scope ) {
Dictionary dict = new Dictionary();
if( this.bCreatedOnAttribute )
return dict;
// walk up the XmlNode parent chain and add all namespace declarations to the dictionary
XmlNode node = curNode;
while ( node != null ) {
if ( node.NodeType == XmlNodeType.Element ) {
XmlElement elem = (XmlElement)node;
if ( elem.HasAttributes ) {
XmlAttributeCollection attrs = elem.Attributes;
for ( int i = 0; i < attrs.Count; i++ ) {
XmlAttribute a = attrs[i];
if ( a.LocalName == "xmlns" && a.Prefix.Length == 0 ) {
if ( !dict.ContainsKey( string.Empty ) ) {
dict.Add( nameTable.Add( string.Empty ), nameTable.Add( a.Value ) );
}
}
else if ( a.Prefix == "xmlns" ) {
string localName = a.LocalName;
if ( !dict.ContainsKey( localName ) ) {
dict.Add( nameTable.Add( localName ), nameTable.Add( a.Value ) );
}
}
}
}
if ( scope == XmlNamespaceScope.Local ) {
break;
}
}
else if ( node.NodeType == XmlNodeType.Attribute ) {
node = ((XmlAttribute)node).OwnerElement;
continue;
}
node = node.ParentNode;
};
if ( scope != XmlNamespaceScope.Local ) {
if ( dict.ContainsKey( string.Empty ) && dict[string.Empty] == string.Empty ) {
dict.Remove( string.Empty );
}
if ( scope == XmlNamespaceScope.All ) {
dict.Add( nameTable.Add( "xml" ), nameTable.Add( XmlReservedNs.NsXml ) );
}
}
return dict;
}
public bool ReadAttributeValue( ref int level, ref bool bResolveEntity, ref XmlNodeType nt ) {
if ( nAttrInd != -1 ) {
Debug.Assert( curNode.NodeType == XmlNodeType.XmlDeclaration || curNode.NodeType == XmlNodeType.DocumentType );
if ( !this.bOnAttrVal ) {
this.bOnAttrVal = true;
level++;
nt = XmlNodeType.Text;
return true;
}
return false;
}
if( curNode.NodeType == XmlNodeType.Attribute ) {
XmlNode firstChild = curNode.FirstChild;
if ( firstChild != null ) {
curNode = firstChild;
nt = curNode.NodeType;
level++;
this.bOnAttrVal = true;
return true;
}
}
else if ( this.bOnAttrVal ) {
XmlNode nextSibling = null;
if ( curNode.NodeType == XmlNodeType.EntityReference && bResolveEntity ) {
//going down to ent ref node
curNode = curNode.FirstChild;
nt = curNode.NodeType;
Debug.Assert( curNode != null );
level++;
bResolveEntity = false;
return true;
}
else
nextSibling = curNode.NextSibling;
if ( nextSibling == null ) {
XmlNode parentNode = curNode.ParentNode;
//Check if its parent is entity ref node is sufficient, because in this senario, ent ref node can't have more than 1 level of children that are not other ent ref nodes
if ( parentNode != null && parentNode.NodeType == XmlNodeType.EntityReference ) {
//come back from ent ref node
curNode = parentNode;
nt = XmlNodeType.EndEntity;
level--;
return true;
}
}
if ( nextSibling != null ) {
curNode = nextSibling;
nt = curNode.NodeType;
return true;
}
else
return false;
}
return false;
}
public XmlDocument Document {
get {
return this.doc;
}
}
}
// Represents a reader that provides fast, non-cached forward only stream access
// to XML data in an XmlDocument or a specific XmlNode within an XmlDocument.
public class XmlNodeReader: XmlReader, IXmlNamespaceResolver
{
XmlNodeReaderNavigator readerNav;
XmlNodeType nodeType; // nodeType of the node that the reader is currently positioned on
int curDepth; // depth of attrNav ( also functions as reader's depth )
ReadState readState; // current reader's state
bool fEOF; // flag to show if reaches the end of file
//mark to the state that EntityReference node is supposed to be resolved
bool bResolveEntity;
bool bStartFromDocument;
bool bInReadBinary;
ReadContentAsBinaryHelper readBinaryHelper;
// Creates an instance of the XmlNodeReader class using the specified XmlNode.
public XmlNodeReader ( XmlNode node ) {
if (node == null) {
throw new ArgumentNullException("node");
}
readerNav = new XmlNodeReaderNavigator( node );
this.curDepth = 0;
readState = ReadState.Initial;
fEOF = false;
nodeType = XmlNodeType.None;
bResolveEntity = false;
bStartFromDocument = false;
}
//function returns if the reader currently in valid reading states
internal bool IsInReadingStates() {
return ( readState == ReadState.Interactive ); // || readState == ReadState.EndOfFile
}
//
// Node Properties
//
// Gets the type of the current node.
public override XmlNodeType NodeType {
get { return ( IsInReadingStates() )? nodeType : XmlNodeType.None; }
}
// Gets the name of
// the current node, including the namespace prefix.
public override string Name {
get {
if ( !IsInReadingStates() )
return String.Empty;
return readerNav.Name;
}
}
// Gets the name of the current node without the namespace prefix.
public override string LocalName {
get {
if ( !IsInReadingStates() )
return String.Empty;
return readerNav.LocalName;
}
}
// Gets the namespace URN (as defined in the W3C Namespace Specification)
// of the current namespace scope.
public override string NamespaceURI {
get {
if ( !IsInReadingStates() )
return String.Empty;
return readerNav.NamespaceURI;
}
}
// Gets the namespace prefix associated with the current node.
public override string Prefix {
get {
if ( !IsInReadingStates() )
return String.Empty;
return readerNav.Prefix;
}
}
// Gets a value indicating whether
// XmlNodeReader.Value has a value to return.
public override bool HasValue {
get {
if ( !IsInReadingStates() )
return false;
return readerNav.HasValue;
}
}
// Gets the text value of the current node.
public override string Value {
get {
if ( !IsInReadingStates() )
return String.Empty;
return readerNav.Value;
}
}
// Gets the depth of the
// current node in the XML element stack.
public override int Depth {
get { return curDepth; }
}
// Gets the base URI of the current node.
public override String BaseURI {
get { return readerNav.BaseURI; }
}
public override bool CanResolveEntity {
get { return true; }
}
// Gets a value indicating whether the current
// node is an empty element (for example, .
public override bool IsEmptyElement {
get {
if ( !IsInReadingStates() )
return false;
return readerNav.IsEmptyElement;
}
}
// Gets a value indicating whether the current node is an
// attribute that was generated from the default value defined
// in the DTD or schema.
public override bool IsDefault {
get {
if ( !IsInReadingStates() )
return false;
return readerNav.IsDefault;
}
}
// Gets the current xml:space scope.
public override XmlSpace XmlSpace {
get {
if ( !IsInReadingStates() )
return XmlSpace.None;
return readerNav.XmlSpace;
}
}
// Gets the current xml:lang scope.
public override string XmlLang {
// Assume everything is in Unicode
get {
if ( !IsInReadingStates() )
return String.Empty;
return readerNav.XmlLang;
}
}
public override IXmlSchemaInfo SchemaInfo {
get {
if (!IsInReadingStates()) {
return null;
}
return readerNav.SchemaInfo;
}
}
//
// Attribute Accessors
//
// Gets the number of attributes on the current node.
public override int AttributeCount {
get {
if ( !IsInReadingStates() || nodeType == XmlNodeType.EndElement )
return 0;
return readerNav.AttributeCount;
}
}
// Gets the value of the attribute with the specified name.
public override string GetAttribute(string name) {
//if not on Attribute, only element node could have attributes
if ( !IsInReadingStates() )
return null;
return readerNav.GetAttribute( name );
}
// Gets the value of the attribute with the specified name and namespace.
public override string GetAttribute(string name, string namespaceURI) {
//if not on Attribute, only element node could have attributes
if ( !IsInReadingStates() )
return null;
String ns = ( namespaceURI == null ) ? String.Empty : namespaceURI;
return readerNav.GetAttribute( name, ns );
}
// Gets the value of the attribute with the specified index.
public override string GetAttribute(int attributeIndex) {
if ( !IsInReadingStates() )
throw new ArgumentOutOfRangeException( "attributeIndex" );
//CheckIndexCondition( i );
//Debug.Assert( nav.NodeType == XmlNodeType.Element );
return readerNav.GetAttribute( attributeIndex );
}
// Moves to the attribute with the specified name.
public override bool MoveToAttribute(string name) {
if ( !IsInReadingStates() )
return false;
readerNav.ResetMove( ref curDepth, ref nodeType );
if ( readerNav.MoveToAttribute( name ) ) { //, ref curDepth ) ) {
curDepth++;
nodeType = readerNav.NodeType;
if ( bInReadBinary ) {
FinishReadBinary();
}
return true;
}
readerNav.RollBackMove(ref curDepth);
return false;
}
// Moves to the attribute with the specified name and namespace.
public override bool MoveToAttribute(string name, string namespaceURI) {
if ( !IsInReadingStates() )
return false;
readerNav.ResetMove( ref curDepth, ref nodeType );
String ns = ( namespaceURI == null ) ? String.Empty : namespaceURI;
if ( readerNav.MoveToAttribute( name, ns ) ) { //, ref curDepth ) ) {
curDepth++;
nodeType = readerNav.NodeType;
if ( bInReadBinary ) {
FinishReadBinary();
}
return true;
}
readerNav.RollBackMove(ref curDepth);
return false;
}
// Moves to the attribute with the specified index.
public override void MoveToAttribute(int attributeIndex) {
if ( !IsInReadingStates() )
throw new ArgumentOutOfRangeException( "attributeIndex" );
readerNav.ResetMove( ref curDepth, ref nodeType );
try {
if (AttributeCount > 0) {
readerNav.MoveToAttribute( attributeIndex );
if ( bInReadBinary ) {
FinishReadBinary();
}
}
else
throw new ArgumentOutOfRangeException( "attributeIndex" );
} catch {
readerNav.RollBackMove(ref curDepth);
throw;
}
curDepth++;
nodeType = readerNav.NodeType;
}
// Moves to the first attribute.
public override bool MoveToFirstAttribute() {
if ( !IsInReadingStates() )
return false;
readerNav.ResetMove( ref curDepth, ref nodeType );
if (AttributeCount > 0) {
readerNav.MoveToAttribute( 0 );
curDepth++;
nodeType = readerNav.NodeType;
if ( bInReadBinary ) {
FinishReadBinary();
}
return true;
}
readerNav.RollBackMove( ref curDepth );
return false;
}
// Moves to the next attribute.
public override bool MoveToNextAttribute() {
if ( !IsInReadingStates() || nodeType == XmlNodeType.EndElement )
return false;
readerNav.LogMove( curDepth );
readerNav.ResetToAttribute( ref curDepth );
if ( readerNav.MoveToNextAttribute( ref curDepth ) ) {
nodeType = readerNav.NodeType;
if ( bInReadBinary ) {
FinishReadBinary();
}
return true;
}
readerNav.RollBackMove( ref curDepth );
return false;
}
// Moves to the element that contains the current attribute node.
public override bool MoveToElement() {
if ( !IsInReadingStates() )
return false;
readerNav.LogMove( curDepth );
readerNav.ResetToAttribute( ref curDepth );
if ( readerNav.MoveToElement() ) {
curDepth--;
nodeType = readerNav.NodeType;
if ( bInReadBinary ) {
FinishReadBinary();
}
return true;
}
readerNav.RollBackMove( ref curDepth );
return false;
}
//
// Moving through the Stream
//
// Reads the next node from the stream.
public override bool Read() {
return Read( false );
}
private bool Read( bool fSkipChildren ) {
if( fEOF )
return false;
if ( readState == ReadState.Initial ) {
// if nav is pointing at the document node, start with its children
// otherwise,start with the node.
if ( ( readerNav.NodeType == XmlNodeType.Document ) || ( readerNav.NodeType == XmlNodeType.DocumentFragment ) ) {
bStartFromDocument = true;
if ( !ReadNextNode(fSkipChildren) ) {
readState = ReadState.Error;
return false;
}
}
ReSetReadingMarks();
readState = ReadState.Interactive;
nodeType = readerNav.NodeType;
//_depth = 0;
curDepth = 0;
return true;
}
if ( bInReadBinary ) {
FinishReadBinary();
}
bool bRead = false;
if( ( readerNav.CreatedOnAttribute ) )
return false;
ReSetReadingMarks();
bRead = ReadNextNode(fSkipChildren);
if ( bRead ) {
return true;
} else {
if ( readState == ReadState.Initial || readState == ReadState.Interactive )
readState = ReadState.Error;
if ( readState == ReadState.EndOfFile )
nodeType = XmlNodeType.None;
return false;
}
}
private bool ReadNextNode( bool fSkipChildren ) {
if ( readState != ReadState.Interactive && readState != ReadState.Initial ) {
nodeType = XmlNodeType.None;
return false;
}
bool bDrillDown = !fSkipChildren;
XmlNodeType nt = readerNav.NodeType;
//only goes down when nav.NodeType is of element or of document at the initial state, other nav.NodeType will not be parsed down
//if nav.NodeType is of EntityReference, ResolveEntity() could be called to get the content parsed;
bDrillDown = bDrillDown
&& ( nodeType != XmlNodeType.EndElement )
&& ( nodeType != XmlNodeType.EndEntity )
&& ( nt == XmlNodeType.Element || ( nt == XmlNodeType.EntityReference && bResolveEntity ) ||
( ( ( readerNav.NodeType == XmlNodeType.Document ) || ( readerNav.NodeType == XmlNodeType.DocumentFragment ) ) && readState == ReadState.Initial) );
//first see if there are children of current node, so to move down
if ( bDrillDown ) {
if ( readerNav.MoveToFirstChild() ) {
nodeType = readerNav.NodeType;
curDepth++;
if ( bResolveEntity )
bResolveEntity = false;
return true;
} else if ( readerNav.NodeType == XmlNodeType.Element
&& !readerNav.IsEmptyElement ) {
nodeType = XmlNodeType.EndElement;
return true;
}
else if (readerNav.NodeType == XmlNodeType.EntityReference && bResolveEntity) {
bResolveEntity = false;
nodeType = XmlNodeType.EndEntity;
return true;
}
// if fails to move to it 1st Child, try to move to next below
return ReadForward( fSkipChildren );
} else {
if ( readerNav.NodeType == XmlNodeType.EntityReference && bResolveEntity ) {
//The only way to get to here is because Skip() is called directly after ResolveEntity()
// in this case, user wants to skip the first Child of EntityRef node and fSkipChildren is true
// We want to pointing to the first child node.
if (readerNav.MoveToFirstChild()) {
nodeType = readerNav.NodeType;
curDepth++;
}
else {
nodeType = XmlNodeType.EndEntity;
}
bResolveEntity = false;
return true;
}
}
return ReadForward( fSkipChildren ); //has to get the next node by moving forward
}
private void SetEndOfFile() {
fEOF = true;
readState = ReadState.EndOfFile;
nodeType = XmlNodeType.None;
}
private bool ReadAtZeroLevel(bool fSkipChildren) {
Debug.Assert( curDepth == 0 );
if ( !fSkipChildren
&& nodeType != XmlNodeType.EndElement
&& readerNav.NodeType == XmlNodeType.Element
&& !readerNav.IsEmptyElement ) {
nodeType = XmlNodeType.EndElement;
return true;
} else {
SetEndOfFile();
return false;
}
}
private bool ReadForward( bool fSkipChildren ) {
if ( readState == ReadState.Error )
return false;
if ( !bStartFromDocument && curDepth == 0 ) {
//already on top most node and we shouldn't move to next
return ReadAtZeroLevel(fSkipChildren);
}
//else either we are not on top level or we are starting from the document at the very beginning in which case
// we will need to read all the "top" most nodes
if ( readerNav.MoveToNext() ) {
nodeType = readerNav.NodeType;
return true;
} else {
//need to check its parent
if ( curDepth == 0 )
return ReadAtZeroLevel(fSkipChildren);
if ( readerNav.MoveToParent() ) {
if ( readerNav.NodeType == XmlNodeType.Element ) {
curDepth--;
nodeType = XmlNodeType.EndElement;
return true;
} else if ( readerNav.NodeType == XmlNodeType.EntityReference ) {
//coming back from entity reference node -- must be getting down through call ResolveEntity()
curDepth--;
nodeType = XmlNodeType.EndEntity;
return true;
}
return true;
}
}
return false;
}
//the function reset the marks used for ReadChars() and MoveToAttribute(...), ReadAttributeValue(...)
private void ReSetReadingMarks() {
//_attrValInd = -1;
readerNav.ResetMove( ref curDepth, ref nodeType );
//attrNav.MoveTo( nav );
//curDepth = _depth;
}
// Gets a value indicating whether the reader is positioned at the
// end of the stream.
public override bool EOF {
get { return (readState != ReadState.Closed) && fEOF; }
}
// Closes the stream, changes the XmlNodeReader.ReadState
// to Closed, and sets all the properties back to zero.
public override void Close() {
readState = ReadState.Closed;
}
// Gets the read state of the stream.
public override ReadState ReadState {
get { return readState; }
}
// Skips to the end tag of the current element.
public override void Skip() {
Read( true );
}
// Reads the contents of an element as a string.
public override string ReadString() {
if ((this.NodeType == XmlNodeType.EntityReference) && bResolveEntity) {
if (! this.Read()) {
throw new InvalidOperationException(Res.GetString(Res.Xml_InvalidOperation));
}
}
return base.ReadString();
}
//
// Partial Content Read Methods
//
// Gets a value indicating whether the current node
// has any attributes.
public override bool HasAttributes {
get {
return ( AttributeCount > 0 );
}
}
//
// Nametable and Namespace Helpers
//
// Gets the XmlNameTable associated with this implementation.
public override XmlNameTable NameTable {
get { return readerNav.NameTable; }
}
// Resolves a namespace prefix in the current element's scope.
public override String LookupNamespace(string prefix) {
if ( !IsInReadingStates() )
return null;
string ns = readerNav.LookupNamespace( prefix );
if (ns != null && ns.Length == 0) {
return null;
}
return ns;
}
// Resolves the entity reference for nodes of NodeType EntityReference.
public override void ResolveEntity() {
if ( !IsInReadingStates() || ( nodeType != XmlNodeType.EntityReference ) )
throw new InvalidOperationException(Res.GetString(Res.Xnr_ResolveEntity));
bResolveEntity = true;;
}
// Parses the attribute value into one or more Text and/or
// EntityReference node types.
public override bool ReadAttributeValue() {
if ( !IsInReadingStates() )
return false;
if ( readerNav.ReadAttributeValue( ref curDepth, ref bResolveEntity, ref nodeType ) ) {
bInReadBinary = false;
return true;
}
return false;
}
public override bool CanReadBinaryContent {
get {
return true;
}
}
public override int ReadContentAsBase64( byte[] buffer, int index, int count ) {
if ( readState != ReadState.Interactive ) {
return 0;
}
// init ReadContentAsBinaryHelper when called first time
if ( !bInReadBinary ) {
readBinaryHelper = ReadContentAsBinaryHelper.CreateOrReset( readBinaryHelper, this );
}
// turn off bInReadBinary in order to have a normal Read() behavior when called from readBinaryHelper
bInReadBinary = false;
// call to the helper
int readCount = readBinaryHelper.ReadContentAsBase64( buffer, index, count );
// turn on bInReadBinary in again and return
bInReadBinary = true;
return readCount;
}
public override int ReadContentAsBinHex( byte[] buffer, int index, int count ) {
if ( readState != ReadState.Interactive ) {
return 0;
}
// init ReadContentAsBinaryHelper when called first time
if ( !bInReadBinary ) {
readBinaryHelper = ReadContentAsBinaryHelper.CreateOrReset( readBinaryHelper, this );
}
// turn off bInReadBinary in order to have a normal Read() behavior when called from readBinaryHelper
bInReadBinary = false;
// call to the helper
int readCount = readBinaryHelper.ReadContentAsBinHex( buffer, index, count );
// turn on bInReadBinary in again and return
bInReadBinary = true;
return readCount;
}
public override int ReadElementContentAsBase64( byte[] buffer, int index, int count ) {
if ( readState != ReadState.Interactive ) {
return 0;
}
// init ReadContentAsBinaryHelper when called first time
if ( !bInReadBinary ) {
readBinaryHelper = ReadContentAsBinaryHelper.CreateOrReset( readBinaryHelper, this );
}
// turn off bInReadBinary in order to have a normal Read() behavior when called from readBinaryHelper
bInReadBinary = false;
// call to the helper
int readCount = readBinaryHelper.ReadElementContentAsBase64( buffer, index, count );
// turn on bInReadBinary in again and return
bInReadBinary = true;
return readCount;
}
public override int ReadElementContentAsBinHex( byte[] buffer, int index, int count ) {
if ( readState != ReadState.Interactive ) {
return 0;
}
// init ReadContentAsBinaryHelper when called first time
if ( !bInReadBinary ) {
readBinaryHelper = ReadContentAsBinaryHelper.CreateOrReset( readBinaryHelper, this );
}
// turn off bInReadBinary in order to have a normal Read() behavior when called from readBinaryHelper
bInReadBinary = false;
// call to the helper
int readCount = readBinaryHelper.ReadElementContentAsBinHex( buffer, index, count );
// turn on bInReadBinary in again and return
bInReadBinary = true;
return readCount;
}
void FinishReadBinary() {
bInReadBinary = false;
readBinaryHelper.Finish();
}
//
// IXmlNamespaceResolver
//
IDictionary IXmlNamespaceResolver.GetNamespacesInScope( XmlNamespaceScope scope ) {
return readerNav.GetNamespacesInScope( scope );
}
string IXmlNamespaceResolver.LookupPrefix( string namespaceName ) {
return readerNav.LookupPrefix( namespaceName );
}
String IXmlNamespaceResolver.LookupNamespace( string prefix ) {
if ( !IsInReadingStates() ) {
return readerNav.DefaultLookupNamespace( prefix );
}
string ns = readerNav.LookupNamespace( prefix );
if ( ns != null ) {
ns = readerNav.NameTable.Add( ns );
}
return ns;
}
// DTD/Schema info used by XmlReader.GetDtdSchemaInfo()
internal override IDtdInfo DtdInfo {
get {
return readerNav.Document.DtdSchemaInfo;
}
}
}
}