Avoid TypeScript Enums
TypeScript enums are not fully type-safe and can cause surprises. Your code should use constant objects instead of introducing a new enum.
Our Recommended Approach
- Use the same name for your type- and value-declaration.
- Use
type
to derive type information from the const object. - Create utilities to convert and identify enums modelled as primitives.
tip
This pattern should simplify the usage of your new objects, improve type safety in files that have adopted TS-strict, and make transitioning an enum to a const object much easier.
Example
Given the following enum:
export enum CipherType = {
Login: 1,
SecureNote: 2,
Card: 3,
Identity: 4,
SshKey: 5,
};
You can redefine it as an object like so:
const _CipherType = {
Login: 1,
SecureNote: 2,
Card: 3,
Identity: 4,
SshKey: 5,
} as const;
type _CipherType = typeof _CipherType;
export type CipherType = _CipherType[keyof _CipherType];
export const CipherType: Readonly<{ [K in keyof typeof _CipherType]: CipherType }> =
Object.freeze(_CipherType);
And use it like so:
// Can be imported together
import { CipherType } from "./cipher-type";
// Used as a type
function doSomething(type: CipherType) {}
// And used as a value (just like a regular `enum`)
doSomething(CipherType.Card);
The following utilities may assist introspection:
import { CipherType } from "./cipher-type";
const namesByCipherType = new Map<CipherType, keyof CipherType>(
Array.fromEntries(Object.entries(CipherType), ([k, v]) => [v, k]),
);
export function isCipherType(value: number): value is CipherType {
return namesByCipherType.has(value);
}
export function asCipherType(value: number): CipherType | undefined {
return isCipherType(value) ? value : undefined;
}
export function nameOfCipherType(value: CipherType): keyof CipherType | undefined {
return namesByCipherType.get(value);
}