123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150 |
- using System;
- using UniRx.Operators;
- namespace UniRx.Operators
- {
- internal class TakeObservable<T> : OperatorObservableBase<T>
- {
- readonly IObservable<T> source;
- readonly int count;
- readonly TimeSpan duration;
- internal readonly IScheduler scheduler; // public for optimization check
- public TakeObservable(IObservable<T> source, int count)
- : base(source.IsRequiredSubscribeOnCurrentThread())
- {
- this.source = source;
- this.count = count;
- }
- public TakeObservable(IObservable<T> source, TimeSpan duration, IScheduler scheduler)
- : base(scheduler == Scheduler.CurrentThread || source.IsRequiredSubscribeOnCurrentThread())
- {
- this.source = source;
- this.duration = duration;
- this.scheduler = scheduler;
- }
- // optimize combiner
- public IObservable<T> Combine(int count)
- {
- // xs = 6
- // xs.Take(5) = 5 | xs.Take(3) = 3
- // xs.Take(5).Take(3) = 3 | xs.Take(3).Take(5) = 3
- // use minimum one
- return (this.count <= count)
- ? this
- : new TakeObservable<T>(source, count);
- }
- public IObservable<T> Combine(TimeSpan duration)
- {
- // xs = 6s
- // xs.Take(5s) = 5s | xs.Take(3s) = 3s
- // xs.Take(5s).Take(3s) = 3s | xs.Take(3s).Take(5s) = 3s
- // use minimum one
- return (this.duration <= duration)
- ? this
- : new TakeObservable<T>(source, duration, scheduler);
- }
- protected override IDisposable SubscribeCore(IObserver<T> observer, IDisposable cancel)
- {
- if (scheduler == null)
- {
- return source.Subscribe(new Take(this, observer, cancel));
- }
- else
- {
- return new Take_(this, observer, cancel).Run();
- }
- }
- class Take : OperatorObserverBase<T, T>
- {
- int rest;
- public Take(TakeObservable<T> parent, IObserver<T> observer, IDisposable cancel) : base(observer, cancel)
- {
- this.rest = parent.count;
- }
- public override void OnNext(T value)
- {
- if (rest > 0)
- {
- rest -= 1;
- base.observer.OnNext(value);
- if (rest == 0)
- {
- try { observer.OnCompleted(); } finally { Dispose(); };
- }
- }
- }
- public override void OnError(Exception error)
- {
- try { observer.OnError(error); } finally { Dispose(); }
- }
- public override void OnCompleted()
- {
- try { observer.OnCompleted(); } finally { Dispose(); }
- }
- }
- class Take_ : OperatorObserverBase<T, T>
- {
- readonly TakeObservable<T> parent;
- readonly object gate = new object();
- public Take_(TakeObservable<T> parent, IObserver<T> observer, IDisposable cancel) : base(observer, cancel)
- {
- this.parent = parent;
- }
- public IDisposable Run()
- {
- var d1 = parent.scheduler.Schedule(parent.duration, Tick);
- var d2 = parent.source.Subscribe(this);
- return StableCompositeDisposable.Create(d1, d2);
- }
- void Tick()
- {
- lock (gate)
- {
- try { observer.OnCompleted(); } finally { Dispose(); };
- }
- }
- public override void OnNext(T value)
- {
- lock (gate)
- {
- base.observer.OnNext(value);
- }
- }
- public override void OnError(Exception error)
- {
- lock (gate)
- {
- try { observer.OnError(error); } finally { Dispose(); };
- }
- }
- public override void OnCompleted()
- {
- lock (gate)
- {
- try { observer.OnCompleted(); } finally { Dispose(); };
- }
- }
- }
- }
- }
|