DotNext by .NET Foundation and Contributors

<PackageReference Include="DotNext" Version="5.3.0" />

 ListSegment<T>

Delimits a section of a list.
using System; using System.Collections; using System.Collections.Generic; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; namespace DotNext.Collections.Generic { [StructLayout(LayoutKind.Auto)] [NullableContext(1)] [Nullable(0)] public readonly struct ListSegment<[Nullable(2)] T> : IList<T>, ICollection<T>, IEnumerable<T>, IEnumerable, IReadOnlyList<T>, IReadOnlyCollection<T> { private readonly int startIndex; private readonly IList<T> list; bool ICollection<T>.IsReadOnly { get { return true; } } public int Count { get; } public T this[int index] { get { return list[ToAbsoluteIndex(index)]; } set { list[ToAbsoluteIndex(index)] = value; } } public ListSegment(IList<T> list, Range range) { (int, int) offsetAndLength = range.GetOffsetAndLength(list.Count); startIndex = offsetAndLength.Item1; Count = offsetAndLength.Item2; this.list = list; } [MethodImpl(MethodImplOptions.AggressiveInlining)] private int ToAbsoluteIndex(int index) { if (!BasicExtensions.IsBetween<int, EnclosedEndpoint<int>, DisclosedEndpoint<int>>(index, BasicExtensions.Enclosed<int>(0), BasicExtensions.Disclosed<int>(Count))) throw new ArgumentOutOfRangeException("index"); return index + startIndex; } [MethodImpl(MethodImplOptions.AggressiveInlining)] private bool ToRelativeIndex(ref int index) { index -= startIndex; return BasicExtensions.IsBetween<int, EnclosedEndpoint<int>, DisclosedEndpoint<int>>(index, BasicExtensions.Enclosed<int>(0), BasicExtensions.Disclosed<int>(Count)); } public int IndexOf(T item) { int index = list.IndexOf(item); if (!ToRelativeIndex(ref index)) return -1; return index; } public bool TryGetSpan([Nullable(new byte[] { 0, 1 })] out Span<T> span) { IList<T> list = this.list; List<T> list2 = list as List<T>; if (list2 == null) { T[] array = list as T[]; if (array == null) { span = default(Span<T>); return false; } span = new Span<T>(array, startIndex, Count); } else span = CollectionsMarshal.AsSpan<T>(list2).Slice(startIndex, Count); return true; } void IList<T>.Insert(int index, T item) { throw new NotSupportedException(); } void IList<T>.RemoveAt(int index) { throw new NotSupportedException(); } void ICollection<T>.Add(T item) { throw new NotSupportedException(); } void ICollection<T>.Clear() { throw new NotSupportedException(); } public bool Contains(T item) { return IndexOf(item) >= 0; } public void CopyTo(T[] array, int arrayIndex) { IList<T> list = this.list; List<T> list2 = list as List<T>; if (list2 == null) { T[] array2 = list as T[]; if (array2 != null) Array.Copy(array2, startIndex, array, arrayIndex, Count); else { for (int i = 0; i < Count; i++) { array[i] = this.list[i + startIndex]; } } } else list2.CopyTo(startIndex, array, arrayIndex, Count); } bool ICollection<T>.Remove(T item) { throw new NotSupportedException(); } public IEnumerator<T> GetEnumerator() { IEnumerator<T> enumerator = list.GetEnumerator(); Enumerator.Skip<T>(enumerator, startIndex); return Enumerator.Limit<T>(enumerator, Count, false); } IEnumerator IEnumerable.GetEnumerator() { return GetEnumerator(); } } }