aboutsummaryrefslogtreecommitdiff
path: root/src/utils/adt.ts
blob: e9a6c518c6f4122d7b2cbef204fd467dfe0404cd (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
type TagValue<T, N> = T extends Tag<N, infer V> ? V : never

export const match =
  <R, T extends Tag<string, any>>(tag: T, pattern: {
    [key in T['tag'] | '_']?: (v: TagValue<T, key>) => R
  }): R => ((pattern as any)[tag.tag] || (pattern._ as any))(tag.value)

type Tag<N, V> = { tag: N; value: V }
export type Enum<T> = { [N in keyof T]: Tag<N, T[N]> }[keyof T]

export const constructors = <T extends Tag<string, any>>(): {
  [N in T['tag']]: TagValue<T, N> extends null | never
    ? (value?: null | never) => T
    : (value: TagValue<T, N>) => T
} =>
  new Proxy(
    {},
    {
      get(_, k) {
        return (value: any) => ({ tag: k, value })
      },
    },
  ) as any