using FishNet.Documenting;
using FishNet.Managing;
using System.Collections;
using System.Collections.Generic;
using System.Runtime.CompilerServices;
using UnityEngine;
namespace FishNet.Utility
{
    /// 
    /// Writes values to a collection of a set size, overwriting old values as needed.
    /// 
    public class RingBuffer
    {
        #region Types.
        /// 
        /// Custom enumerator to prevent garbage collection.
        /// 
        [APIExclude]
        public struct Enumerator : IEnumerator
        {
            #region Public.
            /// 
            /// Current entry in the enumerator. 
            /// 
            public T Current { get; private set; }
            /// 
            /// Actual index of the last enumerated value.
            /// 
            public int ActualIndex
            {
                get
                {
                    int total = (_startIndex + (_read - 1));
                    int capacity = _rollingCollection.Capacity;
                    if (total >= capacity)
                        total -= capacity;
                    return total;
                }
            }
            /// 
            /// Simulated index of the last enumerated value.
            /// 
            public int SimulatedIndex => (_read - 1);
            #endregion
            #region Private.
            /// 
            /// RollingCollection to use.
            /// 
            private RingBuffer _rollingCollection;
            /// 
            /// Collection to iterate.
            /// 
            private readonly T[] _collection;
            /// 
            /// Number of entries read during the enumeration.
            /// 
            private int _read;
            /// 
            /// Start index of enumerations.
            /// 
            private int _startIndex;
            #endregion
            public Enumerator(RingBuffer c)
            {
                _read = 0;
                _startIndex = 0;
                _rollingCollection = c;
                _collection = c.Collection;
                Current = default;
            }
            public bool MoveNext()
            {
                int written = _rollingCollection.Count;
                if (_read >= written)
                {
                    ResetRead();
                    return false;
                }
                int index = (_startIndex + _read);
                int capacity = _rollingCollection.Capacity;
                if (index >= capacity)
                    index -= capacity;
                Current = _collection[index];
                _read++;
                return true;
            }
            /// 
            /// Sets a new start index to begin reading at.
            /// 
            public void SetStartIndex(int index)
            {
                _startIndex = index;
                ResetRead();
            }
            /// 
            /// Sets a new start index to begin reading at.
            /// 
            public void AddStartIndex(int value)
            {
                _startIndex += value;
                int cap = _rollingCollection.Capacity;
                if (_startIndex > cap)
                    _startIndex -= cap;
                else if (_startIndex < 0)
                    _startIndex += cap;
                ResetRead();
            }
            /// 
            /// Resets number of entries read during the enumeration.
            /// 
            public void ResetRead()
            {
                _read = 0;
            }
            /// 
            /// Resets read count.
            /// 
            public void Reset()
            {
                _startIndex = 0;
                ResetRead();
            }
            object IEnumerator.Current => Current;
            public void Dispose() { }
        }
        #endregion
        #region Public.
        /// 
        /// Current write index of the collection.
        /// 
        public int WriteIndex { get; private set; }
        /// 
        /// Number of entries currently written.
        /// 
        public int Count => _written;
        /// 
        /// Maximum size of the collection.
        /// 
        public int Capacity => Collection.Length;
        /// 
        /// Collection being used.
        /// 
        public T[] Collection = new T[0];
        /// 
        /// True if initialized.
        /// 
        public bool Initialized { get; private set; }
        #endregion
        #region Private.
        /// 
        /// Number of entries written. This will never go beyond the capacity but will be less until capacity is filled.
        /// 
        private int _written;
        /// 
        /// Enumerator for the collection.
        /// 
        private Enumerator _enumerator;
        #endregion
        /// 
        /// Initializes the collection at length.
        /// 
        /// Size to initialize the collection as. This cannot be changed after initialized.
        public void Initialize(int capacity)
        {
            if (capacity <= 0)
            {
                NetworkManager.StaticLogError($"Collection length must be larger than 0.");
                return;
            }
            Collection = new T[capacity];
            _enumerator = new Enumerator(this);
            Initialized = true;
        }
        /// 
        /// Clears the collection to default values and resets indexing.
        /// 
        public void Clear()
        {
            for (int i = 0; i < Collection.Length; i++)
                Collection[i] = default;
            Reset();
        }
        /// 
        /// Resets the collection without clearing.
        /// 
        public void Reset()
        {
            _written = 0;
            WriteIndex = 0;
            _enumerator.Reset();
        }
        /// 
        /// Adds an entry to the collection, returning a replaced entry.
        /// 
        /// Data to add.
        /// Replaced entry. Value will be default if no entry was replaced.
        [MethodImpl(MethodImplOptions.AggressiveInlining)]
        public T Add(T data)
        {
            if (!IsInitializedWithError())
                return default;
            int capacity = Capacity;
            T current = Collection[WriteIndex];
            Collection[WriteIndex] = data;
            WriteIndex++;
            _written++;
            //If index would exceed next iteration reset it.
            if (WriteIndex >= capacity)
                WriteIndex = 0;
            /* If written has exceeded capacity
            * then the start index needs to be moved
            * to adjust for overwritten values. */
            if (_written > capacity)
            {
                _written = capacity;
                _enumerator.SetStartIndex(WriteIndex);
            }
            return current;
        }
        /// 
        /// Returns value in actual index as it relates to simulated index.
        /// 
        /// Simulated index to return. A value of 0 would return the first simulated index in the collection.
        /// 
        public T this[int simulatedIndex]
        {
            get
            {
                int offset = (Capacity - _written) + simulatedIndex + WriteIndex;                
                if (offset >= Capacity)
                    offset -= Capacity;
                return Collection[offset];
            }
            set
            {
                int offset = (Capacity - _written) + simulatedIndex + WriteIndex;
                if (offset >= Capacity)
                    offset -= Capacity;
                Collection[offset] = value;
            }
        }
        /// 
        /// Returns Enumerator for the collection.
        /// 
        /// 
        [MethodImpl(MethodImplOptions.AggressiveInlining)]
        public Enumerator GetEnumerator()
        {
            if (!IsInitializedWithError())
                return default;
            _enumerator.ResetRead();
            return _enumerator;
        }
        /// 
        /// Removes values from the simulated start of the collection.
        /// 
        /// True to remove from the start, false to remove from the end.
        /// Number of entries to remove.
        public void RemoveRange(bool fromStart, int length)
        {
            if (length == 0)
                return;
            if (length < 0)
            {
                NetworkManager.StaticLogError($"Negative values cannot be removed.");
                return;
            }
            //Full reset if value is at or more than written.
            if (length >= _written)
            {
                Reset();
                return;
            }
            _written -= length;
            if (fromStart)
            {
                _enumerator.AddStartIndex(length);
            }
            else
            {
                WriteIndex -= length;
                if (WriteIndex < 0)
                    WriteIndex += Capacity;
            }
        }
        /// 
        /// Returns if initialized and errors if not.
        /// 
        /// 
        private bool IsInitializedWithError()
        {
            if (!Initialized)
            {
                NetworkManager.StaticLogError($"RingBuffer has not yet been initialized.");
                return false;
            }
            return true;
        }
    }
}