EnumUtils
using System;
using System.Collections.Generic;
using System.Globalization;
using System.Linq;
using System.Reflection;
using System.Runtime.Serialization;
namespace Newtonsoft.Json.Utilities
{
    internal static class EnumUtils
    {
        private static readonly ThreadSafeStore<Type, BidirectionalDictionary<string, string>> EnumMemberNamesPerType = new ThreadSafeStore<Type, BidirectionalDictionary<string, string>>(InitializeEnumType);
        private static BidirectionalDictionary<string, string> InitializeEnumType(Type type)
        {
            BidirectionalDictionary<string, string> bidirectionalDictionary = new BidirectionalDictionary<string, string>(StringComparer.Ordinal, StringComparer.Ordinal);
            FieldInfo[] fields = type.GetFields(BindingFlags.Static | BindingFlags.Public);
            foreach (FieldInfo fieldInfo in fields) {
                string name = fieldInfo.Name;
                string text = (from EnumMemberAttribute a in fieldInfo.GetCustomAttributes(typeof(EnumMemberAttribute), true)
                select a.Value).SingleOrDefault() ?? fieldInfo.Name;
                if (bidirectionalDictionary.TryGetBySecond(text, out string _))
                    throw new InvalidOperationException("Enum name '{0}' already exists on enum '{1}'.".FormatWith(CultureInfo.InvariantCulture, text, type.Name));
                bidirectionalDictionary.Set(name, text);
            }
            return bidirectionalDictionary;
        }
        public static IList<T> GetFlagsValues<T>(T value) where T : struct
        {
            Type typeFromHandle = typeof(T);
            if (!typeFromHandle.IsDefined(typeof(FlagsAttribute), false))
                throw new ArgumentException("Enum type {0} is not a set of flags.".FormatWith(CultureInfo.InvariantCulture, typeFromHandle));
            Type underlyingType = Enum.GetUnderlyingType(value.GetType());
            ulong num = Convert.ToUInt64(value, CultureInfo.InvariantCulture);
            IList<EnumValue<ulong>> namesAndValues = GetNamesAndValues<T>();
            IList<T> list = new List<T>();
            foreach (EnumValue<ulong> item in (IEnumerable<EnumValue<ulong>>)namesAndValues) {
                if ((num & item.Value) == item.Value && item.Value != 0)
                    ((ICollection<T>)list).Add((T)Convert.ChangeType(item.Value, underlyingType, CultureInfo.CurrentCulture));
            }
            if (((ICollection<T>)list).Count == 0 && namesAndValues.SingleOrDefault((EnumValue<ulong> v) => v.Value == 0) != null)
                ((ICollection<T>)list).Add(default(T));
            return list;
        }
        public static IList<EnumValue<ulong>> GetNamesAndValues<T>() where T : struct
        {
            return GetNamesAndValues<ulong>(typeof(T));
        }
        public static IList<EnumValue<TUnderlyingType>> GetNamesAndValues<TUnderlyingType>(Type enumType) where TUnderlyingType : struct
        {
            if (enumType == (Type)null)
                throw new ArgumentNullException("enumType");
            if (!enumType.IsEnum())
                throw new ArgumentException("Type {0} is not an enum.".FormatWith(CultureInfo.InvariantCulture, enumType.Name), "enumType");
            IList<object> values = GetValues(enumType);
            IList<string> names = GetNames(enumType);
            IList<EnumValue<TUnderlyingType>> list = new List<EnumValue<TUnderlyingType>>();
            for (int i = 0; i < ((ICollection<object>)values).Count; i++) {
                try {
                    ((ICollection<EnumValue<TUnderlyingType>>)list).Add(new EnumValue<TUnderlyingType>(names[i], (TUnderlyingType)Convert.ChangeType(values[i], typeof(TUnderlyingType), CultureInfo.CurrentCulture)));
                } catch (OverflowException innerException) {
                    throw new InvalidOperationException("Value from enum with the underlying type of {0} cannot be added to dictionary with a value type of {1}. Value was too large: {2}".FormatWith(CultureInfo.InvariantCulture, Enum.GetUnderlyingType(enumType), typeof(TUnderlyingType), Convert.ToUInt64(values[i], CultureInfo.InvariantCulture)), innerException);
                }
            }
            return list;
        }
        public static IList<object> GetValues(Type enumType)
        {
            if (!enumType.IsEnum())
                throw new ArgumentException("Type {0} is not an enum.".FormatWith(CultureInfo.InvariantCulture, enumType.Name), "enumType");
            List<object> list = new List<object>();
            FieldInfo[] fields = enumType.GetFields(BindingFlags.Static | BindingFlags.Public);
            for (int i = 0; i < fields.Length; i++) {
                object value = fields[i].GetValue(enumType);
                list.Add(value);
            }
            return list;
        }
        public static IList<string> GetNames(Type enumType)
        {
            if (!enumType.IsEnum())
                throw new ArgumentException("Type {0} is not an enum.".FormatWith(CultureInfo.InvariantCulture, enumType.Name), "enumType");
            List<string> list = new List<string>();
            FieldInfo[] fields = enumType.GetFields(BindingFlags.Static | BindingFlags.Public);
            foreach (FieldInfo fieldInfo in fields) {
                list.Add(fieldInfo.Name);
            }
            return list;
        }
        public static object ParseEnumName(string enumText, bool isNullable, bool disallowValue, Type t)
        {
            if ((enumText == string.Empty) & isNullable)
                return null;
            BidirectionalDictionary<string, string> map = EnumMemberNamesPerType.Get(t);
            string text;
            if (TryResolvedEnumName(map, enumText, out string resolvedEnumName))
                text = resolvedEnumName;
            else if (enumText.IndexOf(',') != -1) {
                string[] array = enumText.Split(new char[1] {
                    ','
                });
                for (int i = 0; i < array.Length; i++) {
                    string text2 = array[i].Trim();
                    array[i] = (TryResolvedEnumName(map, text2, out resolvedEnumName) ? resolvedEnumName : text2);
                }
                text = string.Join(", ", array);
            } else {
                text = enumText;
                if (disallowValue && int.TryParse(text, NumberStyles.AllowLeadingSign, CultureInfo.InvariantCulture, out int _))
                    throw new FormatException("Integer string '{0}' is not allowed.".FormatWith(CultureInfo.InvariantCulture, enumText));
            }
            return Enum.Parse(t, text, true);
        }
        public static string ToEnumName(Type enumType, string enumText, bool camelCaseText)
        {
            BidirectionalDictionary<string, string> bidirectionalDictionary = EnumMemberNamesPerType.Get(enumType);
            string[] array = enumText.Split(new char[1] {
                ','
            });
            for (int i = 0; i < array.Length; i++) {
                string text = array[i].Trim();
                bidirectionalDictionary.TryGetByFirst(text, out string second);
                second = (second ?? text);
                if (camelCaseText)
                    second = StringUtils.ToCamelCase(second);
                array[i] = second;
            }
            return string.Join(", ", array);
        }
        private static bool TryResolvedEnumName(BidirectionalDictionary<string, string> map, string enumText, out string resolvedEnumName)
        {
            if (map.TryGetBySecond(enumText, out resolvedEnumName))
                return true;
            resolvedEnumName = null;
            return false;
        }
    }
}