You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
161 lines
5.8 KiB
161 lines
5.8 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
|
|
namespace NzbDrone.Common.Extensions
|
|
{
|
|
public static class EnumerableExtensions
|
|
{
|
|
public static IEnumerable<TFirst> IntersectBy<TFirst, TSecond, TKey>(this IEnumerable<TFirst> first,
|
|
Func<TFirst, TKey> firstKeySelector,
|
|
IEnumerable<TSecond> second,
|
|
Func<TSecond, TKey> secondKeySelector,
|
|
IEqualityComparer<TKey> keyComparer)
|
|
{
|
|
var keys = new HashSet<TKey>(second.Select(secondKeySelector), keyComparer);
|
|
|
|
foreach (var element in first)
|
|
{
|
|
var key = firstKeySelector(element);
|
|
|
|
// Remove the key so we only yield once
|
|
if (keys.Remove(key))
|
|
{
|
|
yield return element;
|
|
}
|
|
}
|
|
}
|
|
|
|
public static IEnumerable<TFirst> ExceptBy<TFirst, TSecond, TKey>(this IEnumerable<TFirst> first,
|
|
Func<TFirst, TKey> firstKeySelector,
|
|
IEnumerable<TSecond> second,
|
|
Func<TSecond, TKey> secondKeySelector,
|
|
IEqualityComparer<TKey> keyComparer)
|
|
{
|
|
var keys = new HashSet<TKey>(second.Select(secondKeySelector), keyComparer);
|
|
var matchedKeys = new HashSet<TKey>();
|
|
|
|
foreach (var element in first)
|
|
{
|
|
var key = firstKeySelector(element);
|
|
|
|
if (!keys.Contains(key) && !matchedKeys.Contains(key))
|
|
{
|
|
// Store the key so we only yield once
|
|
matchedKeys.Add(key);
|
|
yield return element;
|
|
}
|
|
}
|
|
}
|
|
|
|
public static TSource ExclusiveOrDefault<TSource>(this IEnumerable<TSource> source)
|
|
{
|
|
if (source == null)
|
|
{
|
|
throw new ArgumentNullException("source");
|
|
}
|
|
|
|
var results = source.Take(2).ToArray();
|
|
|
|
return results.Length == 1 ? results[0] : default(TSource);
|
|
}
|
|
|
|
public static TSource ExclusiveOrDefault<TSource>(this IEnumerable<TSource> source, Func<TSource, bool> predicate)
|
|
{
|
|
if (source == null)
|
|
{
|
|
throw new ArgumentNullException("source");
|
|
}
|
|
|
|
if (predicate == null)
|
|
{
|
|
throw new ArgumentNullException("predicate");
|
|
}
|
|
|
|
var results = source.Where(predicate).Take(2).ToArray();
|
|
|
|
return results.Length == 1 ? results[0] : default(TSource);
|
|
}
|
|
|
|
public static Dictionary<TKey, TItem> ToDictionaryIgnoreDuplicates<TItem, TKey>(this IEnumerable<TItem> src, Func<TItem, TKey> keySelector)
|
|
{
|
|
var result = new Dictionary<TKey, TItem>();
|
|
foreach (var item in src)
|
|
{
|
|
var key = keySelector(item);
|
|
if (!result.ContainsKey(key))
|
|
{
|
|
result[key] = item;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
public static Dictionary<TKey, TValue> ToDictionaryIgnoreDuplicates<TItem, TKey, TValue>(this IEnumerable<TItem> src, Func<TItem, TKey> keySelector, Func<TItem, TValue> valueSelector)
|
|
{
|
|
var result = new Dictionary<TKey, TValue>();
|
|
foreach (var item in src)
|
|
{
|
|
var key = keySelector(item);
|
|
if (!result.ContainsKey(key))
|
|
{
|
|
result[key] = valueSelector(item);
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
public static void AddIfNotNull<TSource>(this List<TSource> source, TSource item)
|
|
{
|
|
if (item == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
source.Add(item);
|
|
}
|
|
|
|
public static bool Empty<TSource>(this IEnumerable<TSource> source)
|
|
{
|
|
return !source.Any();
|
|
}
|
|
|
|
public static bool None<TSource>(this IEnumerable<TSource> source, Func<TSource, bool> predicate)
|
|
{
|
|
return !source.Any(predicate);
|
|
}
|
|
|
|
public static bool NotAll<TSource>(this IEnumerable<TSource> source, Func<TSource, bool> predicate)
|
|
{
|
|
return !source.All(predicate);
|
|
}
|
|
|
|
public static List<TResult> SelectList<TSource, TResult>(this IEnumerable<TSource> source, Func<TSource, TResult> predicate)
|
|
{
|
|
return source.Select(predicate).ToList();
|
|
}
|
|
|
|
public static string ConcatToString<TSource>(this IEnumerable<TSource> source, string separator = ", ")
|
|
{
|
|
return string.Join(separator, source.Select(x => x.ToString()));
|
|
}
|
|
|
|
public static string ConcatToString<TSource>(this IEnumerable<TSource> source, Func<TSource, string> predicate, string separator = ", ")
|
|
{
|
|
return string.Join(separator, source.Select(predicate));
|
|
}
|
|
|
|
public static TSource MostCommon<TSource>(this IEnumerable<TSource> items)
|
|
{
|
|
return items.GroupBy(x => x).OrderByDescending(x => x.Count()).First().Key;
|
|
}
|
|
|
|
public static TResult MostCommon<TSource, TResult>(this IEnumerable<TSource> items, Func<TSource, TResult> predicate)
|
|
{
|
|
return items.Select(predicate).GroupBy(x => x).OrderByDescending(x => x.Count()).First().Key;
|
|
}
|
|
}
|
|
}
|