using System; using UniRx.Operators; namespace UniRx.Operators { internal class TakeObservable : OperatorObservableBase { readonly IObservable source; readonly int count; readonly TimeSpan duration; internal readonly IScheduler scheduler; // public for optimization check public TakeObservable(IObservable source, int count) : base(source.IsRequiredSubscribeOnCurrentThread()) { this.source = source; this.count = count; } public TakeObservable(IObservable source, TimeSpan duration, IScheduler scheduler) : base(scheduler == Scheduler.CurrentThread || source.IsRequiredSubscribeOnCurrentThread()) { this.source = source; this.duration = duration; this.scheduler = scheduler; } // optimize combiner public IObservable 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(source, count); } public IObservable 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(source, duration, scheduler); } protected override IDisposable SubscribeCore(IObserver 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 { int rest; public Take(TakeObservable parent, IObserver 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 { readonly TakeObservable parent; readonly object gate = new object(); public Take_(TakeObservable parent, IObserver 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(); }; } } } } }