CollectionUtils
using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Runtime.CompilerServices;
namespace Newtonsoft.Json.Utilities
{
    [System.Runtime.CompilerServices.NullableContext(1)]
    [System.Runtime.CompilerServices.Nullable(0)]
    internal static class CollectionUtils
    {
        [System.Runtime.CompilerServices.NullableContext(0)]
        private static class EmptyArrayContainer<[System.Runtime.CompilerServices.Nullable(2)] T>
        {
            [System.Runtime.CompilerServices.Nullable(1)]
            public static readonly T[] Empty = new T[0];
        }
        public static bool IsNullOrEmpty<[System.Runtime.CompilerServices.Nullable(2)] T>(ICollection<T> collection)
        {
            if (collection != null)
                return collection.Count == 0;
            return true;
        }
        public static void AddRange<[System.Runtime.CompilerServices.Nullable(2)] T>(this IList<T> initial, IEnumerable<T> collection)
        {
            if (initial == null)
                throw new ArgumentNullException("initial");
            if (collection != null) {
                foreach (T item in collection) {
                    ((ICollection<T>)initial).Add(item);
                }
            }
        }
        public static bool IsDictionaryType(Type type)
        {
            ValidationUtils.ArgumentNotNull(type, "type");
            if (typeof(IDictionary).IsAssignableFrom(type))
                return true;
            if (ReflectionUtils.ImplementsGenericDefinition(type, typeof(IDictionary<, >)))
                return true;
            if (ReflectionUtils.ImplementsGenericDefinition(type, typeof(IReadOnlyDictionary<, >)))
                return true;
            return false;
        }
        [return: System.Runtime.CompilerServices.Nullable(2)]
        public static ConstructorInfo ResolveEnumerableCollectionConstructor(Type collectionType, Type collectionItemType)
        {
            Type constructorArgumentType = typeof(IList<>).MakeGenericType(collectionItemType);
            return ResolveEnumerableCollectionConstructor(collectionType, collectionItemType, constructorArgumentType);
        }
        [return: System.Runtime.CompilerServices.Nullable(2)]
        public static ConstructorInfo ResolveEnumerableCollectionConstructor(Type collectionType, Type collectionItemType, Type constructorArgumentType)
        {
            Type type = typeof(IEnumerable<>).MakeGenericType(collectionItemType);
            ConstructorInfo constructorInfo = null;
            foreach (ConstructorInfo constructor in collectionType.GetConstructors(BindingFlags.Instance | BindingFlags.Public)) {
                IList<ParameterInfo> parameters = constructor.GetParameters();
                if (parameters.Count == 1) {
                    Type parameterType = parameters[0].ParameterType;
                    if ((object)type == parameterType)
                        return constructor;
                    if ((object)constructorInfo == null && TypeExtensions.IsAssignableFrom(parameterType, constructorArgumentType))
                        constructorInfo = constructor;
                }
            }
            return constructorInfo;
        }
        public static bool AddDistinct<[System.Runtime.CompilerServices.Nullable(2)] T>(this IList<T> list, T value)
        {
            return list.AddDistinct(value, EqualityComparer<T>.Default);
        }
        public static bool AddDistinct<[System.Runtime.CompilerServices.Nullable(2)] T>(this IList<T> list, T value, IEqualityComparer<T> comparer)
        {
            if (list.ContainsValue(value, comparer))
                return false;
            ((ICollection<T>)list).Add(value);
            return true;
        }
        public static bool ContainsValue<[System.Runtime.CompilerServices.Nullable(2)] TSource>(this IEnumerable<TSource> source, TSource value, IEqualityComparer<TSource> comparer)
        {
            if (comparer == null)
                comparer = EqualityComparer<TSource>.Default;
            if (source == null)
                throw new ArgumentNullException("source");
            foreach (TSource item in source) {
                if (comparer.Equals(item, value))
                    return true;
            }
            return false;
        }
        public static bool AddRangeDistinct<[System.Runtime.CompilerServices.Nullable(2)] T>(this IList<T> list, IEnumerable<T> values, IEqualityComparer<T> comparer)
        {
            bool result = true;
            foreach (T value in values) {
                if (!list.AddDistinct(value, comparer))
                    result = false;
            }
            return result;
        }
        public static int IndexOf<[System.Runtime.CompilerServices.Nullable(2)] T>(this IEnumerable<T> collection, Func<T, bool> predicate)
        {
            int num = 0;
            foreach (T item in collection) {
                if (predicate(item))
                    return num;
                num++;
            }
            return -1;
        }
        public static bool Contains<[System.Runtime.CompilerServices.Nullable(2)] T>(this List<T> list, T value, IEqualityComparer comparer)
        {
            for (int i = 0; i < list.Count; i++) {
                if (comparer.Equals(value, list[i]))
                    return true;
            }
            return false;
        }
        public static int IndexOfReference<[System.Runtime.CompilerServices.Nullable(2)] T>(this List<T> list, T item)
        {
            for (int i = 0; i < list.Count; i++) {
                if ((object)item == (object)list[i])
                    return i;
            }
            return -1;
        }
        private static IList<int> GetDimensions(IList values, int dimensionsCount)
        {
            IList<int> list = new List<int>();
            IList list2 = values;
            while (true) {
                list.Add(list2.Count);
                if (list.Count == dimensionsCount || list2.Count == 0)
                    break;
                IList list3 = list2[0] as IList;
                if (list3 == null)
                    break;
                list2 = list3;
            }
            return list;
        }
        private static void CopyFromJaggedToMultidimensionalArray(IList values, Array multidimensionalArray, int[] indices)
        {
            int num = indices.Length;
            if (num == multidimensionalArray.Rank)
                multidimensionalArray.SetValue(JaggedArrayGetValue(values, indices), indices);
            else {
                int length = multidimensionalArray.GetLength(num);
                if (((IList)JaggedArrayGetValue(values, indices)).Count != length)
                    throw new Exception("Cannot deserialize non-cubical array as multidimensional array.");
                int[] array = new int[num + 1];
                for (int i = 0; i < num; i++) {
                    array[i] = indices[i];
                }
                for (int j = 0; j < multidimensionalArray.GetLength(num); j++) {
                    array[num] = j;
                    CopyFromJaggedToMultidimensionalArray(values, multidimensionalArray, array);
                }
            }
        }
        private static object JaggedArrayGetValue(IList values, int[] indices)
        {
            IList list = values;
            for (int i = 0; i < indices.Length; i++) {
                int index = indices[i];
                if (i == indices.Length - 1)
                    return list[index];
                list = (IList)list[index];
            }
            return list;
        }
        public static Array ToMultidimensionalArray(IList values, Type type, int rank)
        {
            IList<int> dimensions = GetDimensions(values, rank);
            while (dimensions.Count < rank) {
                dimensions.Add(0);
            }
            Array array = Array.CreateInstance(type, dimensions.ToArray());
            CopyFromJaggedToMultidimensionalArray(values, array, ArrayEmpty<int>());
            return array;
        }
        public static T[] ArrayEmpty<[System.Runtime.CompilerServices.Nullable(2)] T>()
        {
            return EmptyArrayContainer<T>.Empty;
        }
    }
}