DotNext by Roman Sakno

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

 ReadOnlyDictionaryView<K, I, O>

Represents lazily converted read-only dictionary.
using System; using System.Collections; using System.Collections.Generic; using System.Diagnostics.CodeAnalysis; using System.Linq; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; namespace DotNext.Collections.Generic { [StructLayout(LayoutKind.Auto)] [System.Runtime.CompilerServices.NullableContext(1)] [System.Runtime.CompilerServices.Nullable(0)] public readonly struct ReadOnlyDictionaryView<K, [System.Runtime.CompilerServices.Nullable(2)] I, [System.Runtime.CompilerServices.Nullable(2)] O> : IReadOnlyDictionary<K, O>, IEnumerable<KeyValuePair<K, O>>, IEnumerable, IReadOnlyCollection<KeyValuePair<K, O>>, IEquatable<ReadOnlyDictionaryView<K, I, O>> { private readonly IReadOnlyDictionary<K, I> source; [System.Runtime.CompilerServices.Nullable(new byte[] { 0, 1, 1 })] private readonly ValueFunc<I, O> mapper; public O this[K key] { get { return mapper.Invoke(source[key]); } } public IEnumerable<K> Keys => source.Keys; public IEnumerable<O> Values => Enumerable.Select<I, O>(source.Values, mapper.ToDelegate()); public int Count => source.Count; public ReadOnlyDictionaryView(IReadOnlyDictionary<K, I> dictionary, [In] [IsReadOnly] [System.Runtime.CompilerServices.Nullable(new byte[] { 0, 1, 1 })] ref ValueFunc<I, O> mapper) { if (dictionary == null) throw new ArgumentNullException("dictionary"); source = dictionary; this.mapper = mapper; } public bool ContainsKey(K key) { return source.ContainsKey(key); } [return: System.Runtime.CompilerServices.Nullable(new byte[] { 1, 0, 1, 1 })] public IEnumerator<KeyValuePair<K, O>> GetEnumerator() { foreach (KeyValuePair<K, I> item in (IEnumerable<KeyValuePair<K, I>>)this.source) { item.Deconstruct(out K key, out I value); K key2 = key; I arg = value; yield return new KeyValuePair<K, O>(key2, this.mapper.Invoke(arg)); } } public bool TryGetValue(K key, [NotNullWhen(true)] out O value) { if (source.TryGetValue(key, out I value2)) { value = mapper.Invoke(value2); return true; } value = default(O); return false; } IEnumerator IEnumerable.GetEnumerator() { return GetEnumerator(); } public bool Equals([System.Runtime.CompilerServices.Nullable(new byte[] { 0, 1, 1, 1 })] ReadOnlyDictionaryView<K, I, O> other) { if (source == other.source) return ref mapper == ref other.mapper; return false; } public override int GetHashCode() { return RuntimeHelpers.GetHashCode(source) ^ mapper.GetHashCode(); } [System.Runtime.CompilerServices.NullableContext(2)] public override bool Equals(object other) { if (!(other is ReadOnlyDictionaryView<K, I, O>)) return object.Equals(source, other); ReadOnlyDictionaryView<K, I, O> other2 = (ReadOnlyDictionaryView<K, I, O>)other; return Equals(other2); } public static bool operator ==([System.Runtime.CompilerServices.Nullable(new byte[] { 0, 1, 1, 1 })] ReadOnlyDictionaryView<K, I, O> first, [System.Runtime.CompilerServices.Nullable(new byte[] { 0, 1, 1, 1 })] ReadOnlyDictionaryView<K, I, O> second) { return first.Equals(second); } public static bool operator !=([System.Runtime.CompilerServices.Nullable(new byte[] { 0, 1, 1, 1 })] ReadOnlyDictionaryView<K, I, O> first, [System.Runtime.CompilerServices.Nullable(new byte[] { 0, 1, 1, 1 })] ReadOnlyDictionaryView<K, I, O> second) { return !first.Equals(second); } } }