/** * Copyright (c) 2016-present, RxJava Contributors. * * 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 io.reactivex.internal.operators.observable; import java.util.concurrent.atomic.*; import io.reactivex.*; import io.reactivex.disposables.Disposable; import io.reactivex.internal.disposables.DisposableHelper; import io.reactivex.observers.SerializedObserver; public final class ObservableSampleWithObservable<T> extends AbstractObservableWithUpstream<T, T> { final ObservableSource<?> other; final boolean emitLast; public ObservableSampleWithObservable(ObservableSource<T> source, ObservableSource<?> other, boolean emitLast) { super(source); this.other = other; this.emitLast = emitLast; } @Override public void subscribeActual(Observer<? super T> t) { SerializedObserver<T> serial = new SerializedObserver<T>(t); if (emitLast) { source.subscribe(new SampleMainEmitLast<T>(serial, other)); } else { source.subscribe(new SampleMainNoLast<T>(serial, other)); } } abstract static class SampleMainObserver<T> extends AtomicReference<T> implements Observer<T>, Disposable { private static final long serialVersionUID = -3517602651313910099L; final Observer<? super T> actual; final ObservableSource<?> sampler; final AtomicReference<Disposable> other = new AtomicReference<Disposable>(); Disposable s; SampleMainObserver(Observer<? super T> actual, ObservableSource<?> other) { this.actual = actual; this.sampler = other; } @Override public void onSubscribe(Disposable s) { if (DisposableHelper.validate(this.s, s)) { this.s = s; actual.onSubscribe(this); if (other.get() == null) { sampler.subscribe(new SamplerObserver<T>(this)); } } } @Override public void onNext(T t) { lazySet(t); } @Override public void onError(Throwable t) { DisposableHelper.dispose(other); actual.onError(t); } @Override public void onComplete() { DisposableHelper.dispose(other); completeMain(); } boolean setOther(Disposable o) { return DisposableHelper.setOnce(other, o); } @Override public void dispose() { DisposableHelper.dispose(other); s.dispose(); } @Override public boolean isDisposed() { return other.get() == DisposableHelper.DISPOSED; } public void error(Throwable e) { s.dispose(); actual.onError(e); } public void complete() { s.dispose(); completeOther(); } void emit() { T value = getAndSet(null); if (value != null) { actual.onNext(value); } } abstract void completeMain(); abstract void completeOther(); abstract void run(); } static final class SamplerObserver<T> implements Observer<Object> { final SampleMainObserver<T> parent; SamplerObserver(SampleMainObserver<T> parent) { this.parent = parent; } @Override public void onSubscribe(Disposable s) { parent.setOther(s); } @Override public void onNext(Object t) { parent.run(); } @Override public void onError(Throwable t) { parent.error(t); } @Override public void onComplete() { parent.complete(); } } static final class SampleMainNoLast<T> extends SampleMainObserver<T> { private static final long serialVersionUID = -3029755663834015785L; SampleMainNoLast(Observer<? super T> actual, ObservableSource<?> other) { super(actual, other); } @Override void completeMain() { actual.onComplete(); } @Override void completeOther() { actual.onComplete(); } @Override void run() { emit(); } } static final class SampleMainEmitLast<T> extends SampleMainObserver<T> { private static final long serialVersionUID = -3029755663834015785L; final AtomicInteger wip; volatile boolean done; SampleMainEmitLast(Observer<? super T> actual, ObservableSource<?> other) { super(actual, other); this.wip = new AtomicInteger(); } @Override void completeMain() { done = true; if (wip.getAndIncrement() == 0) { emit(); actual.onComplete(); } } @Override void completeOther() { done = true; if (wip.getAndIncrement() == 0) { emit(); actual.onComplete(); } } @Override void run() { if (wip.getAndIncrement() == 0) { do { boolean d = done; emit(); if (d) { actual.onComplete(); return; } } while (wip.decrementAndGet() != 0); } } } }