//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // // [....] //------------------------------------------------------------------------------ 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; } } }