1931 lines
73 KiB
C#
Raw Normal View History

//------------------------------------------------------------------------------
// <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
}