using System; using System.Collections.Immutable; public sealed class ImmutableDictionaryValueLens : ILens { public readonly Func, Whole> wrap; public readonly ImmutableDictionary oldDictionary; public readonly TKey oldKey; public ImmutableDictionaryValueLens(Func, Whole> wrap, ImmutableDictionary oldDictionary, TKey oldKey) { // TODO: check that key exists. this.wrap = wrap; this.oldDictionary = oldDictionary; this.oldKey = oldKey; } // Put methods with the following signature here to focus on sub-parts of the list as needed. // public ILens,Whole> sub-part => oldHole.sub-part.ChainLens(value => oldHole.with-sub-part(value)); public Whole Update(Func update) { var oldValue = oldDictionary[oldKey]; return wrap(oldDictionary.SetItem(oldKey, update(oldValue))); } public ImmutableDictionaryValueLens UpdateKey(Func update) { var newKey = update(oldKey); return new ImmutableDictionaryValueLens( wrap, oldDictionary.Remove(oldKey).Add(newKey, oldDictionary[oldKey]), newKey); } } public sealed class ImmutableDictionaryLens : ILens, Whole> { public readonly Func, Whole> wrap; public readonly ImmutableDictionary oldHole; public ImmutableDictionaryLens(Func, Whole> wrap, ImmutableDictionary oldHole) { // TODO: check that key exists. this.wrap = wrap; this.oldHole = oldHole; } // Put methods with the following signature here to focus on sub-parts of the list as needed. public ImmutableDictionaryValueLens this[TKey key] { get => new ImmutableDictionaryValueLens(wrap, oldHole, key); } public Whole Update(Func, ImmutableDictionary> update) { return wrap(update(oldHole)); } } public static class ImmutableDictionaryLensExtensionMethods { public static ImmutableDictionaryLens> lens(this ImmutableDictionary d) => new ImmutableDictionaryLens>(x => x, d); public static ImmutableDictionaryValueLens UpdateKey(this ImmutableDictionaryValueLens lens, TKey newKey) => lens.UpdateKey(oldKey => newKey); // This would need an IFocusable constraint which is hard to get //public static ILens, Whole> ChainLens(this ImmutableDictionary hole, System.Func, Whole> wrap) // => new ImmutableDictionaryLens(wrap: wrap, oldHole: hole); }