Esta é a saída JavaScript desse enum:
var MyEnum;
(function (MyEnum) {
MyEnum[MyEnum["First"] = 0] = "First";
MyEnum[MyEnum["Second"] = 1] = "Second";
MyEnum[MyEnum["Third"] = 2] = "Third";
})(MyEnum || (MyEnum = {}));
Que é um objeto como este:
{
"0": "First",
"1": "Second",
"2": "Third",
"First": 0,
"Second": 1,
"Third": 2
}
Membros Enum com valores de string
TypeScript 2.4 adicionou a capacidade de enums para possivelmente ter valores de membro de enum de string. Portanto, é possível terminar com um enum parecido com o seguinte:
enum MyEnum {
First = "First",
Second = 2,
Other = "Second"
}
var MyEnum;
(function (MyEnum) {
MyEnum["First"] = "First";
MyEnum[MyEnum["Second"] = 2] = "Second";
MyEnum["Other"] = "Second";
})(MyEnum || (MyEnum = {}));
Obtendo nomes de membros
Podemos olhar o exemplo imediatamente acima para tentar descobrir como obter os membros enum:
{
"2": "Second",
"First": "First",
"Second": 2,
"Other": "Second"
}
Aqui está o que eu inventei:
const e = MyEnum as any;
const names = Object.keys(e).filter(k =>
typeof e[k] === "number"
|| e[k] === k
|| e[e[k]]?.toString() !== k
);
Valores de Membro
Assim que tivermos os nomes, podemos percorrê-los para obter o valor correspondente, fazendo:
const values = names.map(k => MyEnum[k]);
Classe de extensão
Acho que a melhor maneira de fazer isso é criar suas próprias funções (ex. EnumEx.getNames(MyEnum)
). Você não pode adicionar uma função a um enum.
class EnumEx {
private constructor() {
}
static getNamesAndValues(e: any) {
return EnumEx.getNames(e).map(n => ({ name: n, value: e[n] as string | number }));
}
static getNames(e: any) {
return Object.keys(e).filter(k =>
typeof e[k] === "number"
|| e[k] === k
|| e[e[k]]?.toString() !== k
);
}
static getValues(e: any) {
return EnumEx.getNames(e).map(n => e[n] as string | number);
}
}
for of
instrução em vez defor in
Com TypeScript> = 2.4, você pode definir enums de string:
enum Color { RED = 'Red', ORANGE = 'Orange', YELLOW = 'Yellow', GREEN = 'Green', BLUE = 'Blue', INDIGO = 'Indigo', VIOLET = 'Violet' }
Saída JavaScript ES5:
var Color; (function (Color) { Color["RED"] = "Red"; Color["ORANGE"] = "Orange"; Color["YELLOW"] = "Yellow"; Color["GREEN"] = "Green"; Color["BLUE"] = "Blue"; Color["INDIGO"] = "Indigo"; Color["VIOLET"] = "Violet"; })(Color || (Color = {}));
Que é um objeto como este:
const Color = { "RED": "Red", "ORANGE": "Orange", "YELLOW": "Yellow", "GREEN": "Green", "BLUE": "Blue", "INDIGO": "Indigo", "VIOLET": "Violet" }
Assim, no caso de enums de string, não há necessidade de filtrar coisas
Object.keys(Color)
eObject.values(Color)
(*) são suficientes:const colorKeys = Object.keys(Color) as (keyof typeof Color)[]; console.log('colorKeys =', colorKeys); // ["RED", "ORANGE", "YELLOW", "GREEN", "BLUE", "INDIGO", "VIOLET"] const colorValues = Object.values(Color); console.log('colorValues =', colorValues); // ["Red", "Orange", "Yellow", "Green", "Blue", "Indigo", "Violet"] colorKeys.map(colorKey => { console.log(`color key = ${colorKey}, value = ${Color[colorKey]}`); }); /* color key = RED, value = Red color key = ORANGE, value = Orange color key = YELLOW, value = Yellow color key = GREEN, value = Green color key = BLUE, value = Blue color key = INDIGO, value = Indigo color key = VIOLET, value = Violet */
Veja o exemplo online no playground do TypeScript
(*) Polyfill necessário para navegadores antigos, consulte https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_objects/Object/values#Browser_compatibility
fonte
Element implicitly has an 'any' type because expression of type 'string' can't be used to index type 'typeof Color'. No index signature with a parameter of type 'string' was found on type 'typeof Color'.
Object.keys(Color) as (keyof typeof Color)[]
Você pode adicionar funções para obter os nomes e índices do enum:
enum MyEnum { First, Second, Third } namespace MyEnum { function isIndex(key):boolean { const n = ~~Number(key); return String(n) === key && n >= 0; } const _names:string[] = Object .keys(MyEnum) .filter(key => !isIndex(key)); const _indices:number[] = Object .keys(MyEnum) .filter(key => isIndex(key)) .map(index => Number(index)); export function names():string[] { return _names; } export function indices():number[] { return _indices; } } console.log("MyEnum names:", MyEnum.names()); // Prints: MyEnum names: ["First", "Second", "Third"] console.log("MyEnum indices:", MyEnum.indices()); // Prints: MyEnum indices: [0, 1, 2]
Observe que você pode apenas exportar as consts
_names
e em_indices
vez de expô-las por meio de uma função exportada, mas como os membros exportados são membros do enum, é indiscutivelmente mais claro tê-los como funções para que não sejam confundidos com os membros reais do enum.Seria bom se o TypeScript gerasse algo assim automaticamente para todos os enums.
fonte
Não há conceito de RTTI (informações de tipo em tempo de execução) no TypeScript (pense: reflexão), portanto, para fazer isso, é necessário conhecimento do JavaScript transpilado. Portanto, assumindo TypeScript 0,95:
torna-se:
var MyEnum; (function(MyEnum) { MyEnum[MyEnum["First"] = 0] = "First"; MyEnum[MyEnum["Second"] = 1] = "Second"; MyEnum[MyEnum["Third"] = 2] = "Third"; }
Então, isso é modelado como um objeto regular em javascript, onde
MyEnum.0 == "First"
eMyEnum.First == 0
. Portanto, para enumerar todos os nomes de enum, você precisa obter todas as propriedades que pertencem ao objeto e que também não são números:for (var prop in MyEnum) { if (MyEnum.hasOwnProperty(prop) && (isNaN(parseInt(prop)))) { console.log("name: " + prop); } }
Ok, agora que disse a você como fazer, posso dizer que é uma má ideia . Você não está escrevendo uma linguagem gerenciada, então não pode trazer esses hábitos. Ainda é um simples JavaScript antigo. Se eu quisesse usar uma estrutura em JavaScript para preencher algum tipo de lista de opções, usaria um array simples e antigo. Um enum não é a escolha certa aqui, trocadilho intencional. O objetivo do TypeScript é gerar um JavaScript bonito e idiomático. Usar enums dessa maneira não preserva esse objetivo.
fonte
Usei a solução proposta por David Sherret e escrevi uma biblioteca npm que você pode usar chamada
enum-values
...Git: enum-values
// Suppose we have an enum enum SomeEnum { VALUE1, VALUE2, VALUE3 } // names will be equal to: ['VALUE1', 'VALUE2', 'VALUE3'] var names = EnumValues.getNames(SomeEnum); // values will be equal to: [0, 1, 2] var values = EnumValues.getValues(SomeEnum);
fonte
Uma linha para obter uma lista de entradas (objetos / pares de valor-chave):
Object.keys(MyEnum).filter(a=>a.match(/^\D/)).map(name=>({name, value: MyEnum[name] as number}));
fonte
enum MyEnum { First, Second, Third, NUM_OF_ENUMS } for(int i = 0; i < MyEnum.NUM_OF_ENUMS; ++i) { // do whatever you need to do. }
fonte
Se você deseja associar valores de strings ao seu enum, esses métodos não funcionam. Para ter uma função genérica, você pode fazer:
function listEnum(enumClass) { var values = []; for (var key in enumClass) { values.push(enum[key]); } values.length = values.length / 2; return values; }
Funciona porque o TypeScript adicionará chaves na primeira etapa e valores na segunda etapa.
No TypeScript é:
var listEnums = <T> (enumClass: any): T[]=> { var values: T[] = []; for (var key in enumClass) { values.push(enumClass[key]); } values.length = values.length / 2; return values; }; var myEnum: TYPE[] = listEnums<TYPE>(TYPE);
fonte
A resposta de joe me fez perceber que é muito mais fácil confiar nas primeiras N teclas numéricas do que fazer testes mais complexos:
function getEnumMembers(myEnum): string[] { let members = [] for(let i:number = 0; true; i++) { if(myEnum[i] === undefined) break members.push(myEnum[i]) } return members } enum Colors { Red, Green, Blue } console.log(getEnumMembers(myEnum))
fonte
para nodejs:
const { isNumber } = require('util'); Object.values(EnumObject) .filter(val => isNumber(val)) .map(val => { // do your stuff })
fonte
Iterando sobre um enum
String Enums são mais usados para isso. Aqui está um exemplo:
// This is a string enum enum MyEnum { First = 'First', Second = 'Second', Third = 'Third', } // An enum is a TS concept // However his MyEnum compiles to JS object: // { // "First": "First", // "Second": "Second", // "Third": "Third" // } // Therefore we can get the keys in the following manner: const keysArray = Object.keys(MyEnum); for (const key of keysArray) { console.log(key) } // [LOG]: "First" // [LOG]: "Second" // [LOG]: "Third"
fonte