Throttle<TSource>
using System.
Reactive.
Concurrency;
using System.
Reactive.
Disposables;
namespace System.
Reactive.
Linq.
ObservableImpl
{
internal sealed class Throttle<
TSource> :
Producer<
TSource,
Throttle<
TSource>.
_>
{
internal sealed class _ :
IdentitySink<
TSource>
{
private readonly TimeSpan _dueTime;
private readonly IScheduler _scheduler;
private readonly object _gate =
new object();
private TSource _value;
private bool _hasValue;
private IDisposable _serialCancelable;
private ulong _id;
public _(
Throttle<
TSource>
parent,
IObserver<
TSource>
observer)
:
base(
observer)
{
_dueTime =
parent.
_dueTime;
_scheduler =
parent.
_scheduler;
}
protected override void Dispose(
bool disposing)
{
if (
disposing)
Disposable.
TryDispose(
ref _serialCancelable);
base.
Dispose(
disposing);
}
public override void OnNext(
TSource value)
{
ulong item =
0;
lock (
_gate) {
_hasValue =
true;
_value =
value;
_id++;
item =
_id;
}
Disposable.
TrySetSerial(
ref _serialCancelable,
null);
Disposable.
TrySetSerial(
ref _serialCancelable,
Scheduler.
ScheduleAction<(
_,
ulong)>(
_scheduler, (
this,
item),
_dueTime, (
Action<(
_,
ulong)>)
delegate((
_ this,
ulong currentid)
tuple) {
tuple.
this.
Propagate(
tuple.
currentid);
}));
}
private void Propagate(
ulong currentid)
{
lock (
_gate) {
if (
_hasValue &&
_id ==
currentid)
ForwardOnNext(
_value);
_hasValue =
false;
}
}
public override void OnError(
Exception error)
{
Disposable.
TryDispose(
ref _serialCancelable);
lock (
_gate) {
ForwardOnError(
error);
_hasValue =
false;
_id++;
}
}
public override void OnCompleted()
{
Disposable.
TryDispose(
ref _serialCancelable);
lock (
_gate) {
if (
_hasValue)
ForwardOnNext(
_value);
ForwardOnCompleted();
_hasValue =
false;
_id++;
}
}
}
private readonly IObservable<
TSource>
_source;
private readonly TimeSpan _dueTime;
private readonly IScheduler _scheduler;
public Throttle(
IObservable<
TSource>
source,
TimeSpan dueTime,
IScheduler scheduler)
{
_source =
source;
_dueTime =
dueTime;
_scheduler =
scheduler;
}
protected override _ CreateSink(
IObserver<
TSource>
observer)
{
return new _(
this,
observer);
}
protected override void Run(
_ sink)
{
sink.
Run(
_source);
}
}
}