Switch<TSource>
using System.
Reactive.
Disposables;
namespace System.
Reactive.
Linq.
ObservableImpl
{
internal sealed class Switch<
TSource> :
Producer<
TSource,
Switch<
TSource>.
_>
{
internal sealed class _ :
Sink<
IObservable<
TSource>,
TSource>
{
private sealed class InnerObserver :
SafeObserver<
TSource>
{
private readonly _ _parent;
private readonly ulong _id;
public InnerObserver(
_ parent,
ulong id)
{
_parent =
parent;
_id =
id;
}
public override void OnNext(
TSource value)
{
lock (
_parent.
_gate) {
if (
_parent.
_latest ==
_id)
_parent.
ForwardOnNext(
value);
}
}
public override void OnError(
Exception error)
{
lock (
_parent.
_gate) {
Dispose();
if (
_parent.
_latest ==
_id)
_parent.
ForwardOnError(
error);
}
}
public override void OnCompleted()
{
lock (
_parent.
_gate) {
Dispose();
if (
_parent.
_latest ==
_id) {
_parent.
_hasLatest =
false;
if (
_parent.
_isStopped)
_parent.
ForwardOnCompleted();
}
}
}
}
private readonly object _gate =
new object();
private IDisposable _innerSerialDisposable;
private bool _isStopped;
private ulong _latest;
private bool _hasLatest;
public _(
IObserver<
TSource>
observer)
:
base(
observer)
{
}
protected override void Dispose(
bool disposing)
{
if (
disposing)
Disposable.
TryDispose(
ref _innerSerialDisposable);
base.
Dispose(
disposing);
}
public override void OnNext(
IObservable<
TSource>
value)
{
ulong id =
0;
lock (
_gate) {
id = ++
_latest;
_hasLatest =
true;
}
InnerObserver innerObserver =
new InnerObserver(
this,
id);
Disposable.
TrySetSerial(
ref _innerSerialDisposable,
innerObserver);
innerObserver.
SetResource(
ObservableExtensions.
SubscribeSafe<
TSource>(
value, (
IObserver<
TSource>)
innerObserver));
}
public override void OnError(
Exception error)
{
lock (
_gate) {
ForwardOnError(
error);
}
}
public override void OnCompleted()
{
lock (
_gate) {
DisposeUpstream();
_isStopped =
true;
if (!
_hasLatest)
ForwardOnCompleted();
}
}
}
private readonly IObservable<
IObservable<
TSource>>
_sources;
public Switch(
IObservable<
IObservable<
TSource>>
sources)
{
_sources =
sources;
}
protected override _ CreateSink(
IObserver<
TSource>
observer)
{
return new _(
observer);
}
protected override void Run(
_ sink)
{
sink.
Run(
_sources);
}
}
}