using System; using UniRx.Operators; namespace UniRx.Operators { internal class AggregateObservable<TSource> : OperatorObservableBase<TSource> { readonly IObservable<TSource> source; readonly Func<TSource, TSource, TSource> accumulator; public AggregateObservable(IObservable<TSource> source, Func<TSource, TSource, TSource> accumulator) : base(source.IsRequiredSubscribeOnCurrentThread()) { this.source = source; this.accumulator = accumulator; } protected override IDisposable SubscribeCore(IObserver<TSource> observer, IDisposable cancel) { return source.Subscribe(new Aggregate(this, observer, cancel)); } class Aggregate : OperatorObserverBase<TSource, TSource> { readonly AggregateObservable<TSource> parent; TSource accumulation; bool seenValue; public Aggregate(AggregateObservable<TSource> parent, IObserver<TSource> observer, IDisposable cancel) : base(observer, cancel) { this.parent = parent; this.seenValue = false; } public override void OnNext(TSource value) { if (!seenValue) { seenValue = true; accumulation = value; } else { try { accumulation = parent.accumulator(accumulation, 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() { if (!seenValue) { throw new InvalidOperationException("Sequence contains no elements."); } observer.OnNext(accumulation); try { observer.OnCompleted(); } finally { Dispose(); } } } } internal class AggregateObservable<TSource, TAccumulate> : OperatorObservableBase<TAccumulate> { readonly IObservable<TSource> source; readonly TAccumulate seed; readonly Func<TAccumulate, TSource, TAccumulate> accumulator; public AggregateObservable(IObservable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> accumulator) : base(source.IsRequiredSubscribeOnCurrentThread()) { this.source = source; this.seed = seed; this.accumulator = accumulator; } protected override IDisposable SubscribeCore(IObserver<TAccumulate> observer, IDisposable cancel) { return source.Subscribe(new Aggregate(this, observer, cancel)); } class Aggregate : OperatorObserverBase<TSource, TAccumulate> { readonly AggregateObservable<TSource, TAccumulate> parent; TAccumulate accumulation; public Aggregate(AggregateObservable<TSource, TAccumulate> parent, IObserver<TAccumulate> observer, IDisposable cancel) : base(observer, cancel) { this.parent = parent; this.accumulation = parent.seed; } public override void OnNext(TSource value) { try { accumulation = parent.accumulator(accumulation, 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(accumulation); try { observer.OnCompleted(); } finally { Dispose(); } } } } internal class AggregateObservable<TSource, TAccumulate, TResult> : OperatorObservableBase<TResult> { readonly IObservable<TSource> source; readonly TAccumulate seed; readonly Func<TAccumulate, TSource, TAccumulate> accumulator; readonly Func<TAccumulate, TResult> resultSelector; public AggregateObservable(IObservable<TSource> source, TAccumulate seed, Func<TAccumulate, TSource, TAccumulate> accumulator, Func<TAccumulate, TResult> resultSelector) : base(source.IsRequiredSubscribeOnCurrentThread()) { this.source = source; this.seed = seed; this.accumulator = accumulator; this.resultSelector = resultSelector; } protected override IDisposable SubscribeCore(IObserver<TResult> observer, IDisposable cancel) { return source.Subscribe(new Aggregate(this, observer, cancel)); } class Aggregate : OperatorObserverBase<TSource, TResult> { readonly AggregateObservable<TSource, TAccumulate, TResult> parent; TAccumulate accumulation; public Aggregate(AggregateObservable<TSource, TAccumulate, TResult> parent, IObserver<TResult> observer, IDisposable cancel) : base(observer, cancel) { this.parent = parent; this.accumulation = parent.seed; } public override void OnNext(TSource value) { try { accumulation = parent.accumulator(accumulation, 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() { TResult result; try { result = parent.resultSelector(accumulation); } catch (Exception ex) { OnError(ex); return; } observer.OnNext(result); try { observer.OnCompleted(); } finally { Dispose(); } } } } }