2016 lines
75 KiB
C#
Raw Normal View History

//------------------------------------------------------------------------------
// <copyright file="DocumentXPathNavigator.cs" company="Microsoft">
// Copyright (c) Microsoft Corporation. All rights reserved.
// </copyright>
// <owner current="true" primary="true">Microsoft</owner>
//------------------------------------------------------------------------------
using System;
using System.Collections;
using System.Collections.Generic;
using System.Text;
using System.Xml.Schema;
using System.Xml.XPath;
using System.Diagnostics;
namespace System.Xml {
internal sealed class DocumentXPathNavigator : XPathNavigator, IHasXmlNode {
private XmlDocument document; // owner document
private XmlNode source; // navigator position
private int attributeIndex; // index in attribute collection for attribute
private XmlElement namespaceParent; // parent for namespace
public DocumentXPathNavigator(XmlDocument document, XmlNode node) {
this.document = document;
ResetPosition(node);
}
public DocumentXPathNavigator(DocumentXPathNavigator other) {
document = other.document;
source = other.source;
attributeIndex = other.attributeIndex;
namespaceParent = other.namespaceParent;
}
public override XPathNavigator Clone() {
return new DocumentXPathNavigator(this);
}
public override void SetValue(string value) {
if (value == null) {
throw new ArgumentNullException("value");
}
XmlNode node = source;
XmlNode end;
switch (node.NodeType) {
case XmlNodeType.Attribute:
if (((XmlAttribute)node).IsNamespace) {
goto default;
}
node.InnerText = value;
break;
case XmlNodeType.Text:
case XmlNodeType.CDATA:
case XmlNodeType.Whitespace:
case XmlNodeType.SignificantWhitespace:
CalibrateText();
node = source;
end = TextEnd(node);
if (node != end) {
if (node.IsReadOnly) {
throw new InvalidOperationException(Res.GetString(Res.Xdom_Node_Modify_ReadOnly));
}
DeleteToFollowingSibling(node.NextSibling, end);
}
goto case XmlNodeType.Element;
case XmlNodeType.Element:
case XmlNodeType.ProcessingInstruction:
case XmlNodeType.Comment:
node.InnerText = value;
break;
default:
throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
}
}
public override XmlNameTable NameTable {
get {
return document.NameTable;
}
}
public override XPathNodeType NodeType {
get {
CalibrateText();
return (XPathNodeType)source.XPNodeType;
}
}
public override string LocalName {
get {
return source.XPLocalName;
}
}
public override string NamespaceURI {
get {
XmlAttribute attribute = source as XmlAttribute;
if (attribute != null
&& attribute.IsNamespace) {
return string.Empty;
}
return source.NamespaceURI;
}
}
public override string Name {
get {
switch (source.NodeType) {
case XmlNodeType.Element:
case XmlNodeType.ProcessingInstruction:
return source.Name;
case XmlNodeType.Attribute:
if (((XmlAttribute)source).IsNamespace) {
string localName = source.LocalName;
if (Ref.Equal(localName, document.strXmlns)) {
return string.Empty; // xmlns declaration
}
return localName; // xmlns:name declaration
}
return source.Name; // attribute
default:
return string.Empty;
}
}
}
public override string Prefix {
get {
XmlAttribute attribute = source as XmlAttribute;
if (attribute != null
&& attribute.IsNamespace) {
return string.Empty;
}
return source.Prefix;
}
}
public override string Value {
get {
switch (source.NodeType) {
case XmlNodeType.Element:
case XmlNodeType.DocumentFragment:
return source.InnerText;
case XmlNodeType.Document:
return ValueDocument;
case XmlNodeType.Text:
case XmlNodeType.CDATA:
case XmlNodeType.Whitespace:
case XmlNodeType.SignificantWhitespace:
return ValueText;
default:
return source.Value;
}
}
}
private string ValueDocument {
get {
XmlElement element = document.DocumentElement;
if (element != null) {
return element.InnerText;
}
return string.Empty;
}
}
private string ValueText {
get {
CalibrateText();
string value = source.Value;
XmlNode nextSibling = NextSibling(source);
if (nextSibling != null
&& nextSibling.IsText) {
StringBuilder builder = new StringBuilder(value);
do {
builder.Append(nextSibling.Value);
nextSibling = NextSibling(nextSibling);
}
while (nextSibling != null
&& nextSibling.IsText);
value = builder.ToString();
}
return value;
}
}
public override string BaseURI {
get {
return source.BaseURI;
}
}
public override bool IsEmptyElement {
get {
XmlElement element = source as XmlElement;
if (element != null) {
return element.IsEmpty;
}
return false;
}
}
public override string XmlLang {
get {
return source.XmlLang;
}
}
public override object UnderlyingObject {
get {
CalibrateText();
return source;
}
}
public override bool HasAttributes {
get {
XmlElement element = source as XmlElement;
if (element != null
&& element.HasAttributes) {
XmlAttributeCollection attributes = element.Attributes;
for (int i = 0; i < attributes.Count; i++) {
XmlAttribute attribute = attributes[i];
if (!attribute.IsNamespace) {
return true;
}
}
}
return false;
}
}
public override string GetAttribute(string localName, string namespaceURI) {
return source.GetXPAttribute(localName, namespaceURI);
}
public override bool MoveToAttribute(string localName, string namespaceURI) {
XmlElement element = source as XmlElement;
if (element != null
&& element.HasAttributes) {
XmlAttributeCollection attributes = element.Attributes;
for (int i = 0; i < attributes.Count; i++) {
XmlAttribute attribute = attributes[i];
if (attribute.LocalName == localName
&& attribute.NamespaceURI == namespaceURI) {
if (!attribute.IsNamespace) {
source = attribute;
attributeIndex = i;
return true;
}
else {
return false;
}
}
}
}
return false;
}
public override bool MoveToFirstAttribute() {
XmlElement element = source as XmlElement;
if (element != null
&& element.HasAttributes) {
XmlAttributeCollection attributes = element.Attributes;
for (int i = 0; i < attributes.Count; i++) {
XmlAttribute attribute = attributes[i];
if (!attribute.IsNamespace) {
source = attribute;
attributeIndex = i;
return true;
}
}
}
return false;
}
public override bool MoveToNextAttribute() {
XmlAttribute attribute = source as XmlAttribute;
if (attribute == null
|| attribute.IsNamespace) {
return false;
}
XmlAttributeCollection attributes;
if (!CheckAttributePosition(attribute, out attributes, attributeIndex)
&& !ResetAttributePosition(attribute, attributes, out attributeIndex)) {
return false;
}
for (int i = attributeIndex + 1; i < attributes.Count; i++) {
attribute = attributes[i];
if (!attribute.IsNamespace) {
source = attribute;
attributeIndex = i;
return true;
}
}
return false;
}
public override string GetNamespace(string name) {
XmlNode node = source;
while (node != null
&& node.NodeType != XmlNodeType.Element) {
XmlAttribute attribute = node as XmlAttribute;
if (attribute != null) {
node = attribute.OwnerElement;
}
else {
node = node.ParentNode;
}
}
XmlElement element = node as XmlElement;
if (element != null) {
string localName;
if (name != null
&& name.Length != 0) {
localName = name;
}
else {
localName = document.strXmlns;
}
string namespaceUri = document.strReservedXmlns;
do
{
XmlAttribute attribute = element.GetAttributeNode(localName, namespaceUri);
if (attribute != null) {
return attribute.Value;
}
element = element.ParentNode as XmlElement;
}
while (element != null);
}
if (name == document.strXml) {
return document.strReservedXml;
}
else if (name == document.strXmlns) {
return document.strReservedXmlns;
}
return string.Empty;
}
public override bool MoveToNamespace(string name) {
if (name == document.strXmlns) {
return false;
}
XmlElement element = source as XmlElement;
if (element != null) {
string localName;
if (name != null
&& name.Length != 0) {
localName = name;
}
else {
localName = document.strXmlns;
}
string namespaceUri = document.strReservedXmlns;
do {
XmlAttribute attribute = element.GetAttributeNode(localName, namespaceUri);
if (attribute != null) {
namespaceParent = (XmlElement)source;
source = attribute;
return true;
}
element = element.ParentNode as XmlElement;
}
while (element != null);
if (name == document.strXml) {
namespaceParent = (XmlElement)source;
source = document.NamespaceXml;
return true;
}
}
return false;
}
public override bool MoveToFirstNamespace(XPathNamespaceScope scope) {
XmlElement element = source as XmlElement;
if (element == null) {
return false;
}
XmlAttributeCollection attributes;
int index = Int32.MaxValue;
switch (scope) {
case XPathNamespaceScope.Local:
if (!element.HasAttributes) {
return false;
}
attributes = element.Attributes;
if (!MoveToFirstNamespaceLocal(attributes, ref index)) {
return false;
}
source = attributes[index];
attributeIndex = index;
namespaceParent = element;
break;
case XPathNamespaceScope.ExcludeXml:
attributes = element.Attributes;
if (!MoveToFirstNamespaceGlobal(ref attributes, ref index)) {
return false;
}
XmlAttribute attribute = attributes[index];
while (Ref.Equal(attribute.LocalName, document.strXml)) {
if (!MoveToNextNamespaceGlobal(ref attributes, ref index)) {
return false;
}
attribute = attributes[index];
}
source = attribute;
attributeIndex = index;
namespaceParent = element;
break;
case XPathNamespaceScope.All:
attributes = element.Attributes;
if (!MoveToFirstNamespaceGlobal(ref attributes, ref index)) {
source = document.NamespaceXml;
// attributeIndex = 0;
}
else {
source = attributes[index];
attributeIndex = index;
}
namespaceParent = element;
break;
default:
Debug.Assert(false);
return false;
}
return true;
}
private static bool MoveToFirstNamespaceLocal(XmlAttributeCollection attributes, ref int index) {
Debug.Assert(attributes != null);
for (int i = attributes.Count - 1; i >= 0; i--) {
XmlAttribute attribute = attributes[i];
if (attribute.IsNamespace) {
index = i;
return true;
}
}
return false;
}
private static bool MoveToFirstNamespaceGlobal(ref XmlAttributeCollection attributes, ref int index) {
if (MoveToFirstNamespaceLocal(attributes, ref index)) {
return true;
}
Debug.Assert(attributes != null && attributes.parent != null);
XmlElement element = attributes.parent.ParentNode as XmlElement;
while (element != null) {
if (element.HasAttributes) {
attributes = element.Attributes;
if (MoveToFirstNamespaceLocal(attributes, ref index)) {
return true;
}
}
element = element.ParentNode as XmlElement;
}
return false;
}
public override bool MoveToNextNamespace(XPathNamespaceScope scope) {
XmlAttribute attribute = source as XmlAttribute;
if (attribute == null
|| !attribute.IsNamespace) {
return false;
}
XmlAttributeCollection attributes;
int index = attributeIndex;
if (!CheckAttributePosition(attribute, out attributes, index)
&& !ResetAttributePosition(attribute, attributes, out index)) {
return false;
}
Debug.Assert(namespaceParent != null);
switch (scope) {
case XPathNamespaceScope.Local:
if (attribute.OwnerElement != namespaceParent) {
return false;
}
if (!MoveToNextNamespaceLocal(attributes, ref index)) {
return false;
}
source = attributes[index];
attributeIndex = index;
break;
case XPathNamespaceScope.ExcludeXml:
string localName;
do {
if (!MoveToNextNamespaceGlobal(ref attributes, ref index)) {
return false;
}
attribute = attributes[index];
localName = attribute.LocalName;
}
while (PathHasDuplicateNamespace(attribute.OwnerElement, namespaceParent, localName)
|| Ref.Equal(localName, document.strXml));
source = attribute;
attributeIndex = index;
break;
case XPathNamespaceScope.All:
do {
if (!MoveToNextNamespaceGlobal(ref attributes, ref index)) {
if (PathHasDuplicateNamespace(null, namespaceParent, document.strXml)) {
return false;
}
else {
source = document.NamespaceXml;
// attributeIndex = 0;
return true;
}
}
attribute = attributes[index];
}
while (PathHasDuplicateNamespace(attribute.OwnerElement, namespaceParent, attribute.LocalName));
source = attribute;
attributeIndex = index;
break;
default:
Debug.Assert(false);
return false;
}
return true;
}
private static bool MoveToNextNamespaceLocal(XmlAttributeCollection attributes, ref int index) {
Debug.Assert(attributes != null);
Debug.Assert(0 <= index && index < attributes.Count);
for (int i = index - 1; i >= 0; i--) {
XmlAttribute attribute = attributes[i];
if (attribute.IsNamespace) {
index = i;
return true;
}
}
return false;
}
private static bool MoveToNextNamespaceGlobal(ref XmlAttributeCollection attributes, ref int index) {
if (MoveToNextNamespaceLocal(attributes, ref index)) {
return true;
}
Debug.Assert(attributes != null && attributes.parent != null);
XmlElement element = attributes.parent.ParentNode as XmlElement;
while (element != null) {
if (element.HasAttributes) {
attributes = element.Attributes;
if (MoveToFirstNamespaceLocal(attributes, ref index)) {
return true;
}
}
element = element.ParentNode as XmlElement;
}
return false;
}
private bool PathHasDuplicateNamespace(XmlElement top, XmlElement bottom, string localName) {
string namespaceUri = document.strReservedXmlns;
while (bottom != null
&& bottom != top) {
XmlAttribute attribute = bottom.GetAttributeNode(localName, namespaceUri);
if (attribute != null) {
return true;
}
bottom = bottom.ParentNode as XmlElement;
}
return false;
}
public override string LookupNamespace(string prefix) {
string ns = base.LookupNamespace(prefix);
if (ns != null) {
ns = this.NameTable.Add(ns);
}
return ns;
}
public override bool MoveToNext() {
XmlNode sibling = NextSibling(source);
if (sibling == null) {
return false;
}
if (sibling.IsText) {
if (source.IsText) {
sibling = NextSibling(TextEnd(sibling));
if (sibling == null) {
return false;
}
}
}
XmlNode parent = ParentNode(sibling);
Debug.Assert(parent != null);
while (!IsValidChild(parent, sibling)) {
sibling = NextSibling(sibling);
if (sibling == null) {
return false;
}
}
source = sibling;
return true;
}
public override bool MoveToPrevious() {
XmlNode sibling = PreviousSibling(source);
if (sibling == null) {
return false;
}
if (sibling.IsText) {
if (source.IsText) {
sibling = PreviousSibling(TextStart(sibling));
if (sibling == null) {
return false;
}
}
else {
sibling = TextStart(sibling);
}
}
XmlNode parent = ParentNode(sibling);
Debug.Assert(parent != null);
while (!IsValidChild(parent, sibling)) {
sibling = PreviousSibling(sibling);
if (sibling == null) {
return false;
}
// if (sibling.IsText) {
// sibling = TextStart(sibling);
// }
}
source = sibling;
return true;
}
public override bool MoveToFirst() {
if (source.NodeType == XmlNodeType.Attribute) {
return false;
}
XmlNode parent = ParentNode(source);
if (parent == null) {
return false;
}
XmlNode sibling = FirstChild(parent);
Debug.Assert(sibling != null);
while (!IsValidChild(parent, sibling)) {
sibling = NextSibling(sibling);
if (sibling == null) {
return false;
}
}
source = sibling;
return true;
}
public override bool MoveToFirstChild() {
XmlNode child;
switch (source.NodeType) {
case XmlNodeType.Element:
child = FirstChild(source);
if (child == null) {
return false;
}
break;
case XmlNodeType.DocumentFragment:
case XmlNodeType.Document:
child = FirstChild(source);
if (child == null) {
return false;
}
while (!IsValidChild(source, child)) {
child = NextSibling(child);
if (child == null) {
return false;
}
}
break;
default:
return false;
}
source = child;
return true;
}
public override bool MoveToParent() {
XmlNode parent = ParentNode(source);
if (parent != null) {
source = parent;
return true;
}
XmlAttribute attribute = source as XmlAttribute;
if (attribute != null) {
parent = attribute.IsNamespace ? namespaceParent : attribute.OwnerElement;
if (parent != null) {
source = parent;
namespaceParent = null;
return true;
}
}
return false;
}
public override void MoveToRoot() {
for (;;) {
XmlNode parent = source.ParentNode;
if (parent == null) {
XmlAttribute attribute = source as XmlAttribute;
if (attribute == null) {
break;
}
parent = attribute.IsNamespace ? namespaceParent : attribute.OwnerElement;
if (parent == null) {
break;
}
}
source = parent;
}
namespaceParent = null;
}
public override bool MoveTo(XPathNavigator other) {
DocumentXPathNavigator that = other as DocumentXPathNavigator;
if (that != null
&& document == that.document) {
source = that.source;
attributeIndex = that.attributeIndex;
namespaceParent = that.namespaceParent;
return true;
}
return false;
}
public override bool MoveToId(string id) {
XmlElement element = document.GetElementById(id);
if (element != null) {
source = element;
namespaceParent = null;
return true;
}
return false;
}
public override bool MoveToChild(string localName, string namespaceUri) {
if (source.NodeType == XmlNodeType.Attribute) {
return false;
}
XmlNode child = FirstChild(source);
if (child != null) {
do {
if (child.NodeType == XmlNodeType.Element
&& child.LocalName == localName
&& child.NamespaceURI == namespaceUri) {
source = child;
return true;
}
child = NextSibling(child);
}
while (child != null);
}
return false;
}
public override bool MoveToChild(XPathNodeType type) {
if (source.NodeType == XmlNodeType.Attribute) {
return false;
}
XmlNode child = FirstChild(source);
if (child != null) {
int mask = GetContentKindMask(type);
if (mask == 0) {
return false;
}
do {
if (((1 << (int)child.XPNodeType) & mask) != 0) {
source = child;
return true;
}
child = NextSibling(child);
}
while (child != null);
}
return false;
}
public override bool MoveToFollowing(string localName, string namespaceUri, XPathNavigator end) {
XmlNode pastFollowing = null;
DocumentXPathNavigator that = end as DocumentXPathNavigator;
if (that != null) {
if (document != that.document) {
return false;
}
switch (that.source.NodeType) {
case XmlNodeType.Attribute:
that = (DocumentXPathNavigator)that.Clone();
if (!that.MoveToNonDescendant()) {
return false;
}
break;
}
pastFollowing = that.source;
}
XmlNode following = source;
if (following.NodeType == XmlNodeType.Attribute) {
following = ((XmlAttribute)following).OwnerElement;
if (following == null) {
return false;
}
}
do {
XmlNode firstChild = following.FirstChild;
if (firstChild != null) {
following = firstChild;
}
else {
for (;;) {
XmlNode nextSibling = following.NextSibling;
if (nextSibling != null) {
following = nextSibling;
break;
}
else {
XmlNode parent = following.ParentNode;
if (parent != null) {
following = parent;
}
else {
return false;
}
}
}
}
if (following == pastFollowing) {
return false;
}
}
while (following.NodeType != XmlNodeType.Element
|| following.LocalName != localName
|| following.NamespaceURI != namespaceUri);
source = following;
return true;
}
public override bool MoveToFollowing(XPathNodeType type, XPathNavigator end) {
XmlNode pastFollowing = null;
DocumentXPathNavigator that = end as DocumentXPathNavigator;
if (that != null) {
if (document != that.document) {
return false;
}
switch (that.source.NodeType) {
case XmlNodeType.Attribute:
that = (DocumentXPathNavigator)that.Clone();
if (!that.MoveToNonDescendant()) {
return false;
}
break;
}
pastFollowing = that.source;
}
int mask = GetContentKindMask(type);
if (mask == 0) {
return false;
}
XmlNode following = source;
switch (following.NodeType) {
case XmlNodeType.Attribute:
following = ((XmlAttribute)following).OwnerElement;
if (following == null) {
return false;
}
break;
case XmlNodeType.Text:
case XmlNodeType.CDATA:
case XmlNodeType.SignificantWhitespace:
case XmlNodeType.Whitespace:
following = TextEnd(following);
break;
}
do {
XmlNode firstChild = following.FirstChild;
if (firstChild != null) {
following = firstChild;
}
else {
for (;;) {
XmlNode nextSibling = following.NextSibling;
if (nextSibling != null) {
following = nextSibling;
break;
}
else {
XmlNode parent = following.ParentNode;
if (parent != null) {
following = parent;
}
else {
return false;
}
}
}
}
if (following == pastFollowing) {
return false;
}
}
while (((1 << (int)following.XPNodeType) & mask) == 0);
source = following;
return true;
}
public override bool MoveToNext(string localName, string namespaceUri) {
XmlNode sibling = NextSibling(source);
if (sibling == null) {
return false;
}
do {
if (sibling.NodeType == XmlNodeType.Element
&& sibling.LocalName == localName
&& sibling.NamespaceURI == namespaceUri) {
source = sibling;
return true;
}
sibling = NextSibling(sibling);
}
while (sibling != null);
return false;
}
public override bool MoveToNext(XPathNodeType type) {
XmlNode sibling = NextSibling(source);
if (sibling == null) {
return false;
}
if (sibling.IsText
&& source.IsText) {
sibling = NextSibling(TextEnd(sibling));
if (sibling == null) {
return false;
}
}
int mask = GetContentKindMask(type);
if (mask == 0) {
return false;
}
do {
if (((1 << (int)sibling.XPNodeType) & mask) != 0) {
source = sibling;
return true;
}
sibling = NextSibling(sibling);
}
while (sibling != null);
return false;
}
public override bool HasChildren {
get {
XmlNode child;
switch (source.NodeType) {
case XmlNodeType.Element:
child = FirstChild(source);
if (child == null) {
return false;
}
return true;
case XmlNodeType.DocumentFragment:
case XmlNodeType.Document:
child = FirstChild(source);
if (child == null) {
return false;
}
while (!IsValidChild(source, child)) {
child = NextSibling(child);
if (child == null) {
return false;
}
}
return true;
default:
return false;
}
}
}
public override bool IsSamePosition(XPathNavigator other) {
DocumentXPathNavigator that = other as DocumentXPathNavigator;
if (that != null) {
this.CalibrateText();
that.CalibrateText();
return this.source == that.source
&& this.namespaceParent == that.namespaceParent;
}
return false;
}
public override bool IsDescendant(XPathNavigator other) {
DocumentXPathNavigator that = other as DocumentXPathNavigator;
if (that != null) {
return IsDescendant(this.source, that.source);
}
return false;
}
public override IXmlSchemaInfo SchemaInfo {
get {
return source.SchemaInfo;
}
}
public override bool CheckValidity(XmlSchemaSet schemas, ValidationEventHandler validationEventHandler) {
XmlDocument ownerDocument;
if (source.NodeType == XmlNodeType.Document) {
ownerDocument = (XmlDocument)source;
}
else {
ownerDocument = source.OwnerDocument;
if (schemas != null) {
throw new ArgumentException(Res.GetString(Res.XPathDocument_SchemaSetNotAllowed, null));
}
}
if (schemas == null && ownerDocument != null) {
schemas = ownerDocument.Schemas;
}
if (schemas == null || schemas.Count == 0) {
throw new InvalidOperationException(Res.GetString(Res.XmlDocument_NoSchemaInfo));
}
DocumentSchemaValidator validator = new DocumentSchemaValidator(ownerDocument, schemas, validationEventHandler);
validator.PsviAugmentation = false;
return validator.Validate(source);
}
private static XmlNode OwnerNode(XmlNode node) {
XmlNode parent = node.ParentNode;
if (parent != null) {
return parent;
}
XmlAttribute attribute = node as XmlAttribute;
if (attribute != null) {
return attribute.OwnerElement;
}
return null;
}
private static int GetDepth(XmlNode node) {
int depth = 0;
XmlNode owner = OwnerNode(node);
while (owner != null) {
depth++;
owner = OwnerNode(owner);
}
return depth;
}
//Assuming that node1 and node2 are in the same level; Except when they are namespace nodes, they should have the same parent node
//the returned value is node2's position corresponding to node1
private XmlNodeOrder Compare( XmlNode node1, XmlNode node2 ) {
Debug.Assert( node1 != null );
Debug.Assert( node2 != null );
Debug.Assert( node1 != node2, "Should be handled by ComparePosition()" );
//Attribute nodes come before other children nodes except namespace nodes
Debug.Assert( OwnerNode(node1) == OwnerNode(node2) );
if (node1.XPNodeType == XPathNodeType.Attribute) {
if (node2.XPNodeType == XPathNodeType.Attribute) {
XmlElement element = ((XmlAttribute)node1).OwnerElement;
if (element.HasAttributes) {
XmlAttributeCollection attributes = element.Attributes;
for (int i = 0; i < attributes.Count; i++) {
XmlAttribute attribute = attributes[i];
if (attribute == node1) {
return XmlNodeOrder.Before;
}
else if (attribute == node2) {
return XmlNodeOrder.After;
}
}
}
return XmlNodeOrder.Unknown;
}
else {
return XmlNodeOrder.Before;
}
}
if (node2.XPNodeType == XPathNodeType.Attribute) {
return XmlNodeOrder.After;
}
//neither of the node is Namespace node or Attribute node
XmlNode nextNode = node1.NextSibling;
while ( nextNode != null && nextNode != node2 )
nextNode = nextNode.NextSibling;
if ( nextNode == null )
//didn't meet node2 in the path to the end, thus it has to be in the front of node1
return XmlNodeOrder.After;
else
//met node2 in the path to the end, so node1 is at front
return XmlNodeOrder.Before;
}
public override XmlNodeOrder ComparePosition(XPathNavigator other) {
DocumentXPathNavigator that = other as DocumentXPathNavigator;
if (that == null) {
return XmlNodeOrder.Unknown;
}
this.CalibrateText();
that.CalibrateText();
if (this.source == that.source
&& this.namespaceParent == that.namespaceParent) {
return XmlNodeOrder.Same;
}
if (this.namespaceParent != null
|| that.namespaceParent != null) {
return base.ComparePosition(other);
}
XmlNode node1 = this.source;
XmlNode node2 = that.source;
XmlNode parent1 = OwnerNode(node1);
XmlNode parent2 = OwnerNode(node2);
if (parent1 == parent2) {
if (parent1 == null) {
return XmlNodeOrder.Unknown;
}
else {
Debug.Assert(node1 != node2);
return Compare(node1, node2);
}
}
int depth1 = GetDepth(node1);
int depth2 = GetDepth(node2);
if (depth2 > depth1) {
while (node2 != null
&& depth2 > depth1) {
node2 = OwnerNode(node2);
depth2--;
}
if (node1 == node2) {
return XmlNodeOrder.Before;
}
parent2 = OwnerNode(node2);
}
else if (depth1 > depth2) {
while (node1 != null
&& depth1 > depth2) {
node1 = OwnerNode(node1);
depth1--;
}
if (node1 == node2) {
return XmlNodeOrder.After;
}
parent1 = OwnerNode(node1);
}
while (parent1 != null
&& parent2 != null) {
if (parent1 == parent2) {
Debug.Assert(node1 != node2);
return Compare(node1, node2);
}
node1 = parent1;
node2 = parent2;
parent1 = OwnerNode(node1);
parent2 = OwnerNode(node2);
}
return XmlNodeOrder.Unknown;
}
//the function just for XPathNodeList to enumerate current Node.
XmlNode IHasXmlNode.GetNode() { return source; }
public override XPathNodeIterator SelectDescendants( string localName, string namespaceURI, bool matchSelf ) {
string nsAtom = document.NameTable.Get( namespaceURI );
if ( nsAtom == null || this.source.NodeType == XmlNodeType.Attribute )
return new DocumentXPathNodeIterator_Empty( this );
Debug.Assert( this.NodeType != XPathNodeType.Attribute && this.NodeType != XPathNodeType.Namespace && this.NodeType != XPathNodeType.All );
string localNameAtom = document.NameTable.Get( localName );
if ( localNameAtom == null )
return new DocumentXPathNodeIterator_Empty( this );
if ( localNameAtom.Length == 0 ) {
if ( matchSelf )
return new DocumentXPathNodeIterator_ElemChildren_AndSelf_NoLocalName( this, nsAtom );
return new DocumentXPathNodeIterator_ElemChildren_NoLocalName( this, nsAtom );
}
if ( matchSelf )
return new DocumentXPathNodeIterator_ElemChildren_AndSelf( this, localNameAtom, nsAtom );
return new DocumentXPathNodeIterator_ElemChildren( this, localNameAtom, nsAtom );
}
public override XPathNodeIterator SelectDescendants( XPathNodeType nt, bool includeSelf ) {
if ( nt == XPathNodeType.Element ) {
XmlNodeType curNT = source.NodeType;
if ( curNT != XmlNodeType.Document && curNT != XmlNodeType.Element ) {
//only Document, Entity, Element node can have Element node as children ( descendant )
//entity nodes should be invisible to XPath data model
return new DocumentXPathNodeIterator_Empty( this );
}
if ( includeSelf )
return new DocumentXPathNodeIterator_AllElemChildren_AndSelf( this );
return new DocumentXPathNodeIterator_AllElemChildren( this );
}
return base.SelectDescendants( nt, includeSelf );
}
public override bool CanEdit {
get {
return true;
}
}
public override XmlWriter PrependChild() {
switch (source.NodeType) {
case XmlNodeType.Element:
case XmlNodeType.Document:
case XmlNodeType.DocumentFragment:
break;
default:
throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
}
DocumentXmlWriter writer = new DocumentXmlWriter(DocumentXmlWriterType.PrependChild, source, document);
writer.NamespaceManager = GetNamespaceManager(source, document);
return new XmlWellFormedWriter(writer, writer.Settings);
}
public override XmlWriter AppendChild() {
switch (source.NodeType) {
case XmlNodeType.Element:
case XmlNodeType.Document:
case XmlNodeType.DocumentFragment:
break;
default:
throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
}
DocumentXmlWriter writer = new DocumentXmlWriter(DocumentXmlWriterType.AppendChild, source, document);
writer.NamespaceManager = GetNamespaceManager(source, document);
return new XmlWellFormedWriter(writer, writer.Settings);
}
public override XmlWriter InsertAfter() {
XmlNode node = source;
switch (node.NodeType) {
case XmlNodeType.Attribute:
case XmlNodeType.Document:
case XmlNodeType.DocumentFragment:
throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
case XmlNodeType.Text:
case XmlNodeType.CDATA:
case XmlNodeType.SignificantWhitespace:
case XmlNodeType.Whitespace:
node = TextEnd(node);
break;
default:
break;
}
DocumentXmlWriter writer = new DocumentXmlWriter(DocumentXmlWriterType.InsertSiblingAfter, node, document);
writer.NamespaceManager = GetNamespaceManager(node.ParentNode, document);
return new XmlWellFormedWriter(writer, writer.Settings);
}
public override XmlWriter InsertBefore() {
switch (source.NodeType) {
case XmlNodeType.Attribute:
case XmlNodeType.Document:
case XmlNodeType.DocumentFragment:
throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
case XmlNodeType.Text:
case XmlNodeType.CDATA:
case XmlNodeType.SignificantWhitespace:
case XmlNodeType.Whitespace:
CalibrateText();
break;
default:
break;
}
DocumentXmlWriter writer = new DocumentXmlWriter(DocumentXmlWriterType.InsertSiblingBefore, source, document);
writer.NamespaceManager = GetNamespaceManager(source.ParentNode, document);
return new XmlWellFormedWriter(writer, writer.Settings);
}
public override XmlWriter CreateAttributes() {
if (source.NodeType != XmlNodeType.Element) {
throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
}
DocumentXmlWriter writer = new DocumentXmlWriter(DocumentXmlWriterType.AppendAttribute, source, document);
writer.NamespaceManager = GetNamespaceManager(source, document);
return new XmlWellFormedWriter(writer, writer.Settings);
}
public override XmlWriter ReplaceRange(XPathNavigator lastSiblingToReplace) {
DocumentXPathNavigator that = lastSiblingToReplace as DocumentXPathNavigator;
if (that == null) {
if (lastSiblingToReplace == null) {
throw new ArgumentNullException("lastSiblingToReplace");
}
else {
throw new NotSupportedException();
}
}
this.CalibrateText();
that.CalibrateText();
XmlNode node = this.source;
XmlNode end = that.source;
if (node == end) {
switch (node.NodeType) {
case XmlNodeType.Attribute:
case XmlNodeType.Document:
case XmlNodeType.DocumentFragment:
throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
case XmlNodeType.Text:
case XmlNodeType.CDATA:
case XmlNodeType.SignificantWhitespace:
case XmlNodeType.Whitespace:
end = that.TextEnd(end);
break;
default:
break;
}
}
else {
if (end.IsText) {
end = that.TextEnd(end);
}
if (!IsFollowingSibling(node, end)) {
throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
}
}
DocumentXmlWriter writer = new DocumentXmlWriter(DocumentXmlWriterType.ReplaceToFollowingSibling, node, document);
writer.NamespaceManager = GetNamespaceManager(node.ParentNode, document);
writer.Navigator = this;
writer.EndNode = end;
return new XmlWellFormedWriter(writer, writer.Settings);
}
public override void DeleteRange(XPathNavigator lastSiblingToDelete) {
DocumentXPathNavigator that = lastSiblingToDelete as DocumentXPathNavigator;
if (that == null) {
if (lastSiblingToDelete == null) {
throw new ArgumentNullException("lastSiblingToDelete");
}
else {
throw new NotSupportedException();
}
}
this.CalibrateText();
that.CalibrateText();
XmlNode node = this.source;
XmlNode end = that.source;
if (node == end) {
switch (node.NodeType) {
case XmlNodeType.Attribute:
XmlAttribute attribute = (XmlAttribute)node;
if (attribute.IsNamespace) {
goto default;
}
XmlNode parent = OwnerNode(attribute);
DeleteAttribute(attribute, attributeIndex);
if (parent != null) {
ResetPosition(parent);
}
break;
case XmlNodeType.Text:
case XmlNodeType.CDATA:
case XmlNodeType.SignificantWhitespace:
case XmlNodeType.Whitespace:
end = that.TextEnd(end);
goto case XmlNodeType.Element;
case XmlNodeType.Element:
case XmlNodeType.ProcessingInstruction:
case XmlNodeType.Comment:
parent = OwnerNode(node);
DeleteToFollowingSibling(node, end);
if (parent != null) {
ResetPosition(parent);
}
break;
default:
throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
}
}
else {
if (end.IsText) {
end = that.TextEnd(end);
}
if (!IsFollowingSibling(node, end)) {
throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
}
XmlNode parent = OwnerNode(node);
DeleteToFollowingSibling(node, end);
if (parent != null) {
ResetPosition(parent);
}
}
}
public override void DeleteSelf() {
XmlNode node = source;
XmlNode end = node;
switch (node.NodeType) {
case XmlNodeType.Attribute:
XmlAttribute attribute = (XmlAttribute)node;
if (attribute.IsNamespace) {
goto default;
}
XmlNode parent = OwnerNode(attribute);
DeleteAttribute(attribute, attributeIndex);
if (parent != null) {
ResetPosition(parent);
}
break;
case XmlNodeType.Text:
case XmlNodeType.CDATA:
case XmlNodeType.SignificantWhitespace:
case XmlNodeType.Whitespace:
CalibrateText();
node = source;
end = TextEnd(node);
goto case XmlNodeType.Element;
case XmlNodeType.Element:
case XmlNodeType.ProcessingInstruction:
case XmlNodeType.Comment:
parent = OwnerNode(node);
DeleteToFollowingSibling(node, end);
if (parent != null) {
ResetPosition(parent);
}
break;
default:
throw new InvalidOperationException(Res.GetString(Res.Xpn_BadPosition));
}
}
private static void DeleteAttribute(XmlAttribute attribute, int index) {
XmlAttributeCollection attributes;
if (!CheckAttributePosition(attribute, out attributes, index)
&& !ResetAttributePosition(attribute, attributes, out index)) {
throw new InvalidOperationException(Res.GetString(Res.Xpn_MissingParent));
}
if (attribute.IsReadOnly) {
throw new InvalidOperationException(Res.GetString(Res.Xdom_Node_Modify_ReadOnly));
}
attributes.RemoveAt(index);
}
internal static void DeleteToFollowingSibling(XmlNode node, XmlNode end) {
XmlNode parent = node.ParentNode;
if (parent == null) {
throw new InvalidOperationException(Res.GetString(Res.Xpn_MissingParent));
}
if (node.IsReadOnly
|| end.IsReadOnly) {
throw new InvalidOperationException(Res.GetString(Res.Xdom_Node_Modify_ReadOnly));
}
while (node != end) {
XmlNode temp = node;
node = node.NextSibling;
parent.RemoveChild(temp);
}
parent.RemoveChild(node);
}
private static XmlNamespaceManager GetNamespaceManager(XmlNode node, XmlDocument document) {
XmlNamespaceManager namespaceManager = new XmlNamespaceManager(document.NameTable);
List<XmlElement> elements = new List<XmlElement>();
while (node != null) {
XmlElement element = node as XmlElement;
if (element != null
&& element.HasAttributes) {
elements.Add(element);
}
node = node.ParentNode;
}
for (int i = elements.Count - 1; i >= 0; i--) {
namespaceManager.PushScope();
XmlAttributeCollection attributes = elements[i].Attributes;
for (int j = 0; j < attributes.Count; j++) {
XmlAttribute attribute = attributes[j];
if (attribute.IsNamespace) {
string prefix = attribute.Prefix.Length == 0 ? string.Empty : attribute.LocalName;
namespaceManager.AddNamespace(prefix, attribute.Value);
}
}
}
return namespaceManager;
}
internal void ResetPosition(XmlNode node) {
Debug.Assert(node != null, "Undefined navigator position");
Debug.Assert(node == document || node.OwnerDocument == document, "Navigator switched documents");
source = node;
XmlAttribute attribute = node as XmlAttribute;
if (attribute != null) {
XmlElement element = attribute.OwnerElement;
if (element != null) {
ResetAttributePosition(attribute, element.Attributes, out attributeIndex);
if (attribute.IsNamespace) {
namespaceParent = element;
}
}
}
}
private static bool ResetAttributePosition(XmlAttribute attribute, XmlAttributeCollection attributes, out int index) {
if (attributes != null) {
for (int i = 0; i < attributes.Count; i++) {
if (attribute == attributes[i]) {
index = i;
return true;
}
}
}
index = 0;
return false;
}
private static bool CheckAttributePosition(XmlAttribute attribute, out XmlAttributeCollection attributes, int index) {
XmlElement element = attribute.OwnerElement;
if (element != null) {
attributes = element.Attributes;
if (index >= 0
&& index < attributes.Count
&& attribute == attributes[index]) {
return true;
}
}
else {
attributes = null;
}
return false;
}
private void CalibrateText() {
XmlNode text = PreviousText(source);
while (text != null) {
ResetPosition(text);
text = PreviousText(text);
}
}
private XmlNode ParentNode(XmlNode node) {
XmlNode parent = node.ParentNode;
if (!document.HasEntityReferences) {
return parent;
}
return ParentNodeTail(parent);
}
private XmlNode ParentNodeTail(XmlNode parent) {
while (parent != null
&& parent.NodeType == XmlNodeType.EntityReference) {
parent = parent.ParentNode;
}
return parent;
}
private XmlNode FirstChild(XmlNode node) {
XmlNode child = node.FirstChild;
if (!document.HasEntityReferences) {
return child;
}
return FirstChildTail(child);
}
private XmlNode FirstChildTail(XmlNode child) {
while (child != null
&& child.NodeType == XmlNodeType.EntityReference) {
child = child.FirstChild;
}
return child;
}
private XmlNode NextSibling(XmlNode node) {
XmlNode sibling = node.NextSibling;
if (!document.HasEntityReferences) {
return sibling;
}
return NextSiblingTail(node, sibling);
}
private XmlNode NextSiblingTail(XmlNode node, XmlNode sibling) {
while (sibling == null) {
node = node.ParentNode;
if (node == null
|| node.NodeType != XmlNodeType.EntityReference) {
return null;
}
sibling = node.NextSibling;
}
while (sibling != null
&& sibling.NodeType == XmlNodeType.EntityReference) {
sibling = sibling.FirstChild;
}
return sibling;
}
private XmlNode PreviousSibling(XmlNode node) {
XmlNode sibling = node.PreviousSibling;
if (!document.HasEntityReferences) {
return sibling;
}
return PreviousSiblingTail(node, sibling);
}
private XmlNode PreviousSiblingTail(XmlNode node, XmlNode sibling) {
while (sibling == null) {
node = node.ParentNode;
if (node == null
|| node.NodeType != XmlNodeType.EntityReference) {
return null;
}
sibling = node.PreviousSibling;
}
while (sibling != null
&& sibling.NodeType == XmlNodeType.EntityReference) {
sibling = sibling.LastChild;
}
return sibling;
}
private XmlNode PreviousText(XmlNode node) {
XmlNode text = node.PreviousText;
if (!document.HasEntityReferences) {
return text;
}
return PreviousTextTail(node, text);
}
private XmlNode PreviousTextTail(XmlNode node, XmlNode text) {
if (text != null) {
return text;
}
if (!node.IsText) {
return null;
}
XmlNode sibling = node.PreviousSibling;
while (sibling == null) {
node = node.ParentNode;
if (node == null
|| node.NodeType != XmlNodeType.EntityReference) {
return null;
}
sibling = node.PreviousSibling;
}
while (sibling != null) {
switch (sibling.NodeType) {
case XmlNodeType.EntityReference:
sibling = sibling.LastChild;
break;
case XmlNodeType.Text:
case XmlNodeType.CDATA:
case XmlNodeType.Whitespace:
case XmlNodeType.SignificantWhitespace:
return sibling;
default:
return null;
}
}
return null;
}
internal static bool IsFollowingSibling(XmlNode left, XmlNode right) {
for (;;) {
left = left.NextSibling;
if (left == null) {
break;
}
if (left == right) {
return true;
}
}
return false;
}
private static bool IsDescendant(XmlNode top, XmlNode bottom) {
for (;;) {
XmlNode parent = bottom.ParentNode;
if (parent == null) {
XmlAttribute attribute = bottom as XmlAttribute;
if (attribute == null) {
break;
}
parent = attribute.OwnerElement;
if (parent == null) {
break;
}
}
bottom = parent;
if (top == bottom) {
return true;
}
}
return false;
}
private static bool IsValidChild(XmlNode parent, XmlNode child) {
switch (parent.NodeType) {
case XmlNodeType.Element:
return true;
case XmlNodeType.DocumentFragment:
switch (child.NodeType) {
case XmlNodeType.Element:
case XmlNodeType.Text:
case XmlNodeType.CDATA:
case XmlNodeType.ProcessingInstruction:
case XmlNodeType.Comment:
case XmlNodeType.Whitespace:
case XmlNodeType.SignificantWhitespace:
return true;
}
break;
case XmlNodeType.Document:
switch (child.NodeType) {
case XmlNodeType.Element:
case XmlNodeType.ProcessingInstruction:
case XmlNodeType.Comment:
return true;
}
break;
default:
break;
}
return false;
}
private XmlNode TextStart(XmlNode node) {
XmlNode start;
do {
start = node;
node = PreviousSibling(node);
}
while (node != null
&& node.IsText);
return start;
}
private XmlNode TextEnd(XmlNode node) {
XmlNode end;
do {
end = node;
node = NextSibling(node);
}
while (node != null
&& node.IsText);
return end;
}
}
// An iterator that matches no nodes
internal sealed class DocumentXPathNodeIterator_Empty : XPathNodeIterator {
private XPathNavigator nav;
internal DocumentXPathNodeIterator_Empty( DocumentXPathNavigator nav ) { this.nav = nav.Clone(); }
internal DocumentXPathNodeIterator_Empty( DocumentXPathNodeIterator_Empty other ) { this.nav = other.nav.Clone(); }
public override XPathNodeIterator Clone() { return new DocumentXPathNodeIterator_Empty( this ); }
public override bool MoveNext() { return false; }
public override XPathNavigator Current { get { return nav; } }
public override int CurrentPosition { get { return 0; } }
public override int Count { get { return 0; } }
}
// An iterator that can match any child elements that match the Match condition (overrided in the derived class)
internal abstract class DocumentXPathNodeIterator_ElemDescendants : XPathNodeIterator {
private DocumentXPathNavigator nav;
private int level;
private int position;
internal DocumentXPathNodeIterator_ElemDescendants( DocumentXPathNavigator nav ) {
this.nav = (DocumentXPathNavigator)(nav.Clone());
this.level = 0;
this.position = 0;
}
internal DocumentXPathNodeIterator_ElemDescendants( DocumentXPathNodeIterator_ElemDescendants other ) {
this.nav = (DocumentXPathNavigator)(other.nav.Clone());
this.level = other.level;
this.position = other.position;
}
protected abstract bool Match( XmlNode node );
public override XPathNavigator Current {
get { return nav; }
}
public override int CurrentPosition {
get { return position; }
}
protected void SetPosition( int pos ) {
position = pos;
}
public override bool MoveNext() {
for (;;) {
if (nav.MoveToFirstChild()) {
level++;
}
else {
if (level == 0) {
return false;
}
while (!nav.MoveToNext()) {
level--;
if (level == 0) {
return false;
}
if (!nav.MoveToParent()) {
return false;
}
}
}
XmlNode node = (XmlNode)nav.UnderlyingObject;
if (node.NodeType == XmlNodeType.Element && Match(node)) {
position++;
return true;
}
}
}
}
// Iterate over all element children irrespective of the localName and namespace
internal class DocumentXPathNodeIterator_AllElemChildren : DocumentXPathNodeIterator_ElemDescendants {
internal DocumentXPathNodeIterator_AllElemChildren( DocumentXPathNavigator nav ) : base( nav ) {
Debug.Assert( ((XmlNode)nav.UnderlyingObject).NodeType != XmlNodeType.Attribute );
}
internal DocumentXPathNodeIterator_AllElemChildren( DocumentXPathNodeIterator_AllElemChildren other ) : base( other ) {
}
public override XPathNodeIterator Clone() {
return new DocumentXPathNodeIterator_AllElemChildren( this );
}
protected override bool Match( XmlNode node ) {
Debug.Assert( node != null );
return ( node.NodeType == XmlNodeType.Element );
}
}
// Iterate over all element children irrespective of the localName and namespace, include the self node when testing for localName/ns
internal sealed class DocumentXPathNodeIterator_AllElemChildren_AndSelf : DocumentXPathNodeIterator_AllElemChildren {
internal DocumentXPathNodeIterator_AllElemChildren_AndSelf( DocumentXPathNavigator nav ) : base( nav ) {
}
internal DocumentXPathNodeIterator_AllElemChildren_AndSelf( DocumentXPathNodeIterator_AllElemChildren_AndSelf other ) : base( other ) {
}
public override XPathNodeIterator Clone() {
return new DocumentXPathNodeIterator_AllElemChildren_AndSelf( this );
}
public override bool MoveNext() {
if( CurrentPosition == 0 ) {
DocumentXPathNavigator nav = (DocumentXPathNavigator)this.Current;
XmlNode node = (XmlNode)nav.UnderlyingObject;
if ( node.NodeType == XmlNodeType.Element && Match( node ) ) {
SetPosition( 1 );
return true;
}
}
return base.MoveNext();
}
}
// Iterate over all element children that have a given namespace but irrespective of the localName
internal class DocumentXPathNodeIterator_ElemChildren_NoLocalName : DocumentXPathNodeIterator_ElemDescendants {
private string nsAtom;
internal DocumentXPathNodeIterator_ElemChildren_NoLocalName( DocumentXPathNavigator nav, string nsAtom ) : base( nav ) {
Debug.Assert( ((XmlNode)nav.UnderlyingObject).NodeType != XmlNodeType.Attribute );
Debug.Assert( Ref.Equal(nav.NameTable.Get( nsAtom ), nsAtom) );
this.nsAtom = nsAtom;
}
internal DocumentXPathNodeIterator_ElemChildren_NoLocalName( DocumentXPathNodeIterator_ElemChildren_NoLocalName other ) : base( other ) {
this.nsAtom = other.nsAtom;
}
public override XPathNodeIterator Clone() {
return new DocumentXPathNodeIterator_ElemChildren_NoLocalName( this );
}
protected override bool Match( XmlNode node ) {
Debug.Assert( node != null );
Debug.Assert( node.NodeType == XmlNodeType.Element );
return Ref.Equal(node.NamespaceURI, nsAtom);
}
}
// Iterate over all element children that have a given namespace but irrespective of the localName, include self node when checking for ns
internal sealed class DocumentXPathNodeIterator_ElemChildren_AndSelf_NoLocalName : DocumentXPathNodeIterator_ElemChildren_NoLocalName {
internal DocumentXPathNodeIterator_ElemChildren_AndSelf_NoLocalName( DocumentXPathNavigator nav, string nsAtom ) : base( nav, nsAtom ) {
}
internal DocumentXPathNodeIterator_ElemChildren_AndSelf_NoLocalName( DocumentXPathNodeIterator_ElemChildren_AndSelf_NoLocalName other ) : base( other ) {
}
public override XPathNodeIterator Clone() {
return new DocumentXPathNodeIterator_ElemChildren_AndSelf_NoLocalName( this );
}
public override bool MoveNext() {
if( CurrentPosition == 0 ) {
DocumentXPathNavigator nav = (DocumentXPathNavigator)this.Current;
XmlNode node = (XmlNode)nav.UnderlyingObject;
if ( node.NodeType == XmlNodeType.Element && Match( node ) ) {
SetPosition( 1 );
return true;
}
}
return base.MoveNext();
}
}
// Iterate over all element children that have a given name and namespace
internal class DocumentXPathNodeIterator_ElemChildren : DocumentXPathNodeIterator_ElemDescendants {
protected string localNameAtom;
protected string nsAtom;
internal DocumentXPathNodeIterator_ElemChildren( DocumentXPathNavigator nav, string localNameAtom, string nsAtom ) : base( nav ) {
Debug.Assert( ((XmlNode)nav.UnderlyingObject).NodeType != XmlNodeType.Attribute );
Debug.Assert( Ref.Equal(nav.NameTable.Get( localNameAtom ), localNameAtom) );
Debug.Assert( Ref.Equal(nav.NameTable.Get( nsAtom ), nsAtom) );
Debug.Assert( localNameAtom.Length > 0 ); // Use DocumentXPathNodeIterator_ElemChildren_NoLocalName class for special magic value of localNameAtom
this.localNameAtom = localNameAtom;
this.nsAtom = nsAtom;
}
internal DocumentXPathNodeIterator_ElemChildren( DocumentXPathNodeIterator_ElemChildren other ) : base( other ) {
this.localNameAtom = other.localNameAtom;
this.nsAtom = other.nsAtom;
}
public override XPathNodeIterator Clone() {
return new DocumentXPathNodeIterator_ElemChildren( this );
}
protected override bool Match( XmlNode node ) {
Debug.Assert( node != null );
Debug.Assert( node.NodeType == XmlNodeType.Element );
return Ref.Equal(node.LocalName, localNameAtom) && Ref.Equal(node.NamespaceURI, nsAtom);
}
}
// Iterate over all elem children and itself and check for the given localName (including the magic value "") and namespace
internal sealed class DocumentXPathNodeIterator_ElemChildren_AndSelf : DocumentXPathNodeIterator_ElemChildren {
internal DocumentXPathNodeIterator_ElemChildren_AndSelf( DocumentXPathNavigator nav, string localNameAtom, string nsAtom )
: base( nav, localNameAtom, nsAtom ) {
Debug.Assert( localNameAtom.Length > 0 ); // Use DocumentXPathNodeIterator_ElemChildren_AndSelf_NoLocalName if localName == String.Empty
}
internal DocumentXPathNodeIterator_ElemChildren_AndSelf( DocumentXPathNodeIterator_ElemChildren_AndSelf other ) : base( other ) {
}
public override XPathNodeIterator Clone() {
return new DocumentXPathNodeIterator_ElemChildren_AndSelf( this );
}
public override bool MoveNext() {
if( CurrentPosition == 0 ) {
DocumentXPathNavigator nav = (DocumentXPathNavigator)this.Current;
XmlNode node = (XmlNode)nav.UnderlyingObject;
if ( node.NodeType == XmlNodeType.Element && Match( node ) ) {
SetPosition( 1 );
return true;
}
}
return base.MoveNext();
}
}
}