// Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the Apache 2.0 License. // See the LICENSE file in the project root for more information. namespace System.Reactive.Linq.ObservableImpl { internal sealed class LastAsync : Producer { private readonly IObservable _source; private readonly Func _predicate; private readonly bool _throwOnEmpty; public LastAsync(IObservable source, Func predicate, bool throwOnEmpty) { _source = source; _predicate = predicate; _throwOnEmpty = throwOnEmpty; } protected override IDisposable Run(IObserver observer, IDisposable cancel, Action setSink) { if (_predicate != null) { var sink = new LastAsyncImpl(this, observer, cancel); setSink(sink); return _source.SubscribeSafe(sink); } else { var sink = new _(this, observer, cancel); setSink(sink); return _source.SubscribeSafe(sink); } } class _ : Sink, IObserver { private readonly LastAsync _parent; private TSource _value; private bool _seenValue; public _(LastAsync parent, IObserver observer, IDisposable cancel) : base(observer, cancel) { _parent = parent; _value = default(TSource); _seenValue = false; } public void OnNext(TSource value) { _value = value; _seenValue = true; } public void OnError(Exception error) { base._observer.OnError(error); base.Dispose(); } public void OnCompleted() { if (!_seenValue && _parent._throwOnEmpty) { base._observer.OnError(new InvalidOperationException(Strings_Linq.NO_ELEMENTS)); } else { base._observer.OnNext(_value); base._observer.OnCompleted(); } base.Dispose(); } } class LastAsyncImpl : Sink, IObserver { private readonly LastAsync _parent; private TSource _value; private bool _seenValue; public LastAsyncImpl(LastAsync parent, IObserver observer, IDisposable cancel) : base(observer, cancel) { _parent = parent; _value = default(TSource); _seenValue = false; } public void OnNext(TSource value) { var b = false; try { b = _parent._predicate(value); } catch (Exception ex) { base._observer.OnError(ex); base.Dispose(); return; } if (b) { _value = value; _seenValue = true; } } public void OnError(Exception error) { base._observer.OnError(error); base.Dispose(); } public void OnCompleted() { if (!_seenValue && _parent._throwOnEmpty) { base._observer.OnError(new InvalidOperationException(Strings_Linq.NO_MATCHING_ELEMENTS)); } else { base._observer.OnNext(_value); base._observer.OnCompleted(); } base.Dispose(); } } } }