e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
1239 lines
47 KiB
C#
1239 lines
47 KiB
C#
//------------------------------------------------------------------------------
|
|
// <copyright file="XmlNode.cs" company="Microsoft">
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
// </copyright>
|
|
// <owner current="true" primary="true">[....]</owner>
|
|
//------------------------------------------------------------------------------
|
|
|
|
namespace System.Xml {
|
|
using System;
|
|
using System.IO;
|
|
using System.Collections;
|
|
using System.Text;
|
|
using System.Diagnostics;
|
|
using System.Xml.Schema;
|
|
using System.Xml.XPath;
|
|
using MS.Internal.Xml.XPath;
|
|
using System.Globalization;
|
|
|
|
// Represents a single node in the document.
|
|
[DebuggerDisplay("{debuggerDisplayProxy}")]
|
|
public abstract class XmlNode : ICloneable, IEnumerable, IXPathNavigable {
|
|
internal XmlNode parentNode; //this pointer is reused to save the userdata information, need to prevent internal user access the pointer directly.
|
|
|
|
internal XmlNode () {
|
|
}
|
|
|
|
internal XmlNode( XmlDocument doc ) {
|
|
if ( doc == null )
|
|
throw new ArgumentException(Res.GetString(Res.Xdom_Node_Null_Doc));
|
|
this.parentNode = doc;
|
|
}
|
|
|
|
public virtual XPathNavigator CreateNavigator() {
|
|
XmlDocument thisAsDoc = this as XmlDocument;
|
|
if ( thisAsDoc != null ) {
|
|
return thisAsDoc.CreateNavigator( this );
|
|
}
|
|
XmlDocument doc = OwnerDocument;
|
|
Debug.Assert( doc != null );
|
|
return doc.CreateNavigator( this );
|
|
}
|
|
|
|
// Selects the first node that matches the xpath expression
|
|
public XmlNode SelectSingleNode( string xpath ) {
|
|
XmlNodeList list = SelectNodes(xpath);
|
|
// SelectNodes returns null for certain node types
|
|
return list != null ? list[0] : null;
|
|
}
|
|
|
|
// Selects the first node that matches the xpath expression and given namespace context.
|
|
public XmlNode SelectSingleNode( string xpath, XmlNamespaceManager nsmgr ) {
|
|
XPathNavigator xn = (this).CreateNavigator();
|
|
//if the method is called on node types like DocType, Entity, XmlDeclaration,
|
|
//the navigator returned is null. So just return null from here for those node types.
|
|
if( xn == null )
|
|
return null;
|
|
XPathExpression exp = xn.Compile(xpath);
|
|
exp.SetContext(nsmgr);
|
|
return new XPathNodeList(xn.Select(exp))[0];
|
|
}
|
|
|
|
// Selects all nodes that match the xpath expression
|
|
public XmlNodeList SelectNodes( string xpath ) {
|
|
XPathNavigator n = (this).CreateNavigator();
|
|
//if the method is called on node types like DocType, Entity, XmlDeclaration,
|
|
//the navigator returned is null. So just return null from here for those node types.
|
|
if( n == null )
|
|
return null;
|
|
return new XPathNodeList( n.Select(xpath) );
|
|
}
|
|
|
|
// Selects all nodes that match the xpath expression and given namespace context.
|
|
public XmlNodeList SelectNodes( string xpath, XmlNamespaceManager nsmgr ) {
|
|
XPathNavigator xn = (this).CreateNavigator();
|
|
//if the method is called on node types like DocType, Entity, XmlDeclaration,
|
|
//the navigator returned is null. So just return null from here for those node types.
|
|
if( xn == null )
|
|
return null;
|
|
XPathExpression exp = xn.Compile(xpath);
|
|
exp.SetContext(nsmgr);
|
|
return new XPathNodeList( xn.Select(exp) );
|
|
}
|
|
|
|
// Gets the name of the node.
|
|
public abstract string Name {
|
|
get;
|
|
}
|
|
|
|
// Gets or sets the value of the node.
|
|
public virtual string Value {
|
|
get { return null;}
|
|
set { throw new InvalidOperationException(string.Format(CultureInfo.InvariantCulture, Res.GetString(Res.Xdom_Node_SetVal), NodeType.ToString()));}
|
|
}
|
|
|
|
// Gets the type of the current node.
|
|
public abstract XmlNodeType NodeType {
|
|
get;
|
|
}
|
|
|
|
// Gets the parent of this node (for nodes that can have parents).
|
|
public virtual XmlNode ParentNode {
|
|
get {
|
|
Debug.Assert(parentNode != null);
|
|
|
|
if (parentNode.NodeType != XmlNodeType.Document) {
|
|
return parentNode;
|
|
}
|
|
|
|
// Linear lookup through the children of the document
|
|
XmlLinkedNode firstChild = parentNode.FirstChild as XmlLinkedNode;
|
|
if (firstChild != null) {
|
|
XmlLinkedNode node = firstChild;
|
|
do {
|
|
if (node == this) {
|
|
return parentNode;
|
|
}
|
|
node = node.next;
|
|
}
|
|
while (node != null
|
|
&& node != firstChild);
|
|
}
|
|
return null;
|
|
}
|
|
}
|
|
|
|
// Gets all children of this node.
|
|
public virtual XmlNodeList ChildNodes {
|
|
get { return new XmlChildNodes(this);}
|
|
}
|
|
|
|
// Gets the node immediately preceding this node.
|
|
public virtual XmlNode PreviousSibling {
|
|
get { return null;}
|
|
}
|
|
|
|
// Gets the node immediately following this node.
|
|
public virtual XmlNode NextSibling {
|
|
get { return null;}
|
|
}
|
|
|
|
// Gets a XmlAttributeCollection containing the attributes
|
|
// of this node.
|
|
public virtual XmlAttributeCollection Attributes {
|
|
get { return null;}
|
|
}
|
|
|
|
// Gets the XmlDocument that contains this node.
|
|
public virtual XmlDocument OwnerDocument {
|
|
get {
|
|
Debug.Assert( parentNode != null );
|
|
if ( parentNode.NodeType == XmlNodeType.Document)
|
|
return (XmlDocument)parentNode;
|
|
return parentNode.OwnerDocument;
|
|
}
|
|
}
|
|
|
|
// Gets the first child of this node.
|
|
public virtual XmlNode FirstChild {
|
|
get {
|
|
XmlLinkedNode linkedNode = LastNode;
|
|
if (linkedNode != null)
|
|
return linkedNode.next;
|
|
|
|
return null;
|
|
}
|
|
}
|
|
|
|
// Gets the last child of this node.
|
|
public virtual XmlNode LastChild {
|
|
get { return LastNode;}
|
|
}
|
|
|
|
internal virtual bool IsContainer {
|
|
get { return false;}
|
|
}
|
|
|
|
internal virtual XmlLinkedNode LastNode {
|
|
get { return null;}
|
|
set {}
|
|
}
|
|
|
|
internal bool AncestorNode(XmlNode node) {
|
|
XmlNode n = this.ParentNode;
|
|
|
|
while (n != null && n != this) {
|
|
if (n == node)
|
|
return true;
|
|
n = n.ParentNode;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
//trace to the top to find out its parent node.
|
|
internal bool IsConnected()
|
|
{
|
|
XmlNode parent = ParentNode;
|
|
while (parent != null && !( parent.NodeType == XmlNodeType.Document ))
|
|
parent = parent.ParentNode;
|
|
return parent != null;
|
|
}
|
|
|
|
// Inserts the specified node immediately before the specified reference node.
|
|
public virtual XmlNode InsertBefore(XmlNode newChild, XmlNode refChild) {
|
|
if (this == newChild || AncestorNode(newChild))
|
|
throw new ArgumentException(Res.GetString(Res.Xdom_Node_Insert_Child));
|
|
|
|
if (refChild == null)
|
|
return AppendChild(newChild);
|
|
|
|
if (!IsContainer)
|
|
throw new InvalidOperationException(Res.GetString(Res.Xdom_Node_Insert_Contain));
|
|
|
|
if (refChild.ParentNode != this)
|
|
throw new ArgumentException(Res.GetString(Res.Xdom_Node_Insert_Path));
|
|
|
|
if (newChild == refChild)
|
|
return newChild;
|
|
|
|
XmlDocument childDoc = newChild.OwnerDocument;
|
|
XmlDocument thisDoc = OwnerDocument;
|
|
if (childDoc != null && childDoc != thisDoc && childDoc != this)
|
|
throw new ArgumentException(Res.GetString(Res.Xdom_Node_Insert_Context));
|
|
|
|
if (!CanInsertBefore( newChild, refChild ))
|
|
throw new InvalidOperationException(Res.GetString(Res.Xdom_Node_Insert_Location));
|
|
|
|
if (newChild.ParentNode != null)
|
|
newChild.ParentNode.RemoveChild( newChild );
|
|
|
|
// special case for doc-fragment.
|
|
if (newChild.NodeType == XmlNodeType.DocumentFragment) {
|
|
XmlNode first = newChild.FirstChild;
|
|
XmlNode node = first;
|
|
if (node != null) {
|
|
newChild.RemoveChild( node );
|
|
InsertBefore( node, refChild );
|
|
// insert the rest of the children after this one.
|
|
InsertAfter( newChild, node );
|
|
}
|
|
return first;
|
|
}
|
|
|
|
if (!(newChild is XmlLinkedNode) || !IsValidChildType(newChild.NodeType))
|
|
throw new InvalidOperationException(Res.GetString(Res.Xdom_Node_Insert_TypeConflict));
|
|
|
|
XmlLinkedNode newNode = (XmlLinkedNode) newChild;
|
|
XmlLinkedNode refNode = (XmlLinkedNode) refChild;
|
|
|
|
string newChildValue = newChild.Value;
|
|
XmlNodeChangedEventArgs args = GetEventArgs( newChild, newChild.ParentNode, this, newChildValue, newChildValue, XmlNodeChangedAction.Insert );
|
|
|
|
if (args != null)
|
|
BeforeEvent( args );
|
|
|
|
if (refNode == FirstChild) {
|
|
newNode.next = refNode;
|
|
LastNode.next = newNode;
|
|
newNode.SetParent(this);
|
|
|
|
if (newNode.IsText) {
|
|
if (refNode.IsText) {
|
|
NestTextNodes(newNode, refNode);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
XmlLinkedNode prevNode = (XmlLinkedNode) refNode.PreviousSibling;
|
|
|
|
newNode.next = refNode;
|
|
prevNode.next = newNode;
|
|
newNode.SetParent(this);
|
|
|
|
if (prevNode.IsText) {
|
|
if (newNode.IsText) {
|
|
NestTextNodes(prevNode, newNode);
|
|
if (refNode.IsText) {
|
|
NestTextNodes(newNode, refNode);
|
|
}
|
|
}
|
|
else {
|
|
if (refNode.IsText) {
|
|
UnnestTextNodes(prevNode, refNode);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
if (newNode.IsText) {
|
|
if (refNode.IsText) {
|
|
NestTextNodes(newNode, refNode);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (args != null)
|
|
AfterEvent( args );
|
|
|
|
return newNode;
|
|
}
|
|
|
|
// Inserts the specified node immediately after the specified reference node.
|
|
public virtual XmlNode InsertAfter(XmlNode newChild, XmlNode refChild) {
|
|
if (this == newChild || AncestorNode(newChild))
|
|
throw new ArgumentException(Res.GetString(Res.Xdom_Node_Insert_Child));
|
|
|
|
if (refChild == null)
|
|
return PrependChild(newChild);
|
|
|
|
if (!IsContainer)
|
|
throw new InvalidOperationException(Res.GetString(Res.Xdom_Node_Insert_Contain));
|
|
|
|
if (refChild.ParentNode != this)
|
|
throw new ArgumentException(Res.GetString(Res.Xdom_Node_Insert_Path));
|
|
|
|
if (newChild == refChild)
|
|
return newChild;
|
|
|
|
XmlDocument childDoc = newChild.OwnerDocument;
|
|
XmlDocument thisDoc = OwnerDocument;
|
|
if (childDoc != null && childDoc != thisDoc && childDoc != this)
|
|
throw new ArgumentException(Res.GetString(Res.Xdom_Node_Insert_Context));
|
|
|
|
if (!CanInsertAfter( newChild, refChild ))
|
|
throw new InvalidOperationException(Res.GetString(Res.Xdom_Node_Insert_Location));
|
|
|
|
if (newChild.ParentNode != null)
|
|
newChild.ParentNode.RemoveChild( newChild );
|
|
|
|
// special case for doc-fragment.
|
|
if (newChild.NodeType == XmlNodeType.DocumentFragment) {
|
|
XmlNode last = refChild;
|
|
XmlNode first = newChild.FirstChild;
|
|
XmlNode node = first;
|
|
while (node != null) {
|
|
XmlNode next = node.NextSibling;
|
|
newChild.RemoveChild( node );
|
|
InsertAfter( node, last );
|
|
last = node;
|
|
node = next;
|
|
}
|
|
return first;
|
|
}
|
|
|
|
if (!(newChild is XmlLinkedNode) || !IsValidChildType(newChild.NodeType))
|
|
throw new InvalidOperationException(Res.GetString(Res.Xdom_Node_Insert_TypeConflict));
|
|
|
|
XmlLinkedNode newNode = (XmlLinkedNode) newChild;
|
|
XmlLinkedNode refNode = (XmlLinkedNode) refChild;
|
|
|
|
string newChildValue = newChild.Value;
|
|
XmlNodeChangedEventArgs args = GetEventArgs( newChild, newChild.ParentNode, this, newChildValue, newChildValue, XmlNodeChangedAction.Insert );
|
|
|
|
if (args != null)
|
|
BeforeEvent( args );
|
|
|
|
if (refNode == LastNode) {
|
|
newNode.next = refNode.next;
|
|
refNode.next = newNode;
|
|
LastNode = newNode;
|
|
newNode.SetParent(this);
|
|
|
|
if (refNode.IsText) {
|
|
if (newNode.IsText) {
|
|
NestTextNodes(refNode, newNode);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
XmlLinkedNode nextNode = refNode.next;
|
|
|
|
newNode.next = nextNode;
|
|
refNode.next = newNode;
|
|
newNode.SetParent(this);
|
|
|
|
if (refNode.IsText) {
|
|
if (newNode.IsText) {
|
|
NestTextNodes(refNode, newNode);
|
|
if (nextNode.IsText) {
|
|
NestTextNodes(newNode, nextNode);
|
|
}
|
|
}
|
|
else {
|
|
if (nextNode.IsText) {
|
|
UnnestTextNodes(refNode, nextNode);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
if (newNode.IsText) {
|
|
if (nextNode.IsText) {
|
|
NestTextNodes(newNode, nextNode);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
if (args != null)
|
|
AfterEvent( args );
|
|
|
|
return newNode;
|
|
}
|
|
|
|
// Replaces the child node oldChild with newChild node.
|
|
public virtual XmlNode ReplaceChild(XmlNode newChild, XmlNode oldChild) {
|
|
XmlNode nextNode = oldChild.NextSibling;
|
|
RemoveChild(oldChild);
|
|
XmlNode node = InsertBefore( newChild, nextNode );
|
|
return oldChild;
|
|
}
|
|
|
|
// Removes specified child node.
|
|
public virtual XmlNode RemoveChild(XmlNode oldChild) {
|
|
if (!IsContainer)
|
|
throw new InvalidOperationException(Res.GetString(Res.Xdom_Node_Remove_Contain));
|
|
|
|
if (oldChild.ParentNode != this)
|
|
throw new ArgumentException(Res.GetString(Res.Xdom_Node_Remove_Child));
|
|
|
|
XmlLinkedNode oldNode = (XmlLinkedNode) oldChild;
|
|
|
|
string oldNodeValue = oldNode.Value;
|
|
XmlNodeChangedEventArgs args = GetEventArgs( oldNode, this, null, oldNodeValue, oldNodeValue, XmlNodeChangedAction.Remove );
|
|
|
|
if (args != null)
|
|
BeforeEvent( args );
|
|
|
|
XmlLinkedNode lastNode = LastNode;
|
|
|
|
if (oldNode == FirstChild) {
|
|
if (oldNode == lastNode) {
|
|
LastNode = null;
|
|
oldNode.next = null;
|
|
oldNode.SetParent( null );
|
|
}
|
|
else {
|
|
XmlLinkedNode nextNode = oldNode.next;
|
|
|
|
if (nextNode.IsText) {
|
|
if (oldNode.IsText) {
|
|
UnnestTextNodes(oldNode, nextNode);
|
|
}
|
|
}
|
|
|
|
lastNode.next = nextNode;
|
|
oldNode.next = null;
|
|
oldNode.SetParent( null );
|
|
}
|
|
}
|
|
else {
|
|
if (oldNode == lastNode) {
|
|
XmlLinkedNode prevNode = (XmlLinkedNode) oldNode.PreviousSibling;
|
|
prevNode.next = oldNode.next;
|
|
LastNode = prevNode;
|
|
oldNode.next = null;
|
|
oldNode.SetParent(null);
|
|
}
|
|
else {
|
|
XmlLinkedNode prevNode = (XmlLinkedNode) oldNode.PreviousSibling;
|
|
XmlLinkedNode nextNode = oldNode.next;
|
|
|
|
if (nextNode.IsText) {
|
|
if (prevNode.IsText) {
|
|
NestTextNodes(prevNode, nextNode);
|
|
}
|
|
else {
|
|
if (oldNode.IsText) {
|
|
UnnestTextNodes(oldNode, nextNode);
|
|
}
|
|
}
|
|
}
|
|
|
|
prevNode.next = nextNode;
|
|
oldNode.next = null;
|
|
oldNode.SetParent(null);
|
|
}
|
|
}
|
|
|
|
if (args != null)
|
|
AfterEvent( args );
|
|
|
|
return oldChild;
|
|
}
|
|
|
|
// Adds the specified node to the beginning of the list of children of this node.
|
|
public virtual XmlNode PrependChild(XmlNode newChild) {
|
|
return InsertBefore(newChild, FirstChild);
|
|
}
|
|
|
|
// Adds the specified node to the end of the list of children of this node.
|
|
public virtual XmlNode AppendChild(XmlNode newChild) {
|
|
XmlDocument thisDoc = OwnerDocument;
|
|
if ( thisDoc == null ) {
|
|
thisDoc = this as XmlDocument;
|
|
}
|
|
if (!IsContainer)
|
|
throw new InvalidOperationException(Res.GetString(Res.Xdom_Node_Insert_Contain));
|
|
|
|
if (this == newChild || AncestorNode(newChild))
|
|
throw new ArgumentException(Res.GetString(Res.Xdom_Node_Insert_Child));
|
|
|
|
if (newChild.ParentNode != null)
|
|
newChild.ParentNode.RemoveChild( newChild );
|
|
|
|
XmlDocument childDoc = newChild.OwnerDocument;
|
|
if (childDoc != null && childDoc != thisDoc && childDoc != this)
|
|
throw new ArgumentException(Res.GetString(Res.Xdom_Node_Insert_Context));
|
|
|
|
// special case for doc-fragment.
|
|
if (newChild.NodeType == XmlNodeType.DocumentFragment) {
|
|
XmlNode first = newChild.FirstChild;
|
|
XmlNode node = first;
|
|
while (node != null) {
|
|
XmlNode next = node.NextSibling;
|
|
newChild.RemoveChild( node );
|
|
AppendChild( node );
|
|
node = next;
|
|
}
|
|
return first;
|
|
}
|
|
|
|
if (!(newChild is XmlLinkedNode) || !IsValidChildType(newChild.NodeType))
|
|
throw new InvalidOperationException(Res.GetString(Res.Xdom_Node_Insert_TypeConflict));
|
|
|
|
|
|
if (!CanInsertAfter( newChild, LastChild ))
|
|
throw new InvalidOperationException(Res.GetString(Res.Xdom_Node_Insert_Location));
|
|
|
|
string newChildValue = newChild.Value;
|
|
XmlNodeChangedEventArgs args = GetEventArgs( newChild, newChild.ParentNode, this, newChildValue, newChildValue, XmlNodeChangedAction.Insert );
|
|
|
|
if (args != null)
|
|
BeforeEvent( args );
|
|
|
|
XmlLinkedNode refNode = LastNode;
|
|
XmlLinkedNode newNode = (XmlLinkedNode) newChild;
|
|
|
|
if (refNode == null) {
|
|
newNode.next = newNode;
|
|
LastNode = newNode;
|
|
newNode.SetParent(this);
|
|
}
|
|
else {
|
|
newNode.next = refNode.next;
|
|
refNode.next = newNode;
|
|
LastNode = newNode;
|
|
newNode.SetParent(this);
|
|
|
|
if (refNode.IsText) {
|
|
if (newNode.IsText) {
|
|
NestTextNodes(refNode, newNode);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (args != null)
|
|
AfterEvent( args );
|
|
|
|
return newNode;
|
|
}
|
|
|
|
//the function is provided only at Load time to speed up Load process
|
|
internal virtual XmlNode AppendChildForLoad(XmlNode newChild, XmlDocument doc) {
|
|
XmlNodeChangedEventArgs args = doc.GetInsertEventArgsForLoad( newChild, this );
|
|
|
|
if (args != null)
|
|
doc.BeforeEvent( args );
|
|
|
|
XmlLinkedNode refNode = LastNode;
|
|
XmlLinkedNode newNode = (XmlLinkedNode) newChild;
|
|
|
|
if (refNode == null) {
|
|
newNode.next = newNode;
|
|
LastNode = newNode;
|
|
newNode.SetParentForLoad(this);
|
|
}
|
|
else {
|
|
newNode.next = refNode.next;
|
|
refNode.next = newNode;
|
|
LastNode = newNode;
|
|
if (refNode.IsText
|
|
&& newNode.IsText) {
|
|
NestTextNodes(refNode, newNode);
|
|
}
|
|
else {
|
|
newNode.SetParentForLoad(this);
|
|
}
|
|
}
|
|
|
|
if (args != null)
|
|
doc.AfterEvent( args );
|
|
|
|
return newNode;
|
|
}
|
|
|
|
internal virtual bool IsValidChildType( XmlNodeType type ) {
|
|
return false;
|
|
}
|
|
|
|
internal virtual bool CanInsertBefore( XmlNode newChild, XmlNode refChild ) {
|
|
return true;
|
|
}
|
|
|
|
internal virtual bool CanInsertAfter( XmlNode newChild, XmlNode refChild ) {
|
|
return true;
|
|
}
|
|
|
|
// Gets a value indicating whether this node has any child nodes.
|
|
public virtual bool HasChildNodes {
|
|
get { return LastNode != null;}
|
|
}
|
|
|
|
// Creates a duplicate of this node.
|
|
public abstract XmlNode CloneNode(bool deep);
|
|
|
|
internal virtual void CopyChildren( XmlDocument doc, XmlNode container, bool deep ) {
|
|
for (XmlNode child = container.FirstChild; child != null; child = child.NextSibling) {
|
|
AppendChildForLoad( child.CloneNode(deep), doc );
|
|
}
|
|
}
|
|
|
|
// DOM Level 2
|
|
|
|
// Puts all XmlText nodes in the full depth of the sub-tree
|
|
// underneath this XmlNode into a "normal" form where only
|
|
// markup (e.g., tags, comments, processing instructions, CDATA sections,
|
|
// and entity references) separates XmlText nodes, that is, there
|
|
// are no adjacent XmlText nodes.
|
|
public virtual void Normalize() {
|
|
XmlNode firstChildTextLikeNode = null;
|
|
StringBuilder sb = new StringBuilder();
|
|
for ( XmlNode crtChild = this.FirstChild; crtChild != null; ) {
|
|
XmlNode nextChild = crtChild.NextSibling;
|
|
switch ( crtChild.NodeType ) {
|
|
case XmlNodeType.Text:
|
|
case XmlNodeType.Whitespace:
|
|
case XmlNodeType.SignificantWhitespace: {
|
|
sb.Append( crtChild.Value );
|
|
XmlNode winner = NormalizeWinner( firstChildTextLikeNode, crtChild );
|
|
if ( winner == firstChildTextLikeNode ) {
|
|
this.RemoveChild( crtChild );
|
|
}
|
|
else {
|
|
if ( firstChildTextLikeNode != null )
|
|
this.RemoveChild( firstChildTextLikeNode );
|
|
firstChildTextLikeNode = crtChild;
|
|
}
|
|
break;
|
|
}
|
|
case XmlNodeType.Element: {
|
|
crtChild.Normalize();
|
|
goto default;
|
|
}
|
|
default : {
|
|
if ( firstChildTextLikeNode != null ) {
|
|
firstChildTextLikeNode.Value = sb.ToString();
|
|
firstChildTextLikeNode = null;
|
|
}
|
|
sb.Remove( 0, sb.Length );
|
|
break;
|
|
}
|
|
}
|
|
crtChild = nextChild;
|
|
}
|
|
if ( firstChildTextLikeNode != null && sb.Length > 0 )
|
|
firstChildTextLikeNode.Value = sb.ToString();
|
|
}
|
|
|
|
private XmlNode NormalizeWinner( XmlNode firstNode, XmlNode secondNode ) {
|
|
//first node has the priority
|
|
if ( firstNode == null )
|
|
return secondNode;
|
|
Debug.Assert( firstNode.NodeType == XmlNodeType.Text
|
|
|| firstNode.NodeType == XmlNodeType.SignificantWhitespace
|
|
|| firstNode.NodeType == XmlNodeType.Whitespace
|
|
|| secondNode.NodeType == XmlNodeType.Text
|
|
|| secondNode.NodeType == XmlNodeType.SignificantWhitespace
|
|
|| secondNode.NodeType == XmlNodeType.Whitespace );
|
|
if ( firstNode.NodeType == XmlNodeType.Text )
|
|
return firstNode;
|
|
if ( secondNode.NodeType == XmlNodeType.Text )
|
|
return secondNode;
|
|
if ( firstNode.NodeType == XmlNodeType.SignificantWhitespace )
|
|
return firstNode;
|
|
if ( secondNode.NodeType == XmlNodeType.SignificantWhitespace )
|
|
return secondNode;
|
|
if ( firstNode.NodeType == XmlNodeType.Whitespace )
|
|
return firstNode;
|
|
if ( secondNode.NodeType == XmlNodeType.Whitespace )
|
|
return secondNode;
|
|
Debug.Assert( true, "shouldn't have fall through here." );
|
|
return null;
|
|
}
|
|
|
|
// Test if the DOM implementation implements a specific feature.
|
|
public virtual bool Supports(string feature, string version) {
|
|
if (String.Compare("XML", feature, StringComparison.OrdinalIgnoreCase) == 0) {
|
|
if (version == null || version == "1.0" || version == "2.0")
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Gets the namespace URI of this node.
|
|
public virtual string NamespaceURI {
|
|
get { return string.Empty;}
|
|
}
|
|
|
|
// Gets or sets the namespace prefix of this node.
|
|
public virtual string Prefix {
|
|
get { return string.Empty;}
|
|
set {}
|
|
}
|
|
|
|
// Gets the name of the node without the namespace prefix.
|
|
public abstract string LocalName {
|
|
get;
|
|
}
|
|
|
|
// Microsoft extensions
|
|
|
|
// Gets a value indicating whether the node is read-only.
|
|
public virtual bool IsReadOnly {
|
|
get {
|
|
XmlDocument doc = OwnerDocument;
|
|
return HasReadOnlyParent( this );
|
|
}
|
|
}
|
|
|
|
internal static bool HasReadOnlyParent( XmlNode n ) {
|
|
while (n != null) {
|
|
switch (n.NodeType) {
|
|
case XmlNodeType.EntityReference:
|
|
case XmlNodeType.Entity:
|
|
return true;
|
|
|
|
case XmlNodeType.Attribute:
|
|
n = ((XmlAttribute)n).OwnerElement;
|
|
break;
|
|
|
|
default:
|
|
n = n.ParentNode;
|
|
break;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Creates a duplicate of this node.
|
|
public virtual XmlNode Clone() {
|
|
return this.CloneNode(true);
|
|
}
|
|
|
|
object ICloneable.Clone() {
|
|
return this.CloneNode(true);
|
|
}
|
|
|
|
// Provides a simple ForEach-style iteration over the
|
|
// collection of nodes in this XmlNamedNodeMap.
|
|
IEnumerator IEnumerable.GetEnumerator() {
|
|
return new XmlChildEnumerator(this);
|
|
}
|
|
|
|
public IEnumerator GetEnumerator() {
|
|
return new XmlChildEnumerator(this);
|
|
}
|
|
|
|
private void AppendChildText( StringBuilder builder ) {
|
|
for (XmlNode child = FirstChild; child != null; child = child.NextSibling) {
|
|
if (child.FirstChild == null) {
|
|
if (child.NodeType == XmlNodeType.Text || child.NodeType == XmlNodeType.CDATA
|
|
|| child.NodeType == XmlNodeType.Whitespace || child.NodeType == XmlNodeType.SignificantWhitespace)
|
|
builder.Append( child.InnerText );
|
|
}
|
|
else {
|
|
child.AppendChildText( builder );
|
|
}
|
|
}
|
|
}
|
|
|
|
// Gets or sets the concatenated values of the node and
|
|
// all its children.
|
|
public virtual string InnerText {
|
|
get {
|
|
XmlNode fc = FirstChild;
|
|
if (fc == null) {
|
|
return string.Empty;
|
|
}
|
|
if (fc.NextSibling == null) {
|
|
XmlNodeType nodeType = fc.NodeType;
|
|
switch (nodeType) {
|
|
case XmlNodeType.Text:
|
|
case XmlNodeType.CDATA:
|
|
case XmlNodeType.Whitespace:
|
|
case XmlNodeType.SignificantWhitespace:
|
|
return fc.Value;
|
|
}
|
|
}
|
|
StringBuilder builder = new StringBuilder();
|
|
AppendChildText( builder );
|
|
return builder.ToString();
|
|
}
|
|
|
|
set {
|
|
XmlNode firstChild = FirstChild;
|
|
if ( firstChild != null //there is one child
|
|
&& firstChild.NextSibling == null // and exactly one
|
|
&& firstChild.NodeType == XmlNodeType.Text )//which is a text node
|
|
{
|
|
//this branch is for perf reason and event fired when TextNode.Value is changed
|
|
firstChild.Value = value;
|
|
}
|
|
else {
|
|
RemoveAll();
|
|
AppendChild( OwnerDocument.CreateTextNode( value ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
// Gets the markup representing this node and all its children.
|
|
public virtual string OuterXml {
|
|
get {
|
|
StringWriter sw = new StringWriter(CultureInfo.InvariantCulture);
|
|
XmlDOMTextWriter xw = new XmlDOMTextWriter( sw );
|
|
try {
|
|
WriteTo( xw );
|
|
}
|
|
finally {
|
|
xw.Close();
|
|
}
|
|
return sw.ToString();
|
|
}
|
|
}
|
|
|
|
// Gets or sets the markup representing just the children of this node.
|
|
public virtual string InnerXml {
|
|
get {
|
|
StringWriter sw = new StringWriter(CultureInfo.InvariantCulture);
|
|
XmlDOMTextWriter xw = new XmlDOMTextWriter( sw );
|
|
try {
|
|
WriteContentTo( xw );
|
|
}
|
|
finally {
|
|
xw.Close();
|
|
}
|
|
return sw.ToString();
|
|
}
|
|
|
|
set {
|
|
throw new InvalidOperationException( Res.GetString(Res.Xdom_Set_InnerXml ) );
|
|
}
|
|
}
|
|
|
|
public virtual IXmlSchemaInfo SchemaInfo {
|
|
get {
|
|
return XmlDocument.NotKnownSchemaInfo;
|
|
}
|
|
}
|
|
|
|
public virtual String BaseURI {
|
|
get {
|
|
XmlNode curNode = this.ParentNode; //save one while loop since if going to here, the nodetype of this node can't be document, entity and entityref
|
|
while ( curNode != null ) {
|
|
XmlNodeType nt = curNode.NodeType;
|
|
//EntityReference's children come from the dtd where they are defined.
|
|
//we need to investigate the same thing for entity's children if they are defined in an external dtd file.
|
|
if ( nt == XmlNodeType.EntityReference )
|
|
return ((XmlEntityReference)curNode).ChildBaseURI;
|
|
if ( nt == XmlNodeType.Document
|
|
|| nt == XmlNodeType.Entity
|
|
|| nt == XmlNodeType.Attribute )
|
|
return curNode.BaseURI;
|
|
curNode = curNode.ParentNode;
|
|
}
|
|
return String.Empty;
|
|
}
|
|
}
|
|
|
|
// Saves the current node to the specified XmlWriter.
|
|
public abstract void WriteTo(XmlWriter w);
|
|
|
|
// Saves all the children of the node to the specified XmlWriter.
|
|
public abstract void WriteContentTo(XmlWriter w);
|
|
|
|
// Removes all the children and/or attributes
|
|
// of the current node.
|
|
public virtual void RemoveAll() {
|
|
XmlNode child = FirstChild;
|
|
XmlNode sibling = null;
|
|
|
|
while (child != null) {
|
|
sibling = child.NextSibling;
|
|
RemoveChild( child );
|
|
child = sibling;
|
|
}
|
|
}
|
|
|
|
internal XmlDocument Document {
|
|
get {
|
|
if (NodeType == XmlNodeType.Document)
|
|
return (XmlDocument)this;
|
|
return OwnerDocument;
|
|
}
|
|
}
|
|
|
|
// Looks up the closest xmlns declaration for the given
|
|
// prefix that is in scope for the current node and returns
|
|
// the namespace URI in the declaration.
|
|
public virtual string GetNamespaceOfPrefix(string prefix) {
|
|
string namespaceName = GetNamespaceOfPrefixStrict(prefix);
|
|
return namespaceName != null ? namespaceName : string.Empty;
|
|
}
|
|
|
|
internal string GetNamespaceOfPrefixStrict(string prefix) {
|
|
XmlDocument doc = Document;
|
|
if (doc != null) {
|
|
prefix = doc.NameTable.Get(prefix);
|
|
if (prefix == null)
|
|
return null;
|
|
|
|
XmlNode node = this;
|
|
while (node != null) {
|
|
if (node.NodeType == XmlNodeType.Element) {
|
|
XmlElement elem = (XmlElement)node;
|
|
if (elem.HasAttributes) {
|
|
XmlAttributeCollection attrs = elem.Attributes;
|
|
if (prefix.Length == 0) {
|
|
for (int iAttr = 0; iAttr < attrs.Count; iAttr++) {
|
|
XmlAttribute attr = attrs[iAttr];
|
|
if (attr.Prefix.Length == 0) {
|
|
if (Ref.Equal(attr.LocalName, doc.strXmlns)) {
|
|
return attr.Value; // found xmlns
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
for (int iAttr = 0; iAttr < attrs.Count; iAttr++) {
|
|
XmlAttribute attr = attrs[iAttr];
|
|
if (Ref.Equal(attr.Prefix, doc.strXmlns)) {
|
|
if (Ref.Equal(attr.LocalName, prefix)) {
|
|
return attr.Value; // found xmlns:prefix
|
|
}
|
|
}
|
|
else if (Ref.Equal(attr.Prefix, prefix)) {
|
|
return attr.NamespaceURI; // found prefix:attr
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (Ref.Equal(node.Prefix, prefix)) {
|
|
return node.NamespaceURI;
|
|
}
|
|
node = node.ParentNode;
|
|
}
|
|
else if (node.NodeType == XmlNodeType.Attribute) {
|
|
node = ((XmlAttribute)node).OwnerElement;
|
|
}
|
|
else {
|
|
node = node.ParentNode;
|
|
}
|
|
}
|
|
if (Ref.Equal(doc.strXml, prefix)) { // xmlns:xml
|
|
return doc.strReservedXml;
|
|
}
|
|
else if (Ref.Equal(doc.strXmlns, prefix)) { // xmlns:xmlns
|
|
return doc.strReservedXmlns;
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
// Looks up the closest xmlns declaration for the given namespace
|
|
// URI that is in scope for the current node and returns
|
|
// the prefix defined in that declaration.
|
|
public virtual string GetPrefixOfNamespace(string namespaceURI) {
|
|
string prefix = GetPrefixOfNamespaceStrict(namespaceURI);
|
|
return prefix != null ? prefix : string.Empty;
|
|
}
|
|
|
|
internal string GetPrefixOfNamespaceStrict(string namespaceURI) {
|
|
XmlDocument doc = Document;
|
|
if (doc != null) {
|
|
namespaceURI = doc.NameTable.Add(namespaceURI);
|
|
|
|
XmlNode node = this;
|
|
while (node != null) {
|
|
if (node.NodeType == XmlNodeType.Element) {
|
|
XmlElement elem = (XmlElement)node;
|
|
if (elem.HasAttributes) {
|
|
XmlAttributeCollection attrs = elem.Attributes;
|
|
for (int iAttr = 0; iAttr < attrs.Count; iAttr++) {
|
|
XmlAttribute attr = attrs[iAttr];
|
|
if (attr.Prefix.Length == 0) {
|
|
if (Ref.Equal(attr.LocalName, doc.strXmlns)) {
|
|
if (attr.Value == namespaceURI) {
|
|
return string.Empty; // found xmlns="namespaceURI"
|
|
}
|
|
}
|
|
}
|
|
else if (Ref.Equal(attr.Prefix, doc.strXmlns)) {
|
|
if (attr.Value == namespaceURI) {
|
|
return attr.LocalName; // found xmlns:prefix="namespaceURI"
|
|
}
|
|
}
|
|
else if (Ref.Equal(attr.NamespaceURI, namespaceURI)) {
|
|
return attr.Prefix; // found prefix:attr
|
|
// with prefix bound to namespaceURI
|
|
}
|
|
}
|
|
}
|
|
if (Ref.Equal(node.NamespaceURI, namespaceURI)) {
|
|
return node.Prefix;
|
|
}
|
|
node = node.ParentNode;
|
|
}
|
|
else if (node.NodeType == XmlNodeType.Attribute) {
|
|
node = ((XmlAttribute)node).OwnerElement;
|
|
}
|
|
else {
|
|
node = node.ParentNode;
|
|
}
|
|
}
|
|
if (Ref.Equal(doc.strReservedXml, namespaceURI)) { // xmlns:xml
|
|
return doc.strXml;
|
|
}
|
|
else if (Ref.Equal(doc.strReservedXmlns, namespaceURI)) { // xmlns:xmlns
|
|
return doc.strXmlns;
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
// Retrieves the first child element with the specified name.
|
|
public virtual XmlElement this[string name]
|
|
{
|
|
get {
|
|
for (XmlNode n = FirstChild; n != null; n = n.NextSibling) {
|
|
if (n.NodeType == XmlNodeType.Element && n.Name == name)
|
|
return(XmlElement) n;
|
|
}
|
|
return null;
|
|
}
|
|
}
|
|
|
|
// Retrieves the first child element with the specified LocalName and
|
|
// NamespaceURI.
|
|
public virtual XmlElement this[string localname, string ns]
|
|
{
|
|
get {
|
|
for (XmlNode n = FirstChild; n != null; n = n.NextSibling) {
|
|
if (n.NodeType == XmlNodeType.Element && n.LocalName == localname && n.NamespaceURI == ns)
|
|
return(XmlElement) n;
|
|
}
|
|
return null;
|
|
}
|
|
}
|
|
|
|
internal virtual void SetParent( XmlNode node ) {
|
|
if (node == null) {
|
|
this.parentNode = OwnerDocument;
|
|
}
|
|
else {
|
|
this.parentNode = node;
|
|
}
|
|
}
|
|
|
|
internal virtual void SetParentForLoad( XmlNode node ) {
|
|
this.parentNode = node;
|
|
}
|
|
|
|
internal static void SplitName( string name, out string prefix, out string localName ) {
|
|
int colonPos = name.IndexOf(':'); // ordinal compare
|
|
if (-1 == colonPos || 0 == colonPos || name.Length-1 == colonPos) {
|
|
prefix = string.Empty;
|
|
localName = name;
|
|
}
|
|
else {
|
|
prefix = name.Substring(0, colonPos);
|
|
localName = name.Substring(colonPos+1);
|
|
}
|
|
}
|
|
|
|
internal virtual XmlNode FindChild( XmlNodeType type ) {
|
|
for (XmlNode child = FirstChild; child != null; child = child.NextSibling) {
|
|
if (child.NodeType == type) {
|
|
return child;
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
internal virtual XmlNodeChangedEventArgs GetEventArgs( XmlNode node, XmlNode oldParent, XmlNode newParent, string oldValue, string newValue, XmlNodeChangedAction action ) {
|
|
XmlDocument doc = OwnerDocument;
|
|
if (doc != null) {
|
|
if ( ! doc.IsLoading ) {
|
|
if ( ( (newParent != null && newParent.IsReadOnly) || ( oldParent != null && oldParent.IsReadOnly ) ) )
|
|
throw new InvalidOperationException( Res.GetString(Res.Xdom_Node_Modify_ReadOnly));
|
|
}
|
|
return doc.GetEventArgs( node, oldParent, newParent, oldValue, newValue, action );
|
|
}
|
|
return null;
|
|
}
|
|
|
|
internal virtual void BeforeEvent( XmlNodeChangedEventArgs args ) {
|
|
if (args != null)
|
|
OwnerDocument.BeforeEvent( args );
|
|
}
|
|
|
|
internal virtual void AfterEvent( XmlNodeChangedEventArgs args ) {
|
|
if (args != null)
|
|
OwnerDocument.AfterEvent( args );
|
|
}
|
|
|
|
internal virtual XmlSpace XmlSpace {
|
|
get {
|
|
XmlNode node = this;
|
|
XmlElement elem = null;
|
|
do {
|
|
elem = node as XmlElement;
|
|
if (elem != null && elem.HasAttribute("xml:space")) {
|
|
switch (XmlConvert.TrimString(elem.GetAttribute("xml:space"))) {
|
|
case "default":
|
|
return XmlSpace.Default;
|
|
case "preserve":
|
|
return XmlSpace.Preserve;
|
|
default:
|
|
//should we throw exception if value is otherwise?
|
|
break;
|
|
}
|
|
}
|
|
node = node.ParentNode;
|
|
}
|
|
while (node != null);
|
|
return XmlSpace.None;
|
|
}
|
|
}
|
|
|
|
internal virtual String XmlLang {
|
|
get {
|
|
XmlNode node = this;
|
|
XmlElement elem = null;
|
|
do {
|
|
elem = node as XmlElement;
|
|
if ( elem != null ) {
|
|
if ( elem.HasAttribute( "xml:lang" ) )
|
|
return elem.GetAttribute( "xml:lang" );
|
|
}
|
|
node = node.ParentNode;
|
|
} while ( node != null );
|
|
return String.Empty;
|
|
}
|
|
}
|
|
|
|
internal virtual XPathNodeType XPNodeType {
|
|
get {
|
|
return (XPathNodeType)(-1);
|
|
}
|
|
}
|
|
|
|
internal virtual string XPLocalName {
|
|
get {
|
|
return string.Empty;
|
|
}
|
|
}
|
|
|
|
internal virtual string GetXPAttribute(string localName, string namespaceURI) {
|
|
return String.Empty;
|
|
}
|
|
|
|
internal virtual bool IsText {
|
|
get {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public virtual XmlNode PreviousText {
|
|
get {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
internal static void NestTextNodes(XmlNode prevNode, XmlNode nextNode) {
|
|
Debug.Assert(prevNode.IsText);
|
|
Debug.Assert(nextNode.IsText);
|
|
|
|
nextNode.parentNode = prevNode;
|
|
}
|
|
|
|
internal static void UnnestTextNodes(XmlNode prevNode, XmlNode nextNode) {
|
|
Debug.Assert(prevNode.IsText);
|
|
Debug.Assert(nextNode.IsText);
|
|
|
|
nextNode.parentNode = prevNode.ParentNode;
|
|
}
|
|
private object debuggerDisplayProxy { get { return new DebuggerDisplayXmlNodeProxy(this); } }
|
|
}
|
|
|
|
[DebuggerDisplay("{ToString()}")]
|
|
internal struct DebuggerDisplayXmlNodeProxy {
|
|
private XmlNode node;
|
|
|
|
public DebuggerDisplayXmlNodeProxy(XmlNode node) {
|
|
this.node = node;
|
|
}
|
|
|
|
public override string ToString() {
|
|
XmlNodeType nodeType = node.NodeType;
|
|
string result = nodeType.ToString();
|
|
switch (nodeType) {
|
|
case XmlNodeType.Element:
|
|
case XmlNodeType.EntityReference:
|
|
result += ", Name=\"" + node.Name + "\"";
|
|
break;
|
|
case XmlNodeType.Attribute:
|
|
case XmlNodeType.ProcessingInstruction:
|
|
result += ", Name=\"" + node.Name + "\", Value=\"" + XmlConvert.EscapeValueForDebuggerDisplay(node.Value) + "\"";
|
|
break;
|
|
case XmlNodeType.Text:
|
|
case XmlNodeType.CDATA:
|
|
case XmlNodeType.Comment:
|
|
case XmlNodeType.Whitespace:
|
|
case XmlNodeType.SignificantWhitespace:
|
|
case XmlNodeType.XmlDeclaration:
|
|
result += ", Value=\"" + XmlConvert.EscapeValueForDebuggerDisplay(node.Value) + "\"";
|
|
break;
|
|
case XmlNodeType.DocumentType:
|
|
XmlDocumentType documentType = (XmlDocumentType)node;
|
|
result += ", Name=\"" + documentType.Name + "\", SYSTEM=\"" + documentType.SystemId + "\", PUBLIC=\"" + documentType.PublicId + "\", Value=\"" + XmlConvert.EscapeValueForDebuggerDisplay(documentType.InternalSubset) + "\"";
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return result;
|
|
}
|
|
}
|
|
}
|