using Cysharp.Threading.Tasks.Internal; using System; using System.Threading; namespace Cysharp.Threading.Tasks.Linq { public static partial class UniTaskAsyncEnumerable { public static UniTask LastAsync(this IUniTaskAsyncEnumerable source, CancellationToken cancellationToken = default) { Error.ThrowArgumentNullException(source, nameof(source)); return Last.LastAsync(source, cancellationToken, false); } public static UniTask LastAsync(this IUniTaskAsyncEnumerable source, Func predicate, CancellationToken cancellationToken = default) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(predicate, nameof(predicate)); return Last.LastAsync(source, predicate, cancellationToken, false); } public static UniTask LastAwaitAsync(this IUniTaskAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken = default) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(predicate, nameof(predicate)); return Last.LastAwaitAsync(source, predicate, cancellationToken, false); } public static UniTask LastAwaitWithCancellationAsync(this IUniTaskAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken = default) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(predicate, nameof(predicate)); return Last.LastAwaitWithCancellationAsync(source, predicate, cancellationToken, false); } public static UniTask LastOrDefaultAsync(this IUniTaskAsyncEnumerable source, CancellationToken cancellationToken = default) { Error.ThrowArgumentNullException(source, nameof(source)); return Last.LastAsync(source, cancellationToken, true); } public static UniTask LastOrDefaultAsync(this IUniTaskAsyncEnumerable source, Func predicate, CancellationToken cancellationToken = default) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(predicate, nameof(predicate)); return Last.LastAsync(source, predicate, cancellationToken, true); } public static UniTask LastOrDefaultAwaitAsync(this IUniTaskAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken = default) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(predicate, nameof(predicate)); return Last.LastAwaitAsync(source, predicate, cancellationToken, true); } public static UniTask LastOrDefaultAwaitWithCancellationAsync(this IUniTaskAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken = default) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(predicate, nameof(predicate)); return Last.LastAwaitWithCancellationAsync(source, predicate, cancellationToken, true); } } internal static class Last { public static async UniTask LastAsync(IUniTaskAsyncEnumerable source, CancellationToken cancellationToken, bool defaultIfEmpty) { var e = source.GetAsyncEnumerator(cancellationToken); try { TSource value = default; if (await e.MoveNextAsync()) { value = e.Current; } else { if (defaultIfEmpty) { return value; } else { throw Error.NoElements(); } } while (await e.MoveNextAsync()) { value = e.Current; } return value; } finally { if (e != null) { await e.DisposeAsync(); } } } public static async UniTask LastAsync(IUniTaskAsyncEnumerable source, Func predicate, CancellationToken cancellationToken, bool defaultIfEmpty) { var e = source.GetAsyncEnumerator(cancellationToken); try { TSource value = default; bool found = false; while (await e.MoveNextAsync()) { var v = e.Current; if (predicate(v)) { found = true; value = v; } } if (defaultIfEmpty) { return value; } else { if (found) { return value; } else { throw Error.NoElements(); } } } finally { if (e != null) { await e.DisposeAsync(); } } } public static async UniTask LastAwaitAsync(IUniTaskAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken, bool defaultIfEmpty) { var e = source.GetAsyncEnumerator(cancellationToken); try { TSource value = default; bool found = false; while (await e.MoveNextAsync()) { var v = e.Current; if (await predicate(v)) { found = true; value = v; } } if (defaultIfEmpty) { return value; } else { if (found) { return value; } else { throw Error.NoElements(); } } } finally { if (e != null) { await e.DisposeAsync(); } } } public static async UniTask LastAwaitWithCancellationAsync(IUniTaskAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken, bool defaultIfEmpty) { var e = source.GetAsyncEnumerator(cancellationToken); try { TSource value = default; bool found = false; while (await e.MoveNextAsync()) { var v = e.Current; if (await predicate(v, cancellationToken)) { found = true; value = v; } } if (defaultIfEmpty) { return value; } else { if (found) { return value; } else { throw Error.NoElements(); } } } finally { if (e != null) { await e.DisposeAsync(); } } } } }