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

rx.internal.operators.OperatorWithLatestFrom Maven / Gradle / Ivy

/**
 * 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.internal.operators;

import java.util.concurrent.atomic.AtomicReference;

import rx.*;
import rx.Observable.Operator;
import rx.functions.Func2;
import rx.observers.SerializedSubscriber;

/**
 * Combines values from two sources only when the main source emits.
 * @param  the element type of the main observable
 * @param  the element type of the other observable that is merged into the main
 * @param  the result element type
 */
public final class OperatorWithLatestFrom implements Operator  {
    final Func2 resultSelector;
    final Observable other;
    /** Indicates the other has not yet emitted a value. */
    static final Object EMPTY = new Object();
    
    public OperatorWithLatestFrom(Observable other, Func2 resultSelector) {
        this.other = other;
        this.resultSelector = resultSelector;
    }
    @Override
    public Subscriber call(Subscriber child) {
        // onError and onCompleted may happen either from the main or from other.
        final SerializedSubscriber s = new SerializedSubscriber(child, false);
        child.add(s);
        
        final AtomicReference current = new AtomicReference(EMPTY);
        
        final Subscriber subscriber = new Subscriber(s, true) {
            @Override
            public void onNext(T t) {
                Object o = current.get();
                if (o != EMPTY) {
                    try {
                        @SuppressWarnings("unchecked")
                        U u = (U)o;
                        R result = resultSelector.call(t, u);
                        
                        s.onNext(result);
                    } catch (Throwable e) {
                        onError(e);
                        return;
                    }
                }
            }
            @Override
            public void onError(Throwable e) {
                s.onError(e);
                s.unsubscribe();
            }
            @Override
            public void onCompleted() {
                s.onCompleted();
                s.unsubscribe();
            }
        };
        
        Subscriber otherSubscriber = new Subscriber() {
            @Override
            public void onNext(U t) {
                current.set(t);
            }
            @Override
            public void onError(Throwable e) {
                s.onError(e);
                s.unsubscribe();
            }
            @Override
            public void onCompleted() {
                if (current.get() == EMPTY) {
                    s.onCompleted();
                    s.unsubscribe();
                }
            }
        };
        s.add(subscriber);
        s.add(otherSubscriber);
        
        other.unsafeSubscribe(otherSubscriber);
        
        return subscriber;
    }
}