Skip to content

Commit 9757a73

Browse files
Allow readonly string arrays in types.enumeration (#2059)
* feat: Allow readonly string arrays in types.enumeration * fix type a bit
1 parent 789ff2c commit 9757a73

File tree

2 files changed

+35
-7
lines changed

2 files changed

+35
-7
lines changed

packages/mobx-state-tree/__tests__/core/enum.test.ts

+26
Original file line numberDiff line numberDiff line change
@@ -46,3 +46,29 @@ test("should support optional enums inside a model", () => {
4646
const l = TrafficLight.create({})
4747
expect(l.color).toBe(ColorEnum.Orange)
4848
})
49+
test("should support plain string[] arrays", () => {
50+
const colorOptions: string[] = ["Red", "Orange", "Green"]
51+
const TrafficLight = types.model({ color: types.enumeration(colorOptions) })
52+
const l = TrafficLight.create({ color: "Orange" })
53+
unprotect(l)
54+
l.color = "Red"
55+
expect(TrafficLight.describe()).toBe('{ color: ("Red" | "Orange" | "Green") }')
56+
if (process.env.NODE_ENV !== "production") {
57+
expect(() => (l.color = "Blue" as any)).toThrowError(
58+
/Error while converting `"Blue"` to `"Red" | "Orange" | "Green"`/
59+
)
60+
}
61+
})
62+
test("should support readonly enums as const", () => {
63+
const colorOptions = ["Red", "Orange", "Green"] as const
64+
const TrafficLight = types.model({ color: types.enumeration(colorOptions) })
65+
const l = TrafficLight.create({ color: "Orange" })
66+
unprotect(l)
67+
l.color = "Red"
68+
expect(TrafficLight.describe()).toBe('{ color: ("Red" | "Orange" | "Green") }')
69+
if (process.env.NODE_ENV !== "production") {
70+
expect(() => (l.color = "Blue" as any)).toThrowError(
71+
/Error while converting `"Blue"` to `"Red" | "Orange" | "Green"`/
72+
)
73+
}
74+
})

packages/mobx-state-tree/src/types/utility-types/enumeration.ts

+9-7
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,16 @@
11
import { ISimpleType, union, literal, assertIsString, devMode } from "../../internal"
22

33
/** @hidden */
4-
export type UnionStringArray<T extends string[]> = T[number]
4+
export type UnionStringArray<T extends readonly string[]> = T[number]
55

6-
// strongly typed enumeration forms for plain string arrays (when passed directly to the function)
7-
// with these overloads we also get correct typing for native TS string enums when we use Object.values(Enum) as Enum[] as options
8-
// also these overloads make types.enumeration<Enum>(Object.values(Enum)) possible
9-
// the only case where this doesn't work is when passing to the function an arrays variable, for these cases
10-
// it will just fallback and assume the type is a generic string
11-
export function enumeration<T extends string>(options: T[]): ISimpleType<UnionStringArray<T[]>>
6+
// strongly typed enumeration forms for plain and readonly string arrays (when passed directly to the function).
7+
// with these overloads, we get correct typing for native TS string enums when we use Object.values(Enum) as Enum[] as options.
8+
// these overloads also allow both mutable and immutable arrays, making types.enumeration<Enum>(Object.values(Enum)) possible.
9+
// the only case where this doesn't work is when passing to the function an array variable with a mutable type constraint;
10+
// for these cases, it will just fallback and assume the type is a generic string.
11+
export function enumeration<T extends readonly string[]>(
12+
options: T
13+
): ISimpleType<UnionStringArray<T>>
1214
export function enumeration<T extends string>(
1315
name: string,
1416
options: T[]

0 commit comments

Comments
 (0)