// Code quality of this file: low. using System; using System.Collections.Immutable; public interface ILens { Whole Update(Func update); } public sealed class ImmutableListLens : ILens, Whole> { public readonly System.Func, Whole> wrap; public readonly ImmutableList oldHole; public ImmutableListLens(System.Func, Whole> wrap, ImmutableList oldHole) { this.wrap = wrap; this.oldHole = oldHole; } // 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, ImmutableList> update) => wrap(update(oldHole)); } // Lenses for primitive types and other types that are not // interesting to further focus. public sealed class LeafLens : ILens { public readonly System.Func wrap; public readonly T oldHole; public LeafLens(System.Func wrap, T oldHole) { this.wrap = wrap; this.oldHole = oldHole; } public Whole Update(Func update) => wrap(update(oldHole)); } public static class LensExtensionMethods { public static Whole Update(this ILens lens, Hole newHole) => lens.Update(oldHole => newHole); public static Whole Cons(this ILens, Whole> lens, T value) => lens.Update(oldHole => oldHole.Cons(value)); public static ImmutableListLens ChainLens( this ImmutableList hole, System.Func, Whole> wrap) => new ImmutableListLens(wrap: wrap, oldHole: hole); public static ILens ChainLens(this string hole, System.Func wrap) => new LeafLens(wrap: wrap, oldHole: hole); public static ILens, Whole> ChainLens(this Func hole, System.Func, Whole> wrap) => new LeafLens, Whole>(wrap: wrap, oldHole: hole); public class FocusableLeaf { private readonly T value; public FocusableLeaf(T value) { this.value = value; } public LeafLens ChainLens(Func wrap) => new LeafLens(wrap: wrap, oldHole: value); } public static ILens, Whole> ChainLens(this ImmutableList hole, System.Func, Whole> wrap) => new ImmutableListLens(wrap: wrap, oldHole: hole); }