|
| 1 | +package com.jnape.palatable.lambda.traversable; |
| 2 | + |
| 3 | +import com.jnape.palatable.lambda.functions.Fn2; |
| 4 | +import com.jnape.palatable.lambda.functor.Applicative; |
| 5 | +import com.jnape.palatable.lambda.functor.Functor; |
| 6 | + |
| 7 | +import java.util.Collections; |
| 8 | +import java.util.HashMap; |
| 9 | +import java.util.Map; |
| 10 | +import java.util.Objects; |
| 11 | +import java.util.function.Function; |
| 12 | + |
| 13 | +import static com.jnape.palatable.lambda.adt.hlist.HList.tuple; |
| 14 | +import static com.jnape.palatable.lambda.functions.builtin.fn2.Into.into; |
| 15 | +import static com.jnape.palatable.lambda.functions.builtin.fn2.Map.map; |
| 16 | +import static com.jnape.palatable.lambda.functions.builtin.fn2.ToMap.toMap; |
| 17 | +import static com.jnape.palatable.lambda.functions.builtin.fn3.FoldLeft.foldLeft; |
| 18 | +import static java.util.Collections.emptyMap; |
| 19 | + |
| 20 | +/** |
| 21 | + * Extension point for {@link Map} to adapt lambda core types like {@link Functor} and {@link Traversable}. |
| 22 | + * |
| 23 | + * @param <A> the {@link Map} element type |
| 24 | + * @see LambdaIterable |
| 25 | + */ |
| 26 | +public final class LambdaMap<A, B> implements Functor<B, LambdaMap<A, ?>>, Traversable<B, LambdaMap<A, ?>> { |
| 27 | + private final Map<A, B> map; |
| 28 | + |
| 29 | + private LambdaMap(Map<A, B> map) { |
| 30 | + this.map = map; |
| 31 | + } |
| 32 | + |
| 33 | + /** |
| 34 | + * Unwrap the underlying {@link Map}. |
| 35 | + * |
| 36 | + * @return the wrapped {@link Map} |
| 37 | + */ |
| 38 | + public Map<A, B> unwrap() { |
| 39 | + return map; |
| 40 | + } |
| 41 | + |
| 42 | + @Override |
| 43 | + public <C> LambdaMap<A, C> fmap(Function<? super B, ? extends C> fn) { |
| 44 | + return wrap(toMap(HashMap::new, map(entry -> tuple(entry.getKey(), fn.apply(entry.getValue())), map.entrySet()))); |
| 45 | + } |
| 46 | + |
| 47 | + @Override |
| 48 | + @SuppressWarnings("unchecked") |
| 49 | + public <C, App extends Applicative, TravC extends Traversable<C, LambdaMap<A, ?>>, AppC extends Applicative<C, App>, AppTrav extends Applicative<TravC, App>> AppTrav traverse( |
| 50 | + Function<? super B, ? extends AppC> fn, Function<? super TravC, ? extends AppTrav> pure) { |
| 51 | + return foldLeft(Fn2.<AppTrav, Map.Entry<A, AppC>, AppTrav>fn2(appTrav -> into((k, appV) -> (AppTrav) appTrav.<TravC>zip(appV.fmap(v -> m -> { |
| 52 | + ((LambdaMap<A, C>) m).unwrap().put(k, v); |
| 53 | + return (TravC) m; |
| 54 | + })))).toBiFunction(), |
| 55 | + pure.apply((TravC) LambdaMap.<A, C>wrap(new HashMap<>())), |
| 56 | + this.<AppC>fmap(fn).unwrap().entrySet()); |
| 57 | + } |
| 58 | + |
| 59 | + @Override |
| 60 | + public boolean equals(Object other) { |
| 61 | + return other instanceof LambdaMap && Objects.equals(map, ((LambdaMap) other).map); |
| 62 | + } |
| 63 | + |
| 64 | + @Override |
| 65 | + public int hashCode() { |
| 66 | + return Objects.hash(map); |
| 67 | + } |
| 68 | + |
| 69 | + @Override |
| 70 | + public String toString() { |
| 71 | + return "LambdaMap{map=" + map + '}'; |
| 72 | + } |
| 73 | + |
| 74 | + /** |
| 75 | + * Wrap a {@link Map} in a {@link LambdaMap}. |
| 76 | + * |
| 77 | + * @param map the {@link Map} |
| 78 | + * @param <A> the key type |
| 79 | + * @param <B> the value type |
| 80 | + * @return the {@link Map} wrapped in a {@link LambdaMap} |
| 81 | + */ |
| 82 | + public static <A, B> LambdaMap<A, B> wrap(Map<A, B> map) { |
| 83 | + return new LambdaMap<>(map); |
| 84 | + } |
| 85 | + |
| 86 | + /** |
| 87 | + * Construct an empty {@link LambdaMap} by wrapping {@link Collections#emptyMap()} |
| 88 | + * |
| 89 | + * @param <A> the key type |
| 90 | + * @param <B> the value type |
| 91 | + * @return an empty {@link LambdaMap} |
| 92 | + */ |
| 93 | + public static <A, B> LambdaMap<A, B> empty() { |
| 94 | + return wrap(emptyMap()); |
| 95 | + } |
| 96 | +} |
0 commit comments