namespace System.Workflow.ComponentModel.Serialization
{
    using System;
    using System.CodeDom;
    using System.ComponentModel;
    using System.ComponentModel.Design;
    using System.ComponentModel.Design.Serialization;
    using System.Collections;
    using System.Resources;
    using System.Workflow.ComponentModel.Design;
    using System.Collections.Generic;
    using Microsoft.CSharp;
    using System.Workflow.ComponentModel;
    using System.Workflow.ComponentModel.Compiler;
    using System.CodeDom.Compiler;
    using System.IO;
    using System.Reflection;
    using System.Diagnostics;
    using System.Diagnostics.CodeAnalysis;
    #region Class ActivityCodeDomSerializationManager
    /// 
    /// Serialization manager class for code only serialization which inherits from 
    /// WorkflowMarkupSerializationManager and overrides the GetService method to return the
    /// code-only specific reference service.
    /// 
    /// 
    /// Due to the fact that the base classes used by the code-only serializer from 
    /// System.Design rely on the reference service to generate code variable ids, we
    /// need to supply our own version of the reference service, since the default
    /// generates invalid variable ids for Activity based classes.
    /// 
    [Obsolete("The System.Workflow.* types are deprecated.  Instead, please use the new types from System.Activities.*")]
    public class ActivityCodeDomSerializationManager : IDesignerSerializationManager
    {
        private ServiceContainer serviceContainer = null;
        private IDesignerSerializationManager serializationManager;
        private ResolveNameEventHandler resolveNameEventHandler;
        private EventHandler serializationCompleteEventHandler;
        public ActivityCodeDomSerializationManager(IDesignerSerializationManager manager)
        {
            if (manager == null)
                throw new ArgumentNullException("manager");
            this.serializationManager = manager;
            this.serviceContainer = new ServiceContainer();
            this.serializationManager.ResolveName += new ResolveNameEventHandler(OnResolveName);
            this.serializationManager.SerializationComplete += new EventHandler(OnSerializationComplete);
            if (this.serializationManager is DesignerSerializationManager)
                ((DesignerSerializationManager)this.serializationManager).SessionDisposed += new EventHandler(OnSessionDisposed);
        }
        void OnResolveName(object sender, ResolveNameEventArgs e)
        {
            if (this.resolveNameEventHandler != null)
                this.resolveNameEventHandler(this, e);
        }
        void OnSerializationComplete(object sender, EventArgs e)
        {
            if (this.serializationCompleteEventHandler != null)
                this.serializationCompleteEventHandler(this, e);
        }
        void OnSessionDisposed(object sender, EventArgs e)
        {
            try
            {
                if (this.serializationCompleteEventHandler != null)
                    this.serializationCompleteEventHandler(this, EventArgs.Empty);
            }
            finally
            {
                this.resolveNameEventHandler = null;
                this.serializationCompleteEventHandler = null;
            }
        }
        public event System.EventHandler SerializationComplete
        {
            add
            {
                this.serializationCompleteEventHandler += value;
            }
            remove
            {
                this.serializationCompleteEventHandler -= value;
            }
        }
        public void SetName(object instance, string name)
        {
            // Need to check this since code dom deserialization tries to 
            // set name twice on all object creation statements.
            if (GetInstance(name) != instance)
                this.serializationManager.SetName(instance, name);
        }
        [SuppressMessage("Microsoft.Globalization", "CA1307:SpecifyStringComparison", MessageId = "System.String.LastIndexOf(System.String)", Justification = "This is not a security threat since it is called only in XAML design time scenarios")]
        public string GetName(object value)
        {
            string name = null;
            Activity a = value as Activity;
            if (a != null)
            {
                if (a.Parent == null)
                {
                    name = a.GetValue(WorkflowMarkupSerializer.XClassProperty) as string;
                    if (name != null && name.LastIndexOf(".") > 0)
                        name = name.Substring(name.LastIndexOf('.') + 1);
                }
                else
                    name = a.QualifiedName.Replace('.', '_');
            }
            if (name == null)
                name = this.serializationManager.GetName(value);
            return name;
        }
        /// 
        /// Overridded to force siteing of all Activity based instances created, enabling
        /// code-only programmers to write activity.Activities.Add(new activityType());
        /// 
        public object CreateInstance(Type type, ICollection arguments, string name, bool addToContainer)
        {
            return this.serializationManager.CreateInstance(type, arguments, name, false);
        }
        public object GetService(Type serviceType)
        {
            object service = null;
            if (serviceType == typeof(IReferenceService))
                service = new ActivityCodeDomReferenceService(this.serializationManager.GetService(serviceType) as IReferenceService);
            if (serviceType == typeof(IServiceContainer))
            {
                service = serializationManager.GetService(serviceType);
                if (service == null)
                {
                    service = serviceContainer;
                }
            }
            // If we haven't provded the service, try the serialization manager we wrap
            if (service == null)
                service = this.serializationManager.GetService(serviceType);
            // Last ditch effort, see if our consumer had added the service
            if (service == null)
                service = serviceContainer.GetService(serviceType);
            return service;
        }
        // work around : PD7's PrimitiveCodeDomSerializer does not handle well strings bigger than 200 characters, 
        //       we push our own version to fix it.
        public object GetSerializer(Type objectType, Type serializerType)
        {
            if (objectType == typeof(string))
            {
                return PrimitiveCodeDomSerializer.Default;
            }
            else if (objectType != null && TypeProvider.IsAssignable(typeof(ICollection), objectType) && !objectType.IsArray && serializerType == typeof(CodeDomSerializer))
            {
                PropertyDescriptor pd = Context[typeof(PropertyDescriptor)] as PropertyDescriptor;
                if (pd != null)
                {
                    if (string.Equals(pd.Name, "SynchronizationHandles", StringComparison.Ordinal) && TypeProvider.IsAssignable(typeof(Activity), pd.ComponentType))
                        return new SynchronizationHandlesCodeDomSerializer();
                }
                else
                {
                    // If property descriptor is not available, we then look at the expression context.
                    ExpressionContext context = Context[typeof(ExpressionContext)] as ExpressionContext;
                    if (context != null && context.Expression is CodePropertyReferenceExpression &&
                        string.Equals(((CodePropertyReferenceExpression)context.Expression).PropertyName, "SynchronizationHandles", StringComparison.Ordinal))
                        return new SynchronizationHandlesCodeDomSerializer();
                }
            }
            object serializer = this.serializationManager.GetSerializer(objectType, serializerType);
            if (!UseUserDefinedSerializer(objectType, serializerType))
                serializer = new SerializableTypeCodeDomSerializer(serializer as CodeDomSerializer);
            return serializer;
        }
        // We add this custom serializable to handler serializable types so they don't go into the resources
        // as majority of our types will be serializable per WinOE runtime requirements.
        // Note: we will not overwrite any custom serializer behaviors.  If the serializer does not come
        // from the Sytem.ComponentModel.dll, we will not overwrite it.
        private bool UseUserDefinedSerializer(Type objectType, Type serializerType)
        {
            if (objectType == null || serializerType == null)
                return true;
            //If objectType is not serializable or we are not looking for codedomserializer then use user defined serializer
            if (!objectType.IsSerializable || serializerType != typeof(CodeDomSerializer))
                return true;
            //For primitives always honor user defined serializer
            if (objectType.IsPrimitive || objectType.IsEnum || objectType == typeof(string) || typeof(Activity).IsAssignableFrom(objectType))
                return true;
            //If user has defined instance descriptor then we always serialize to a create expression so we honor
            //user defined serializer
            TypeConverter converter = TypeDescriptor.GetConverter(objectType);
            if (converter != null && converter.CanConvertTo(typeof(InstanceDescriptor)))
                return true;
            //If the serializer is a custom codedom serializer ie it does not come from our or system  assembly
            //defining codedomserializer then always honor user's serializer as user needs a way to override the
            //serializartion process
            object serializer = this.serializationManager.GetSerializer(objectType, serializerType);
            if (serializer.GetType().Assembly != typeof(CodeDomSerializer).Assembly &&
                serializer.GetType().Assembly != Assembly.GetExecutingAssembly() &&
                serializer.GetType().Assembly != Assembly.Load(AssemblyRef.ActivitiesAssemblyRef))
                return true;
            //Special case for UI objects they need to always come from the resource only if we are not compiling
            Activity activity = this.serializationManager.Context[typeof(Activity)] as Activity;
            if (activity != null && activity.Site != null && activity.Site.Container != null &&
                objectType.Namespace != null && objectType.Namespace.Equals(typeof(System.Drawing.Image).Namespace))
                return true;
            return false;
        }
        public object GetInstance(string name)
        {
            return this.serializationManager.GetInstance(name);
        }
        public void AddSerializationProvider(IDesignerSerializationProvider provider)
        {
            this.serializationManager.AddSerializationProvider(provider);
        }
        public void RemoveSerializationProvider(IDesignerSerializationProvider provider)
        {
            this.serializationManager.RemoveSerializationProvider(provider);
        }
        public void ReportError(object errorInformation)
        {
            this.serializationManager.ReportError(errorInformation);
        }
        public ContextStack Context
        {
            get
            {
                return this.serializationManager.Context;
            }
        }
        public event ResolveNameEventHandler ResolveName
        {
            add
            {
                this.resolveNameEventHandler += value;
            }
            remove
            {
                this.resolveNameEventHandler -= value;
            }
        }
        public PropertyDescriptorCollection Properties
        {
            get
            {
                return this.serializationManager.Properties;
            }
        }
        public Type GetType(string typeName)
        {
            Type type = this.serializationManager.GetType(typeName);
            if (type == null)
            {
                // if this is a design time time
                ITypeProvider typeProvider = this.GetService(typeof(ITypeProvider)) as ITypeProvider;
                if (typeProvider != null)
                    type = typeProvider.GetType(typeName);
            }
            return type;
        }
        protected IDesignerSerializationManager SerializationManager
        {
            get
            {
                return this.serializationManager;
            }
            set
            {
                this.serializationManager = value;
            }
        }
    }
    #endregion
}