12

因此,在我创建的使用自定义元素的库中,您显然需要在CustomElementsRegistry实例化它之前定义类。

截至目前,这正在通过装饰器解决:

class Component extends HTMLElement {

    static register (componentName) {
        return component => {
            window.customElements.define(componentName, component);
            return component;
        }
    }
}

@Component.register('my-element')
class MyElement extends Component { }

document.body.appendChild(new MyElement());

这可行,但是,我想在类的实例化时自动注册自定义元素(这样作者就不必将装饰器添加到他们编写的每个组件中)。这可以通过Proxy.


但是,我的问题是,当我尝试在构造函数上使用 Proxy 并尝试返回目标实例时,我仍然得到Illegal Constructor,就好像该元素从未在注册表中定义过一样。

这显然与我在代理内部实例化类的方式有关,但我不确定如何做到这一点。我的代码如下:

请在最新的 Chrome 中运行:

class Component extends HTMLElement {

    static get componentName () {
        return this.name.replace(/[A-Z]/g, char => `-${ char.toLowerCase() }`).substring(1);
    }
}

const ProxiedComponent = new Proxy(Component, {

    construct (target, args, extender) {
        const { componentName } = extender;
  	
        if (!window.customElements.get(componentName)) {
            window.customElements.define(componentName, extender);
        }
    
        return new target(); // culprit
    }
});

class MyElement extends ProxiedComponent { }

document.body.appendChild(new MyElement());

MyElement我怎样才能继续代理内部的继承链,而不会丢失我正在实例化类以使其不会引发Illegal Constructor异常这一事实的上下文?

4

2 回答 2

11

有2个问题:

  • new target()创建LibElement的实例,未注册为自定义元素。在这里你得到了Illegal Constructor错误。
  • 即使您注册LibElement生成的 DOM 元素也会<lib-element>,因为您调用new target并且此时 javascript 不知道子类。

我发现的唯一方法是使用ReflectAPI 创建正确的对象实例。

class LibElement extends HTMLElement {
    static get componentName () {
        return this.name.replace(/[A-Z]/g, char => `-${ char.toLowerCase() }`).substring(1);
    }
}

const LibElementProxy = new Proxy(LibElement, {
    construct (base, args, extended) {
        if (!customElements.get(extended.componentName)) {
            customElements.define(extended.componentName, extended);
        }
    
        return Reflect.construct(base, args, extended);
    }
});

class MyCustomComponent extends LibElementProxy {}
class MyCustomComponentExtended extends MyCustomComponent {}

document.body.appendChild(new MyCustomComponent());
document.body.appendChild(new MyCustomComponentExtended());

我真的很喜欢这种使用代理构造函数来自动注册自定义元素的想法)

于 2017-12-12T21:36:01.250 回答
0

您在这里非常接近解决方案,唯一的问题是原生HTMLElement无法使用new关键字实例化,必须通过document.createElement. 您可以重用您拥有的所有内容,并且只替换construct代理中方法的返回值:

class Component extends HTMLElement {

  static get componentName() {
    return this.name.replace(/[A-Z]/g, char => `-${ char.toLowerCase() }`).substring(1);
  }
}

const ProxiedComponent = new Proxy(Component, {

  construct(target, arguments, extender) {
    const {
      componentName
    } = target;

    if (!window.customElements.get(componentName)) {
      window.customElements.define(componentName, extender);
    }

    return document.createElement(target.componentName); // Properly constructs the new element
  }
});

class MyElement extends ProxiedComponent {}

document.body.appendChild(new MyElement());

于 2017-12-12T21:16:23.500 回答