Because enums are bad, and const enums are even worse.
The simplest way:
import { constantsUntagged, InferUnion } from "ts-consts";
const ACTIONS = constantsUntagged([
"SAVE",
"RESET",
"CANCEL",
])
ACTIONS.SAVE // "SAVE"
ACTIONS.RESET // "RESET"
ACTIONS.CANCEL // "CANCEL"
type ACTIONS = typeof ACTIONS;
// {
// readonly SAVE: "SAVE";
// readonly RESET: "RESET";
// readonly CANCEL: "CANCEL";
// }
type ACTIONS_u = InferUnion<ACTIONS>;
// "SAVE" | "RESET" | "CANCEL"
To customize keys and values:
import { constantsUntagged, InferUnion } from "ts-consts";
const ACTIONS = constantsUntagged({
SAVE: "save",
RESET: "reset",
CANCEL: "cancel",
})
ACTIONS.SAVE // "save"
ACTIONS.RESET // "reset"
ACTIONS.CANCEL // "cancel"
type ACTIONS = typeof ACTIONS;
// {
// readonly SAVE: "save";
// readonly RESET: "reset";
// readonly CANCEL: "cancel";
// }
type ACTIONS_u = InferUnion<ACTIONS>;
// "save" | "reset" | "cancel"
import { constantsTagged, InferUnion } from "ts-consts";
const ACTIONS = constantsTagged("ACTIONS", {
SAVE: "save",
RESET: "reset",
CANCEL: "cancel",
})
ACTIONS.SAVE // Tagged<"save", "ACTIONS">
ACTIONS.RESET // Tagged<"reset", "ACTIONS">
ACTIONS.CANCEL // Tagged<"cancel", "ACTIONS">
type ACTIONS = typeof ACTIONS;
// {
// readonly SAVE: Tagged<"save", "ACTIONS">;
// readonly RESET: Tagged<"reset", "ACTIONS">;
// readonly CANCEL: Tagged<"cancel", "ACTIONS">;
// }
type ACTIONS_u = InferUnion<ACTIONS>
// Tagged<"save", "ACTIONS"> | Tagged<"reset", "ACTIONS"> | Tagged<"cancel", "ACTIONS">
// so now the following is an error:
let a_v0: ACTIONS_u = "save" // Type '"save"' is not assignable to type 'ACTIONS_u'
// only this is allowed now:
let a_v1: ACTIONS_u = ACTIONS.SAVE;
import { constants, InferUnions } from "ts-consts";
const ACTIONS = constants("ACTIONS", {
SAVE: "save",
RESET: "reset",
CANCEL: "cancel",
});
ACTIONS.untagged;
// {
// readonly SAVE: "save";
// readonly RESET: "reset";
// readonly CANCEL: "cancel";
// }
ACTIONS.tagged;
// {
// readonly SAVE: Tagged<"save", "ACTIONS">;
// readonly RESET: Tagged<"reset", "ACTIONS">;
// readonly CANCEL: Tagged<"cancel", "ACTIONS">;
// }
type ACTIONS = typeof ACTIONS;
type ACTIONS_us = InferUnions<typeof ACTIONS>;
ACTIONS_us["untagged"]
// "save" | "reset" | "cancel"
ACTIONS_us["tagged"]
// Tagged<"save", "ACTIONS"> | Tagged<"reset", "ACTIONS"> | Tagged<"cancel", "ACTIONS">
From the keys of an object.
import { constantsUntagged, deriveUntaggedConstants } from "ts-consts";
const ACTIONS = constantsUntagged({
SAVE: "save",
RESET: "reset",
CANCEL: "cancel",
});
const ACTIONS_keys = deriveUntaggedConstants(ACTIONS);
// {
// readonly SAVE: "SAVE";
// readonly RESET: "RESET";
// readonly CANCEL: "CANCEL";
// }
import { constantsTagged, removeTags } from "ts-consts";
const ACTIONS = constantsTagged("ACTIONS", {
SAVE: "save",
RESET: "reset",
CANCEL: "cancel",
});
const ACTIONS_untagged = removeTags(ACTIONS);
// {
// readonly SAVE: "save";
// readonly RESET: "reset";
// readonly CANCEL: "cancel";
// }
ProjectUntaggedConstants
help us to create a type that is safely indexable only by a record of constants.
Given a record type containing only untagged PropertyKeys
, as the ACTIONS
type below, and a type which uses the properties' types of the first as keys, it forces the latter to use ALL AND ONLY the properties' types of the first as keys.
import { constantsUntagged, ProjectUntaggedConstants } from "ts-consts";
const ACTIONS = constantsUntagged({
SAVE: "save",
RESET: "reset",
CANCEL: "cancel",
});
type ACTIONS = typeof ACTIONS;
type ActionHandlers = ProjectUntaggedConstants<
ACTIONS,
// if you either remove a key or add something extraneous
// you'll get an error
{
[ACTIONS.SAVE]: (...args: any) => any;
[ACTIONS.RESET]: (...args: any) => any;
[ACTIONS.CANCEL]: (...args: any) => any;
}
>;
// {
// save: (...args: any) => any;
// reset: (...args: any) => any;
// cancel: (...args: any) => any;
// }
declare const ahs: ActionHandlers;
// safely use of ACTIONS members as keys
ahs[ACTIONS.SAVE];
ahs[ACTIONS.RESET];
ahs[ACTIONS.CANCEL];