// 
//   Copyright (c) Microsoft Corporation.  All rights reserved.
// 
namespace System.Activities.Expressions
{
    using System;
    using System.Collections.Generic;
    using System.Collections.ObjectModel;
    using System.ComponentModel;
    using System.Reflection;
    using System.Runtime;
    using System.Xaml;
    public static class TextExpression
    {
        private static readonly AttachableMemberIdentifier namespacesProperty = 
            new AttachableMemberIdentifier(typeof(TextExpression), "Namespaces");
        private static readonly AttachableMemberIdentifier namespacesForImplementationProperty =
            new AttachableMemberIdentifier(typeof(TextExpression), "NamespacesForImplementation");
        private static readonly AttachableMemberIdentifier referencesProperty =
            new AttachableMemberIdentifier(typeof(TextExpression), "References");
        private static readonly AttachableMemberIdentifier referencesForImplementationProperty =
            new AttachableMemberIdentifier(typeof(TextExpression), "ReferencesForImplementation");
        // This should be kept consistent with VisualBasicSettings.defaultImportReferences
        private static readonly ReadOnlyCollection defaultNamespaces = new ReadOnlyCollection(new string[]
        {
            "System",
            "System.Collections",
            "System.Collections.Generic",
            "System.Activities",
            "System.Activities.Expressions",
            "System.Activities.Statements"
        });
        private static readonly ReadOnlyCollection defaultReferences = new ReadOnlyCollection(new AssemblyReference[]
        {
            new AssemblyName("mscorlib"),
            new AssemblyName("System"),
            new AssemblyName("System.Activities"),
            new AssemblyName("System.Core")
        });
        public static IList DefaultNamespaces
        {
            get
            {
                return TextExpression.defaultNamespaces;
            }
        }
        public static IList DefaultReferences
        {
            get
            {
                return TextExpression.defaultReferences;
            }
        }
        public static IList GetNamespacesInScope(Activity activity)
        {
            bool isImplementation;
            Activity root = GetRoot(activity, out isImplementation);
            IList result = isImplementation ? GetNamespacesForImplementation(root) : GetNamespaces(root);
            if (result.Count == 0 && !isImplementation && GetReferences(root).Count == 0)
            {
                // If this is a public child, but there are no public setings, this activity was
                // probably a default set as part of the activity definition; so fall back to the
                // implementation settings.
                result = GetNamespacesForImplementation(root);
            }
            return result;
        }
        public static IList GetNamespaces(object target)
        {
            return GetCollection(target, namespacesProperty);
        }
        public static void SetNamespaces(object target, IList namespaces)
        {
            SetCollection(target, namespacesProperty, namespaces);
        }
        public static void SetNamespaces(object target, params string[] namespaces)
        {
            SetCollection(target, namespacesProperty, namespaces);
        }
        public static bool ShouldSerializeNamespaces(object target)
        {
            return ShouldSerializeCollection(target, namespacesProperty);
        }
        public static IList GetNamespacesForImplementation(object target)
        {
            return GetCollection(target, namespacesForImplementationProperty);
        }
        public static void SetNamespacesForImplementation(object target, IList namespaces)
        {
            SetCollection(target, namespacesForImplementationProperty, namespaces);
        }
        public static void SetNamespacesForImplementation(object target, params string[] namespaces)
        {
            SetCollection(target, namespacesForImplementationProperty, namespaces);
        }
        // Implementation namespaces only serialize when the activity is being defined (target is ActivityBuilder),
        // not when it is being consumed (target is Activity)
        public static bool ShouldSerializeNamespacesForImplementation(object target)
        {
            return !(target is Activity) && ShouldSerializeCollection(target, namespacesForImplementationProperty);
        }
        public static IList GetReferencesInScope(Activity activity)
        {
            bool isImplementation;
            Activity root = GetRoot(activity, out isImplementation);
            IList result = isImplementation ? GetReferencesForImplementation(root) : GetReferences(root);
            if (result.Count == 0 && !isImplementation && GetNamespaces(root).Count == 0)
            {
                // If this is a public child, but there are no public setings, this activity was
                // probably a default set as part of the activity definition; so fall back to the
                // implementation settings.
                result = GetReferencesForImplementation(root);
            }
            return result;
        }
        public static IList GetReferences(object target)
        {
            return GetCollection(target, referencesProperty);
        }
        public static void SetReferences(object target, IList references)
        {
            SetCollection(target, referencesProperty, references);
        }
        public static void SetReferences(object target, params AssemblyReference[] references)
        {
            SetCollection(target, referencesProperty, references);
        }
        public static bool ShouldSerializeReferences(object target)
        {
            return ShouldSerializeCollection(target, referencesProperty);
        }
        public static IList GetReferencesForImplementation(object target)
        {
            return GetCollection(target, referencesForImplementationProperty);
        }
        public static void SetReferencesForImplementation(object target, IList references)
        {
            SetCollection(target, referencesForImplementationProperty, references);
        }
        public static void SetReferencesForImplementation(object target, params AssemblyReference[] references)
        {
            SetCollection(target, referencesForImplementationProperty, references);
        }
        // Implementation references only serialize when the activity is being defined (target is ActivityBuilder),
        // not when it is being consumed (target is Activity)
        public static bool ShouldSerializeReferencesForImplementation(object target)
        {
            return !(target is Activity) && ShouldSerializeCollection(target, referencesForImplementationProperty);
        }
        internal static bool LanguagesAreEqual(string left, string right)
        {
            // CodeDOM languages are case-insensitive
            return string.Equals(left, right, StringComparison.OrdinalIgnoreCase);
        }
        private static Activity GetRoot(Activity activity, out bool isImplementation)
        {
            isImplementation = false;
            Activity root = null;
            if (activity != null)
            {
                if (activity.MemberOf == null)
                {
                    throw FxTrace.Exception.Argument("activity", SR.ActivityIsUncached);
                }
                LocationReferenceEnvironment environment = activity.GetParentEnvironment();
                isImplementation = activity.MemberOf != activity.RootActivity.MemberOf;
                root = environment.Root;
            }
            return root;
        }
        private static IList GetCollection(object target, AttachableMemberIdentifier property)
        {
            IList result;
            if (!AttachablePropertyServices.TryGetProperty(target, property, out result))
            {
                result = new Collection();
                AttachablePropertyServices.SetProperty(target, property, result);
            }
            return result;
        }
        private static void SetCollection(object target, AttachableMemberIdentifier property, IList collection)
        {
            if (collection == null)
            {
                AttachablePropertyServices.RemoveProperty(target, property);
            }
            else
            {
                if (collection is Array)
                {
                    collection = new Collection(collection);
                }
                AttachablePropertyServices.SetProperty(target, property, collection);
            }
        }
        // We need this explicit check because otherwise, an empty collection might get serialized
        // just because the getter was accessed
        private static bool ShouldSerializeCollection(object target, AttachableMemberIdentifier property)
        {
            IList result;
            if (AttachablePropertyServices.TryGetProperty(target, property, out result))
            {
                return result.Count > 0;
            }
            return false;
        }
    }
}