200 lines
6.0 KiB
C#
200 lines
6.0 KiB
C#
// Licensed to the .NET Foundation under one or more agreements.
|
|
// The .NET Foundation licenses this file to you under the MIT License.
|
|
// See the LICENSE file in the project root for more information.
|
|
|
|
using System.Reactive.Concurrency;
|
|
using System.Reactive.Disposables;
|
|
|
|
namespace System.Reactive.Linq.ObservableImpl
|
|
{
|
|
internal sealed class SkipUntil<TSource, TOther> : Producer<TSource, SkipUntil<TSource, TOther>._>
|
|
{
|
|
private readonly IObservable<TSource> _source;
|
|
private readonly IObservable<TOther> _other;
|
|
|
|
public SkipUntil(IObservable<TSource> source, IObservable<TOther> other)
|
|
{
|
|
_source = source;
|
|
_other = other;
|
|
}
|
|
|
|
protected override _ CreateSink(IObserver<TSource> observer) => new(observer);
|
|
|
|
protected override void Run(_ sink) => sink.Run(this);
|
|
|
|
internal sealed class _ : IdentitySink<TSource>
|
|
{
|
|
private SingleAssignmentDisposableValue _otherDisposable;
|
|
private bool _forward;
|
|
private int _halfSerializer;
|
|
private Exception? _error;
|
|
|
|
public _(IObserver<TSource> observer)
|
|
: base(observer)
|
|
{
|
|
}
|
|
|
|
public void Run(SkipUntil<TSource, TOther> parent)
|
|
{
|
|
_otherDisposable.Disposable = parent._other.Subscribe(new OtherObserver(this));
|
|
Run(parent._source);
|
|
}
|
|
|
|
protected override void Dispose(bool disposing)
|
|
{
|
|
if (disposing)
|
|
{
|
|
if (!_otherDisposable.IsDisposed)
|
|
{
|
|
_otherDisposable.Dispose();
|
|
}
|
|
}
|
|
|
|
base.Dispose(disposing);
|
|
}
|
|
|
|
public override void OnNext(TSource value)
|
|
{
|
|
if (_forward)
|
|
{
|
|
HalfSerializer.ForwardOnNext(this, value, ref _halfSerializer, ref _error);
|
|
}
|
|
}
|
|
|
|
public override void OnError(Exception ex)
|
|
{
|
|
HalfSerializer.ForwardOnError(this, ex, ref _halfSerializer, ref _error);
|
|
}
|
|
|
|
public override void OnCompleted()
|
|
{
|
|
if (_forward)
|
|
{
|
|
HalfSerializer.ForwardOnCompleted(this, ref _halfSerializer, ref _error);
|
|
}
|
|
else
|
|
{
|
|
DisposeUpstream();
|
|
}
|
|
}
|
|
|
|
private void OtherComplete()
|
|
{
|
|
_forward = true;
|
|
}
|
|
|
|
private sealed class OtherObserver : IObserver<TOther>, IDisposable
|
|
{
|
|
private readonly _ _parent;
|
|
|
|
public OtherObserver(_ parent)
|
|
{
|
|
_parent = parent;
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
if (!_parent._otherDisposable.IsDisposed)
|
|
{
|
|
_parent._otherDisposable.Dispose();
|
|
}
|
|
}
|
|
|
|
public void OnCompleted()
|
|
{
|
|
Dispose();
|
|
}
|
|
|
|
public void OnError(Exception error)
|
|
{
|
|
HalfSerializer.ForwardOnError(_parent, error, ref _parent._halfSerializer, ref _parent._error);
|
|
}
|
|
|
|
public void OnNext(TOther value)
|
|
{
|
|
_parent.OtherComplete();
|
|
Dispose();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
internal sealed class SkipUntil<TSource> : Producer<TSource, SkipUntil<TSource>._>
|
|
{
|
|
private readonly IObservable<TSource> _source;
|
|
private readonly DateTimeOffset _startTime;
|
|
internal readonly IScheduler _scheduler;
|
|
|
|
public SkipUntil(IObservable<TSource> source, DateTimeOffset startTime, IScheduler scheduler)
|
|
{
|
|
_source = source;
|
|
_startTime = startTime;
|
|
_scheduler = scheduler;
|
|
}
|
|
|
|
public IObservable<TSource> Combine(DateTimeOffset startTime)
|
|
{
|
|
//
|
|
// Maximum semantics:
|
|
//
|
|
// t 0--1--2--3--4--5--6--7-> t 0--1--2--3--4--5--6--7->
|
|
//
|
|
// xs --o--o--o--o--o--o--| xs --o--o--o--o--o--o--|
|
|
// xs.SU(5AM) xxxxxxxxxxxxxxxx-o--| xs.SU(3AM) xxxxxxxxxx-o--o--o--|
|
|
// xs.SU(5AM).SU(3AM) xxxxxxxxx--------o--| xs.SU(3AM).SU(5AM) xxxxxxxxxxxxxxxx-o--|
|
|
//
|
|
if (startTime <= _startTime)
|
|
{
|
|
return this;
|
|
}
|
|
|
|
return new SkipUntil<TSource>(_source, startTime, _scheduler);
|
|
}
|
|
|
|
protected override _ CreateSink(IObserver<TSource> observer) => new(observer);
|
|
|
|
protected override void Run(_ sink) => sink.Run(this);
|
|
|
|
internal sealed class _ : IdentitySink<TSource>
|
|
{
|
|
private bool _open;
|
|
|
|
public _(IObserver<TSource> observer)
|
|
: base(observer)
|
|
{
|
|
}
|
|
|
|
private SingleAssignmentDisposableValue _task;
|
|
|
|
public void Run(SkipUntil<TSource> parent)
|
|
{
|
|
_task.Disposable = parent._scheduler.ScheduleAction(this, parent._startTime, static state => state.Tick());
|
|
Run(parent._source);
|
|
}
|
|
|
|
protected override void Dispose(bool disposing)
|
|
{
|
|
if (disposing)
|
|
{
|
|
_task.Dispose();
|
|
}
|
|
|
|
base.Dispose(disposing);
|
|
}
|
|
|
|
private void Tick()
|
|
{
|
|
_open = true;
|
|
}
|
|
|
|
public override void OnNext(TSource value)
|
|
{
|
|
if (_open)
|
|
{
|
|
ForwardOnNext(value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|