using Cysharp.Threading.Tasks.Internal; using System; using System.Threading; namespace Cysharp.Threading.Tasks.Linq { public static partial class UniTaskAsyncEnumerable { public static IUniTaskAsyncEnumerable Append(this IUniTaskAsyncEnumerable source, TSource element) { Error.ThrowArgumentNullException(source, nameof(source)); return new AppendPrepend(source, element, true); } public static IUniTaskAsyncEnumerable Prepend(this IUniTaskAsyncEnumerable source, TSource element) { Error.ThrowArgumentNullException(source, nameof(source)); return new AppendPrepend(source, element, true); } } internal sealed class AppendPrepend : IUniTaskAsyncEnumerable { readonly IUniTaskAsyncEnumerable source; readonly TSource element; readonly bool append; // or prepend public AppendPrepend(IUniTaskAsyncEnumerable source, TSource element, bool append) { this.source = source; this.element = element; this.append = append; } public IUniTaskAsyncEnumerator GetAsyncEnumerator(CancellationToken cancellationToken = default) { return new Enumerator(source, element, append, cancellationToken); } sealed class Enumerator : MoveNextSource, IUniTaskAsyncEnumerator { enum State : byte { None, RequirePrepend, RequireAppend, Completed } static readonly Action MoveNextCoreDelegate = MoveNextCore; readonly IUniTaskAsyncEnumerable source; readonly TSource element; CancellationToken cancellationToken; State state; IUniTaskAsyncEnumerator enumerator; UniTask.Awaiter awaiter; public Enumerator(IUniTaskAsyncEnumerable source, TSource element, bool append, CancellationToken cancellationToken) { this.source = source; this.element = element; this.state = append ? State.RequireAppend : State.RequirePrepend; this.cancellationToken = cancellationToken; } public TSource Current { get; private set; } public UniTask MoveNextAsync() { cancellationToken.ThrowIfCancellationRequested(); completionSource.Reset(); if (enumerator == null) { if (state == State.RequireAppend) { Current = element; state = State.None; return CompletedTasks.True; } enumerator = source.GetAsyncEnumerator(cancellationToken); } if (state == State.Completed) { return CompletedTasks.False; } awaiter = enumerator.MoveNextAsync().GetAwaiter(); if (awaiter.IsCompleted) { MoveNextCoreDelegate(this); } else { awaiter.SourceOnCompleted(MoveNextCoreDelegate, this); } return new UniTask(this, completionSource.Version); } static void MoveNextCore(object state) { var self = (Enumerator)state; if (self.TryGetResult(self.awaiter, out var result)) { if (result) { self.Current = self.enumerator.Current; self.completionSource.TrySetResult(true); } else { if (self.state == State.RequireAppend) { self.state = State.Completed; self.Current = self.element; self.completionSource.TrySetResult(true); } else { self.state = State.Completed; self.completionSource.TrySetResult(false); } } } } public UniTask DisposeAsync() { if (enumerator != null) { return enumerator.DisposeAsync(); } return default; } } } }