You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
207 lines
6.8 KiB
207 lines
6.8 KiB
1 year ago
|
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(); }
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|