Files
UniTask/src/UniTask.NetCore/Linq/Distinct.cs

76 lines
2.5 KiB
C#
Raw Normal View History

2020-05-11 15:53:27 +09:00
using Cysharp.Threading.Tasks.Internal;
using System;
using System.Collections.Generic;
using System.Threading;
namespace Cysharp.Threading.Tasks.Linq
2020-05-08 03:23:14 +09:00
{
2020-05-11 15:53:27 +09:00
public static partial class UniTaskAsyncEnumerable
2020-05-08 03:23:14 +09:00
{
2020-05-11 15:53:27 +09:00
public static IUniTaskAsyncEnumerable<TSource> Distinct<TSource>(this IUniTaskAsyncEnumerable<TSource> source)
{
Error.ThrowArgumentNullException(source, nameof(source));
2020-05-08 03:23:14 +09:00
2020-05-11 15:53:27 +09:00
return Distinct(source, EqualityComparer<TSource>.Default);
}
2020-05-08 03:23:14 +09:00
2020-05-11 15:53:27 +09:00
public static IUniTaskAsyncEnumerable<TSource> Distinct<TSource>(this IUniTaskAsyncEnumerable<TSource> source, IEqualityComparer<TSource> comparer)
{
Error.ThrowArgumentNullException(source, nameof(source));
Error.ThrowArgumentNullException(comparer, nameof(comparer));
2020-05-08 03:23:14 +09:00
2020-05-11 15:53:27 +09:00
return new Distinct<TSource>(source, comparer);
}
}
2020-05-08 03:23:14 +09:00
2020-05-11 15:53:27 +09:00
internal sealed class Distinct<TSource> : IUniTaskAsyncEnumerable<TSource>
{
readonly IUniTaskAsyncEnumerable<TSource> source;
readonly IEqualityComparer<TSource> comparer;
public Distinct(IUniTaskAsyncEnumerable<TSource> source, IEqualityComparer<TSource> comparer)
{
this.source = source;
this.comparer = comparer;
}
public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
{
return new Enumerator(source, comparer, cancellationToken);
}
class Enumerator : AsyncEnumeratorBase<TSource, TSource>
{
readonly HashSet<TSource> set;
public Enumerator(IUniTaskAsyncEnumerable<TSource> source, IEqualityComparer<TSource> comparer, CancellationToken cancellationToken)
: base(source, cancellationToken)
{
this.set = new HashSet<TSource>(comparer);
}
protected override bool TryMoveNextCore(bool sourceHasCurrent, out bool result)
{
if (sourceHasCurrent)
{
var v = SourceCurrent;
if (set.Add(v))
{
Current = v;
result = true;
return true;
}
else
{
result = default;
return false;
}
}
result = false;
return true;
}
}
}
}