// 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.Collections.Generic; namespace System.Reactive.Linq.ObservableImpl { internal sealed class Min : Producer._> { private readonly IObservable _source; private readonly IComparer _comparer; public Min(IObservable source, IComparer comparer) { _source = source; _comparer = comparer; } protected override _ CreateSink(IObserver observer) => default(TSource) == null ? (_)new Null(_comparer, observer) : new NonNull(_comparer, observer); protected override void Run(_ sink) => sink.Run(_source); internal abstract class _ : IdentitySink { protected readonly IComparer _comparer; protected _(IComparer comparer, IObserver observer) : base(observer) { _comparer = comparer; } } private sealed class NonNull : _ { private bool _hasValue; private TSource? _lastValue; public NonNull(IComparer comparer, IObserver observer) : base(comparer, observer) { } public override void OnNext(TSource value) { if (_hasValue) { int comparison; try { comparison = _comparer.Compare(value, _lastValue!); } catch (Exception ex) { ForwardOnError(ex); return; } if (comparison < 0) { _lastValue = value; } } else { _hasValue = true; _lastValue = value; } } public override void OnError(Exception error) { ForwardOnError(error); } public override void OnCompleted() { if (!_hasValue) { try { throw new InvalidOperationException(Strings_Linq.NO_ELEMENTS); } catch (Exception e) { ForwardOnError(e); } } else { ForwardOnNext(_lastValue!); ForwardOnCompleted(); } } } private sealed class Null : _ { private TSource? _lastValue; public Null(IComparer comparer, IObserver observer) : base(comparer, observer) { } public override void OnNext(TSource value) { if (value != null) { if (_lastValue == null) { _lastValue = value; } else { int comparison; try { comparison = _comparer.Compare(value, _lastValue); } catch (Exception ex) { ForwardOnError(ex); return; } if (comparison < 0) { _lastValue = value; } } } } public override void OnCompleted() { ForwardOnNext(_lastValue!); ForwardOnCompleted(); } } } internal sealed class MinDouble : Producer { private readonly IObservable _source; public MinDouble(IObservable source) { _source = source; } protected override _ CreateSink(IObserver observer) => new(observer); protected override void Run(_ sink) => sink.Run(_source); internal sealed class _ : IdentitySink { private bool _hasValue; private double _lastValue; public _(IObserver observer) : base(observer) { } public override void OnNext(double value) { if (_hasValue) { if (value < _lastValue || double.IsNaN(value)) { _lastValue = value; } } else { _lastValue = value; _hasValue = true; } } public override void OnCompleted() { if (!_hasValue) { try { throw new InvalidOperationException(Strings_Linq.NO_ELEMENTS); } catch (Exception e) { ForwardOnError(e); } } else { ForwardOnNext(_lastValue); ForwardOnCompleted(); } } } } internal sealed class MinSingle : Producer { private readonly IObservable _source; public MinSingle(IObservable source) { _source = source; } protected override _ CreateSink(IObserver observer) => new(observer); protected override void Run(_ sink) => sink.Run(_source); internal sealed class _ : IdentitySink { private bool _hasValue; private float _lastValue; public _(IObserver observer) : base(observer) { } public override void OnNext(float value) { if (_hasValue) { if (value < _lastValue || float.IsNaN(value)) { _lastValue = value; } } else { _lastValue = value; _hasValue = true; } } public override void OnCompleted() { if (!_hasValue) { try { throw new InvalidOperationException(Strings_Linq.NO_ELEMENTS); } catch (Exception e) { ForwardOnError(e); } } else { ForwardOnNext(_lastValue); ForwardOnCompleted(); } } } } internal sealed class MinDecimal : Producer { private readonly IObservable _source; public MinDecimal(IObservable source) { _source = source; } protected override _ CreateSink(IObserver observer) => new(observer); protected override void Run(_ sink) => sink.Run(_source); internal sealed class _ : IdentitySink { private bool _hasValue; private decimal _lastValue; public _(IObserver observer) : base(observer) { } public override void OnNext(decimal value) { if (_hasValue) { if (value < _lastValue) { _lastValue = value; } } else { _lastValue = value; _hasValue = true; } } public override void OnCompleted() { if (!_hasValue) { try { throw new InvalidOperationException(Strings_Linq.NO_ELEMENTS); } catch (Exception e) { ForwardOnError(e); } } else { ForwardOnNext(_lastValue); ForwardOnCompleted(); } } } } internal sealed class MinInt32 : Producer { private readonly IObservable _source; public MinInt32(IObservable source) { _source = source; } protected override _ CreateSink(IObserver observer) => new(observer); protected override void Run(_ sink) => sink.Run(_source); internal sealed class _ : IdentitySink { private bool _hasValue; private int _lastValue; public _(IObserver observer) : base(observer) { } public override void OnNext(int value) { if (_hasValue) { if (value < _lastValue) { _lastValue = value; } } else { _lastValue = value; _hasValue = true; } } public override void OnCompleted() { if (!_hasValue) { try { throw new InvalidOperationException(Strings_Linq.NO_ELEMENTS); } catch (Exception e) { ForwardOnError(e); } } else { ForwardOnNext(_lastValue); ForwardOnCompleted(); } } } } internal sealed class MinInt64 : Producer { private readonly IObservable _source; public MinInt64(IObservable source) { _source = source; } protected override _ CreateSink(IObserver observer) => new(observer); protected override void Run(_ sink) => sink.Run(_source); internal sealed class _ : IdentitySink { private bool _hasValue; private long _lastValue; public _(IObserver observer) : base(observer) { } public override void OnNext(long value) { if (_hasValue) { if (value < _lastValue) { _lastValue = value; } } else { _lastValue = value; _hasValue = true; } } public override void OnCompleted() { if (!_hasValue) { try { throw new InvalidOperationException(Strings_Linq.NO_ELEMENTS); } catch (Exception e) { ForwardOnError(e); } } else { ForwardOnNext(_lastValue); ForwardOnCompleted(); } } } } internal sealed class MinDoubleNullable : Producer { private readonly IObservable _source; public MinDoubleNullable(IObservable source) { _source = source; } protected override _ CreateSink(IObserver observer) => new(observer); protected override void Run(_ sink) => sink.Run(_source); internal sealed class _ : IdentitySink { private double? _lastValue; public _(IObserver observer) : base(observer) { } public override void OnNext(double? value) { if (!value.HasValue) { return; } if (_lastValue.HasValue) { if (value < _lastValue || double.IsNaN((double)value)) { _lastValue = value; } } else { _lastValue = value; } } public override void OnCompleted() { ForwardOnNext(_lastValue); ForwardOnCompleted(); } } } internal sealed class MinSingleNullable : Producer { private readonly IObservable _source; public MinSingleNullable(IObservable source) { _source = source; } protected override _ CreateSink(IObserver observer) => new(observer); protected override void Run(_ sink) => sink.Run(_source); internal sealed class _ : IdentitySink { private float? _lastValue; public _(IObserver observer) : base(observer) { } public override void OnNext(float? value) { if (!value.HasValue) { return; } if (_lastValue.HasValue) { if (value < _lastValue || float.IsNaN((float)value)) { _lastValue = value; } } else { _lastValue = value; } } public override void OnCompleted() { ForwardOnNext(_lastValue); ForwardOnCompleted(); } } } internal sealed class MinDecimalNullable : Producer { private readonly IObservable _source; public MinDecimalNullable(IObservable source) { _source = source; } protected override _ CreateSink(IObserver observer) => new(observer); protected override void Run(_ sink) => sink.Run(_source); internal sealed class _ : IdentitySink { private decimal? _lastValue; public _(IObserver observer) : base(observer) { } public override void OnNext(decimal? value) { if (!value.HasValue) { return; } if (_lastValue.HasValue) { if (value < _lastValue) { _lastValue = value; } } else { _lastValue = value; } } public override void OnCompleted() { ForwardOnNext(_lastValue); ForwardOnCompleted(); } } } internal sealed class MinInt32Nullable : Producer { private readonly IObservable _source; public MinInt32Nullable(IObservable source) { _source = source; } protected override _ CreateSink(IObserver observer) => new(observer); protected override void Run(_ sink) => sink.Run(_source); internal sealed class _ : IdentitySink { private int? _lastValue; public _(IObserver observer) : base(observer) { } public override void OnNext(int? value) { if (!value.HasValue) { return; } if (_lastValue.HasValue) { if (value < _lastValue) { _lastValue = value; } } else { _lastValue = value; } } public override void OnCompleted() { ForwardOnNext(_lastValue); ForwardOnCompleted(); } } } internal sealed class MinInt64Nullable : Producer { private readonly IObservable _source; public MinInt64Nullable(IObservable source) { _source = source; } protected override _ CreateSink(IObserver observer) => new(observer); protected override void Run(_ sink) => sink.Run(_source); internal sealed class _ : IdentitySink { private long? _lastValue; public _(IObserver observer) : base(observer) { } public override void OnNext(long? value) { if (!value.HasValue) { return; } if (_lastValue.HasValue) { if (value < _lastValue) { _lastValue = value; } } else { _lastValue = value; } } public override void OnCompleted() { ForwardOnNext(_lastValue); ForwardOnCompleted(); } } } }