//------------------------------------------------------------------------------
// 
//     Copyright (c) Microsoft Corporation.  All rights reserved.
//                                                                 
//------------------------------------------------------------------------------
namespace System.ComponentModel {
    using System;
    using System.ComponentModel;
    using System.Globalization;
    using System.IO;
    using System.Security.Permissions;
    
    /// 
    ///     A nested container is a container that is owned by another component.  Nested
    ///     containers can be found by querying a component site's services for NestedConainter.
    ///     Nested containers are a useful tool to establish owner relationships among components.
    ///     All components within a nested container are named with the owning component's name
    ///     as a prefix.
    /// 
    [HostProtection(SharedState = true)]
    public class NestedContainer : Container, INestedContainer {
        private IComponent _owner;
        /// 
        ///     Creates a new NestedContainer.
        /// 
        public NestedContainer(IComponent owner) {
            if (owner == null) {
                throw new ArgumentNullException("owner");
            }
            _owner = owner;
            _owner.Disposed += new EventHandler(OnOwnerDisposed);
        }
        /// 
        ///     The component that owns this nested container.
        /// 
        public IComponent Owner {
            get {
                return _owner;
            }
        }
        /// 
        ///     Retrieves the name of the owning component.  This may be overridden to
        ///     provide a custom owner name.  The default searches the owner's site for
        ///     INestedSite and calls FullName, or ISite.Name if there is no nested site.
        ///     If neither is available, this returns null.
        /// 
        protected virtual string OwnerName {
            get {
                string ownerName = null;
                if (_owner != null && _owner.Site != null) {
                    INestedSite nestedOwnerSite = _owner.Site as INestedSite;
                    if (nestedOwnerSite != null) {
                        ownerName = nestedOwnerSite.FullName;
                    }
                    else {
                        ownerName = _owner.Site.Name;
                    }
                }
                return ownerName;
            }
        }
        /// 
        ///     Creates a site for the component within the container.
        /// 
        protected override ISite CreateSite(IComponent component, string name) {
            if (component == null) {
                throw new ArgumentNullException("component");
            }
            return new Site(component, this, name);
        }
        /// 
        ///    Override of Container's dispose.
        /// 
        protected override void Dispose(bool disposing) {
            if (disposing) {
                _owner.Disposed -= new EventHandler(OnOwnerDisposed);
            }
            base.Dispose(disposing);
        }
        /// 
        ///    [To be supplied.]
        /// 
        protected override object GetService(Type service) {
            if (service == typeof(INestedContainer)) {
                return this;
            }
            else {
                return base.GetService(service);
            }
        }
        /// 
        ///     Called when our owning component is destroyed.
        /// 
        private void OnOwnerDisposed(object sender, EventArgs e) {
            Dispose();
        }
        /// 
        ///     Simple site implementation.  We do some special processing to name the site, but 
        ///     that's about it.
        /// 
        private class Site : INestedSite
        {
            private IComponent component;
            private NestedContainer container;
            private string name;
            internal Site(IComponent component, NestedContainer container, string name) {
                this.component = component;
                this.container = container;
                this.name = name;
            }
            // The component sited by this component site.
            public IComponent Component {
                get {
                    return component;
                }
            }
            // The container in which the component is sited.
            public IContainer Container {
                get {
                    return container;
                }
            }
            public Object GetService(Type service) {
                return((service == typeof(ISite)) ? this : container.GetService(service));
            }
            // Indicates whether the component is in design mode.
            public bool DesignMode {
                get {
                    IComponent owner = container.Owner;
                    if (owner != null && owner.Site != null) {
                        return owner.Site.DesignMode;
                    }
                    return false;
                }
            }
            public string FullName {
                get {
                    if (name != null) {
                        string ownerName = container.OwnerName;
                        string childName = name;
                        if (ownerName != null) {
                            childName = string.Format(CultureInfo.InvariantCulture, "{0}.{1}", ownerName, childName);
                        }
                        return childName;
                    }
                    return name;
                }
            }
            // The name of the component.
            //
            public String Name 
            {
                get { 
                    return name;
                }
                set { 
                    if (value == null || name == null || !value.Equals(name)) {
                        container.ValidateName(component, value);
                        name = value;
                    }
                }
            }
        }
    }
}