| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129 | // Licensed to the .NET Foundation under one or more agreements.// The .NET Foundation licenses this file to you under the Apache 2.0 License.// See the LICENSE file in the project root for more information. #if !NO_PERFusing System;using System.Reactive.Concurrency;using System.Reactive.Disposables;namespace System.Reactive.Linq.ObservableImpl{    class Repeat<TResult> : Producer<TResult>    {        private readonly TResult _value;        private readonly int? _repeatCount;        private readonly IScheduler _scheduler;        public Repeat(TResult value, int? repeatCount, IScheduler scheduler)        {            _value = value;            _repeatCount = repeatCount;            _scheduler = scheduler;        }        protected override IDisposable Run(IObserver<TResult> observer, IDisposable cancel, Action<IDisposable> setSink)        {            var sink = new _(this, observer, cancel);            setSink(sink);            return sink.Run();        }        class _ : Sink<TResult>        {            private readonly Repeat<TResult> _parent;            public _(Repeat<TResult> parent, IObserver<TResult> observer, IDisposable cancel)                : base(observer, cancel)            {                _parent = parent;            }            public IDisposable Run()            {                var longRunning = _parent._scheduler.AsLongRunning();                if (longRunning != null)                {                    return Run(longRunning);                }                else                {                    return Run(_parent._scheduler);                }            }            private IDisposable Run(IScheduler scheduler)            {                if (_parent._repeatCount == null)                {                    return scheduler.Schedule(LoopRecInf);                }                else                {                    return scheduler.Schedule(_parent._repeatCount.Value, LoopRec);                }            }            private void LoopRecInf(Action recurse)            {                base._observer.OnNext(_parent._value);                recurse();            }            private void LoopRec(int n, Action<int> recurse)            {                if (n > 0)                {                    base._observer.OnNext(_parent._value);                    n--;                }                if (n == 0)                {                    base._observer.OnCompleted();                    base.Dispose();                    return;                }                recurse(n);            }            private IDisposable Run(ISchedulerLongRunning scheduler)            {                if (_parent._repeatCount == null)                {                    return scheduler.ScheduleLongRunning(LoopInf);                }                else                {                    return scheduler.ScheduleLongRunning(_parent._repeatCount.Value, Loop);                }            }            private void LoopInf(ICancelable cancel)            {                var value = _parent._value;                while (!cancel.IsDisposed)                    base._observer.OnNext(value);                base.Dispose();            }            private void Loop(int n, ICancelable cancel)            {                var value = _parent._value;                while (n > 0 && !cancel.IsDisposed)                {                    base._observer.OnNext(value);                    n--;                }                if (!cancel.IsDisposed)                    base._observer.OnCompleted();                base.Dispose();            }        }    }}#endif
 |