File: System\Xml\Dom\XmlEntityReference.cs
Project: ndp\fx\src\Xml\System.Xml.csproj (System.Xml)
//------------------------------------------------------------------------------
// <copyright file="XmlEntityReference.cs" company="Microsoft">
//     Copyright (c) Microsoft Corporation.  All rights reserved.
// </copyright>
// <owner current="true" primary="true">Microsoft</owner>
//------------------------------------------------------------------------------
 
// <code>EntityReference</code> objects may be inserted into the structure 
// model when an entity reference is in the source document, or when the user 
// wishes to insert an entity reference. Note that  character references and 
// references to predefined entities are considered to be expanded by the 
// HTML or XML processor so that characters are represented by their Unicode 
// equivalent rather than by an entity reference. Moreover, the XML  
// processor may completely expand references to entities while building the 
// structure model, instead of providing <code>EntityReference</code> 
// objects. If it does provide such objects, then for a given 
// <code>EntityReference</code> node, it may be that there is no 
// <code>Entity</code> node representing the referenced entity; but if such 
// an <code>Entity</code> exists, then the child list of the 
// <code>EntityReference</code> node is the same as that of the 
// <code>Entity</code> node. As with the <code>Entity</code> node, all 
// descendants of the <code>EntityReference</code> are readonly.
// <p>The resolution of the children of the <code>EntityReference</code> (the  
// replacement value of the referenced <code>Entity</code>) may be lazily  
// evaluated; actions by the user (such as calling the  
// <code>childNodes</code> method on the <code>EntityReference</code> node)  
// are assumed to trigger the evaluation.
 
namespace System.Xml {
 
    using System.Diagnostics;
 
    // Represents an entity reference node.
    public class XmlEntityReference : XmlLinkedNode {
        string name;
        XmlLinkedNode lastChild;
 
        protected internal XmlEntityReference( string name, XmlDocument doc ) : base( doc ) {
            if ( !doc.IsLoading ) {
                if ( name.Length > 0 && name[0] == '#' ) {
                    throw new ArgumentException( Res.GetString( Res.Xdom_InvalidCharacter_EntityReference ) );
                }
            }
            this.name = doc.NameTable.Add(name);
            doc.fEntRefNodesPresent = true;
        }
 
        // Gets the name of the node.
        public override string Name {
            get { return name;}
        }
 
        // Gets the name of the node without the namespace prefix.
        public override string LocalName {
            get { return name;}
        }
 
        // Gets or sets the value of the node.
        public override String Value { 
            get {
                return null;
            }
 
            set {
                throw new InvalidOperationException(Res.GetString(Res.Xdom_EntRef_SetVal));
            }
        }
 
        // Gets the type of the node.
        public override XmlNodeType NodeType {
            get { return XmlNodeType.EntityReference;}
        }
 
        // Creates a duplicate of this node.
        public override XmlNode CloneNode(bool deep) {
            Debug.Assert( OwnerDocument != null );
            XmlEntityReference eref = OwnerDocument.CreateEntityReference( name );
            return eref;
        }
 
        //
        // Microsoft extensions
        //
 
        // Gets a value indicating whether the node is read-only.
        public override bool IsReadOnly {
            get { 
                return true;        // Make entity references readonly
            }
        }
 
        internal override bool IsContainer {
            get { return true;}
        }
 
        internal override void SetParent( XmlNode node ) {
            base.SetParent(node);
            if ( LastNode == null && node != null && node != OwnerDocument ) {
                //first time insert the entity reference into the tree, we should expand its children now
                XmlLoader loader = new XmlLoader();
                loader.ExpandEntityReference(this);
                
            }
        }
 
        internal override void SetParentForLoad( XmlNode node ) {
         this.SetParent(  node );
        }
 
        internal override XmlLinkedNode LastNode {
            get { 
                return lastChild;
            }
            set { lastChild = value;}
        }
 
        internal override bool IsValidChildType( XmlNodeType type ) {
            switch (type) {
                case XmlNodeType.Element:
                case XmlNodeType.Text:
                case XmlNodeType.EntityReference:
                case XmlNodeType.Comment:
                case XmlNodeType.Whitespace:
                case XmlNodeType.SignificantWhitespace:
                case XmlNodeType.ProcessingInstruction:
                case XmlNodeType.CDATA:
                    return true;
 
                default:
                    return false;
            }
        }
 
        // Saves the node to the specified XmlWriter.
        public override void WriteTo(XmlWriter w) {
            w.WriteEntityRef(name);
        }
 
        // Saves all the children of the node to the specified XmlWriter.
        public override void WriteContentTo(XmlWriter w) {
            // -- eventually will the fix. commented out waiting for finalizing on the issue.
            foreach( XmlNode n in this ) {
                n.WriteTo( w );
            } //still use the old code to generate the output
            /*
            foreach( XmlNode n in this ) {
                if ( n.NodeType != XmlNodeType.EntityReference )
                n.WriteTo( w );
                else
                    n.WriteContentTo( w );
            }*/
        }
 
        public override String BaseURI {
            get {
                return OwnerDocument.BaseURI;
            }
        }
 
        private string ConstructBaseURI( string baseURI, string systemId ) {
            if ( baseURI == null )
                return systemId;
            int nCount = baseURI.LastIndexOf('/')+1;
            string buf = baseURI;
            if ( nCount > 0 && nCount < baseURI.Length )
                buf = baseURI.Substring(0, nCount);
            else if ( nCount == 0 )
                buf = buf + "\\";
            return (buf + systemId.Replace('\\', '/')); 
        }
 
        //childrenBaseURI returns where the entity reference node's children come from
        internal String ChildBaseURI {
            get {
                //get the associate entity and return its baseUri
                XmlEntity ent = OwnerDocument.GetEntityNode( name );
                if ( ent != null ) {
                    if ( ent.SystemId != null && ent.SystemId.Length > 0 )
                        return ConstructBaseURI(ent.BaseURI, ent.SystemId); 
                    else
                        return ent.BaseURI;
                }
                return String.Empty;
            }
        }
    }
}