猿问

无法使用代理在自定义元素上捕获访问器调用?

我正在使用 customElements.define 注册一些自定义元素,并希望在成员访问器上自动设置陷阱,以便在事件更改时发出事件


class State extends HTMLElement {

    public someValue = 1;


    public constructor() {

        super();

        console.log('State constructor');

    }

}


const oProxy = new Proxy(State, {

    get(target, prop: string) {

        console.log(`GET trap ${prop}`);

        return Reflect.get(target, prop);

    },

    set(target, prop: string, value: any) {

        console.log(`SET trap ${prop}`);

        return Reflect.set(target, prop, value);

    }

});


customElements.define('my-state', oProxy);


const oStateEl = document.querySelector('my-state');

console.log(oStateEl.someValue);

console.log(oStateEl.someValue = 2);

console.log(oStateEl.someValue);

我的浏览器似乎没有上述代码的问题,我可以看到一些陷阱输出,因为元素已设置


GET trap prototype

GET trap disabledFeatures

GET trap formAssociated

GET trap prototype

但是,当我手动获取/设置值时,陷阱不会被触发。这有可能吗?


慕侠2389804
浏览 90回答 1
1回答

慕尼黑的夜晚无繁华

我最终所做的是将所有成员变量值移动到一个私有对象,并在自定义元素挂载到DOM上后立即为每个对象动态定义一个 getter/setter,如下所示...//class State extends HTMLElement {    protected _data: object = {}    public connectedCallback() {        // Loop over member vars        Object.getOwnPropertyNames(this).forEach(sPropertyKey => {            // Ignore private            if(sPropertyKey.startsWith('_')) {                return;            }            // Copy member var to data object            Reflect.set(this._data, sPropertyKey, Reflect.get(this, sPropertyKey));            // Remove member var            Reflect.deleteProperty(this, sPropertyKey);            // Define getter/setter to access data object            Object.defineProperty(this, sPropertyKey, {                set: function(mValue: any) {                    console.log(`setting ${sPropertyKey}`);                    Reflect.set(this._data, sPropertyKey, mValue);                },                get: function() {                    return this._data[sPropertyKey];                }            });        });    }}// class SubState extends State {    public foobar = 'foobar_val';    public flipflop = 'flipflop_val';    public SubStateMethod() { }}//window.customElements.define('sub-state', SubState);//const oState = document.querySelector('sub-state') as SubState;oState.foobar = 'foobar_new_val';这样,我仍然可以像往常一样获取/设置对象的值,typescript很高兴成员变量存在,并且我可以在访问成员时触发自定义事件 - 同时允许自定义元素存在于DOM就绪的标记中。
随时随地看视频慕课网APP

相关分类

JavaScript
我要回答