|
- using Cysharp.Threading.Tasks.Internal;
- using System;
- using System.Collections.Generic;
- using System.Threading;
-
- namespace Cysharp.Threading.Tasks.Linq
- {
- public static partial class UniTaskAsyncEnumerable
- {
- public static IUniTaskAsyncEnumerable<TSource> DistinctUntilChanged<TSource>(this IUniTaskAsyncEnumerable<TSource> source)
- {
- return DistinctUntilChanged(source, EqualityComparer<TSource>.Default);
- }
-
- public static IUniTaskAsyncEnumerable<TSource> DistinctUntilChanged<TSource>(this IUniTaskAsyncEnumerable<TSource> source, IEqualityComparer<TSource> comparer)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(comparer, nameof(comparer));
-
- return new DistinctUntilChanged<TSource>(source, comparer);
- }
-
- public static IUniTaskAsyncEnumerable<TSource> DistinctUntilChanged<TSource, TKey>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector)
- {
- return DistinctUntilChanged(source, keySelector, EqualityComparer<TKey>.Default);
- }
-
- public static IUniTaskAsyncEnumerable<TSource> DistinctUntilChanged<TSource, TKey>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- Error.ThrowArgumentNullException(comparer, nameof(comparer));
-
- return new DistinctUntilChanged<TSource, TKey>(source, keySelector, comparer);
- }
-
- public static IUniTaskAsyncEnumerable<TSource> DistinctUntilChangedAwait<TSource, TKey>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector)
- {
- return DistinctUntilChangedAwait(source, keySelector, EqualityComparer<TKey>.Default);
- }
-
- public static IUniTaskAsyncEnumerable<TSource> DistinctUntilChangedAwait<TSource, TKey>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- Error.ThrowArgumentNullException(comparer, nameof(comparer));
-
- return new DistinctUntilChangedAwait<TSource, TKey>(source, keySelector, comparer);
- }
-
- public static IUniTaskAsyncEnumerable<TSource> DistinctUntilChangedAwaitWithCancellation<TSource, TKey>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector)
- {
- return DistinctUntilChangedAwaitWithCancellation(source, keySelector, EqualityComparer<TKey>.Default);
- }
-
- public static IUniTaskAsyncEnumerable<TSource> DistinctUntilChangedAwaitWithCancellation<TSource, TKey>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- Error.ThrowArgumentNullException(keySelector, nameof(keySelector));
- Error.ThrowArgumentNullException(comparer, nameof(comparer));
-
- return new DistinctUntilChangedAwaitWithCancellation<TSource, TKey>(source, keySelector, comparer);
- }
- }
-
- internal sealed class DistinctUntilChanged<TSource> : IUniTaskAsyncEnumerable<TSource>
- {
- readonly IUniTaskAsyncEnumerable<TSource> source;
- readonly IEqualityComparer<TSource> comparer;
-
- public DistinctUntilChanged(IUniTaskAsyncEnumerable<TSource> source, IEqualityComparer<TSource> comparer)
- {
- this.source = source;
- this.comparer = comparer;
- }
-
- public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
- {
- return new _DistinctUntilChanged(source, comparer, cancellationToken);
- }
-
- sealed class _DistinctUntilChanged : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
- {
- readonly IUniTaskAsyncEnumerable<TSource> source;
- readonly IEqualityComparer<TSource> comparer;
- readonly CancellationToken cancellationToken;
-
- int state = -1;
- IUniTaskAsyncEnumerator<TSource> enumerator;
- UniTask<bool>.Awaiter awaiter;
- Action moveNextAction;
-
- public _DistinctUntilChanged(IUniTaskAsyncEnumerable<TSource> source, IEqualityComparer<TSource> comparer, CancellationToken cancellationToken)
- {
- this.source = source;
- this.comparer = comparer;
- this.cancellationToken = cancellationToken;
- this.moveNextAction = MoveNext;
- }
-
- public TSource Current { get; private set; }
-
- public UniTask<bool> MoveNextAsync()
- {
- if (state == -2) return default;
-
- completionSource.Reset();
- MoveNext();
- return new UniTask<bool>(this, completionSource.Version);
- }
-
- void MoveNext()
- {
- REPEAT:
- try
- {
- switch (state)
- {
- case -1: // init
- enumerator = source.GetAsyncEnumerator(cancellationToken);
- awaiter = enumerator.MoveNextAsync().GetAwaiter();
- if (awaiter.IsCompleted)
- {
- goto case -3;
- }
- else
- {
- state = -3;
- awaiter.UnsafeOnCompleted(moveNextAction);
- return;
- }
- case -3: // first
- if (awaiter.GetResult())
- {
- Current = enumerator.Current;
- goto CONTINUE;
- }
- else
- {
- goto DONE;
- }
- case 0: // normal
- awaiter = enumerator.MoveNextAsync().GetAwaiter();
- if (awaiter.IsCompleted)
- {
- goto case 1;
- }
- else
- {
- state = 1;
- awaiter.UnsafeOnCompleted(moveNextAction);
- return;
- }
- case 1:
- if (awaiter.GetResult())
- {
- var v = enumerator.Current;
- if (!comparer.Equals(Current, v))
- {
- Current = v;
- goto CONTINUE;
- }
- else
- {
- state = 0;
- goto REPEAT;
- }
- }
- else
- {
- goto DONE;
- }
- case -2:
- default:
- goto DONE;
- }
- }
- catch (Exception ex)
- {
- state = -2;
- completionSource.TrySetException(ex);
- return;
- }
-
- DONE:
- state = -2;
- completionSource.TrySetResult(false);
- return;
-
- CONTINUE:
- state = 0;
- completionSource.TrySetResult(true);
- return;
- }
-
- public UniTask DisposeAsync()
- {
- return enumerator.DisposeAsync();
- }
- }
- }
-
- internal sealed class DistinctUntilChanged<TSource, TKey> : IUniTaskAsyncEnumerable<TSource>
- {
- readonly IUniTaskAsyncEnumerable<TSource> source;
- readonly Func<TSource, TKey> keySelector;
- readonly IEqualityComparer<TKey> comparer;
-
- public DistinctUntilChanged(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer)
- {
- this.source = source;
- this.keySelector = keySelector;
- this.comparer = comparer;
- }
-
- public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
- {
- return new _DistinctUntilChanged(source, keySelector, comparer, cancellationToken);
- }
-
- sealed class _DistinctUntilChanged : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
- {
- readonly IUniTaskAsyncEnumerable<TSource> source;
- readonly Func<TSource, TKey> keySelector;
- readonly IEqualityComparer<TKey> comparer;
- readonly CancellationToken cancellationToken;
-
- int state = -1;
- IUniTaskAsyncEnumerator<TSource> enumerator;
- UniTask<bool>.Awaiter awaiter;
- Action moveNextAction;
- TKey prev;
-
- public _DistinctUntilChanged(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
- {
- this.source = source;
- this.keySelector = keySelector;
- this.comparer = comparer;
- this.cancellationToken = cancellationToken;
- this.moveNextAction = MoveNext;
- }
-
- public TSource Current { get; private set; }
-
- public UniTask<bool> MoveNextAsync()
- {
- if (state == -2) return default;
-
- completionSource.Reset();
- MoveNext();
- return new UniTask<bool>(this, completionSource.Version);
- }
-
- void MoveNext()
- {
- REPEAT:
- try
- {
- switch (state)
- {
- case -1: // init
- enumerator = source.GetAsyncEnumerator(cancellationToken);
- awaiter = enumerator.MoveNextAsync().GetAwaiter();
- if (awaiter.IsCompleted)
- {
- goto case -3;
- }
- else
- {
- state = -3;
- awaiter.UnsafeOnCompleted(moveNextAction);
- return;
- }
- case -3: // first
- if (awaiter.GetResult())
- {
- Current = enumerator.Current;
- goto CONTINUE;
- }
- else
- {
- goto DONE;
- }
- case 0: // normal
- awaiter = enumerator.MoveNextAsync().GetAwaiter();
- if (awaiter.IsCompleted)
- {
- goto case 1;
- }
- else
- {
- state = 1;
- awaiter.UnsafeOnCompleted(moveNextAction);
- return;
- }
- case 1:
- if (awaiter.GetResult())
- {
- var v = enumerator.Current;
- var key = keySelector(v);
- if (!comparer.Equals(prev, key))
- {
- prev = key;
- Current = v;
- goto CONTINUE;
- }
- else
- {
- state = 0;
- goto REPEAT;
- }
- }
- else
- {
- goto DONE;
- }
- case -2:
- default:
- goto DONE;
- }
- }
- catch (Exception ex)
- {
- state = -2;
- completionSource.TrySetException(ex);
- return;
- }
-
- DONE:
- state = -2;
- completionSource.TrySetResult(false);
- return;
-
- CONTINUE:
- state = 0;
- completionSource.TrySetResult(true);
- return;
- }
-
- public UniTask DisposeAsync()
- {
- return enumerator.DisposeAsync();
- }
- }
- }
-
- internal sealed class DistinctUntilChangedAwait<TSource, TKey> : IUniTaskAsyncEnumerable<TSource>
- {
- readonly IUniTaskAsyncEnumerable<TSource> source;
- readonly Func<TSource, UniTask<TKey>> keySelector;
- readonly IEqualityComparer<TKey> comparer;
-
- public DistinctUntilChangedAwait(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer)
- {
- this.source = source;
- this.keySelector = keySelector;
- this.comparer = comparer;
- }
-
- public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
- {
- return new _DistinctUntilChangedAwait(source, keySelector, comparer, cancellationToken);
- }
-
- sealed class _DistinctUntilChangedAwait : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
- {
- readonly IUniTaskAsyncEnumerable<TSource> source;
- readonly Func<TSource, UniTask<TKey>> keySelector;
- readonly IEqualityComparer<TKey> comparer;
- readonly CancellationToken cancellationToken;
-
- int state = -1;
- IUniTaskAsyncEnumerator<TSource> enumerator;
- UniTask<bool>.Awaiter awaiter;
- UniTask<TKey>.Awaiter awaiter2;
- Action moveNextAction;
- TSource enumeratorCurrent;
- TKey prev;
-
- public _DistinctUntilChangedAwait(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
- {
- this.source = source;
- this.keySelector = keySelector;
- this.comparer = comparer;
- this.cancellationToken = cancellationToken;
- this.moveNextAction = MoveNext;
- }
-
- public TSource Current { get; private set; }
-
- public UniTask<bool> MoveNextAsync()
- {
- if (state == -2) return default;
-
- completionSource.Reset();
- MoveNext();
- return new UniTask<bool>(this, completionSource.Version);
- }
-
- void MoveNext()
- {
- REPEAT:
- try
- {
- switch (state)
- {
- case -1: // init
- enumerator = source.GetAsyncEnumerator(cancellationToken);
- awaiter = enumerator.MoveNextAsync().GetAwaiter();
- if (awaiter.IsCompleted)
- {
- goto case -3;
- }
- else
- {
- state = -3;
- awaiter.UnsafeOnCompleted(moveNextAction);
- return;
- }
- case -3: // first
- if (awaiter.GetResult())
- {
- Current = enumerator.Current;
- goto CONTINUE;
- }
- else
- {
- goto DONE;
- }
- case 0: // normal
- awaiter = enumerator.MoveNextAsync().GetAwaiter();
- if (awaiter.IsCompleted)
- {
- goto case 1;
- }
- else
- {
- state = 1;
- awaiter.UnsafeOnCompleted(moveNextAction);
- return;
- }
- case 1:
- if (awaiter.GetResult())
- {
- enumeratorCurrent = enumerator.Current;
- awaiter2 = keySelector(enumeratorCurrent).GetAwaiter();
- if (awaiter2.IsCompleted)
- {
- goto case 2;
- }
- else
- {
- state = 2;
- awaiter2.UnsafeOnCompleted(moveNextAction);
- return;
- }
- }
- else
- {
- goto DONE;
- }
- case 2:
- var key = awaiter2.GetResult();
- if (!comparer.Equals(prev, key))
- {
- prev = key;
- Current = enumeratorCurrent;
- goto CONTINUE;
- }
- else
- {
- state = 0;
- goto REPEAT;
- }
- case -2:
- default:
- goto DONE;
- }
- }
- catch (Exception ex)
- {
- state = -2;
- completionSource.TrySetException(ex);
- return;
- }
-
- DONE:
- state = -2;
- completionSource.TrySetResult(false);
- return;
-
- CONTINUE:
- state = 0;
- completionSource.TrySetResult(true);
- return;
- }
-
- public UniTask DisposeAsync()
- {
- return enumerator.DisposeAsync();
- }
- }
- }
-
- internal sealed class DistinctUntilChangedAwaitWithCancellation<TSource, TKey> : IUniTaskAsyncEnumerable<TSource>
- {
- readonly IUniTaskAsyncEnumerable<TSource> source;
- readonly Func<TSource, CancellationToken, UniTask<TKey>> keySelector;
- readonly IEqualityComparer<TKey> comparer;
-
- public DistinctUntilChangedAwaitWithCancellation(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer)
- {
- this.source = source;
- this.keySelector = keySelector;
- this.comparer = comparer;
- }
-
- public IUniTaskAsyncEnumerator<TSource> GetAsyncEnumerator(CancellationToken cancellationToken = default)
- {
- return new _DistinctUntilChangedAwaitWithCancellation(source, keySelector, comparer, cancellationToken);
- }
-
- sealed class _DistinctUntilChangedAwaitWithCancellation : MoveNextSource, IUniTaskAsyncEnumerator<TSource>
- {
- readonly IUniTaskAsyncEnumerable<TSource> source;
- readonly Func<TSource, CancellationToken, UniTask<TKey>> keySelector;
- readonly IEqualityComparer<TKey> comparer;
- readonly CancellationToken cancellationToken;
-
- int state = -1;
- IUniTaskAsyncEnumerator<TSource> enumerator;
- UniTask<bool>.Awaiter awaiter;
- UniTask<TKey>.Awaiter awaiter2;
- Action moveNextAction;
- TSource enumeratorCurrent;
- TKey prev;
-
- public _DistinctUntilChangedAwaitWithCancellation(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<TKey>> keySelector, IEqualityComparer<TKey> comparer, CancellationToken cancellationToken)
- {
- this.source = source;
- this.keySelector = keySelector;
- this.comparer = comparer;
- this.cancellationToken = cancellationToken;
- this.moveNextAction = MoveNext;
- }
-
- public TSource Current { get; private set; }
-
- public UniTask<bool> MoveNextAsync()
- {
- if (state == -2) return default;
-
- completionSource.Reset();
- MoveNext();
- return new UniTask<bool>(this, completionSource.Version);
- }
-
- void MoveNext()
- {
- REPEAT:
- try
- {
- switch (state)
- {
- case -1: // init
- enumerator = source.GetAsyncEnumerator(cancellationToken);
- awaiter = enumerator.MoveNextAsync().GetAwaiter();
- if (awaiter.IsCompleted)
- {
- goto case -3;
- }
- else
- {
- state = -3;
- awaiter.UnsafeOnCompleted(moveNextAction);
- return;
- }
- case -3: // first
- if (awaiter.GetResult())
- {
- Current = enumerator.Current;
- goto CONTINUE;
- }
- else
- {
- goto DONE;
- }
- case 0: // normal
- awaiter = enumerator.MoveNextAsync().GetAwaiter();
- if (awaiter.IsCompleted)
- {
- goto case 1;
- }
- else
- {
- state = 1;
- awaiter.UnsafeOnCompleted(moveNextAction);
- return;
- }
- case 1:
- if (awaiter.GetResult())
- {
- enumeratorCurrent = enumerator.Current;
- awaiter2 = keySelector(enumeratorCurrent, cancellationToken).GetAwaiter();
- if (awaiter2.IsCompleted)
- {
- goto case 2;
- }
- else
- {
- state = 2;
- awaiter2.UnsafeOnCompleted(moveNextAction);
- return;
- }
- }
- else
- {
- goto DONE;
- }
- case 2:
- var key = awaiter2.GetResult();
- if (!comparer.Equals(prev, key))
- {
- prev = key;
- Current = enumeratorCurrent;
- goto CONTINUE;
- }
- else
- {
- state = 0;
- goto REPEAT;
- }
- case -2:
- default:
- goto DONE;
- }
- }
- catch (Exception ex)
- {
- state = -2;
- completionSource.TrySetException(ex);
- return;
- }
-
- DONE:
- state = -2;
- completionSource.TrySetResult(false);
- return;
-
- CONTINUE:
- state = 0;
- completionSource.TrySetResult(true);
- return;
- }
-
- public UniTask DisposeAsync()
- {
- return enumerator.DisposeAsync();
- }
- }
- }
- }
|