| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555 |
- //
- // System.Xml.XmlNode
- //
- // Author:
- // Kral Ferch <[email protected]>
- // Atsushi Enomoto <[email protected]>
- //
- // (C) 2002 Kral Ferch
- // (C) 2002 Atsushi Enomoto
- //
- using System;
- using System.Collections;
- using System.IO;
- using System.Text;
- using System.Xml.XPath;
- namespace System.Xml
- {
- public abstract class XmlNode : ICloneable, IEnumerable, IXPathNavigable
- {
- #region Fields
- XmlDocument ownerDocument;
- XmlNode parentNode;
- #endregion
- #region Constructors
- internal XmlNode (XmlDocument ownerDocument)
- {
- this.ownerDocument = ownerDocument;
- }
- #endregion
- #region Properties
- public virtual XmlAttributeCollection Attributes {
- get { return null; }
- }
- public virtual string BaseURI {
- get {
- // Isn't it conformant to W3C XML Base Recommendation?
- // As far as I tested, there are not...
- return (ParentNode != null) ? ParentNode.BaseURI : OwnerDocument.BaseURI;
- }
- }
- public virtual XmlNodeList ChildNodes {
- get {
- return new XmlNodeListChildren (this);
- }
- }
- public virtual XmlNode FirstChild {
- get {
- if (LastChild != null) {
- return LastLinkedChild.NextLinkedSibling;
- }
- else {
- return null;
- }
- }
- }
- public virtual bool HasChildNodes {
- get { return LastChild != null; }
- }
- [MonoTODO("confirm whether this way is right for each not-overriden types.")]
- public virtual string InnerText {
- get {
- StringBuilder builder = new StringBuilder ();
- AppendChildValues (this, builder);
- return builder.ToString ();
- }
- set { throw new NotImplementedException (); }
- }
- private void AppendChildValues (XmlNode parent, StringBuilder builder)
- {
- XmlNode node = parent.FirstChild;
- while (node != null) {
- if (node.NodeType == XmlNodeType.Text)
- builder.Append (node.Value);
- AppendChildValues (node, builder);
- node = node.NextSibling;
- }
- }
- [MonoTODO("Setter.")]
- public virtual string InnerXml {
- get {
- StringWriter sw = new StringWriter ();
- XmlTextWriter xtw = new XmlTextWriter (sw);
- WriteContentTo (xtw);
- return sw.GetStringBuilder ().ToString ();
- }
- set { throw new NotImplementedException (); }
- }
- public virtual bool IsReadOnly {
- get { return false; }
- }
- [System.Runtime.CompilerServices.IndexerName("Item")]
- public virtual XmlElement this [string name] {
- get {
- foreach (XmlNode node in ChildNodes) {
- if ((node.NodeType == XmlNodeType.Element) &&
- (node.Name == name)) {
- return (XmlElement) node;
- }
- }
- return null;
- }
- }
- [System.Runtime.CompilerServices.IndexerName("Item")]
- public virtual XmlElement this [string localname, string ns] {
- get {
- foreach (XmlNode node in ChildNodes) {
- if ((node.NodeType == XmlNodeType.Element) &&
- (node.LocalName == localname) &&
- (node.NamespaceURI == ns)) {
- return (XmlElement) node;
- }
- }
- return null;
- }
- }
- public virtual XmlNode LastChild {
- get { return LastLinkedChild; }
- }
- internal virtual XmlLinkedNode LastLinkedChild {
- get { return null; }
- set { }
- }
- public abstract string LocalName { get; }
- public abstract string Name { get; }
- public virtual string NamespaceURI {
- get { return String.Empty; }
- }
- public virtual XmlNode NextSibling {
- get { return null; }
- }
- public abstract XmlNodeType NodeType { get; }
- internal virtual XPathNodeType XPathNodeType {
- get {
- return (XPathNodeType) (-1);
- }
- }
- public virtual string OuterXml {
- get {
- StringWriter sw = new StringWriter ();
- XmlTextWriter xtw = new XmlTextWriter (sw);
- WriteTo (xtw);
- return sw.GetStringBuilder ().ToString ();
- }
- }
- public virtual XmlDocument OwnerDocument {
- get { return ownerDocument; }
- }
- public virtual XmlNode ParentNode {
- get { return parentNode; }
- }
- public virtual string Prefix {
- get { return String.Empty; }
- set {}
- }
- public virtual XmlNode PreviousSibling {
- get { return null; }
- }
- public virtual string Value {
- get { return null; }
- set { throw new InvalidOperationException ("This node does not have a value"); }
- }
- internal virtual string XmlLang {
- get {
- if(Attributes != null)
- foreach(XmlAttribute attr in Attributes)
- if(attr.Name == "xml:lang")
- return attr.Value;
- return (ParentNode != null) ? ParentNode.XmlLang : OwnerDocument.XmlLang;
- }
- }
- internal virtual XmlSpace XmlSpace {
- get {
- if(Attributes != null) {
- foreach(XmlAttribute attr in Attributes) {
- if(attr.Name == "xml:space") {
- switch(attr.Value) {
- case "preserve": return XmlSpace.Preserve;
- case "default": return XmlSpace.Default;
- }
- break;
- }
- }
- }
- return (ParentNode != null) ? ParentNode.XmlSpace : OwnerDocument.XmlSpace;
- }
- }
- #endregion
- #region Methods
- public virtual XmlNode AppendChild (XmlNode newChild)
- {
- // I assume that AppendChild(n) equals to InsertAfter(n, this.LastChild) or InsertBefore(n, null)
- return InsertBefore (newChild, null);
- }
- public virtual XmlNode Clone ()
- {
- // By MS document, it is equivalent to CloneNode(true).
- return this.CloneNode (true);
- }
- public abstract XmlNode CloneNode (bool deep);
- [MonoTODO]
- public XPathNavigator CreateNavigator ()
- {
- return new XmlDocumentNavigator (this);
- }
- public IEnumerator GetEnumerator ()
- {
- return new XmlNodeListChildren (this).GetEnumerator ();
- }
- [MonoTODO("performance problem.")]
- public virtual string GetNamespaceOfPrefix (string prefix)
- {
- XmlNamespaceManager nsmgr = ConstructNamespaceManager ();
- return nsmgr.LookupNamespace (prefix);
- }
- [MonoTODO("performance problem.")]
- public virtual string GetPrefixOfNamespace (string namespaceURI)
- {
- XmlNamespaceManager nsmgr = ConstructNamespaceManager ();
- string ns = nsmgr.LookupPrefix (namespaceURI);
- return (ns != null) ? ns : String.Empty;
- }
- object ICloneable.Clone ()
- {
- return Clone ();
- }
- IEnumerator IEnumerable.GetEnumerator ()
- {
- return GetEnumerator ();
- }
- public virtual XmlNode InsertAfter (XmlNode newChild, XmlNode refChild)
- {
- // I assume that insertAfter(n1, n2) equals to InsertBefore(n1, n2.PreviousSibling).
- // I took this way because rather than calling InsertAfter() from InsertBefore()
- // because current implementation of 'NextSibling' looks faster than 'PreviousSibling'.
- XmlNode argNode = null;
- if(refChild != null)
- argNode = refChild.NextSibling;
- else if(ChildNodes.Count > 0)
- argNode = FirstChild;
- return InsertBefore (newChild, argNode);
- }
- [MonoTODO("If inserted node is entity reference, then check conforming entity. Wait for DTD implementation.")]
- public virtual XmlNode InsertBefore (XmlNode newChild, XmlNode refChild)
- {
- XmlDocument ownerDoc = (NodeType == XmlNodeType.Document) ? (XmlDocument)this : OwnerDocument;
- if (NodeType == XmlNodeType.Document ||
- NodeType == XmlNodeType.Element ||
- NodeType == XmlNodeType.Attribute ||
- NodeType == XmlNodeType.DocumentFragment) {
- if (IsReadOnly)
- throw new ArgumentException ("The specified node is readonly.");
- if (newChild.OwnerDocument != ownerDoc)
- throw new ArgumentException ("Can't append a node created by another document.");
- if (refChild != null && newChild.OwnerDocument != refChild.OwnerDocument)
- throw new ArgumentException ("argument nodes are on the different documents.");
- // This check is done by MS.NET 1.0, but isn't done for MS.NET 1.1.
- // Skip this check in the meantime...
- // if(this == ownerDoc && ownerDoc.DocumentElement != null && (newChild is XmlElement))
- // throw new XmlException ("multiple document element not allowed.");
- // checking validity finished. then appending...
- return insertBeforeIntern (newChild, refChild);
- }
- else
- throw new InvalidOperationException ();
- }
- internal XmlNode insertBeforeIntern (XmlNode newChild, XmlNode refChild)
- {
- XmlDocument ownerDoc = (NodeType == XmlNodeType.Document) ? (XmlDocument)this : OwnerDocument;
- ownerDoc.onNodeInserting (newChild, this);
- if(newChild.ParentNode != null)
- newChild.ParentNode.RemoveChild (newChild);
- if(newChild.NodeType == XmlNodeType.DocumentFragment) {
- int x = newChild.ChildNodes.Count;
- for(int i=0; i<x; i++) {
- XmlNode n = newChild.ChildNodes [0];
- this.InsertBefore (n, refChild); // recursively invokes events. (It is compatible with MS implementation.)
- }
- }
- else {
- XmlLinkedNode newLinkedChild = (XmlLinkedNode) newChild;
- XmlLinkedNode lastLinkedChild = LastLinkedChild;
- newLinkedChild.parentNode = this;
- if(refChild == null) {
- // append last, so:
- // * set nextSibling of previous lastchild to newChild
- // * set lastchild = newChild
- // * set next of newChild to firstChild
- if(LastLinkedChild != null) {
- XmlLinkedNode formerFirst = FirstChild as XmlLinkedNode;
- LastLinkedChild.NextLinkedSibling = newLinkedChild;
- LastLinkedChild = newLinkedChild;
- newLinkedChild.NextLinkedSibling = formerFirst;
- }
- else {
- LastLinkedChild = newLinkedChild;
- LastLinkedChild.NextLinkedSibling = newLinkedChild; // FirstChild
- }
- }
- else {
- // append not last, so:
- // * if newchild is first, then set next of lastchild is newChild.
- // otherwise, set next of previous sibling to newChild
- // * set next of newChild to refChild
- XmlLinkedNode prev = refChild.PreviousSibling as XmlLinkedNode;
- if(prev == null)
- LastLinkedChild.NextLinkedSibling = newLinkedChild;
- else
- prev.NextLinkedSibling = newLinkedChild;
- newLinkedChild.NextLinkedSibling = refChild as XmlLinkedNode;
- }
- ownerDoc.onNodeInserted (newChild, newChild.ParentNode);
- }
- return newChild;
- }
- [MonoTODO]
- public virtual void Normalize ()
- {
- throw new NotImplementedException ();
- }
- public virtual XmlNode PrependChild (XmlNode newChild)
- {
- return InsertAfter (newChild, null);
- }
- public virtual void RemoveAll ()
- {
- XmlNode next = null;
- for (XmlNode node = FirstChild; node != null; node = next) {
- next = node.NextSibling;
- RemoveChild (node);
- }
- }
- public virtual XmlNode RemoveChild (XmlNode oldChild)
- {
- XmlDocument ownerDoc = (NodeType == XmlNodeType.Document) ? (XmlDocument)this : OwnerDocument;
- if(oldChild.ParentNode != this)
- throw new XmlException ("specified child is not child of this node.");
- ownerDoc.onNodeRemoving (oldChild, oldChild.ParentNode);
- if (NodeType == XmlNodeType.Document || NodeType == XmlNodeType.Element || NodeType == XmlNodeType.Attribute || NodeType == XmlNodeType.DocumentFragment) {
- if (IsReadOnly)
- throw new ArgumentException ();
- if (Object.ReferenceEquals (LastLinkedChild, LastLinkedChild.NextLinkedSibling) && Object.ReferenceEquals (LastLinkedChild, oldChild))
- LastLinkedChild = null;
- else {
- XmlLinkedNode oldLinkedChild = (XmlLinkedNode)oldChild;
- XmlLinkedNode beforeLinkedChild = LastLinkedChild;
-
- while (!Object.ReferenceEquals (beforeLinkedChild.NextLinkedSibling, LastLinkedChild) && !Object.ReferenceEquals (beforeLinkedChild.NextLinkedSibling, oldLinkedChild))
- beforeLinkedChild = beforeLinkedChild.NextLinkedSibling;
- if (!Object.ReferenceEquals (beforeLinkedChild.NextLinkedSibling, oldLinkedChild))
- throw new ArgumentException ();
- beforeLinkedChild.NextLinkedSibling = oldLinkedChild.NextLinkedSibling;
- oldLinkedChild.NextLinkedSibling = null;
- }
- ownerDoc.onNodeRemoved (oldChild, oldChild.ParentNode);
- oldChild.parentNode = null; // clear parent 'after' above logic.
- return oldChild;
- }
- else
- throw new ArgumentException ();
- }
- public virtual XmlNode ReplaceChild (XmlNode newChild, XmlNode oldChild)
- {
- if(oldChild.ParentNode != this)
- throw new InvalidOperationException ("oldChild is not a child of this node.");
- XmlNode parent = this.ParentNode;
- while(parent != null) {
- if(newChild == parent)
- throw new InvalidOperationException ("newChild is ancestor of this node.");
- parent = parent.ParentNode;
- }
- foreach(XmlNode n in ChildNodes) {
- if(n == oldChild) {
- XmlNode prev = oldChild.PreviousSibling;
- RemoveChild (oldChild);
- InsertAfter (newChild, prev);
- break;
- }
- }
- return oldChild;
- }
- public XmlNodeList SelectNodes (string xpath)
- {
- return SelectNodes (xpath, null);
- }
- [MonoTODO]
- public XmlNodeList SelectNodes (string xpath, XmlNamespaceManager nsmgr)
- {
- XPathNavigator nav = CreateNavigator ();
- XPathExpression expr = nav.Compile (xpath);
- if (nsmgr != null)
- expr.SetContext (nsmgr);
- XPathNodeIterator iter = nav.Select (expr);
- ArrayList rgNodes = new ArrayList ();
- while (iter.MoveNext ())
- {
- rgNodes.Add (((XmlDocumentNavigator) iter.Current).Node);
- }
- return new XmlNodeArrayList (rgNodes);
- }
- public XmlNode SelectSingleNode (string xpath)
- {
- return SelectSingleNode (xpath, null);
- }
- [MonoTODO]
- public XmlNode SelectSingleNode (string xpath, XmlNamespaceManager nsmgr)
- {
- XPathNavigator nav = CreateNavigator ();
- XPathExpression expr = nav.Compile (xpath);
- if (nsmgr != null)
- expr.SetContext (nsmgr);
- XPathNodeIterator iter = nav.Select (expr);
- if (!iter.MoveNext ())
- return null;
- return ((XmlDocumentNavigator) iter.Current).Node;
- }
- internal void SetParentNode (XmlNode parent)
- {
- parentNode = parent;
- }
- [MonoTODO]
- public virtual bool Supports (string feature, string version)
- {
- throw new NotImplementedException ();
- }
- public abstract void WriteContentTo (XmlWriter w);
- public abstract void WriteTo (XmlWriter w);
- // It parses this and all the ancestor elements,
- // find 'xmlns' declarations, stores and then return them.
- // TODO: tests
- internal XmlNamespaceManager ConstructNamespaceManager ()
- {
- XmlDocument doc = this is XmlDocument ? (XmlDocument)this : this.OwnerDocument;
- XmlNamespaceManager nsmgr = new XmlNamespaceManager (doc.NameTable);
- XmlElement el = null;
- switch(this.NodeType) {
- case XmlNodeType.Attribute:
- el = ((XmlAttribute)this).OwnerElement;
- break;
- case XmlNodeType.Element:
- el = this as XmlElement;
- break;
- default:
- el = this.ParentNode as XmlElement;
- break;
- }
- while(el != null) {
- foreach(XmlAttribute attr in el.Attributes) {
- if(attr.Prefix == "xmlns") {
- if (nsmgr.LookupNamespace (attr.LocalName) == null)
- nsmgr.AddNamespace (attr.LocalName, attr.Value);
- } else if(attr.Name == "xmlns") {
- if(nsmgr.LookupNamespace (String.Empty) == null)
- nsmgr.AddNamespace (String.Empty, attr.Value);
- }
- }
- // When reached to document, then it will set null value :)
- el = el.ParentNode as XmlElement;
- }
- return nsmgr;
- }
- #endregion
- }
- }
|