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

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

There is a newer version: 1.3.8
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.internal.operators;

import java.util.concurrent.atomic.AtomicReference;

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

/**
 * Sample with the help of another observable.
 * 
 * @see MSDN: Observable.Sample
 * 
 * @param  the source and result value type
 * @param  the element type of the sampler Observable
 */
public final class OperatorSampleWithObservable implements Operator {
    final Observable sampler;
    /** Indicates that no value is available. */
    static final Object EMPTY_TOKEN = new Object();

    public OperatorSampleWithObservable(Observable sampler) {
        this.sampler = sampler;
    }

    @Override
    public Subscriber call(Subscriber child) {
        final SerializedSubscriber s = new SerializedSubscriber(child);
    
        final AtomicReference value = new AtomicReference(EMPTY_TOKEN);
        
        final AtomicReference main = new AtomicReference();
        
        final Subscriber samplerSub = new Subscriber() {
            @Override
            public void onNext(U t) {
                Object localValue = value.getAndSet(EMPTY_TOKEN);
                if (localValue != EMPTY_TOKEN) {
                    @SuppressWarnings("unchecked")
                    T v = (T)localValue;
                    s.onNext(v);
                }
            }

            @Override
            public void onError(Throwable e) {
                s.onError(e);
                // no need to null check, main is assigned before any of the two gets subscribed
                main.get().unsubscribe();
            }

            @Override
            public void onCompleted() {
                onNext(null);
                s.onCompleted();
                // no need to null check, main is assigned before any of the two gets subscribed
                main.get().unsubscribe();
            }
        };
        
        Subscriber result = new Subscriber() {
            @Override
            public void onNext(T t) {
                value.set(t);
            }

            @Override
            public void onError(Throwable e) {
                s.onError(e);
                
                samplerSub.unsubscribe();
            }

            @Override
            public void onCompleted() {
                samplerSub.onNext(null);
                s.onCompleted();

                samplerSub.unsubscribe();
            }
        };
        
        main.lazySet(result);
        
        child.add(result);
        child.add(samplerSub);
        
        sampler.unsafeSubscribe(samplerSub);
        
        return result;
    }
}