UNPKG

enum-plus

Version:

A drop-in replacement for native enum. Like native enum but much better!

1,323 lines (930 loc) 61.2 kB
<!-- markdownlint-disable MD001 MD009 MD033 MD041 --> [English](./README.md) | [中文](./README.zh-CN.md) | [CHANGELOG](./CHANGELOG.md) <p align="center"> <a href="https://github.com/shijistar/enum-plus" target="blank"> <img src="https://cdn.jsdelivr.net/gh/shijistar/enum-plus@v3.1.1/public/enum-plus.svg" width="240" alt="enum-plus" /> </a> </p> <p align="center"> <strong>Like native enum, but much better!</strong> </p> <br/> [![npm latest version](https://img.shields.io/npm/v/enum-plus.svg?cacheSeconds=86400)](https://www.npmjs.com/package/enum-plus) [![npm package minimized gzipped size](https://img.shields.io/bundlejs/size/enum-plus?logo=javascript&label=Minzipped&color=44cc11&cacheSeconds=86400)](https://www.npmjs.com/package/enum-plus?activeTab=code) [![npm downloads](https://img.shields.io/npm/dm/enum-plus.svg?logo=rolldown&logoColor=fd7b24&label=Downloads&color=007ec6&cacheSeconds=86400)](https://www.npmjs.com/package/enum-plus) [![Ask DeepWiki](https://deepwiki.com/badge.svg)](https://deepwiki.com/shijistar/enum-plus) [![code coverage](https://codecov.io/gh/shijistar/enum-plus/graph/badge.svg?token=JMCDJKLT0B)](https://app.codecov.io/gh/shijistar/enum-plus) ![GitHub License](https://img.shields.io/github/license/shijistar/enum-plus?label=License&color=ff8000&cacheSeconds=86400) **Supported Platforms** [![Node compatibility](https://img.shields.io/node/v/enum-plus?logo=nodedotjs&label=Node.js&color=2185D0&cacheSeconds=86400)](https://github.com/shijistar/enum-plus) [![Web Browsers](https://img.shields.io/badge/Web%20Browser-2185D0?logo=)](https://www.npmjs.com/package/enum-plus) [![React Native](https://img.shields.io/badge/ReactNative-2185D0?logo=react)](https://reactnative.dev) [![MiniProgram](https://img.shields.io/badge/MiniProgram-2185D0?logo=wechat)](https://developers.weixin.qq.com/miniprogram/en/dev/framework) [![Taro](https://img.shields.io/badge/Taro-18BCF2?logo=)](https://docs.taro.zone/en/docs) ⬇️ &nbsp;[Introduction](#introduction) | [Features](#user-content-features) | [Installation](#user-content-installation) | [Enum Initialization](#user-content-enum-initialization) | [API](#user-content-api) | [Static Methods](#user-content-static-methods) | [Global Configuration](#user-content-global-configuration) | [User Stories](#user-content-user-stories) | [Plugin System](#user-content-plugin-system) | [Localization](#user-content-localization) | [Extensibility](#user-content-extensibility) | [Naming Conflicts](#user-content-naming-conflicts) | [Best Practices](#user-content-best-practices) | [Compatibility](#user-content-compatibility) | [Q&A](#user-content-qa) | [Support](#user-content-support)&nbsp; ⬇️ > **🎉 v3.0 is Released!** > > The new version is a major milestone that brings many exciting features and improvements. Please refer to the [Release Notes](./docs/release-v3.md) and [Migration Guide](./docs/migration-guide-v2-to-v3.md) for details. > > If all enum types become `any` after upgrading to v3.0, please see [here](#user-content-why-did-all-enum-types-become-any-after-upgrading-to-version-30). ## Introduction `enum-plus` is an enhanced enumeration library that is fully compatible with the native `enum` and serves as a drop-in replacement library. It supports adding display text to enum items and extending more custom metadata fields. Enums can be generated into various UI components such as dropdowns and menus, which greatly simplifies front-end development. It also provides many extension methods for iterating over enum items and data transformations. You can convert values directly into names in the current language, since it supports internationalization. It's very useful for displaying business data in the UI. It is a lightweight, zero-dependency library, and it works with any front-end framework, including vanilla projects. What other exciting features are there? Please continue to explore! Or you can check out this video first. <p align="center"> <a href="https://cdn.jsdelivr.net/gh/shijistar/enum-plus@v3.1.1/public/usage-screenshot-high-v3.mp4" target="_blank"> <img src="https://cdn.jsdelivr.net/gh/shijistar/enum-plus@v3.1.1/public/usage-screenshot-v3.gif" width="500" alt="usage video" /> </a> </p> <details> <summary>Here are some hot questions, feel free to check them out</summary> - [Why do I need this library? TypeScript already has the built-in enums](#user-content-why-do-i-need-this-library-typescript-already-has-the-built-in-enums) - [Do I have to install TypeScript? What if my project is in JavaScript?](#user-content-do-i-have-to-install-typescript-what-if-my-project-is-in-javascript) - [How about the performance of this library?](#user-content-how-about-the-performance-of-this-library) - [It seems that TypeScript is going to deprecate enum?](#user-content-it-seems-that-typescript-is-going-to-deprecate-enum) - [I have a great idea and would like to contribute to this project. What should I do?](#user-content-i-have-a-great-idea-and-would-like-to-contribute-to-this-project-what-should-i-do) </details> ## Features - Compatible with the usage of native enums - Supports multiple data types such as `number` and `string` - Enhanced enum items with display names - Internationalization support for display names, can be integrated with any i18n library - Converts values directly into display names, simplifying data display in the UI - Extensible design, allowing custom metadata fields for enum items - Plugin system design, extending enum functionality through plugin installations - Supports type narrowing to enhance type safety<sup>_&nbsp;&nbsp;TypeScript_</sup> - Generates dropdowns from enums, compatible with UI libraries like [Ant Design](https://ant.design/components/overview), [Element Plus](https://element-plus.org/en-US/component/overview), [Material-UI](https://mui.com/material-ui) - Compatible with various environments including Browsers, Node.js, React Native, Taro, and mini-programs - Supports server-side rendering (SSR) - Compatible with any front-end development framework, including vanilla projects - TypeScript‑oriented, providing excellent type inference and code completion capabilities - Zero dependencies - Lightweight (min+gzip 2KB+ only) ## Installation Install using npm: ```bash npm install enum-plus ``` Install using pnpm: ```bash pnpm add enum-plus ``` Install using bun: ```bash bun add enum-plus ``` Or using yarn: ```bash yarn add enum-plus ``` **Runs in the Browser**: - The specific version: ```html <!-- Modern version, ES2020 compatible --> <script src="https://cdn.jsdelivr.net/npm/enum-plus@v3.1.1/umd/enum-plus.min.js"></script> <!-- Legacy version, ES2015 compatible --> <script src="https://cdn.jsdelivr.net/npm/enum-plus@v3.1.1/umd/enum-plus-legacy.min.js"></script> ``` - The latest version: ```html <!-- Modern version, ES2020 compatible --> <script src="https://cdn.jsdelivr.net/npm/enum-plus/umd/enum-plus.min.js"></script> <!-- Legacy version, ES2015 compatible --> <script src="https://cdn.jsdelivr.net/npm/enum-plus/umd/enum-plus-legacy.min.js"></script> ``` ⬇️ **Download**: - [enum-plus.umd.min.js.gz](https://github.com/shijistar/enum-plus/releases/download/v3.1.1/enum-plus.umd.min.js.gz) - [enum-plus.umd.tar.gz](https://github.com/shijistar/enum-plus/releases/download/v3.1.1/enum-plus.umd.tar.gz) (Full package with sourcemap) - [enum-plus-legacy.umd.min.js.gz](https://github.com/shijistar/enum-plus/releases/download/v3.1.1/enum-plus-legacy.umd.min.js.gz) - [enum-plus-legacy.umd.tar.gz](https://github.com/shijistar/enum-plus/releases/download/v3.1.1/enum-plus-legacy.umd.tar.gz) (Full package with sourcemap) > You can also download these files from the [GitHub Releases](https://github.com/shijistar/enum-plus/releases/tag/v3.0.1). ## Enum Initialization This section shows the various ways to initialize enums using the `Enum` function. Understanding these different initialization formats allows you to choose the most convenient approach for your specific use case. ### 1. Key-Value Format The simplest format is a direct mapping of keys to values. This is similar to the native enum format. ```js import { Enum } from 'enum-plus'; // With number values const WeekEnum = Enum({ Sunday: 0, Monday: 1, }); WeekEnum.Monday; // 1 // With string values const WeekEnum2 = Enum({ Sunday: 'Sun', Monday: 'Mon', }); WeekEnum2.Monday; // 'Mon' ``` > If your project uses TypeScript and the version is below 5.0, it is recommended to add the `as const` type assertion to the parameter of the `Enum` method. In this way, the enum values will remain their original literal values instead of being converted to `number` or `string`. For more details, please refer to [here](#user-content-do-i-have-to-upgrade-typescript-to-version-50). ### 2. Standard Format (Recommended) The standard format includes both a `value` and a `label` for each enum item. This is the most commonly used format and is recommended for most cases. This format allows you to specify a display name for each enum item, which can be used in UI components. Please refer to the [Localization](#user-content-localization) section for internationalization support. ```js import { Enum } from 'enum-plus'; const WeekEnum = Enum({ Sunday: { value: 0, label: 'I love Sunday' }, Monday: { value: 1, label: 'I hate Monday' }, }); WeekEnum.Sunday; // 0 WeekEnum.items[0].key; // 'Sunday' WeekEnum.items[0].label; // I love Sunday ``` > Want to enable code completion when entering `label`? Please refer to the [Enable Code Intelligence for Enum Item Labels](#user-content--enable-code-intelligence-for-enum-item-labels) section for more details. ### 3. Label-Only Format Create an enumeration that only provides the `key` and `label` fields. If the `value` field is omitted, it defaults to the same value as the `key` field. ```js import { Enum } from 'enum-plus'; const WeekEnum = Enum({ Sunday: { label: 'I love Sunday' }, Monday: { label: 'I hate Monday' }, }); WeekEnum.Sunday; // 'Sunday' WeekEnum.items[0].key; // 'Sunday' WeekEnum.items[0].label; // I love Sunday ``` ### 4. Array Format The array format is useful when you need to create enums with dynamic data, for example the data from an API. You can also use dynamic field mapping rules to adapt to various different data structures. Please refer to the [Custom Field Mapping](#user-content--custom-field-mapping-in-array-format-initialization) section for more details. ```js import { Enum } from 'enum-plus'; const pets = [ { value: 1, key: 'Dog', label: 'Dog' }, { value: 2, key: 'Cat', label: 'Cat' }, { value: 3, key: 'Rabbit', label: 'Rabbit' }, ]; const PetEnum = Enum(pets); PetEnum.Dog; // 1 PetEnum.label(1); // Dog ``` ### 5. Native Enum Format Create from native enums. It benefits from the native enum's `auto-incrementing` behavior. ```ts import { Enum } from 'enum-plus'; enum WeekNative { Sunday = 0, Monday, Tuesday, Wednesday, Thursday, Friday, Saturday, } const WeekEnum = Enum(WeekNative); WeekEnum.Sunday; // 0 WeekEnum.Monday; // 1 WeekEnum.Saturday; // 6 ``` > You can pass in some optional configuration options to better control the behavior of the enum. Please refer to the [Enum Configuration Options](#user-content--enum-configuration-options) section for details. ## API ### 💎 &nbsp; Pick enum values `Enum.XXX` Works like native `enum`, allowing you to access enum values directly. ```js WeekEnum.Sunday; // 0 WeekEnum.Monday; // 1 ``` --- ### 💎 &nbsp; named `Record<string, EnumItemClass>` An object that aggregates all enum items, allowing quick access to a specific enum item object through its `key`. ```js WeekEnum.named.Monday; // { key: 'Monday', value: 1, label: 'Monday' } ``` ### 💎 &nbsp; items `{ value, label, key, raw }[]` Returns a read-only array of all enum items. ```js WeekEnum.items; // [ { value: 0, label: 'Sunday', key: 'Sunday' }, { value: 1, label: 'Monday', key: 'Monday' }, ... ] ``` --- ### 💎 &nbsp; values `(string | number)[]` Returns an array of all enum item `value`(s). ```js WeekEnum.values; // [0, 1, 2, 3, 4, 5, 6] ``` --- ### 💎 &nbsp; labels `string[]` Returns an array of all enum item `label`(s). If [localization](#user-content-localization) has been enabled, the localized texts of each enum item will be returned. ```js WeekEnum.labels; // ['Sunday', 'Monday', ... 'Friday', 'Saturday'] ``` --- ### 💎 &nbsp; keys `string[]` Returns an array of all enum item `key`(s) ```js WeekEnum.keys; // ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'] ``` --- ### 💎 &nbsp; meta `Record<string, any[]>` Returns an object containing arrays of custom field values for all enum items. Each key in the object corresponds to a custom field name, and the value is an array of that field's values across all enum items. ```js const ColorEnum = Enum({ Red: { value: 1, label: 'Red', hex: '#FF0000' }, Green: { value: 2, label: 'Green', hex: '#00FF00' }, Blue: { value: 3, label: 'Blue', hex: '#0000FF' }, }); ColorEnum.meta.hex; // ['#FF0000', '#00FF00', '#0000FF'] ``` Additionally, you can quickly access custom fields of an enum item through the `named` and `raw` properties. ```js ColorEnum.named.Red.raw.hex; // '#FF0000' ``` --- ### 💎 &nbsp; has <sup>**_\[F]_**</sup> &nbsp; `has(keyOrValue?: string | number): boolean` Check if the value or key of an enum item exists. ```js WeekEnum.has(1); // true WeekEnum.has('Sunday'); // true WeekEnum.has(9); // false WeekEnum.has('Birthday'); // false ``` --- ### 💎 &nbsp; findBy <sup>**_\[F]_**</sup> &nbsp; `findBy(field: string, value: any): EnumItemClass | undefined` Find an enum item by a specific field and its value. Returns the enum item object if found, otherwise returns `undefined`. The `field` parameter can be one of the built-in fields: `key`, `value`, `label`, or any meta field defined in the enum item. ```js ColorEnum.findBy('value', 1); // { key: 'Red', value: 1, label: 'Red', hex: '#FF0000' } ColorEnum.findBy('key', 'Red'); // { key: 'Red', value: 1, label: 'Red', hex: '#FF0000' } ColorEnum.findBy('hex', '#FF0000'); // { key: 'Red', value: 1, label: 'Red', hex: '#FF0000' } ``` > If you need to get the meta fields of a known enum item, it is recommended to use the `named` and `raw` property, for example: `ColorEnum.named.Red.raw.hex`. --- ### 💎 &nbsp; label <sup>**_\[F]_**</sup> &nbsp; `label(keyOrValue?: string | number): string | undefined` Gets the display name of an enum item according to its value or key. If [localization](#user-content-localization) is enabled, the localized text will be returned. ```js WeekEnum.label(1); // Monday WeekEnum.label('Monday'); // Monday, this is label, not key ``` --- ### 💎 &nbsp; key <sup>**_\[F]_**</sup> &nbsp; `key(value?: string | number): string | undefined` Find the key of an enum item by its value. It's also known as [reverse mapping](https://www.typescriptlang.org/docs/handbook/enums.html#reverse-mappings). If not found, `undefined` is returned. ```js WeekEnum.key(1); // Monday (this is key, not label) ``` --- ### 💎 &nbsp; raw <sup>**_\[F^1]_**</sup> &nbsp; `raw(): Record<K, T[K]>` <br/> <sup>**_\[F^2]_**</sup> &nbsp; `raw(keyOrValue: V | K): T[K]` The `raw` method has two overloads. The first one is to return the original initialization object of the whole enum collection, i.e. the first parameter of the `Enum` method. The second one is to return the original initialization object of a single enum item, i.e. the sub-object of the corresponding field in the first parameter of the `Enum` method. The main purpose of the `raw` method is to get the extended custom fields of the enum items. ```js const WeekEnum = Enum({ Sunday: { value: 0, label: 'Sunday', happy: true }, Monday: { value: 1, label: 'Monday', happy: false }, }); WeekEnum.raw(0).happy; // true WeekEnum.raw(0); // { value: 0, label: 'Sunday', happy: true } WeekEnum.raw('Monday'); // { value: 1, label: 'Monday', happy: false } WeekEnum.raw(); // { Sunday: { value: 0, label: 'Sunday', happy: true }, Monday: { value: 1, label: 'Monday', happy: false } } ``` > Tips: If you want to access the metadata fields of a known enum item, using `enum.named.XXX.raw` is a good option, for example: `WeekEnum.named.Sunday.raw.happy`. --- ### 💎 &nbsp; toList <sup>**_\[F^1]_**</sup> &nbsp; `toList(): { value, label }[]` <br/> <sup>**_\[F^2]_**</sup> &nbsp; `toList(options?: { valueField?: string; labelField?: string }): { [key: string]: any }[]` Converts the enum items to an array of objects, each containing `value` and `label` fields by default. You can customize the field names using the `options` parameter. ```js WeekEnum.toList(); // [ // { value: 0, label: 'Sunday' }, // { value: 1, label: 'Monday' }, // ... // { value: 6, label: 'Saturday' } // ] WeekEnum.toList({ valueField: 'id', labelField: 'name' }); // [ // { id: 0, name: 'Sunday' }, // { id: 1, name: 'Monday' }, // ... // { id: 6, name: 'Saturday' } // ] ``` --- ### 💎 &nbsp; toMap <sup>**_\[F^1]_**</sup> &nbsp; `toMap(): Record<string, string | number>` <br/> <sup>**_\[F^2]_**</sup> &nbsp; `toMap(options?: { keySelector?: string; valueSelector?: string }): Record<string, any>` Converts the enum items to a key-value map object, where the keys are the enum values and the values are the enum labels by default. You can customize the key and value fields using the `options` parameter. ```js WeekEnum.toMap(); // { // "0": 'Sunday', // "1": 'Monday', // ... // "6": 'Saturday' // } WeekEnum.toMap({ keySelector: 'key', valueSelector: 'value' }); // { // "Sunday": 0, // "Monday": 1, // ... // "Saturday": 6 // } ``` --- ### 💎 &nbsp; name `string` The display name of the whole enum collection. It's optional and can be set as the second parameter of the `Enum` method. This name can be either a plain string or a localization key to support internationalized text. Please refer to the [Localization](#user-content-localization) section for more details. ```js const WeekEnum = Enum( { Sunday: { value: 0, label: 'Sunday' }, Monday: { value: 1, label: 'Monday' }, }, { name: 'i18n.enums.week', // A regular string or localization key } ); WeekEnum.name; // Week or 周, depending on the current language setting ``` > Enums are usually used to generate dropdown menus in forms, or show item text in table cells. The display name of the enum type often serves as the form field label or table caption. By utilizing the `name` property, you can centralize the management of both the enum type's display name and its items' labels, simplifying maintenance and ensuring consistency across your application. --- ### ⚡️ &nbsp; valueType &nbsp; <sup>**_\[TypeScript Only]_**</sup> `value1 | value2 | ...` In TypeScript, provides a union type containing all enum values. It's usually used for type constraints on variables and method parameters, or used in component properties. ```ts const weekValue: typeof WeekEnum.valueType = 1; function setToday(day: typeof WeekEnum.valueType) { // ... } const MyComponent = (props: { day: typeof WeekEnum.valueType }) => { // ... }; ``` > Note: This is a TypeScript type and cannot be called at runtime. Calling it at runtime will throw an error. --- ### ⚡️ &nbsp; keyType &nbsp; <sup>**_\[TypeScript Only]_**</sup> `key1 | key2 | ...` Similar to `valueType`, provides a union type of all enum keys in TypeScript. ```ts type WeekKeys = typeof WeekEnum.keyType; // 'Sunday' | 'Monday' const weekKey: typeof WeekEnum.keyType = 'Monday'; const weekKeys: (typeof WeekEnum.keyType)[] = ['Sunday', 'Monday']; ``` > Note: This is a TypeScript type and cannot be called at runtime. Calling it at runtime will throw an error. --- ### ⚡️ &nbsp; rawType &nbsp; <sup>**_\[TypeScript Only]_**</sup> `{ value: V, label: string, [...] }` Provides a type of the original initialization object of the whole enum collection, i.e. the type of the first parameter of the `Enum` method. Do not confuse it with the `raw` method. The `raw` method is a runtime method, while `rawType` is a TypeScript type. ```ts type WeekRaw = typeof WeekEnum.rawType; // { Sunday: { value: 0, label: string }, Monday: { value: 1, label: string } } ``` > Note: This is a TypeScript type and cannot be called at runtime. Calling it at runtime will throw an error. --- ## Static Methods ### 💎 &nbsp; Enum.isEnum <sup>**_\[F]_**</sup> &nbsp; `isEnum(obj: any): boolean` Check if a given object is an instance created by the `Enum` function. ```js Enum.isEnum(WeekEnum); // true Enum.isEnum({}); // false ``` --- ### 💎 &nbsp; Enum.localize <sup>**_\[F]_**</sup> &nbsp; `(key: string) => string` Set a global localization function for all enums. This function will be used to get the localized text for enum items and enum type names. Please refer to the [Localization](#user-content-localization) section for more details. ```js import i18n from 'i18next'; Enum.localize = (key) => i18n.t(key); ``` --- ### 💎 &nbsp; Enum.extends <sup>**_\[F]_**</sup> &nbsp; `(obj: Record<string, Function>) => void` Extend the `Enum` objects with custom methods. More details can be found in the [Extensibility](#user-content-extensibility) section. ```js Enum.extends({ sayHello() { return `Hello EnumPlus!`; }, }); ``` --- ### 💎 &nbsp; Enum.install <sup>**_\[F]_**</sup> &nbsp; `(plugin: Plugin, options?: any) => void` Install a plugin to extend the functionality of all enums. More details can be found in the [Plugin System](#user-content-plugin-system) section. ```js import i18nextPlugin from '@enum-plus/plugin-i18next'; Enum.install(i18nextPlugin); ``` ## Global Configuration `Enum.config` provides some global configuration options that affect the behavior and features of enums. ### autoLabel `Enum.config.autoLabel` is a global configuration option used to automatically generate labels for enum items. It allows you to set the `options.labelPrefix` option when defining an enum, which sets a `label` prefix for all enum items. Enum items only need to set the base value and can even omit the `label` field (when same as the `key` field). This reduces repetitive code and improves the conciseness of enum definitions. `Enum.config.autoLabel` can be a boolean value or a function type to implement more complex logic. - `true` - The default value, enabling the automatic label generation feature. The final value of the enum item's `label` will be automatically set to `labelPrefix` + `label`. If the `label` field is omitted, the `labelPrefix` + `key` rule will be used. Of course, if the `labelPrefix` is not set when creating the enum, this option will have no effect. - `function` - A custom function used to customize the `label` generation rule for each enum item. This function accepts an options object parameter that contains: `item` (the enum item object) and `labelPrefix`, and returns a string as the final `label` value. ```js Enum.config.autoLabel = ({ item, labelPrefix }) => { return `${labelPrefix}.${item.key.lowerFirst()}`; }; ``` - `false` - Disables the automatic label generation feature. Enum items must explicitly provide the `label` field. > Note that `autoLabel` is also an option that can be set when creating an enum as `options.autoLabel`. The usage is the same as `Enum.config.autoLabel`, and it overrides the global configuration for that specific enum. ## User Stories ### 💡 Prevent Magic Numbers ```js const WeekEnum = Enum({ Sunday: { value: 0, label: 'Sunday' }, Monday: { value: 1, label: 'Monday' }, }); if (today === WeekEnum.Sunday) { // It's Sunday, enjoy your day! } else if (today === WeekEnum.Monday) { // Oh no, it's Monday again... } ``` --- ### 💡 Check for Valid Enum Values ```js if (WeekEnum.has(value)) { // It's a valid enum value, safe to use } else { // Throw an error or use a default value } ``` --- ### 💡 Check for Enum Objects ```ts let values: number[] | undefined; if (Enum.isEnum(data)) { values = data.values; } else if (Array.isArray(data)) { values = data; } else { // Throw an error or use a default value } ``` --- ### 💡 Generate UI Controls Take React and Ant Design as examples. Please refer to the [Supports Various Frontend Frameworks](#user-content--supports-various-frontend-frameworks) section for more examples. ```jsx import { Menu, Select, Table } from 'antd'; import { ProFormCheckbox, ProFormSelect } from '@ant-design/pro-components'; const App = () => { return ( <> <Select options={WeekEnum.items} /> <Menu items={WeekEnum.toMenu()} /> <Table columns={[{ filters: WeekEnum.toFilter() }]} /> <ProFormSelect valueEnum={WeekEnum.toValueMap()} /> <ProFormCheckbox valueEnum={WeekEnum.toValueMap()} /> </> ); }; ``` > Need to install [@enum-plus/plugin-antd](https://github.com/shijistar/enum-plus/tree/master/packages/plugin-antd) plugin --- ### 💡 Internationalization for Enum Names and Labels Internationalization support. Set the `label` field to a localization key, so that it displays the corresponding text based on the current language environment. Please refer to the [Localization](#user-content-localization) section for more details. ```js WeekEnum.label(1); // Monday or 星期一, depending on the current locale WeekEnum.named.Monday.label; // Monday or 星期一, depending on the current locale WeekEnum.name; // Week or 周, depending on the current locale ``` --- ### 💡 Type Narrowing &nbsp; <sup>**_\[TypeScript Only]_**</sup> Type narrowing is a TypeScript-specific feature that allows you to use `valueType` to constrain the types of variables, function parameters, or component props. This prevents invalid assignments and enhances type safety in your code. - Variables ```ts type WeekValues = typeof WeekEnum.valueType; // 0 | 1 | ... | 5 | 6 const weekValue: WeekValues = 1; // ✅ Correct, 1 is a valid week enum value const weeks: WeekValues[] = [0, 1]; // ✅ Correct, 0 and 1 are valid week enum values const badWeekValue1: WeekValues = 'Weekend'; // ❌ Type error, "Weekend" is not a number const badWeekValue2: WeekValues = 8; // ❌ Error, 8 is not a valid week enum value const badWeeks: WeekValues[] = [0, 8]; // ❌ Error, 8 is not a valid week enum value ``` - Function Parameters ```ts function setDay(day: typeof WeekEnum.valueType) { // The type of day is narrowed to 0 | 1 | ... | 5 | 6 } setDay(1); // ✅ Correct setDay('Monday'); // ❌ Type error setDay(8); // ❌ Error, 8 is not a valid enum value ``` - Component Props ```ts type MyComponentProps = { day: typeof WeekEnum.valueType; // 0 | 1 | ... | 5 | 6 }; const MyComponent = (props: MyComponentProps) => { return <div>Today is {WeekEnum.label(props.day)}</div>; }; <MyComponent day={1} />; // ✅ Correct <MyComponent day="Monday" />; // ❌ Type error, "Monday" is not a number <MyComponent day={8} />; // ❌ Error, 8 is not a valid enum value ``` --- ### 💡 Support Metadata Fields That Can Serve as a Static Configuration System ```js const ColorEnum = Enum({ Red: { value: 1, hex: '#FF0000', icon: '🔥' }, Green: { value: 2, hex: '#00FF00', icon: '🍏' }, Blue: { value: 3, hex: '#0000FF', icon: '🔵' }, }); ColorEnum.values; // [1, 2, 3] ColorEnum.keys; // ['Red', 'Green', 'Blue'] ColorEnum.meta.hex; // ['#FF0000', '#00FF00', '#0000FF'] ColorEnum.meta.icon; // ['🔥', '🍏', '🔵'] ColorEnum.named.Red.raw.hex; // '#FF0000' ColorEnum.named.Red.raw.icon; // '🔥' ``` --- ### 💡 Supports Traversing Enum Items in a Read-Only Manner ```js Array.isArray(WeekEnum.items); // true WeekEnum.items.map((item) => item.value); // [0, 1, ..., 5, 6] WeekEnum.items.forEach((item) => { // ✅ Traversable }); for (const item of WeekEnum.items) { // ✅ Traversable } WeekEnum.items.push({ value: 2, label: 'Tuesday' }); // ❌ Not allowed, read-only WeekEnum.items.splice(0, 1); // ❌ Not allowed, read-only WeekEnum.items[0].label = 'foo'; // ❌ Not allowed, read-only ``` --- ### 💡 Enum Composition (Merging) ```js const PrimaryColorEnum = Enum({ Red: { value: 1, hex: '#FF0000' }, Green: { value: 2, hex: '#00FF00' }, Blue: { value: 3, hex: '#0000FF' }, }); const SecondaryColorEnum = Enum({ Yellow: { value: 4, hex: '#FFFF00' }, Cyan: { value: 5, hex: '#00FFFF' }, Magenta: { value: 6, hex: '#FF00FF' }, }); const AllColorEnum = Enum({ ...PrimaryColorEnum.raw(), ...SecondaryColorEnum.raw(), }); ``` --- ### 💡 Enable Code Intelligence for Enum Item Labels If you have internationalization enabled, you may want to enable code intelligence when entering enum item labels, listing all available localization resource key values to simplify the input process. You can achieve this by adding a new property to the `EnumLocaleExtends` interface. _index.ts_ ```ts declare module 'enum-plus/extension' { type EN = typeof import('./packages/locals/en').default; interface EnumLocaleExtends { LocaleKeys: keyof EN; } } ``` --- ### 💡 Supports JSDoc Comments on Enum Items, Enabling Code Intelligence Supports inline documentation through JSDoc, allowing engineers to view detailed comments by simply hovering over enum values in the editor. Please refer to the [Best Practices](./docs/best-practices.md) section for how to write good code. ```js const WeekEnum = Enum({ /** Represents Sunday */ Sunday: { value: 0, label: 'Sunday' }, /** Represents Monday */ Monday: { value: 1, label: 'Monday' }, }); WeekEnum.Monday; // Hover over Monday ``` ![JSDoc](https://cdn.jsdelivr.net/gh/shijistar/enum-plus@v3.1.1/public/jsdoc-en.png) We can see that both the enumeration value and the description of the enumeration item can be displayed at the same time, when the cursor hovers over an enumeration item. There is no need to jump away from the current position in the code to check the definitions. --- ### 💡 Supports Various Frontend Frameworks `Enum.items` can be consumed as the data source of UI components. Take `Select` component as examples. - **React-based UI libraries** [Ant Design](https://ant.design/components/select) | [Arco Design](https://arco.design/react/en-US/components/select) Select ```tsx import { Select } from 'antd'; <Select options={WeekEnum.items} />; ``` [Material-UI](https://mui.com/material-ui/react-select/) Select ```tsx import { MenuItem, Select } from '@mui/material'; <Select> {WeekEnum.items.map((item) => ( <MenuItem key={item.value} value={item.value}> {item.label} </MenuItem> ))} </Select>; ``` [Kendo UI](https://www.telerik.com/kendo-react-ui/components/dropdowns/dropdownlist) Select ```tsx import { DropDownList } from '@progress/kendo-react-dropdowns'; <DropDownList data={WeekEnum.items} textField="label" dataItemKey="value" />; ``` - **Vue-based UI libraries** [Element Plus](https://element-plus.org/en-US/component/select.html) Select ```html <el-select> <el-option v-for="item in WeekEnum.items" v-bind="item" /> </el-select> ``` [Ant Design Vue](https://antdv.com/components/select) | [Arco Design](https://arco.design/vue/en-US/component/select) Select ```html <a-select :options="WeekEnum.items" /> ``` [Vuetify](https://vuetifyjs.com/en/components/selects/) Select ```html <v-select :items="WeekEnum.items" item-title="label" /> ``` - **Angular-based UI libraries** [Angular Material](https://material.angular.io/components/select/overview) Select ```html <mat-select> @for (item of WeekEnum.items; track item.value) { <mat-option [value]="item.value">{{ item.label }}</mat-option> } </mat-select> ``` [NG-ZORRO](https://ng.ant.design/components/select/en) Select ```jsx <nz-select> @for (item of WeekEnum.items; track item.value) { <nz-option [nzValue]="item.value">{{ item.label }}</nz-option> } </nz-select> ``` --- ### 💡 Enum Configuration Options You can pass in an optional configuration object when creating an enum to customize its behavior and features. Here are some commonly used configuration options: ```ts interface EnumOptions { /** The name of the enum type, which can be a plain string or a localization key */ name?: string; /** * Set a label prefix for all enum items. For more details, please refer to the [Global * Configuration] section */ labelPrefix?: string; /** * The rule for automatically generating enum item labels. For more details, please refer to the * [Global Configuration] section */ autoLabel?: boolean | ((params: { item: EnumItemClass; labelPrefix?: string }) => string); /** * Enum instance-level localization function, which overrides the Enum.localize global * configuration function */ localize?: (localeKey: string) => string; } ``` For more configuration options, please refer to the next section. --- ### 💡 Custom Field Mapping in Array Format Initialization In the [4. Array Format](#user-content-4-array-format) section, we know that enums can be created from dynamic data arrays. However, the field names in the real world may be different from the default `value`, `label`, and `key`. In such cases, you can pass in a custom option to map these to other field names. ```js import { Enum } from 'enum-plus'; const data = await getPetsData(); // [ { id: 1, code: 'dog', name: 'Dog' }, // { id: 2, code: 'cat', name: 'Cat' }, // { id: 3, code: 'rabbit', name: 'Rabbit' } ]; const PetTypeEnum = Enum(data, { getValue: 'id', getLabel: 'name', getKey: 'code', // getKey is optional }); PetTypeEnum.items; // The output is: // [ { value: 1, label: 'Dog', key: 'dog' }, // { value: 2, label: 'Cat', key: 'cat' }, // { value: 3, label: 'Rabbit', key: 'rabbit' } ] ``` `getValue`, `getLabel`, `getKey` can also be a function to handle more complex business logic, for example: ```js const PetTypeEnum = Enum(petTypes, { getValue: (item) => item.id, getLabel: (item) => `${item.name} (${item.code})`, getKey: (item) => item.code, }); ``` --- ## Plugin System `enum-plus` provides a plugin system that allows you to extend the functionality of all enums. Plugins can add new methods or properties to all enum instances, greatly enhancing their capabilities. You can choose to install only the plugins you need, keeping the core library lightweight and efficient. ```ts import antdPlugin from '@enum-plus/plugin-antd'; import { Enum } from 'enum-plus'; Enum.install(antdPlugin); ``` After installing a plugin, it will add new methods or properties to all enum instances. For example, after installing the `@enum-plus/plugin-antd` plugin, you can use the `enum.toSelect` method to generate a Select component from the enum. Optionally, you can provide configuration options to customize the behavior of the plugin. For details on the configuration options for each plugin, please refer to the documentation of the respective plugins. ```ts import antdPlugin from '@enum-plus/plugin-antd'; import { Enum } from 'enum-plus'; Enum.install(antdPlugin, { toSelect: { valueField: 'id', // Sets the field representing the value in the data object generated by the toSelect method labelField: 'name', // Sets the field representing the label in the data object generated by the toSelect method }, }); ``` ### Awesome Plugins The following plugins are available. You can choose to install them based on your needs: - [@enum-plus/plugin-antd](https://github.com/shijistar/enum-plus/tree/master/packages/plugin-antd) [Ant Design](https://ant.design) oriented features, including `enum.toSelect`, `enum.toMenu`, `enum.toFilter`, and `enum.toValueMap`. With these methods, you can directly bind enums to the corresponding Ant Design components, greatly simplifying your code. - [@enum-plus/plugin-i18next](https://github.com/shijistar/enum-plus/tree/master/packages/plugin-i18next) Integrates [i18next](https://www.i18next.com) to enable internationalization of enum labels. - [@enum-plus/plugin-react-i18next](https://github.com/shijistar/enum-plus/tree/master/packages/plugin-react-i18next) Integrates [react-i18next](https://react.i18next.com) to enable internationalization of enum labels. - [@enum-plus/plugin-react](https://github.com/shijistar/enum-plus/tree/master/packages/plugin-react) React integration, including support for `Enum.localize` to return React components, and listening for language changes to auto update components. - [@enum-plus/plugin-i18next-vue](https://github.com/shijistar/enum-plus/tree/master/packages/plugin-i18next-vue) Integrates [i18next-vue](https://github.com/shijistar/enum-plus/tree/master/packages/plugin-i18next-vue) to enable internationalization of enum labels and listen for language changes to auto update components. - [@enum-plus/plugin-vue-i18n](https://github.com/shijistar/enum-plus/tree/master/packages/plugin-vue-i18n) Integrates [vue-i18n](https://vue-i18n.intlify.dev) to enable internationalization of enum labels and listen for language changes to auto update components. We are working on the following plugins: - [@enum-plus/plugin-angular](https://github.com/shijistar/enum-plus/tree/master/packages/plugin-angular): Angular integration, including support for `Enum.localize` to return Angular components, and listening for language changes to auto update components. _We need your help to develop this plugin!_ > If the plugin you are searching for is not available, or if you want to develop your own plugin, please refer to the [Plugin Development Guide](./docs/plugin-development.md). You can develop new plugins in the official enum-plus repository or publish your developed plugins to npm and share your plugin links here. We sincerely need your help to enrich the plugin ecosystem! --- ## Localization `enum-plus` does not include built-in internationalization capabilities by default. Therefore, the `label` field of enum items is treated as a plain string and returns the original text directly. To add localization support to `enum-plus`, the simplest way is to install the corresponding [i18n plugin](#user-content-plugin-system), such as `@enum-plus/plugin-i18next`, which automatically passes the values of the `label` and `name` fields to i18next for translation. ```bash npm install @enum-plus/plugin-i18next i18next ``` Then install the plugin in the project entry file: _index.js_ ```js import i18nextPlugin from '@enum-plus/plugin-i18next'; import { Enum } from 'enum-plus'; Enum.install(i18nextPlugin); ``` After installing the plugin, the `label` and `name` fields of the enum will be automatically translated through i18next. ```js const WeekEnum = Enum( { Sunday: { value: 0, label: 'week.sunday' }, Monday: { value: 1, label: 'week.monday' }, }, { name: 'weekDays.name' } ); WeekEnum.label(1); // Monday or 星期一, depending on the current locale WeekEnum.named.Monday.label; // Monday or 星期一, depending on the current locale WeekEnum.name; // Week or 周, depending on the current locale ``` This plugin also supports custom i18next options, and even allows complete control over the `localize` method. Please refer to the [plugin documentation](https://github.com/shijistar/enum-plus/tree/master/packages/plugin-i18next#plugin-options) for more details. If you need to automatically update the UI after switching languages, this requires the capabilities of frameworks like React, Vue, or Angular. Please consider using plugins such as [@enum-plus/plugin-react](https://github.com/shijistar/enum-plus/tree/master/packages/plugin-react) or [@enum-plus/plugin-vue](https://github.com/shijistar/enum-plus/tree/master/packages/plugin-vue). If you are using other internationalization libraries, such as `react-intl`, `vue-i18next`, or `ngx-translate`, you can integrate these libraries by overwriting the `Enum.localize` method. _my-extension.js_ ```js import { Enum } from 'enum-plus'; Enum.localize = (key) => { // This is a pseudo code, please adjust it according to your own logic return intl.formatMessage({ id: key }); }; ``` > Once you have completed this feature, it is recommended that you consider publishing it as an npm package and share it in the [Awesome Plugins](#user-content-awesome-plugins) section, so that others can benefit from your work. If you believe that this project is very general, you can also consider submitting it to the official [enum-plus](https://github.com/shijistar/enum-plus/tree/master/packages) plugin repository. For specific development rules, please refer to the [Plugin Development Guide](./docs/plugin-development.md). --- ## Extensibility Enum provides a wealth of built-in methods and properties that can satisfy most common use cases. However, if they are not sufficient, you can use `Enum.extends` to add more custom methods. These extensions will be globally applied to all enum instances, including those created before the extension was applied, and will take effect immediately without any additional setup. > In fact, the entire [Plugin System](#user-content-plugin-system) and the `Enum.install` method are implemented using `Enum.extends` at the underlying level. - **TypeScript Projects** _my-enum-extension.ts_ ```ts // Implementation code Enum.extends({ toMySelect() { return this.items.map((item) => ({ value: item.value, title: item.label })); }, reversedItems() { return this.items.toReversed(); }, }); // Type declaration for better type hints declare module 'enum-plus/extension' { export interface EnumExtension<T, K, V> { toMySelect: () => { value: V; title: string }[]; reversedItems: () => EnumItemClass<EnumItemInit<V>, K, V>[]; } } ``` _index.ts_ Then import this file in the entry file of your project: ```ts import './my-enum-extension'; WeekEnum.toMySelect(); // [{ value: 0, title: 'Sunday' }, { value: 1, title: 'Monday' }] ``` - **JavaScript Projects** _my-enum-extension.js_ ```js import { Enum } from 'enum-plus'; // Implementation code Enum.extends({ toMySelect() { return this.items.map((item) => ({ value: item.value, title: item.label })); }, reversedItems() { return this.items.toReversed(); }, }); ``` _my-enum-extension.js.d.ts_ ```ts import { EnumExtension, EnumItemClass, EnumItemInit } from 'enum-plus'; // Type declaration for better type hints declare module 'enum-plus/extension' { export interface EnumExtension<T, K, V> { toMySelect: () => { value: V; title: string }[]; reversedItems: () => EnumItemClass<EnumItemInit<V>, K, V>[]; } } ``` _index.js_ Then import this file in the entry file of your project: ```js import './my-enum-extension'; WeekEnum.toMySelect(); // [{ value: 0, title: 'Sunday' }, { value: 1, title: 'Monday' }] ``` Please note that `EnumExtension` is a generic interface that accepts three type parameters, which represent: - `T`: Initialization object of the enum type (e.g., the object passed to `Enum()`) - `K`: Key of the enum item (e.g., Sunday, Monday) - `V`: Value of the enum items > If you want to provide more friendly type hints in the extension methods, you may need to use these type parameters. However these are all optional, if you don't need them, you can omit them. --- ## Naming Conflicts? `enum-plus` is designed with naming conflicts in mind. The namespace of enum items is separate from the methods and properties of the enum instance, minimizing the chances of conflicts. For example, when an enum item's name conflicts with a method name, you can access the overridden methods through the `items` property. ```js import { KEYS, VALUES } from 'enum-plus'; const WeekEnum = Enum({ foo: { value: 1 }, bar: { value: 2 }, keys: { value: 3 }, // Naming conflict values: { value: 4 }, // Naming conflict label: { value: 5 }, // Naming conflict named: { value: 6 }, // Naming conflict toList: { value: 7 }, // Naming conflict }); WeekEnum.foo; // 1 WeekEnum.bar; // 2 // Below are all enum items, which take precedence and override the original methods WeekEnum.keys; // 3 WeekEnum.values; // 4 WeekEnum.label; // 5 WeekEnum.named; // 6 WeekEnum.toList; // 7 // You can access these overridden methods via .items 🙂 WeekEnum.items[KEYS]; // ['foo', 'bar', 'keys', 'values', 'label', 'named', 'toList'] WeekEnum.items[VALUES]; // [1, 2, 3, 4, 5, 6, 7] WeekEnum.items.label(1); // 'foo' WeekEnum.items.named.foo; // { value: 1, label: 'foo', key: 'foo' } WeekEnum.items.toList(); // [{ value: 1, label: 'foo' }, ...] ``` > Note that `keys` and `values` are special because they are built-in methods of JavaScript arrays. To avoid altering the behavior of the `items` array, you need to use the `KEYS` and `VALUES` symbols as aliases to access them. For an even more extreme case, what if `items` conflicts with an enum item name? Don't worry, you can still access it via the `ITEMS` alias. ```js import { ITEMS } from 'enum-plus'; const WeekEnum = Enum({ foo: { value: 1 }, bar: { value: 2 }, items: { value: 3 }, // Naming conflict toList: { value: 4 }, // Naming conflict }); WeekEnum.items; // 3, enum item takes precedence and overrides items WeekEnum[ITEMS].toList(); // But you can access it via the ITEMS alias ``` --- ## Best Practices When using `enum-plus`, following these best practices can help ensure consistency, maintainability, and clarity in your codebase: 1. **Enum Type Naming:** Use `PascalCase` and end with `Enum` (e.g., _WeekEnum_, _ColorEnum_). 2. **Enum Item Naming:** Use `PascalCase` (e.g., _Sunday_, _Red_). This naming approach highlights the static and immutable nature of enumeration members. Moreover, in the IDE's intelligent prompting, they will be displayed at the top instead of being mixed with other method names, making it more convenient for viewing and selection. 3. **Semantic Clarity:** Ensure enum and item names have clear semantics. Good semantic naming serves as self-documentation, making code intent explicit and reducing cognitive overhead. 4. **Single Responsibility Principle:** Each enum type should represent a single, cohesive set of related constants. Avoid overlapping responsibilities between different enum types. 5. **Provide JSDoc Comments:** Provide JSDoc comments for each enum item and the enum type itself, explaining their purpose and usage. Comprehensive documentation enables IDE hover tooltips and improves code readability and maintainability. 6. **Internationalization Architecture:** Plan for internationalization from the outset by leveraging the library's [localization](#user-content-localization) features. A well-designed internationalization architecture minimizes future refactoring and facilitates global scalability. Here is an example that combines the above best practices to define an enum: ```js /** Represents the days of the week */ const WeekEnum = Enum( { /** Sunday */ Sunday: { value: 0, label: 'enums.week.sunday' }, /** Monday */ Monday: { value: 1, label: 'enums.week.monday' }, // ... /** Friday */ Friday: { value: 5, label: 'enums.week.friday' }, /** Saturday */ Saturday: { value: 6, label: 'enums.week.saturday' }, }, { name: 'enums.week.name' } ); ``` --- ## Compatibility enum-plus is designed to be compatible with a wide range of environments, including modern browsers, Node.js, and various build tools. Below are the compatibility details for different environments: ### Browser Environments - **Modern Bundlers**: For bundlers supporting the [exports](https://nodejs.org/api/packages.html#exports-sugar) configuration (such as webpack 5+, vite, rollup), imports