You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			1931 lines
		
	
	
		
			73 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			1931 lines
		
	
	
		
			73 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //------------------------------------------------------------------------------
 | |
| // <copyright file="XPathNavigator.cs" company="Microsoft">
 | |
| //     Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| // </copyright>
 | |
| // <owner current="true" primary="true">[....]</owner>
 | |
| //------------------------------------------------------------------------------
 | |
| 
 | |
| using System.ComponentModel;
 | |
| using System.IO;
 | |
| using System.Collections;
 | |
| using System.Collections.Generic;
 | |
| using System.Globalization;
 | |
| using System.Xml.Schema;
 | |
| using System.Diagnostics;
 | |
| using System.Diagnostics.Contracts;
 | |
| using System.Security;
 | |
| using System.Security.Policy;
 | |
| using System.Security.Permissions;
 | |
| using System.Text;
 | |
| using System.Xml;
 | |
| using MS.Internal.Xml.Cache;
 | |
| using MS.Internal.Xml.XPath;
 | |
| 
 | |
| namespace System.Xml.XPath {
 | |
|     // Provides a navigation interface API using XPath data model.
 | |
|     [DebuggerDisplay("{debuggerDisplayProxy}")]
 | |
| #if CONTRACTS_FULL
 | |
|     [ContractClass(typeof(XPathNavigatorContract))]
 | |
| #endif
 | |
|     public abstract class XPathNavigator : XPathItem, ICloneable, IXPathNavigable, IXmlNamespaceResolver {
 | |
|         internal static readonly XPathNavigatorKeyComparer comparer = new XPathNavigatorKeyComparer();
 | |
| 
 | |
|         //-----------------------------------------------
 | |
|         // Object
 | |
|         //-----------------------------------------------
 | |
| 
 | |
|         public override string ToString() {
 | |
|             return Value;
 | |
|         }
 | |
| 
 | |
|         //-----------------------------------------------
 | |
|         // XPathItem
 | |
|         //-----------------------------------------------
 | |
| 
 | |
|         public override sealed bool IsNode {
 | |
|             get { return true; }
 | |
|         }
 | |
| 
 | |
|         public override XmlSchemaType XmlType {
 | |
|             get {
 | |
|                 IXmlSchemaInfo schemaInfo = SchemaInfo;
 | |
|                 if (schemaInfo != null) {
 | |
|                     if (schemaInfo.Validity == XmlSchemaValidity.Valid) {
 | |
|                         XmlSchemaType memberType = schemaInfo.MemberType; 
 | |
|                         if (memberType != null) {
 | |
|                             return memberType; 
 | |
|                         }
 | |
|                         return schemaInfo.SchemaType;
 | |
|                     }
 | |
|                 }
 | |
|                 return null;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public virtual void SetValue(string value) {
 | |
|             throw new NotSupportedException();
 | |
|         }
 | |
| 
 | |
|         public override object TypedValue {
 | |
|             get {
 | |
|                 IXmlSchemaInfo schemaInfo = SchemaInfo;
 | |
|                 XmlSchemaType schemaType;
 | |
|                 XmlSchemaDatatype datatype;
 | |
|                 if (schemaInfo != null) {
 | |
|                     if (schemaInfo.Validity == XmlSchemaValidity.Valid) {
 | |
|                         schemaType = schemaInfo.MemberType;
 | |
|                         if (schemaType == null) {
 | |
|                             schemaType = schemaInfo.SchemaType;
 | |
|                         }
 | |
|                         if (schemaType != null) {
 | |
|                             datatype = schemaType.Datatype;
 | |
|                             if (datatype != null) {
 | |
|                                 return schemaType.ValueConverter.ChangeType(Value, datatype.ValueType, this);
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                     else {
 | |
|                         schemaType = schemaInfo.SchemaType;
 | |
|                         if (schemaType != null) {
 | |
|                             datatype = schemaType.Datatype;
 | |
|                             if (datatype != null) {
 | |
|                                 return schemaType.ValueConverter.ChangeType(datatype.ParseValue(Value, NameTable, this), datatype.ValueType, this);
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 return Value; 
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public virtual void SetTypedValue(object typedValue) {
 | |
|             if (typedValue == null) {
 | |
|                 throw new ArgumentNullException("typedValue");
 | |
|             }
 | |
|             switch (NodeType) {
 | |
|                 case XPathNodeType.Element:
 | |
|                 case XPathNodeType.Attribute:
 | |
|                     break;
 | |
|                 default:
 | |
|                     throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
 | |
|             }
 | |
|             string value = null; 
 | |
|             IXmlSchemaInfo schemaInfo = SchemaInfo;
 | |
|             if (schemaInfo != null) {
 | |
|                 XmlSchemaType schemaType = schemaInfo.SchemaType;
 | |
|                 if (schemaType != null) {
 | |
|                     value = schemaType.ValueConverter.ToString(typedValue, this);
 | |
|                     XmlSchemaDatatype datatype = schemaType.Datatype;
 | |
|                     if (datatype != null) {
 | |
|                         datatype.ParseValue(value, NameTable, this);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             if (value == null) {
 | |
|                 value = XmlUntypedConverter.Untyped.ToString(typedValue, this);
 | |
|             }
 | |
|             SetValue(value);
 | |
|         }
 | |
| 
 | |
|         public override Type ValueType {
 | |
|             get {
 | |
|                 IXmlSchemaInfo schemaInfo = SchemaInfo;
 | |
|                 XmlSchemaType schemaType;
 | |
|                 XmlSchemaDatatype datatype;
 | |
|                 if (schemaInfo != null) {
 | |
|                     if (schemaInfo.Validity == XmlSchemaValidity.Valid) {
 | |
|                         schemaType = schemaInfo.MemberType;
 | |
|                         if (schemaType == null) {
 | |
|                             schemaType = schemaInfo.SchemaType;
 | |
|                         }
 | |
|                         if (schemaType != null) {
 | |
|                             datatype = schemaType.Datatype;
 | |
|                             if (datatype != null) {
 | |
|                                 return datatype.ValueType;
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                     else {
 | |
|                         schemaType = schemaInfo.SchemaType;
 | |
|                         if (schemaType != null) {
 | |
|                             datatype = schemaType.Datatype;
 | |
|                             if (datatype != null) {
 | |
|                                 return datatype.ValueType;
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 return typeof(string);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public override bool ValueAsBoolean {
 | |
|             get {
 | |
|                 IXmlSchemaInfo schemaInfo = SchemaInfo;
 | |
|                 XmlSchemaType schemaType;
 | |
|                 XmlSchemaDatatype datatype;
 | |
|                 if (schemaInfo != null) {
 | |
|                     if (schemaInfo.Validity == XmlSchemaValidity.Valid) {
 | |
|                         schemaType = schemaInfo.MemberType;
 | |
|                         if (schemaType == null) {
 | |
|                             schemaType = schemaInfo.SchemaType;
 | |
|                         }
 | |
|                         if (schemaType != null) {
 | |
|                             return schemaType.ValueConverter.ToBoolean(Value);
 | |
|                         }
 | |
|                     }
 | |
|                     else {
 | |
|                         schemaType = schemaInfo.SchemaType;
 | |
|                         if (schemaType != null) {
 | |
|                             datatype = schemaType.Datatype;
 | |
|                             if (datatype != null) {
 | |
|                                 return schemaType.ValueConverter.ToBoolean(datatype.ParseValue(Value, NameTable, this));
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 return XmlUntypedConverter.Untyped.ToBoolean(Value);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public override DateTime ValueAsDateTime {
 | |
|             get {
 | |
|                 IXmlSchemaInfo schemaInfo = SchemaInfo;
 | |
|                 XmlSchemaType schemaType;
 | |
|                 XmlSchemaDatatype datatype;
 | |
|                 if (schemaInfo != null) {
 | |
|                     if (schemaInfo.Validity == XmlSchemaValidity.Valid) {
 | |
|                         schemaType = schemaInfo.MemberType;
 | |
|                         if (schemaType == null) {
 | |
|                             schemaType = schemaInfo.SchemaType;
 | |
|                         }
 | |
|                         if (schemaType != null) {
 | |
|                             return schemaType.ValueConverter.ToDateTime(Value);
 | |
|                         }
 | |
|                     }
 | |
|                     else {
 | |
|                         schemaType = schemaInfo.SchemaType;
 | |
|                         if (schemaType != null) {
 | |
|                             datatype = schemaType.Datatype;
 | |
|                             if (datatype != null) {
 | |
|                                 return schemaType.ValueConverter.ToDateTime(datatype.ParseValue(Value, NameTable, this));
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 return XmlUntypedConverter.Untyped.ToDateTime(Value);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public override double ValueAsDouble {
 | |
|             get {
 | |
|                 IXmlSchemaInfo schemaInfo = SchemaInfo;
 | |
|                 XmlSchemaType schemaType;
 | |
|                 XmlSchemaDatatype datatype;
 | |
|                 if (schemaInfo != null) {
 | |
|                     if (schemaInfo.Validity == XmlSchemaValidity.Valid) {
 | |
|                         schemaType = schemaInfo.MemberType;
 | |
|                         if (schemaType == null) {
 | |
|                             schemaType = schemaInfo.SchemaType;
 | |
|                         }
 | |
|                         if (schemaType != null) {
 | |
|                             return schemaType.ValueConverter.ToDouble(Value);
 | |
|                         }
 | |
|                     }
 | |
|                     else {
 | |
|                         schemaType = schemaInfo.SchemaType;
 | |
|                         if (schemaType != null) {
 | |
|                             datatype = schemaType.Datatype;
 | |
|                             if (datatype != null) {
 | |
|                                 return schemaType.ValueConverter.ToDouble(datatype.ParseValue(Value, NameTable, this));
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 return XmlUntypedConverter.Untyped.ToDouble(Value);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public override int ValueAsInt {
 | |
|             get {
 | |
|                 IXmlSchemaInfo schemaInfo = SchemaInfo;
 | |
|                 XmlSchemaType schemaType;
 | |
|                 XmlSchemaDatatype datatype;
 | |
|                 if (schemaInfo != null) {
 | |
|                     if (schemaInfo.Validity == XmlSchemaValidity.Valid) {
 | |
|                         schemaType = schemaInfo.MemberType;
 | |
|                         if (schemaType == null) {
 | |
|                             schemaType = schemaInfo.SchemaType;
 | |
|                         }
 | |
|                         if (schemaType != null) {
 | |
|                             return schemaType.ValueConverter.ToInt32(Value);
 | |
|                         }
 | |
|                     }
 | |
|                     else {
 | |
|                         schemaType = schemaInfo.SchemaType;
 | |
|                         if (schemaType != null) {
 | |
|                             datatype = schemaType.Datatype;
 | |
|                             if (datatype != null) {
 | |
|                                 return schemaType.ValueConverter.ToInt32(datatype.ParseValue(Value, NameTable, this));
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 return XmlUntypedConverter.Untyped.ToInt32(Value);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public override long ValueAsLong {
 | |
|             get {
 | |
|                 IXmlSchemaInfo schemaInfo = SchemaInfo;
 | |
|                 XmlSchemaType schemaType;
 | |
|                 XmlSchemaDatatype datatype;
 | |
|                 if (schemaInfo != null) {
 | |
|                     if (schemaInfo.Validity == XmlSchemaValidity.Valid) {
 | |
|                         schemaType = schemaInfo.MemberType;
 | |
|                         if (schemaType == null) {
 | |
|                             schemaType = schemaInfo.SchemaType;
 | |
|                         }
 | |
|                         if (schemaType != null) {
 | |
|                             return schemaType.ValueConverter.ToInt64(Value);
 | |
|                         }
 | |
|                     }
 | |
|                     else {
 | |
|                         schemaType = schemaInfo.SchemaType;
 | |
|                         if (schemaType != null) {
 | |
|                             datatype = schemaType.Datatype;
 | |
|                             if (datatype != null) {
 | |
|                                 return schemaType.ValueConverter.ToInt64(datatype.ParseValue(Value, NameTable, this));
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 return XmlUntypedConverter.Untyped.ToInt64(Value);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public override object ValueAs(Type returnType, IXmlNamespaceResolver nsResolver) {
 | |
|             if (nsResolver == null) {
 | |
|                 nsResolver = this;
 | |
|             }
 | |
|             IXmlSchemaInfo schemaInfo = SchemaInfo;
 | |
|             XmlSchemaType schemaType;
 | |
|             XmlSchemaDatatype datatype;
 | |
|             if (schemaInfo != null) {
 | |
|                 if (schemaInfo.Validity == XmlSchemaValidity.Valid) {
 | |
|                     schemaType = schemaInfo.MemberType;
 | |
|                     if (schemaType == null) {
 | |
|                         schemaType = schemaInfo.SchemaType;
 | |
|                     }
 | |
|                     if (schemaType != null) {
 | |
|                         return schemaType.ValueConverter.ChangeType(Value, returnType, nsResolver);
 | |
|                     }
 | |
|                 }
 | |
|                 else {
 | |
|                     schemaType = schemaInfo.SchemaType;
 | |
|                     if (schemaType != null) {
 | |
|                         datatype = schemaType.Datatype;
 | |
|                         if (datatype != null) {
 | |
|                             return schemaType.ValueConverter.ChangeType(datatype.ParseValue(Value, NameTable, nsResolver), returnType, nsResolver);
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             return XmlUntypedConverter.Untyped.ChangeType(Value, returnType, nsResolver);
 | |
|         }
 | |
| 
 | |
|         //-----------------------------------------------
 | |
|         // ICloneable
 | |
|         //-----------------------------------------------
 | |
| 
 | |
|         object ICloneable.Clone() {
 | |
|             return Clone();
 | |
|         }
 | |
| 
 | |
|         //-----------------------------------------------
 | |
|         // IXPathNavigable
 | |
|         //-----------------------------------------------
 | |
| 
 | |
|         public virtual XPathNavigator CreateNavigator() {
 | |
|             return Clone();
 | |
|         }
 | |
| 
 | |
|         //-----------------------------------------------
 | |
|         // IXmlNamespaceResolver
 | |
|         //-----------------------------------------------
 | |
| 
 | |
|         public abstract XmlNameTable NameTable { get; }
 | |
| 
 | |
|         public virtual string LookupNamespace(string prefix) {
 | |
|             if (prefix == null)
 | |
|                 return null;
 | |
| 
 | |
|             if (NodeType != XPathNodeType.Element) {
 | |
|                 XPathNavigator navSave = Clone();
 | |
| 
 | |
|                 // If current item is not an element, then try parent
 | |
|                 if (navSave.MoveToParent())
 | |
|                     return navSave.LookupNamespace(prefix);
 | |
|             }
 | |
|             else if (MoveToNamespace(prefix)) {
 | |
|                 string namespaceURI = Value;
 | |
|                 MoveToParent();
 | |
|                 return namespaceURI;
 | |
|             }
 | |
| 
 | |
|             // Check for "", "xml", and "xmlns" prefixes
 | |
|             if (prefix.Length == 0)
 | |
|                 return string.Empty;
 | |
|             else if (prefix == "xml")
 | |
|                 return XmlReservedNs.NsXml;
 | |
|             else if (prefix == "xmlns")
 | |
|                 return XmlReservedNs.NsXmlNs;
 | |
| 
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         public virtual string LookupPrefix(string namespaceURI) {
 | |
|             if (namespaceURI == null)
 | |
|                 return null;
 | |
| 
 | |
|             XPathNavigator navClone = Clone();
 | |
| 
 | |
|             if (NodeType != XPathNodeType.Element) {
 | |
|                 // If current item is not an element, then try parent
 | |
|                 if (navClone.MoveToParent())
 | |
|                     return navClone.LookupPrefix(namespaceURI);
 | |
|             }
 | |
|             else {
 | |
|                 if (navClone.MoveToFirstNamespace(XPathNamespaceScope.All)) {
 | |
|                     // Loop until a matching namespace is found
 | |
|                     do {
 | |
|                         if (namespaceURI == navClone.Value)
 | |
|                             return navClone.LocalName;
 | |
|                     }
 | |
|                     while (navClone.MoveToNextNamespace(XPathNamespaceScope.All));
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // Check for default, "xml", and "xmlns" namespaces
 | |
|             if (namespaceURI == LookupNamespace(string.Empty))
 | |
|                 return string.Empty;
 | |
|             else if (namespaceURI == XmlReservedNs.NsXml)
 | |
|                 return "xml";
 | |
|             else if (namespaceURI == XmlReservedNs.NsXmlNs)
 | |
|                 return "xmlns";
 | |
| 
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
| // This pragma disables a warning that the return type is not CLS-compliant, but generics are part of CLS in Whidbey. 
 | |
| #pragma warning disable 3002
 | |
|         public virtual IDictionary<string,string> GetNamespacesInScope(XmlNamespaceScope scope) {
 | |
| #pragma warning restore 3002
 | |
|             XPathNodeType nt = NodeType;
 | |
|             if ((nt != XPathNodeType.Element && scope != XmlNamespaceScope.Local) || nt == XPathNodeType.Attribute || nt == XPathNodeType.Namespace) {
 | |
|                 XPathNavigator navSave = Clone();
 | |
| 
 | |
|                 // If current item is not an element, then try parent
 | |
|                 if (navSave.MoveToParent())
 | |
|                     return navSave.GetNamespacesInScope(scope);
 | |
|             }
 | |
| 
 | |
|             Dictionary<string,string> dict = new Dictionary<string,string>();
 | |
| 
 | |
|             // "xml" prefix always in scope
 | |
|             if (scope == XmlNamespaceScope.All)
 | |
|                 dict["xml"] = XmlReservedNs.NsXml;
 | |
| 
 | |
|             // Now add all in-scope namespaces
 | |
|             if (MoveToFirstNamespace((XPathNamespaceScope) scope)) {
 | |
|                 do {
 | |
|                     string prefix = LocalName;
 | |
|                     string ns = Value;
 | |
| 
 | |
|                     // Exclude xmlns="" declarations unless scope = Local
 | |
|                     if (prefix.Length != 0 || ns.Length != 0 || scope == XmlNamespaceScope.Local)
 | |
|                         dict[prefix] = ns;
 | |
|                 }
 | |
|                 while (MoveToNextNamespace((XPathNamespaceScope) scope));
 | |
| 
 | |
|                 MoveToParent();
 | |
|             }
 | |
| 
 | |
|             return dict;
 | |
|         }
 | |
| 
 | |
|         //-----------------------------------------------
 | |
|         // XPathNavigator
 | |
|         //-----------------------------------------------
 | |
| 
 | |
|         // Returns an object of type IKeyComparer. Using this the navigators can be hashed
 | |
|         // on the basis of actual position it represents rather than the clr reference of 
 | |
|         // the navigator object.
 | |
|         public static IEqualityComparer NavigatorComparer {
 | |
|             get { return comparer; }
 | |
|         }
 | |
| 
 | |
|         public abstract XPathNavigator Clone();
 | |
| 
 | |
|         public abstract XPathNodeType NodeType { get; }
 | |
| 
 | |
|         public abstract string LocalName { get; }
 | |
| 
 | |
|         public abstract string Name { get; }
 | |
| 
 | |
|         public abstract string NamespaceURI { get; }
 | |
| 
 | |
|         public abstract string Prefix { get; }
 | |
| 
 | |
|         public abstract string BaseURI { get; }
 | |
| 
 | |
|         public abstract bool IsEmptyElement { get; }
 | |
| 
 | |
|         public virtual string XmlLang {
 | |
|             get {
 | |
|                 XPathNavigator navClone = Clone();
 | |
|                 do {
 | |
|                     if (navClone.MoveToAttribute("lang", XmlReservedNs.NsXml))
 | |
|                         return navClone.Value;
 | |
|                 }
 | |
|                 while (navClone.MoveToParent());
 | |
| 
 | |
|                 return string.Empty;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public virtual XmlReader ReadSubtree() {
 | |
|             switch (NodeType) {
 | |
|                 case XPathNodeType.Root:
 | |
|                 case XPathNodeType.Element:
 | |
|                     break;
 | |
|                 default:
 | |
|                     throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
 | |
|             }
 | |
|             return CreateReader(); 
 | |
|         }
 | |
| 
 | |
|         public virtual void WriteSubtree(XmlWriter writer) {
 | |
|             if (null == writer)
 | |
|                 throw new ArgumentNullException("writer");
 | |
|             writer.WriteNode(this, true);
 | |
|         }
 | |
| 
 | |
|         public virtual object UnderlyingObject {
 | |
|             get { return null; }
 | |
|         }
 | |
| 
 | |
|         public virtual bool HasAttributes {
 | |
|             get {
 | |
|                 if (!MoveToFirstAttribute())
 | |
|                     return false;
 | |
| 
 | |
|                 MoveToParent();
 | |
|                 return true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public virtual string GetAttribute(string localName, string namespaceURI) {
 | |
|             string value;
 | |
| 
 | |
|             if (!MoveToAttribute(localName, namespaceURI))
 | |
|                 return "";
 | |
| 
 | |
|             value = Value;
 | |
|             MoveToParent();
 | |
| 
 | |
|             return value;
 | |
|         }
 | |
| 
 | |
|         public virtual bool MoveToAttribute(string localName, string namespaceURI) {
 | |
|             if (MoveToFirstAttribute()) {
 | |
|                 do {
 | |
|                     if (localName == LocalName && namespaceURI == NamespaceURI)
 | |
|                         return true;
 | |
|                 }
 | |
|                 while (MoveToNextAttribute());
 | |
| 
 | |
|                 MoveToParent();
 | |
|             }
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         public abstract bool MoveToFirstAttribute();
 | |
| 
 | |
|         public abstract bool MoveToNextAttribute();
 | |
| 
 | |
|         public virtual string GetNamespace(string name) {
 | |
|             string value;
 | |
| 
 | |
|             if (!MoveToNamespace(name)) {
 | |
|                 if (name == "xml")
 | |
|                     return XmlReservedNs.NsXml;
 | |
|                 if (name == "xmlns")
 | |
|                     return XmlReservedNs.NsXmlNs;
 | |
|                 return string.Empty;
 | |
|             }
 | |
| 
 | |
|             value = Value;
 | |
|             MoveToParent();
 | |
| 
 | |
|             return value;
 | |
|         }
 | |
| 
 | |
|         public virtual bool MoveToNamespace(string name) {
 | |
|             if (MoveToFirstNamespace(XPathNamespaceScope.All)) {
 | |
| 
 | |
|                 do {
 | |
|                     if (name == LocalName)
 | |
|                         return true;
 | |
|                 }
 | |
|                 while (MoveToNextNamespace(XPathNamespaceScope.All));
 | |
| 
 | |
|                 MoveToParent();
 | |
|             }
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         public abstract bool MoveToFirstNamespace(XPathNamespaceScope namespaceScope);
 | |
| 
 | |
|         public abstract bool MoveToNextNamespace(XPathNamespaceScope namespaceScope);
 | |
| 
 | |
|         public bool MoveToFirstNamespace() { return MoveToFirstNamespace(XPathNamespaceScope.All); }
 | |
| 
 | |
|         public bool MoveToNextNamespace() { return MoveToNextNamespace(XPathNamespaceScope.All); }
 | |
| 
 | |
|         public abstract bool MoveToNext();
 | |
| 
 | |
|         public abstract bool MoveToPrevious();
 | |
| 
 | |
|         public virtual bool MoveToFirst() {
 | |
|             switch (NodeType) {
 | |
|                 case XPathNodeType.Attribute:
 | |
|                 case XPathNodeType.Namespace:
 | |
|                     // MoveToFirst should only succeed for content-typed nodes
 | |
|                     return false;
 | |
|             }
 | |
| 
 | |
|             if (!MoveToParent())
 | |
|                 return false;
 | |
| 
 | |
|             return MoveToFirstChild();
 | |
|         }
 | |
| 
 | |
|         public abstract bool MoveToFirstChild();
 | |
| 
 | |
|         public abstract bool MoveToParent();
 | |
| 
 | |
|         public virtual void MoveToRoot() {
 | |
|             while (MoveToParent())
 | |
|                 ;
 | |
|         }
 | |
| 
 | |
|         public abstract bool MoveTo(XPathNavigator other);
 | |
| 
 | |
|         public abstract bool MoveToId(string id);
 | |
| 
 | |
|         public virtual bool MoveToChild(string localName, string namespaceURI) {
 | |
|             if (MoveToFirstChild()) {
 | |
|                 do {
 | |
|                     if (NodeType == XPathNodeType.Element && localName == LocalName && namespaceURI == NamespaceURI)
 | |
|                         return true;
 | |
|                 }
 | |
|                 while (MoveToNext());
 | |
|                 MoveToParent();
 | |
|             }
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         public virtual bool MoveToChild(XPathNodeType type) {
 | |
|             if (MoveToFirstChild()) {
 | |
|                 int mask = GetContentKindMask(type);
 | |
|                 do {
 | |
|                     if (((1 << (int) NodeType) & mask) != 0)
 | |
|                         return true;
 | |
|                 }
 | |
|                 while (MoveToNext());
 | |
| 
 | |
|                 MoveToParent();
 | |
|             }
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         public virtual bool MoveToFollowing(string localName, string namespaceURI) {
 | |
|             return MoveToFollowing(localName, namespaceURI, null);
 | |
|         }
 | |
| 
 | |
|         public virtual bool MoveToFollowing(string localName, string namespaceURI, XPathNavigator end) {
 | |
|             XPathNavigator navSave = Clone();
 | |
| 
 | |
|             if (end != null) {
 | |
|                 switch (end.NodeType) {
 | |
|                     case XPathNodeType.Attribute:
 | |
|                     case XPathNodeType.Namespace:
 | |
|                         // Scan until we come to the next content-typed node 
 | |
|                         // after the attribute or namespace node
 | |
|                         end = end.Clone();
 | |
|                         end.MoveToNonDescendant();
 | |
|                         break;
 | |
|                 }
 | |
|             }
 | |
|             switch (NodeType) {
 | |
|                 case XPathNodeType.Attribute:
 | |
|                 case XPathNodeType.Namespace:
 | |
|                     if (!MoveToParent()) {
 | |
|                         // Restore previous position and return false
 | |
|                         // MoveTo(navSave);
 | |
|                         return false;
 | |
|                     }
 | |
|                     break;
 | |
|             }
 | |
|             do {
 | |
|                 if (!MoveToFirstChild()) {
 | |
|                     // Look for next sibling
 | |
|                     while (true) {
 | |
|                         if (MoveToNext())
 | |
|                             break;
 | |
| 
 | |
|                         if (!MoveToParent()) {
 | |
|                             // Restore previous position and return false
 | |
|                             MoveTo(navSave);
 | |
|                             return false;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 // Have we reached the end of the scan?
 | |
|                 if (end != null && IsSamePosition(end)) {
 | |
|                     // Restore previous position and return false
 | |
|                     MoveTo(navSave);
 | |
|                     return false;
 | |
|                 }
 | |
|             }
 | |
|             while (NodeType != XPathNodeType.Element 
 | |
|                    || localName != LocalName 
 | |
|                    || namespaceURI != NamespaceURI);
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         public virtual bool MoveToFollowing(XPathNodeType type) {
 | |
|             return MoveToFollowing(type, null);
 | |
|         }
 | |
| 
 | |
|         public virtual bool MoveToFollowing(XPathNodeType type, XPathNavigator end) {
 | |
|             XPathNavigator navSave = Clone();
 | |
|             int mask = GetContentKindMask(type);
 | |
| 
 | |
|             if (end != null) {
 | |
|                 switch (end.NodeType) {
 | |
|                     case XPathNodeType.Attribute:
 | |
|                     case XPathNodeType.Namespace:
 | |
|                         // Scan until we come to the next content-typed node 
 | |
|                         // after the attribute or namespace node
 | |
|                         end = end.Clone();
 | |
|                         end.MoveToNonDescendant();
 | |
|                         break;
 | |
|                 }
 | |
|             }
 | |
|             switch (NodeType) {
 | |
|                 case XPathNodeType.Attribute:
 | |
|                 case XPathNodeType.Namespace:
 | |
|                     if (!MoveToParent()) {
 | |
|                         // Restore previous position and return false
 | |
|                         // MoveTo(navSave);
 | |
|                         return false;
 | |
|                     }
 | |
|                     break;
 | |
|             }
 | |
|             do {
 | |
|                 if (!MoveToFirstChild()) {
 | |
|                     // Look for next sibling
 | |
|                     while (true) {
 | |
|                         if (MoveToNext())
 | |
|                             break;
 | |
| 
 | |
|                         if (!MoveToParent()) {
 | |
|                             // Restore previous position and return false
 | |
|                             MoveTo(navSave);
 | |
|                             return false;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 // Have we reached the end of the scan?
 | |
|                 if (end != null && IsSamePosition(end)) {
 | |
|                     // Restore previous position and return false
 | |
|                     MoveTo(navSave);
 | |
|                     return false;
 | |
|                 }
 | |
|             }
 | |
|             while (((1 << (int) NodeType) & mask) == 0);
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         public virtual bool MoveToNext(string localName, string namespaceURI) {
 | |
|             XPathNavigator navClone = Clone();
 | |
| 
 | |
|             while (MoveToNext()) {
 | |
|                 if (NodeType == XPathNodeType.Element && localName == LocalName && namespaceURI == NamespaceURI)
 | |
|                     return true;
 | |
|             }
 | |
|             MoveTo(navClone);
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         public virtual bool MoveToNext(XPathNodeType type) {
 | |
|             XPathNavigator navClone = Clone();
 | |
|             int mask = GetContentKindMask(type);
 | |
| 
 | |
|             while (MoveToNext()) {
 | |
|                 if (((1 << (int) NodeType) & mask) != 0)
 | |
|                     return true;
 | |
|             }
 | |
| 
 | |
|             MoveTo(navClone);
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         public virtual bool HasChildren {
 | |
|             get {
 | |
|                 if (MoveToFirstChild()) {
 | |
|                     MoveToParent();
 | |
|                     return true;
 | |
|                 }
 | |
|                 return false;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public abstract bool IsSamePosition(XPathNavigator other);
 | |
| 
 | |
|         public virtual bool IsDescendant(XPathNavigator nav) {
 | |
|             if (nav != null){
 | |
|                 nav = nav.Clone();
 | |
|                 while ( nav.MoveToParent() )
 | |
|                     if (nav.IsSamePosition(this))
 | |
|                         return true;
 | |
|             }
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         public virtual XmlNodeOrder ComparePosition( XPathNavigator nav ) {
 | |
|             if (nav == null) {
 | |
|                 return XmlNodeOrder.Unknown;
 | |
|             }
 | |
| 
 | |
|             if( IsSamePosition( nav ) )
 | |
|                 return XmlNodeOrder.Same;
 | |
| 
 | |
|             XPathNavigator n1 = this.Clone();
 | |
|             XPathNavigator n2 = nav.Clone();
 | |
| 
 | |
|             int depth1 = GetDepth( n1.Clone() );
 | |
|             int depth2 = GetDepth( n2.Clone() );
 | |
| 
 | |
|             if( depth1 > depth2 ) {
 | |
|                 while( depth1 > depth2 ) {
 | |
|                     n1.MoveToParent();
 | |
|                     depth1--;
 | |
|                 }
 | |
|                 if( n1.IsSamePosition(n2) )
 | |
|                     return XmlNodeOrder.After;
 | |
|             }
 | |
| 
 | |
|             if( depth2 > depth1 ) {
 | |
|                 while( depth2 > depth1 ) {
 | |
|                     n2.MoveToParent();
 | |
|                     depth2 --;
 | |
|                 }
 | |
|                 if( n1.IsSamePosition(n2) )
 | |
|                     return XmlNodeOrder.Before;
 | |
|             }
 | |
| 
 | |
|             XPathNavigator parent1 = n1.Clone();
 | |
|             XPathNavigator parent2 = n2.Clone();
 | |
| 
 | |
|             while( true ) {
 | |
|                 if( !parent1.MoveToParent() || !parent2.MoveToParent() )
 | |
|                     return XmlNodeOrder.Unknown;
 | |
| 
 | |
|                 if( parent1.IsSamePosition( parent2 ) ) {
 | |
|                     if (n1.GetType().ToString() != "Microsoft.VisualStudio.Modeling.StoreNavigator") {
 | |
|                         Debug.Assert( CompareSiblings(n1.Clone(), n2.Clone()) != CompareSiblings(n2.Clone(), n1.Clone()), "IsSamePosition() on custom navigator returns incosistent results" );
 | |
|                     }
 | |
|                     return CompareSiblings(n1, n2);
 | |
|                 }
 | |
| 
 | |
|                 n1.MoveToParent();
 | |
|                 n2.MoveToParent();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public virtual IXmlSchemaInfo SchemaInfo {
 | |
|             get { return this as IXmlSchemaInfo; }
 | |
|         }
 | |
| 
 | |
|         public virtual bool CheckValidity(XmlSchemaSet schemas, ValidationEventHandler validationEventHandler) {
 | |
|             IXmlSchemaInfo schemaInfo;
 | |
|             XmlSchemaType schemaType = null;
 | |
|             XmlSchemaElement schemaElement = null;
 | |
|             XmlSchemaAttribute schemaAttribute = null;
 | |
| 
 | |
|             switch (NodeType) {
 | |
|                 case XPathNodeType.Root:
 | |
|                     if (schemas == null) {
 | |
|                         throw new InvalidOperationException(Res.GetString(Res.XPathDocument_MissingSchemas));
 | |
|                     }
 | |
|                     schemaType = null;
 | |
|                     break;
 | |
|                 case XPathNodeType.Element:
 | |
|                     if (schemas == null) {
 | |
|                         throw new InvalidOperationException(Res.GetString(Res.XPathDocument_MissingSchemas)); 
 | |
|                     }
 | |
|                     schemaInfo = SchemaInfo;
 | |
|                     if (schemaInfo != null) {
 | |
|                         schemaType = schemaInfo.SchemaType;
 | |
|                         schemaElement = schemaInfo.SchemaElement;
 | |
|                     }
 | |
|                     if (schemaType == null
 | |
|                         && schemaElement == null) {
 | |
|                         throw new InvalidOperationException(Res.GetString(Res.XPathDocument_NotEnoughSchemaInfo, null));
 | |
|                     }
 | |
|                     break;
 | |
|                 case XPathNodeType.Attribute:
 | |
|                     if (schemas == null) {
 | |
|                         throw new InvalidOperationException(Res.GetString(Res.XPathDocument_MissingSchemas)); 
 | |
|                     }
 | |
|                     schemaInfo = SchemaInfo;
 | |
|                     if (schemaInfo != null) {
 | |
|                         schemaType = schemaInfo.SchemaType;
 | |
|                         schemaAttribute = schemaInfo.SchemaAttribute;
 | |
|                     }
 | |
|                     if (schemaType == null
 | |
|                         && schemaAttribute == null) {
 | |
|                         throw new InvalidOperationException(Res.GetString(Res.XPathDocument_NotEnoughSchemaInfo, null));
 | |
|                     }
 | |
|                     break;
 | |
|                 default:
 | |
|                     throw new InvalidOperationException(Res.GetString(Res.XPathDocument_ValidateInvalidNodeType, null));
 | |
| 
 | |
|             }
 | |
|            
 | |
|             Debug.Assert( schemaType != null  || this.NodeType == XPathNodeType.Root, "schemaType != null  || this.NodeType == XPathNodeType.Root" );
 | |
| 
 | |
|             XmlReader reader = CreateReader();
 | |
| 
 | |
|             CheckValidityHelper validityTracker = new CheckValidityHelper( validationEventHandler, reader as XPathNavigatorReader );
 | |
|             validationEventHandler = new ValidationEventHandler( validityTracker.ValidationCallback );
 | |
|             XmlReader validatingReader = GetValidatingReader( reader, schemas, validationEventHandler, schemaType, schemaElement, schemaAttribute );
 | |
| 
 | |
|             while( validatingReader.Read() )
 | |
|                 ;
 | |
| 
 | |
|             return validityTracker.IsValid;
 | |
|         }
 | |
| 
 | |
|         private XmlReader GetValidatingReader( XmlReader reader, XmlSchemaSet schemas, ValidationEventHandler validationEvent, XmlSchemaType schemaType, XmlSchemaElement schemaElement, XmlSchemaAttribute schemaAttribute ) {
 | |
|             if (schemaAttribute != null) {
 | |
|                 return schemaAttribute.Validate(reader, null, schemas, validationEvent);
 | |
|             }
 | |
|             else if (schemaElement != null) {
 | |
|                 return schemaElement.Validate(reader, null, schemas, validationEvent);
 | |
|             }
 | |
|             else if (schemaType != null) {
 | |
|                 return schemaType.Validate(reader, null, schemas, validationEvent);
 | |
|             }
 | |
|             Debug.Assert( schemas != null, "schemas != null" );
 | |
|             XmlReaderSettings readerSettings = new XmlReaderSettings();
 | |
|             readerSettings.ConformanceLevel = ConformanceLevel.Auto;
 | |
|             readerSettings.ValidationType = ValidationType.Schema;
 | |
|             readerSettings.Schemas = schemas;
 | |
|             readerSettings.ValidationEventHandler += validationEvent;
 | |
|             return XmlReader.Create( reader, readerSettings );
 | |
|         }
 | |
| 
 | |
|         class CheckValidityHelper {
 | |
|             bool isValid;
 | |
|             ValidationEventHandler nextEventHandler;
 | |
|             XPathNavigatorReader reader;
 | |
|             
 | |
|             internal CheckValidityHelper( ValidationEventHandler nextEventHandler, XPathNavigatorReader reader ) {
 | |
|                 this.isValid = true;
 | |
|                 this.nextEventHandler = nextEventHandler;
 | |
|                 this.reader = reader;
 | |
|             }
 | |
| 
 | |
|             internal void ValidationCallback( object sender, ValidationEventArgs args ) {
 | |
|                 Debug.Assert( args != null );
 | |
|                 if ( args.Severity == XmlSeverityType.Error )
 | |
|                     this.isValid = false;
 | |
|                 XmlSchemaValidationException exception = args.Exception as XmlSchemaValidationException;
 | |
|                 if (exception != null && reader != null)
 | |
|                     exception.SetSourceObject(reader.UnderlyingObject);
 | |
| 
 | |
|                 if (this.nextEventHandler != null) {
 | |
|                     this.nextEventHandler( sender, args );
 | |
|                 }
 | |
|                 else if (exception != null && args.Severity == XmlSeverityType.Error) {
 | |
|                     throw exception;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             internal bool IsValid {
 | |
|                 get { return this.isValid; }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public virtual XPathExpression Compile(string xpath) {
 | |
|             return XPathExpression.Compile(xpath);
 | |
|         }
 | |
| 
 | |
|         public virtual XPathNavigator SelectSingleNode(string xpath) {
 | |
|             return SelectSingleNode(XPathExpression.Compile(xpath));
 | |
|         }
 | |
| 
 | |
|         public virtual XPathNavigator SelectSingleNode(string xpath, IXmlNamespaceResolver resolver) {
 | |
|             return SelectSingleNode(XPathExpression.Compile(xpath, resolver));
 | |
|         }
 | |
| 
 | |
|         public virtual XPathNavigator SelectSingleNode(XPathExpression expression) {
 | |
|             // 
 | |
|             XPathNodeIterator iter = this.Select(expression);
 | |
|             if (iter.MoveNext()) {
 | |
|                 return iter.Current;
 | |
|             }
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         public virtual XPathNodeIterator Select(string xpath) {
 | |
|             Contract.Ensures(Contract.Result<XPathNodeIterator>() != null);
 | |
| 
 | |
|             return this.Select(XPathExpression.Compile(xpath));
 | |
|         }
 | |
| 
 | |
|         public virtual XPathNodeIterator Select(string xpath, IXmlNamespaceResolver resolver) {
 | |
|             Contract.Ensures(Contract.Result<XPathNodeIterator>() != null);
 | |
| 
 | |
|             return this.Select(XPathExpression.Compile(xpath, resolver));
 | |
|         }
 | |
| 
 | |
|         public virtual XPathNodeIterator Select(XPathExpression expr) {
 | |
|             Contract.Ensures(Contract.Result<XPathNodeIterator>() != null);
 | |
| 
 | |
|             XPathNodeIterator result = Evaluate(expr) as XPathNodeIterator;
 | |
|             if (result == null) {
 | |
|                 throw XPathException.Create(Res.Xp_NodeSetExpected);
 | |
|             }
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         public virtual object Evaluate(string xpath) {
 | |
|             return Evaluate(XPathExpression.Compile(xpath), null);
 | |
|         }
 | |
| 
 | |
|         public virtual object Evaluate(string xpath, IXmlNamespaceResolver resolver) {
 | |
|             return this.Evaluate(XPathExpression.Compile(xpath, resolver));
 | |
|         }
 | |
| 
 | |
|         public virtual object Evaluate(XPathExpression expr) {
 | |
|             return Evaluate(expr, null);
 | |
|         }
 | |
| 
 | |
|         public virtual object Evaluate(XPathExpression expr, XPathNodeIterator context) {
 | |
|             CompiledXpathExpr cexpr = expr as CompiledXpathExpr;
 | |
|             if (cexpr == null) {
 | |
|                 throw XPathException.Create(Res.Xp_BadQueryObject);
 | |
|             }
 | |
|             Query query = Query.Clone(cexpr.QueryTree);
 | |
|             query.Reset();
 | |
| 
 | |
|             if (context == null) {
 | |
|                 context = new XPathSingletonIterator(this.Clone(), /*moved:*/true);
 | |
|             }
 | |
| 
 | |
|             object result = query.Evaluate(context);
 | |
| 
 | |
|             if (result is XPathNodeIterator) {
 | |
|                 return new XPathSelectionIterator(context.Current, query);
 | |
|             }
 | |
| 
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         public virtual bool Matches( XPathExpression expr ) {
 | |
|             CompiledXpathExpr cexpr = expr as CompiledXpathExpr;
 | |
|             if( cexpr == null )
 | |
|                 throw XPathException.Create(Res.Xp_BadQueryObject);
 | |
| 
 | |
|             // We should clone query because some Query.MatchNode() alter expression state and this may brake
 | |
|             // SelectionIterators that are runing using this Query
 | |
|             // Excample of MatchNode() that alret the state is FilterQuery.MatchNode()
 | |
|             Query query = Query.Clone(cexpr.QueryTree);  
 | |
| 
 | |
|             try {
 | |
|                 return query.MatchNode(this) != null;
 | |
|             }
 | |
|             catch(XPathException) {
 | |
|                 throw XPathException.Create(Res.Xp_InvalidPattern, cexpr.Expression);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public virtual bool Matches(string xpath) {
 | |
|             return Matches(CompileMatchPattern(xpath));
 | |
|         }
 | |
| 
 | |
|         public virtual XPathNodeIterator SelectChildren( XPathNodeType type ) {
 | |
|             return new XPathChildIterator( this.Clone(), type );
 | |
|         }
 | |
| 
 | |
|         public virtual XPathNodeIterator SelectChildren( string name, string namespaceURI ) {
 | |
|             return new XPathChildIterator( this.Clone(), name, namespaceURI );
 | |
|         }
 | |
| 
 | |
|         public virtual XPathNodeIterator SelectAncestors( XPathNodeType type, bool matchSelf ) {
 | |
|             return new XPathAncestorIterator( this.Clone(), type, matchSelf );
 | |
|         }
 | |
| 
 | |
|         public virtual XPathNodeIterator SelectAncestors( string name, string namespaceURI, bool matchSelf ) {
 | |
|             return new XPathAncestorIterator( this.Clone(), name, namespaceURI, matchSelf );
 | |
|         }
 | |
| 
 | |
|         public virtual XPathNodeIterator SelectDescendants( XPathNodeType type, bool matchSelf ) {
 | |
|             return new XPathDescendantIterator( this.Clone(), type, matchSelf );
 | |
|         }
 | |
| 
 | |
|         public virtual XPathNodeIterator SelectDescendants( string name, string namespaceURI, bool matchSelf ) {
 | |
|             return new XPathDescendantIterator( this.Clone(), name, namespaceURI, matchSelf );
 | |
|         }
 | |
| 
 | |
|         public virtual bool CanEdit {
 | |
|             get {
 | |
|                 return false;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public virtual XmlWriter PrependChild() {
 | |
|             throw new NotSupportedException();
 | |
|         }
 | |
| 
 | |
|         public virtual XmlWriter AppendChild() {
 | |
|             throw new NotSupportedException();
 | |
|         }
 | |
| 
 | |
|         public virtual XmlWriter InsertAfter() {
 | |
|             throw new NotSupportedException();
 | |
|         }
 | |
| 
 | |
|         public virtual XmlWriter InsertBefore() {
 | |
|             throw new NotSupportedException();
 | |
|         }
 | |
| 
 | |
|         public virtual XmlWriter CreateAttributes() {
 | |
|             throw new NotSupportedException();
 | |
|         }
 | |
| 
 | |
|         public virtual XmlWriter ReplaceRange(XPathNavigator lastSiblingToReplace) {
 | |
|             throw new NotSupportedException();
 | |
|         }
 | |
| 
 | |
|         public virtual void ReplaceSelf(string newNode) {
 | |
|             XmlReader reader = CreateContextReader(newNode, false);
 | |
|             ReplaceSelf(reader);
 | |
|         }
 | |
| 
 | |
|         public virtual void ReplaceSelf(XmlReader newNode) {
 | |
|             if (newNode == null) {
 | |
|                 throw new ArgumentNullException("newNode");
 | |
|             }
 | |
|             XPathNodeType type = NodeType;
 | |
|             if (type == XPathNodeType.Root
 | |
|                 || type == XPathNodeType.Attribute
 | |
|                 || type == XPathNodeType.Namespace) {
 | |
|                 throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
 | |
|             }
 | |
|             XmlWriter writer = ReplaceRange(this);
 | |
|             BuildSubtree(newNode, writer);
 | |
|             writer.Close();
 | |
|         }
 | |
| 
 | |
|         public virtual void ReplaceSelf(XPathNavigator newNode) {
 | |
|             if (newNode == null) {
 | |
|                 throw new ArgumentNullException("newNode");
 | |
|             }
 | |
|             XmlReader reader = newNode.CreateReader();
 | |
|             ReplaceSelf(reader);
 | |
|         }
 | |
| 
 | |
|         // Returns the markup representing the current node and all of its children.
 | |
|         public virtual string OuterXml {
 | |
|             get {
 | |
|                 StringWriter stringWriter;
 | |
|                 XmlWriterSettings writerSettings;
 | |
|                 XmlWriter xmlWriter;
 | |
| 
 | |
|                 // Attributes and namespaces are not allowed at the top-level by the well-formed writer
 | |
|                 if (NodeType == XPathNodeType.Attribute) {
 | |
|                     return string.Concat(Name, "=\"", Value, "\"");
 | |
|                 }
 | |
|                 else if (NodeType == XPathNodeType.Namespace) {
 | |
|                     if (LocalName.Length == 0)
 | |
|                         return string.Concat("xmlns=\"", Value, "\"");
 | |
|                     else
 | |
|                         return string.Concat("xmlns:", LocalName, "=\"", Value, "\"");
 | |
|                 }
 | |
| 
 | |
|                 stringWriter = new StringWriter(CultureInfo.InvariantCulture);
 | |
| 
 | |
|                 writerSettings = new XmlWriterSettings();
 | |
|                 writerSettings.Indent = true;
 | |
|                 writerSettings.OmitXmlDeclaration = true;
 | |
|                 writerSettings.ConformanceLevel = ConformanceLevel.Auto;
 | |
| 
 | |
|                 xmlWriter = XmlWriter.Create(stringWriter, writerSettings);
 | |
|                 try {
 | |
|                     xmlWriter.WriteNode(this, true);
 | |
|                 }
 | |
|                 finally {
 | |
|                     xmlWriter.Close();
 | |
|                 }
 | |
| 
 | |
|                 return stringWriter.ToString();
 | |
|             }
 | |
| 
 | |
|             set {
 | |
|                 ReplaceSelf(value);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Returns the markup representing just the children of the current node.
 | |
|         public virtual string InnerXml {
 | |
|             get {
 | |
|                 switch (NodeType) {
 | |
|                     case XPathNodeType.Root:
 | |
|                     case XPathNodeType.Element:
 | |
|                         StringWriter stringWriter;
 | |
|                         XmlWriterSettings writerSettings;
 | |
|                         XmlWriter xmlWriter;
 | |
| 
 | |
|                         stringWriter = new StringWriter(CultureInfo.InvariantCulture);
 | |
| 
 | |
|                         writerSettings = new XmlWriterSettings();
 | |
|                         writerSettings.Indent = true;
 | |
|                         writerSettings.OmitXmlDeclaration = true;
 | |
|                         writerSettings.ConformanceLevel = ConformanceLevel.Auto;
 | |
|                         xmlWriter = XmlWriter.Create(stringWriter, writerSettings);
 | |
| 
 | |
|                         try {
 | |
|                             if (MoveToFirstChild()) {
 | |
|                                 do {
 | |
|                                     xmlWriter.WriteNode(this, true);
 | |
|                                 }
 | |
|                                 while (MoveToNext());
 | |
| 
 | |
|                                 // Restore position
 | |
|                                 MoveToParent();
 | |
|                             }
 | |
|                         }
 | |
|                         finally {
 | |
|                             xmlWriter.Close();
 | |
|                         }
 | |
|                         return stringWriter.ToString();
 | |
|                     case XPathNodeType.Attribute:
 | |
|                     case XPathNodeType.Namespace:
 | |
|                         return Value;
 | |
|                     default:
 | |
|                         return string.Empty;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             set {
 | |
|                 if (value == null) {
 | |
|                     throw new ArgumentNullException("value");
 | |
|                 }
 | |
| 
 | |
|                 switch (NodeType) {
 | |
|                     case XPathNodeType.Root:
 | |
|                     case XPathNodeType.Element:
 | |
|                         XPathNavigator edit = CreateNavigator();
 | |
|                         while (edit.MoveToFirstChild()) {
 | |
|                             edit.DeleteSelf();
 | |
|                         }
 | |
|                         if (value.Length != 0) {
 | |
|                             edit.AppendChild(value);
 | |
|                         }
 | |
|                         break;
 | |
|                     case XPathNodeType.Attribute:
 | |
|                         SetValue(value);
 | |
|                         break;
 | |
|                     default:
 | |
|                         throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public virtual void AppendChild(string newChild) {
 | |
|             XmlReader reader = CreateContextReader(newChild, true);
 | |
|             AppendChild(reader);
 | |
|         }
 | |
| 
 | |
|         public virtual void AppendChild(XmlReader newChild) {
 | |
|             if (newChild == null) {
 | |
|                 throw new ArgumentNullException("newChild");
 | |
|             }
 | |
|             XmlWriter writer = AppendChild();
 | |
|             BuildSubtree(newChild, writer);
 | |
|             writer.Close();
 | |
|         }
 | |
| 
 | |
|         public virtual void AppendChild(XPathNavigator newChild) {
 | |
|             if (newChild == null) {
 | |
|                 throw new ArgumentNullException("newChild");
 | |
|             }
 | |
|             if (!IsValidChildType(newChild.NodeType)) {
 | |
|                 throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
 | |
|             }
 | |
|             XmlReader reader = newChild.CreateReader();
 | |
|             AppendChild(reader);
 | |
|         }
 | |
| 
 | |
|         public virtual void PrependChild(string newChild) {
 | |
|             XmlReader reader = CreateContextReader(newChild, true);
 | |
|             PrependChild(reader);
 | |
|         }
 | |
| 
 | |
|         public virtual void PrependChild(XmlReader newChild) {
 | |
|             if (newChild == null) {
 | |
|                 throw new ArgumentNullException("newChild");
 | |
|             }
 | |
|             XmlWriter writer = PrependChild();
 | |
|             BuildSubtree(newChild, writer);
 | |
|             writer.Close();
 | |
|         }
 | |
| 
 | |
|         public virtual void PrependChild(XPathNavigator newChild) {
 | |
|             if (newChild == null) {
 | |
|                 throw new ArgumentNullException("newChild");
 | |
|             }
 | |
|             if (!IsValidChildType(newChild.NodeType)) {
 | |
|                 throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
 | |
|             }
 | |
|             XmlReader reader = newChild.CreateReader();
 | |
|             PrependChild(reader);
 | |
|         }
 | |
| 
 | |
|         public virtual void InsertBefore(string newSibling) {
 | |
|             XmlReader reader = CreateContextReader(newSibling, false);
 | |
|             InsertBefore(reader);
 | |
|         }
 | |
| 
 | |
|         public virtual void InsertBefore(XmlReader newSibling) {
 | |
|             if (newSibling == null) {
 | |
|                 throw new ArgumentNullException("newSibling");
 | |
|             }
 | |
|             XmlWriter writer = InsertBefore();
 | |
|             BuildSubtree(newSibling, writer);
 | |
|             writer.Close();
 | |
|         }
 | |
| 
 | |
|         public virtual void InsertBefore(XPathNavigator newSibling) {
 | |
|             if (newSibling == null) {
 | |
|                 throw new ArgumentNullException("newSibling");
 | |
|             }
 | |
|             if (!IsValidSiblingType(newSibling.NodeType)) {
 | |
|                 throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
 | |
|             }
 | |
|             XmlReader reader = newSibling.CreateReader();
 | |
|             InsertBefore(reader);
 | |
|         }
 | |
| 
 | |
|         public virtual void InsertAfter(string newSibling) {
 | |
|             XmlReader reader = CreateContextReader(newSibling, false);
 | |
|             InsertAfter(reader);
 | |
|         }
 | |
| 
 | |
|         public virtual void InsertAfter(XmlReader newSibling) {
 | |
|             if (newSibling == null) {
 | |
|                 throw new ArgumentNullException("newSibling");
 | |
|             }
 | |
|             XmlWriter writer = InsertAfter();
 | |
|             BuildSubtree(newSibling, writer);
 | |
|             writer.Close();
 | |
|         }
 | |
| 
 | |
|         public virtual void InsertAfter(XPathNavigator newSibling) {
 | |
|             if (newSibling == null) {
 | |
|                 throw new ArgumentNullException("newSibling");
 | |
|             }
 | |
|             if (!IsValidSiblingType(newSibling.NodeType)) {
 | |
|                 throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
 | |
|             }
 | |
|             XmlReader reader = newSibling.CreateReader();
 | |
|             InsertAfter(reader);
 | |
|         }
 | |
| 
 | |
|         public virtual void DeleteRange(XPathNavigator lastSiblingToDelete) {
 | |
|             throw new NotSupportedException();
 | |
|         }
 | |
| 
 | |
|         public virtual void DeleteSelf() {
 | |
|             DeleteRange(this);
 | |
|         }
 | |
| 
 | |
|         public virtual void PrependChildElement(string prefix, string localName, string namespaceURI, string value) {
 | |
|             XmlWriter writer = PrependChild();
 | |
|             writer.WriteStartElement(prefix, localName, namespaceURI);
 | |
|             if (value != null) {
 | |
|                 writer.WriteString(value);
 | |
|             }
 | |
|             writer.WriteEndElement();
 | |
|             writer.Close();
 | |
|         }
 | |
| 
 | |
|         public virtual void AppendChildElement(string prefix, string localName, string namespaceURI, string value) {
 | |
|             XmlWriter writer = AppendChild();
 | |
|             writer.WriteStartElement(prefix, localName, namespaceURI);
 | |
|             if (value != null) {
 | |
|                 writer.WriteString(value);
 | |
|             }
 | |
|             writer.WriteEndElement();
 | |
|             writer.Close();
 | |
|         }
 | |
| 
 | |
|         public virtual void InsertElementBefore(string prefix, string localName, string namespaceURI, string value) {
 | |
|             XmlWriter writer = InsertBefore();
 | |
|             writer.WriteStartElement(prefix, localName, namespaceURI);
 | |
|             if (value != null) {
 | |
|                 writer.WriteString(value);
 | |
|             }
 | |
|             writer.WriteEndElement();
 | |
|             writer.Close();
 | |
|         }
 | |
| 
 | |
|         public virtual void InsertElementAfter(string prefix, string localName, string namespaceURI, string value) {
 | |
|             XmlWriter writer = InsertAfter();
 | |
|             writer.WriteStartElement(prefix, localName, namespaceURI);
 | |
|             if (value != null) {
 | |
|                 writer.WriteString(value);
 | |
|             }
 | |
|             writer.WriteEndElement();
 | |
|             writer.Close();
 | |
|         }
 | |
| 
 | |
|         public virtual void CreateAttribute(string prefix, string localName, string namespaceURI, string value) {
 | |
|             XmlWriter writer = CreateAttributes();
 | |
|             writer.WriteStartAttribute(prefix, localName, namespaceURI);
 | |
|             if (value != null) {
 | |
|                 writer.WriteString(value);
 | |
|             }
 | |
|             writer.WriteEndAttribute();
 | |
|             writer.Close();
 | |
|         }
 | |
| 
 | |
|         //-----------------------------------------------
 | |
|         // Internal
 | |
|         //-----------------------------------------------
 | |
| 
 | |
|         internal bool MoveToPrevious(string localName, string namespaceURI) {
 | |
|             XPathNavigator navClone = Clone();
 | |
| 
 | |
|             localName = (localName != null) ? NameTable.Get(localName) : null;
 | |
|             while (MoveToPrevious()) {
 | |
|                 if (NodeType == XPathNodeType.Element && (object) localName == (object) LocalName && namespaceURI == NamespaceURI)
 | |
|                     return true;
 | |
|             }
 | |
| 
 | |
|             MoveTo(navClone);
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         internal bool MoveToPrevious(XPathNodeType type) {
 | |
|             XPathNavigator navClone = Clone();
 | |
|             int mask = GetContentKindMask(type);
 | |
| 
 | |
|             while (MoveToPrevious()) {
 | |
|                 if (((1 << (int) NodeType) & mask) != 0)
 | |
|                     return true;
 | |
|             }
 | |
| 
 | |
|             MoveTo(navClone);
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         internal bool MoveToNonDescendant() {
 | |
|             // If current node is document, there is no next non-descendant
 | |
|             if (NodeType == XPathNodeType.Root)
 | |
|                 return false;
 | |
| 
 | |
|             // If sibling exists, it is the next non-descendant
 | |
|             if (MoveToNext())
 | |
|                 return true;
 | |
| 
 | |
|             // The current node is either an attribute, namespace, or last child node
 | |
|             XPathNavigator navSave = Clone();
 | |
| 
 | |
|             if (!MoveToParent())
 | |
|                 return false;
 | |
| 
 | |
|             switch (navSave.NodeType) {
 | |
|                 case XPathNodeType.Attribute:
 | |
|                 case XPathNodeType.Namespace:
 | |
|                     // Next node in document order is first content-child of parent
 | |
|                     if (MoveToFirstChild())
 | |
|                         return true;
 | |
|                     break;
 | |
|             }
 | |
| 
 | |
|             while (!MoveToNext()) {
 | |
|                 if (!MoveToParent()) {
 | |
|                     // Restore original position and return false
 | |
|                     MoveTo(navSave);
 | |
|                     return false;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Returns ordinal number of attribute, namespace or child node within its parent.
 | |
|         /// Order is reversed for attributes and child nodes to avoid O(N**2) running time.
 | |
|         /// This property is useful for debugging, and also used in UniqueId implementation.
 | |
|         /// </summary>
 | |
|         internal uint IndexInParent {
 | |
|             get {
 | |
|                 XPathNavigator nav = this.Clone();
 | |
|                 uint idx = 0;
 | |
| 
 | |
|                 switch (NodeType) {
 | |
|                     case XPathNodeType.Attribute:
 | |
|                         while (nav.MoveToNextAttribute()) {
 | |
|                             idx ++;
 | |
|                         }
 | |
|                         break;
 | |
|                     case XPathNodeType.Namespace:
 | |
|                         while (nav.MoveToNextNamespace()) {
 | |
|                             idx ++;
 | |
|                         }
 | |
|                         break;
 | |
|                     default:
 | |
|                         while (nav.MoveToNext()) {
 | |
|                             idx ++;
 | |
|                         }
 | |
|                         break;
 | |
|                 }
 | |
|                 return idx;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal static readonly char[] NodeTypeLetter = new char[] {
 | |
|             'R',    // Root
 | |
|             'E',    // Element
 | |
|             'A',    // Attribute
 | |
|             'N',    // Namespace
 | |
|             'T',    // Text
 | |
|             'S',    // SignificantWhitespace
 | |
|             'W',    // Whitespace
 | |
|             'P',    // ProcessingInstruction
 | |
|             'C',    // Comment
 | |
|             'X',    // All
 | |
|         };
 | |
| 
 | |
|         internal static readonly char[] UniqueIdTbl = new char[] {
 | |
|             'A',  'B',  'C',  'D',  'E',  'F',  'G',  'H',  'I',  'J',
 | |
|             'K',  'L',  'M',  'N',  'O',  'P',  'Q',  'R',  'S',  'T',
 | |
|             'U',  'V',  'W',  'X',  'Y',  'Z',  '1',  '2',  '3',  '4',
 | |
|             '5',  '6'
 | |
|         };
 | |
| 
 | |
|         // Requirements for id:
 | |
|         //  1. must consist of alphanumeric characters only
 | |
|         //  2. must begin with an alphabetic character
 | |
|         //  3. same id is generated for the same node
 | |
|         //  4. ids are unique
 | |
|         //
 | |
|         //  id = node type letter + reverse path to root in terms of encoded IndexInParent integers from node to root seperated by 0's if needed
 | |
|         internal virtual string UniqueId {
 | |
|             get {
 | |
|                 XPathNavigator  nav = this.Clone();
 | |
|                 StringBuilder sb = new StringBuilder();
 | |
| 
 | |
|                 // Ensure distinguishing attributes, namespaces and child nodes
 | |
|                 sb.Append(NodeTypeLetter[(int)NodeType]);
 | |
| 
 | |
|                 while (true) {
 | |
|                     uint idx = nav.IndexInParent;
 | |
|                     if (!nav.MoveToParent()) {
 | |
|                         break;
 | |
|                     }
 | |
|                     if (idx <= 0x1f) {
 | |
|                         sb.Append(UniqueIdTbl[idx]);
 | |
|                     } else {
 | |
|                         sb.Append('0');
 | |
|                         do {
 | |
|                             sb.Append(UniqueIdTbl[idx & 0x1f]);
 | |
|                             idx >>= 5;
 | |
|                         } while (idx != 0);
 | |
|                         sb.Append('0');
 | |
|                     }
 | |
|                 }
 | |
|                 return sb.ToString();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private static XPathExpression CompileMatchPattern(string xpath) {
 | |
|             bool hasPrefix;
 | |
|             Query query = new QueryBuilder().BuildPatternQuery(xpath, out hasPrefix);
 | |
|             return new CompiledXpathExpr(query, xpath, hasPrefix);
 | |
|         }
 | |
| 
 | |
|         private static int GetDepth(XPathNavigator nav) {
 | |
|             int depth = 0;
 | |
|             while (nav.MoveToParent()) {
 | |
|                 depth++;
 | |
|             }
 | |
|             return depth;
 | |
|         }
 | |
| 
 | |
|         // XPath based comparison for namespaces, attributes and other 
 | |
|         // items with the same parent element.
 | |
|         //
 | |
|         //                 n2
 | |
|         //                 namespace(0)    attribute(-1)   other(-2)
 | |
|         // n1
 | |
|         // namespace(0)    ?(0)            before(-1)      before(-2)
 | |
|         // attribute(1)    after(1)        ?(0)            before(-1)
 | |
|         // other    (2)    after(2)        after(1)        ?(0)
 | |
|         private XmlNodeOrder CompareSiblings(XPathNavigator n1, XPathNavigator n2) {
 | |
|             int cmp = 0;
 | |
| 
 | |
| #if DEBUG
 | |
|             Debug.Assert(!n1.IsSamePosition(n2));
 | |
|             XPathNavigator p1 = n1.Clone(), p2 = n2.Clone();
 | |
|             Debug.Assert(p1.MoveToParent() && p2.MoveToParent() && p1.IsSamePosition(p2));
 | |
| #endif
 | |
|             switch (n1.NodeType) {
 | |
|                 case XPathNodeType.Namespace: 
 | |
|                     break;
 | |
|                 case XPathNodeType.Attribute: 
 | |
|                     cmp += 1; 
 | |
|                     break;
 | |
|                 default:
 | |
|                     cmp += 2; 
 | |
|                     break;
 | |
|             }
 | |
|             switch (n2.NodeType) {
 | |
|                 case XPathNodeType.Namespace: 
 | |
|                     if (cmp == 0) {
 | |
|                         while (n1.MoveToNextNamespace()) {
 | |
|                             if (n1.IsSamePosition(n2)) {
 | |
|                                 return XmlNodeOrder.Before;
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                     break; 
 | |
|                 case XPathNodeType.Attribute: 
 | |
|                     cmp -= 1; 
 | |
|                     if (cmp == 0) {
 | |
|                         while (n1.MoveToNextAttribute()) {
 | |
|                             if (n1.IsSamePosition(n2)) {
 | |
|                                 return XmlNodeOrder.Before;
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                     break;
 | |
|                 default:
 | |
|                     cmp -= 2; 
 | |
|                     if (cmp == 0) {
 | |
|                         while (n1.MoveToNext()) {
 | |
|                             if (n1.IsSamePosition(n2)) {
 | |
|                                 return XmlNodeOrder.Before;
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                     break;
 | |
|             }
 | |
|             return cmp < 0 ? XmlNodeOrder.Before : XmlNodeOrder.After;
 | |
|         }
 | |
| 
 | |
|         internal static XmlNamespaceManager GetNamespaces( IXmlNamespaceResolver resolver ) {
 | |
|             XmlNamespaceManager mngr = new XmlNamespaceManager(new NameTable());
 | |
|             IDictionary<string,string> dictionary = resolver.GetNamespacesInScope( XmlNamespaceScope.All );
 | |
|             foreach ( KeyValuePair<string,string> pair in dictionary ) {
 | |
|                 //"xmlns " is always in the namespace manager so adding it would throw an exception
 | |
|                 if( pair.Key != "xmlns" )
 | |
|                     mngr.AddNamespace( pair.Key, pair.Value );
 | |
|             }
 | |
|             return mngr;
 | |
|         }
 | |
| 
 | |
|         // Get mask that will allow XPathNodeType content matching to be performed using only a shift and an and operation
 | |
|         internal const int AllMask = 0x7FFFFFFF;
 | |
|         internal const int NoAttrNmspMask = AllMask & ~(1 << (int) XPathNodeType.Attribute) & ~(1 << (int) XPathNodeType.Namespace);
 | |
|         internal const int TextMask = (1 << (int) XPathNodeType.Text) | (1 << (int) XPathNodeType.SignificantWhitespace) | (1 << (int) XPathNodeType.Whitespace);
 | |
|         internal static readonly int[] ContentKindMasks = {
 | |
|             (1 << (int) XPathNodeType.Root),                        // Root
 | |
|             (1 << (int) XPathNodeType.Element),                     // Element
 | |
|             0,                                                      // Attribute (not content)
 | |
|             0,                                                      // Namespace (not content)
 | |
|             TextMask,                                               // Text
 | |
|             (1 << (int) XPathNodeType.SignificantWhitespace),       // SignificantWhitespace
 | |
|             (1 << (int) XPathNodeType.Whitespace),                  // Whitespace
 | |
|             (1 << (int) XPathNodeType.ProcessingInstruction),       // ProcessingInstruction
 | |
|             (1 << (int) XPathNodeType.Comment),                     // Comment
 | |
|             NoAttrNmspMask,                                         // All
 | |
|         };
 | |
| 
 | |
|         internal static int GetContentKindMask(XPathNodeType type) {
 | |
|             return ContentKindMasks[(int) type];
 | |
|         }
 | |
| 
 | |
|         internal static int GetKindMask(XPathNodeType type) {
 | |
|             if (type == XPathNodeType.All)
 | |
|                 return AllMask;
 | |
|             else if (type == XPathNodeType.Text)
 | |
|                 return TextMask;
 | |
| 
 | |
|             return (1 << (int) type);
 | |
|         }
 | |
| 
 | |
|         internal static bool IsText(XPathNodeType type) {
 | |
|             //return ((1 << (int) type) & TextMask) != 0;
 | |
|             return (uint)(type - XPathNodeType.Text) <= (XPathNodeType.Whitespace - XPathNodeType.Text);
 | |
|         }
 | |
| 
 | |
|         // Lax check for potential child item.
 | |
|         private bool IsValidChildType(XPathNodeType type) {
 | |
|             switch (NodeType) {
 | |
|                 case XPathNodeType.Root:
 | |
|                     switch (type) {
 | |
|                         case XPathNodeType.Element:
 | |
|                         case XPathNodeType.SignificantWhitespace:
 | |
|                         case XPathNodeType.Whitespace:
 | |
|                         case XPathNodeType.ProcessingInstruction:
 | |
|                         case XPathNodeType.Comment:
 | |
|                             return true;
 | |
|                     }
 | |
|                     break;
 | |
|                 case XPathNodeType.Element:
 | |
|                     switch (type) {
 | |
|                         case XPathNodeType.Element:
 | |
|                         case XPathNodeType.Text:
 | |
|                         case XPathNodeType.SignificantWhitespace:
 | |
|                         case XPathNodeType.Whitespace:
 | |
|                         case XPathNodeType.ProcessingInstruction:
 | |
|                         case XPathNodeType.Comment:
 | |
|                             return true;
 | |
|                     }
 | |
|                     break;
 | |
|             }
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         // Lax check for potential sibling item. 
 | |
|         private bool IsValidSiblingType(XPathNodeType type) {
 | |
|             switch (NodeType) {
 | |
|                 case XPathNodeType.Element:
 | |
|                 case XPathNodeType.Text:
 | |
|                 case XPathNodeType.SignificantWhitespace:
 | |
|                 case XPathNodeType.Whitespace:
 | |
|                 case XPathNodeType.ProcessingInstruction:
 | |
|                 case XPathNodeType.Comment:
 | |
|                     switch (type) {
 | |
|                         case XPathNodeType.Element:
 | |
|                         case XPathNodeType.Text:
 | |
|                         case XPathNodeType.SignificantWhitespace:
 | |
|                         case XPathNodeType.Whitespace:
 | |
|                         case XPathNodeType.ProcessingInstruction:
 | |
|                         case XPathNodeType.Comment:
 | |
|                             return true;
 | |
|                     }
 | |
|                     break;
 | |
|             }
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         private XmlReader CreateReader() {
 | |
|             return XPathNavigatorReader.Create(this);
 | |
|         }
 | |
| 
 | |
|         private XmlReader CreateContextReader(string xml, bool fromCurrentNode) {
 | |
|             if (xml == null) {
 | |
|                 throw new ArgumentNullException("xml");
 | |
|             }
 | |
| 
 | |
|             // We have to set the namespace context for the reader.
 | |
|             XPathNavigator editor = CreateNavigator();
 | |
|             // scope starts from parent.
 | |
|             XmlNamespaceManager mgr = new XmlNamespaceManager( NameTable );
 | |
|             if (!fromCurrentNode) {
 | |
|                 editor.MoveToParent(); // should always succeed.
 | |
|             }
 | |
|             if (editor.MoveToFirstNamespace(XPathNamespaceScope.All)) {
 | |
|                 do {
 | |
|                     mgr.AddNamespace(editor.LocalName, editor.Value);
 | |
|                 } 
 | |
|                 while (editor.MoveToNextNamespace(XPathNamespaceScope.All));
 | |
|             }
 | |
|             // 
 | |
|             XmlParserContext context = new XmlParserContext(NameTable, mgr, null, XmlSpace.Default);
 | |
|             XmlTextReader reader = new XmlTextReader(xml, XmlNodeType.Element, context);
 | |
|             // 
 | |
|             reader.WhitespaceHandling = WhitespaceHandling.Significant;
 | |
|             return reader;
 | |
|         }
 | |
| 
 | |
|         internal void BuildSubtree(XmlReader reader, XmlWriter writer) {
 | |
|             // important (perf) string literal...
 | |
|             string xmlnsUri = XmlReservedNs.NsXmlNs; // http://www.w3.org/2000/xmlns/
 | |
|             ReadState readState = reader.ReadState;
 | |
| 
 | |
|             if (readState != ReadState.Initial
 | |
|                 && readState != ReadState.Interactive) {
 | |
|                 throw new ArgumentException(Res.GetString(Res.Xml_InvalidOperation), "reader");
 | |
|             }
 | |
|             int level = 0;
 | |
|             if ( readState == ReadState.Initial ) {
 | |
|                 if( !reader.Read() )
 | |
|                     return;
 | |
|                 level++; // if start in initial, read everything (not just first)
 | |
|             }
 | |
|             do {
 | |
|                 switch (reader.NodeType) {
 | |
|                     case XmlNodeType.Element:
 | |
|                         writer.WriteStartElement( reader.Prefix,  reader.LocalName, reader.NamespaceURI );
 | |
|                         bool isEmptyElement = reader.IsEmptyElement;
 | |
| 
 | |
|                         while (reader.MoveToNextAttribute()) {                                
 | |
|                             if ((object) reader.NamespaceURI == (object) xmlnsUri) {
 | |
|                                 if (reader.Prefix.Length == 0) {
 | |
|                                     // Default namespace declaration "xmlns"
 | |
|                                     Debug.Assert(reader.LocalName == "xmlns");
 | |
|                                     writer.WriteAttributeString( "", "xmlns", xmlnsUri, reader.Value );
 | |
|                                 }
 | |
|                                 else {
 | |
|                                     Debug.Assert(reader.Prefix == "xmlns");
 | |
|                                     writer.WriteAttributeString( "xmlns", reader.LocalName, xmlnsUri, reader.Value );
 | |
|                                 }
 | |
|                             }
 | |
|                             else {
 | |
|                                 writer.WriteStartAttribute(reader.Prefix, reader.LocalName, reader.NamespaceURI);
 | |
|                                 writer.WriteString(reader.Value);
 | |
|                                 writer.WriteEndAttribute();
 | |
|                             }
 | |
|                         }
 | |
| 
 | |
|                         reader.MoveToElement();
 | |
|                         if (isEmptyElement) {
 | |
|                             // there might still be a value, if there is a default value specified in the schema
 | |
|                             writer.WriteEndElement();
 | |
|                         }
 | |
|                         else {
 | |
|                             level++;
 | |
|                         }
 | |
|                         break;
 | |
|                     case XmlNodeType.EndElement:
 | |
|                         writer.WriteFullEndElement();
 | |
|                         //should not read beyond the level of the reader's original position.
 | |
|                         level--;
 | |
|                         break;
 | |
|                     case XmlNodeType.Text:
 | |
|                     case XmlNodeType.CDATA:
 | |
|                         writer.WriteString( reader.Value );
 | |
|                         break;
 | |
|                     case XmlNodeType.SignificantWhitespace:
 | |
|                     case XmlNodeType.Whitespace:
 | |
|                         // 
 | |
|                         writer.WriteString( reader.Value );
 | |
|                         break;
 | |
|                     case XmlNodeType.Comment:
 | |
|                         writer.WriteComment( reader.Value );
 | |
|                         break;
 | |
|                     case XmlNodeType.ProcessingInstruction:
 | |
|                         writer.WriteProcessingInstruction( reader.LocalName , reader.Value);
 | |
|                         break;
 | |
|                     case XmlNodeType.EntityReference:
 | |
|                         reader.ResolveEntity(); // 
 | |
|                         break;
 | |
|                     case XmlNodeType.EndEntity:
 | |
|                     case XmlNodeType.None:
 | |
|                     case XmlNodeType.DocumentType:
 | |
|                     case XmlNodeType.XmlDeclaration:
 | |
|                         break;                    
 | |
|                     case XmlNodeType.Attribute:
 | |
|                         if ((object) reader.NamespaceURI == (object) xmlnsUri) {
 | |
|                             if (reader.Prefix.Length == 0) {
 | |
|                                 // Default namespace declaration "xmlns"
 | |
|                                 Debug.Assert(reader.LocalName == "xmlns");
 | |
|                                 writer.WriteAttributeString( "", "xmlns", xmlnsUri, reader.Value );
 | |
|                             }
 | |
|                             else {
 | |
|                                 Debug.Assert(reader.Prefix == "xmlns");
 | |
|                                 writer.WriteAttributeString( "xmlns", reader.LocalName, xmlnsUri, reader.Value );
 | |
|                             }
 | |
|                         }
 | |
|                         else {
 | |
|                             writer.WriteStartAttribute(reader.Prefix, reader.LocalName, reader.NamespaceURI);
 | |
|                             writer.WriteString(reader.Value);
 | |
|                             writer.WriteEndAttribute();
 | |
|                         }
 | |
|                         break;
 | |
|                 }
 | |
|             } 
 | |
|             while( reader.Read() && ( level > 0 ) );
 | |
|         }
 | |
| 
 | |
|         private object debuggerDisplayProxy { get { return new DebuggerDisplayProxy(this); } }
 | |
| 
 | |
|         [DebuggerDisplay("{ToString()}")]
 | |
|         internal struct DebuggerDisplayProxy {
 | |
|             XPathNavigator nav;
 | |
|             public DebuggerDisplayProxy(XPathNavigator nav) {
 | |
|                 this.nav = nav;
 | |
|             }
 | |
|             public override string ToString() {
 | |
|                 string result = nav.NodeType.ToString();
 | |
|                 switch (nav.NodeType) {
 | |
|                 case XPathNodeType.Element              :
 | |
|                     result += ", Name=\"" + nav.Name + '"';
 | |
|                     break;
 | |
|                 case XPathNodeType.Attribute:
 | |
|                 case XPathNodeType.Namespace            :
 | |
|                 case XPathNodeType.ProcessingInstruction:
 | |
|                     result += ", Name=\"" + nav.Name + '"';
 | |
|                     result += ", Value=\"" + XmlConvert.EscapeValueForDebuggerDisplay(nav.Value) + '"';
 | |
|                     break;
 | |
|                 case XPathNodeType.Text                 :
 | |
|                 case XPathNodeType.Whitespace           :
 | |
|                 case XPathNodeType.SignificantWhitespace:
 | |
|                 case XPathNodeType.Comment              :
 | |
|                     result += ", Value=\"" + XmlConvert.EscapeValueForDebuggerDisplay(nav.Value) + '"';
 | |
|                     break;
 | |
|                 }
 | |
|                 return result;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
| #if CONTRACTS_FULL
 | |
|     [ContractClassFor(typeof(XPathNavigator))]
 | |
|     internal abstract class XPathNavigatorContract : XPathNavigator
 | |
|     {
 | |
|         public override XPathNavigator Clone()
 | |
|         {
 | |
|             Contract.Ensures(Contract.Result<XPathNavigator>() != null);
 | |
|             return default(XPathNavigator);
 | |
|         }
 | |
| 
 | |
|         public override XmlNameTable NameTable { 
 | |
|             get {
 | |
|                 Contract.Ensures(Contract.Result<XmlNameTable>() != null);
 | |
|                 return default(XmlNameTable);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| #endif
 | |
| }
 |