All Downloads are FREE. Search and download functionalities are using the official Maven repository.

rx.operators.OperationTakeWhile Maven / Gradle / Ivy

There is a newer version: 0.20.7
Show newest version
/**
 * Copyright 2014 Netflix, Inc.
 * 
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 * http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package rx.operators;

import java.util.concurrent.atomic.AtomicInteger;

import rx.Observable;
import rx.Observable.OnSubscribeFunc;
import rx.Observer;
import rx.Subscription;
import rx.functions.Func1;
import rx.functions.Func2;

/**
 * Returns an Observable that emits items emitted by the source Observable as long as a specified
 * condition is true.
 * 

* */ public final class OperationTakeWhile { /** * Returns values from an observable sequence as long as a specified condition is true, and then skips the remaining values. * * @param items * @param predicate * a function to test each source element for a condition * @return sequence of observable values from the start as long as the predicate is true */ public static OnSubscribeFunc takeWhile(final Observable items, final Func1 predicate) { return takeWhileWithIndex(items, OperationTakeWhile. skipIndex(predicate)); } /** * Returns values from an observable sequence as long as a specified condition is true, and then skips the remaining values. * * @param items * @param predicate * a function to test each element for a condition; the second parameter of the function represents the index of the source element; otherwise, false. * @return sequence of observable values from the start as long as the predicate is true */ public static OnSubscribeFunc takeWhileWithIndex(final Observable items, final Func2 predicate) { // wrap in a Func so that if a chain is built up, then asynchronously subscribed to twice we will have 2 instances of Take rather than 1 handing both, which is not thread-safe. return new OnSubscribeFunc() { @Override public Subscription onSubscribe(Observer observer) { return new TakeWhile(items, predicate).onSubscribe(observer); } }; } private static Func2 skipIndex(final Func1 underlying) { return new Func2() { @Override public Boolean call(T input, Integer index) { return underlying.call(input); } }; } /** * This class is NOT thread-safe if invoked and referenced multiple times. In other words, don't subscribe to it multiple times from different threads. *

* It IS thread-safe from within it while receiving onNext events from multiple threads. *

* This should all be fine as long as it's kept as a private class and a new instance created from static factory method above. *

* Note how the takeWhileWithIndex() factory method above protects us from a single instance being exposed with the Observable wrapper handling the subscribe flow. * * @param */ private static class TakeWhile implements OnSubscribeFunc { private final Observable items; private final Func2 predicate; private final SafeObservableSubscription subscription = new SafeObservableSubscription(); private TakeWhile(Observable items, Func2 predicate) { this.items = items; this.predicate = predicate; } @Override public Subscription onSubscribe(Observer observer) { return subscription.wrap(items.subscribe(new ItemObserver(observer))); } private class ItemObserver implements Observer { private final Observer observer; private final AtomicInteger counter = new AtomicInteger(); public ItemObserver(Observer observer) { // Using AtomicObserver because the unsubscribe, onCompleted, onError and error handling behavior // needs "isFinished" logic to not send duplicated events // The 'testTakeWhile1' and 'testTakeWhile2' tests fail without this. this.observer = new SafeObserver(subscription, observer); } @Override public void onCompleted() { observer.onCompleted(); } @Override public void onError(Throwable e) { observer.onError(e); } @Override public void onNext(T args) { Boolean isSelected; try { isSelected = predicate.call(args, counter.getAndIncrement()); } catch (Throwable e) { observer.onError(e); return; } if (isSelected) { observer.onNext(args); } else { observer.onCompleted(); // this will work if the sequence is asynchronous, it will have no effect on a synchronous observable subscription.unsubscribe(); } } } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy