如何为属性创建TypeScript @enumerable(false)装饰器

mve*_*and 5 properties decorator enumerable typescript

我想在TypeScript中创建一个装饰器,以便能够使类属性不可枚举.

我在这里找到了@enumerable的一个例子:https://www.typescriptlang.org/docs/handbook/decorators.html#method-decorators 但这似乎只适用于方法,而不是属性:

https://www.typescriptlang.org/docs/handbook/decorators.html#property-decorators

注意由于在TypeScript中如何初始化属性修饰符,因此不提供属性描述符作为属性修饰符的参数.这是因为当定义原型的成员时,当前没有机制来描述实例属性,也无法观察或修改属性的初始化器.因此,属性装饰器只能用于观察已为类声明特定名称的属性.

有没有办法为类属性创建一个@enumerable装饰器?

谢谢

mve*_*and 14

我最终得到了这个解决方案:

/**
 * @enumerable decorator that sets the enumerable property of a class field to false.
 * @param value true|false
 */
function enumerable(value: boolean) {
    return function (target: any, propertyKey: string) {
        let descriptor = Object.getOwnPropertyDescriptor(target, propertyKey) || {};
        if (descriptor.enumerable != value) {
            descriptor.enumerable = value;
            descriptor.writable= true;
            Object.defineProperty(target, propertyKey, descriptor)
        }
    };
}
Run Code Online (Sandbox Code Playgroud)

用法:

class User {
    id:string;

    @enumerable(false)
    name: string;
}
Run Code Online (Sandbox Code Playgroud)

测试:

   var user = new User();
   user.id = 1;
   user.name = 'John Doe';
   for (key in user){ console.log(key, user[key]);}
Run Code Online (Sandbox Code Playgroud)

产量

id 1
Run Code Online (Sandbox Code Playgroud)

没有使用装饰器的相同测试

id 1
name John Doe
Run Code Online (Sandbox Code Playgroud)


小智 6

该解决方案实际上不起作用,或者不适用于现代打字稿。然而,以下内容确实如此:

const enumerable: {
    (target: any, name: string): void;
    (target: any, name: string, desc: PropertyDescriptor): PropertyDescriptor;
} = (target: any, name: string, desc?: any) => {
    if(desc) {
        desc.enumerable = true;
        return desc;
    }
    Object.defineProperty(target, name,  {
        set(value) {
            Object.defineProperty(this, name, {
                value, enumerable: true, writable: true, configurable: true,
            });
        },
        enumerable: true,
        configurable: true,
    });
};

const nonenumerable: {
    (target: any, name: string): void;
    (target: any, name: string, desc: PropertyDescriptor): PropertyDescriptor;
} = (target: any, name: string, desc?: any) => {
    if(desc) {
        desc.enumerable = false;
        return desc;
    }
    Object.defineProperty(target, name,  {
        set(value) {
            Object.defineProperty(this, name, {
                value, writable: true, configurable: true,
            });
        },
        configurable: true,
    });
};
Run Code Online (Sandbox Code Playgroud)