// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the Apache 2.0 License.
// See the LICENSE file in the project root for more information. 
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
namespace System.Linq
{
    public static partial class EnumerableEx
    {
        /// 
        ///     Returns the minimum value in the enumerable sequence by using the specified comparer to compare values.
        /// 
        /// Source sequence element type.
        /// Source sequence.
        /// Comparer used to determine the minimum value.
        /// Minimum value in the sequence.
        public static TSource Min(this IEnumerable source, IComparer comparer)
        {
            if (source == null)
                throw new ArgumentNullException(nameof(source));
            if (comparer == null)
                throw new ArgumentNullException(nameof(comparer));
            return MinBy(source, x => x, comparer)
                .First();
        }
        /// 
        ///     Returns the elements with the minimum key value by using the default comparer to compare key values.
        /// 
        /// Source sequence element type.
        /// Key type.
        /// Source sequence.
        /// Key selector used to extract the key for each element in the sequence.
        /// List with the elements that share the same minimum key value.
        public static IList MinBy(this IEnumerable source, Func keySelector)
        {
            if (source == null)
                throw new ArgumentNullException(nameof(source));
            if (keySelector == null)
                throw new ArgumentNullException(nameof(keySelector));
            return MinBy(source, keySelector, Comparer.Default);
        }
        /// 
        ///     Returns the elements with the minimum key value by using the specified comparer to compare key values.
        /// 
        /// Source sequence element type.
        /// Key type.
        /// Source sequence.
        /// Key selector used to extract the key for each element in the sequence.
        /// Comparer used to determine the minimum key value.
        /// List with the elements that share the same minimum key value.
        public static IList MinBy(this IEnumerable source, Func keySelector, IComparer comparer)
        {
            if (source == null)
                throw new ArgumentNullException(nameof(source));
            if (keySelector == null)
                throw new ArgumentNullException(nameof(keySelector));
            if (comparer == null)
                throw new ArgumentNullException(nameof(comparer));
            return ExtremaBy(source, keySelector, (key, minValue) => -comparer.Compare(key, minValue));
        }
    }
}