using System; namespace UniRx.Operators { internal class WhereObservable : OperatorObservableBase { readonly IObservable source; readonly Func predicate; readonly Func predicateWithIndex; public WhereObservable(IObservable source, Func predicate) : base(source.IsRequiredSubscribeOnCurrentThread()) { this.source = source; this.predicate = predicate; } public WhereObservable(IObservable source, Func predicateWithIndex) : base(source.IsRequiredSubscribeOnCurrentThread()) { this.source = source; this.predicateWithIndex = predicateWithIndex; } // Optimize for .Where().Where() public IObservable CombinePredicate(Func combinePredicate) { if (this.predicate != null) { return new WhereObservable(source, x => this.predicate(x) && combinePredicate(x)); } else { return new WhereObservable(this, combinePredicate); } } // Optimize for .Where().Select() public IObservable CombineSelector(Func selector) { if (this.predicate != null) { return new WhereSelectObservable(source, predicate, selector); } else { return new SelectObservable(this, selector); // can't combine } } protected override IDisposable SubscribeCore(IObserver observer, IDisposable cancel) { if (predicate != null) { return source.Subscribe(new Where(this, observer, cancel)); } else { return source.Subscribe(new Where_(this, observer, cancel)); } } class Where : OperatorObserverBase { readonly WhereObservable parent; public Where(WhereObservable parent, IObserver observer, IDisposable cancel) : base(observer, cancel) { this.parent = parent; } public override void OnNext(T value) { var isPassed = false; try { isPassed = parent.predicate(value); } catch (Exception ex) { try { observer.OnError(ex); } finally { Dispose(); } return; } if (isPassed) { observer.OnNext(value); } } public override void OnError(Exception error) { try { observer.OnError(error); } finally { Dispose(); } } public override void OnCompleted() { try { observer.OnCompleted(); } finally { Dispose(); } } } class Where_ : OperatorObserverBase { readonly WhereObservable parent; int index; public Where_(WhereObservable parent, IObserver observer, IDisposable cancel) : base(observer, cancel) { this.parent = parent; this.index = 0; } public override void OnNext(T value) { var isPassed = false; try { isPassed = parent.predicateWithIndex(value, index++); } catch (Exception ex) { try { observer.OnError(ex); } finally { Dispose(); } return; } if (isPassed) { observer.OnNext(value); } } public override void OnError(Exception error) { try { observer.OnError(error); } finally { Dispose(); } } public override void OnCompleted() { try { observer.OnCompleted(); } finally { Dispose(); } } } } }