using System; using UniRx.Operators; namespace UniRx.Operators { internal class ForEachAsyncObservable : OperatorObservableBase { readonly IObservable source; readonly Action onNext; readonly Action onNextWithIndex; public ForEachAsyncObservable(IObservable source, Action onNext) : base(source.IsRequiredSubscribeOnCurrentThread()) { this.source = source; this.onNext = onNext; } public ForEachAsyncObservable(IObservable source, Action onNext) : base(source.IsRequiredSubscribeOnCurrentThread()) { this.source = source; this.onNextWithIndex = onNext; } protected override IDisposable SubscribeCore(IObserver observer, IDisposable cancel) { if (onNext != null) { return source.Subscribe(new ForEachAsync(this, observer, cancel)); } else { return source.Subscribe(new ForEachAsync_(this, observer, cancel)); } } class ForEachAsync : OperatorObserverBase { readonly ForEachAsyncObservable parent; public ForEachAsync(ForEachAsyncObservable parent, IObserver observer, IDisposable cancel) : base(observer, cancel) { this.parent = parent; } public override void OnNext(T value) { try { parent.onNext(value); } catch (Exception ex) { try { observer.OnError(ex); } finally { Dispose(); } return; } } public override void OnError(Exception error) { try { observer.OnError(error); } finally { Dispose(); } } public override void OnCompleted() { observer.OnNext(Unit.Default); try { observer.OnCompleted(); } finally { Dispose(); } } } // with index class ForEachAsync_ : OperatorObserverBase { readonly ForEachAsyncObservable parent; int index = 0; public ForEachAsync_(ForEachAsyncObservable parent, IObserver observer, IDisposable cancel) : base(observer, cancel) { this.parent = parent; } public override void OnNext(T value) { try { parent.onNextWithIndex(value, index++); } catch (Exception ex) { try { observer.OnError(ex); } finally { Dispose(); } return; } } public override void OnError(Exception error) { try { observer.OnError(error); } finally { Dispose(); } } public override void OnCompleted() { observer.OnNext(Unit.Default); try { observer.OnCompleted(); } finally { Dispose(); } } } } }