Search code examples
javascriptdomweb-componentshadow-domcustom-element

Shorthand for extending native DOM element?


As explained in mdn, custom elements may be based on a native element. For Example, <button is="my-button">. What I'm looking for is kind of the opposite syntax. I'd like a shorthand for creating an element whose tag is custom and is based on a native element, such as <purchase-button is="button">. I'm not looking at add functionality to this custom element over what HTMLButtonElement provides. Just the name.

Yet, this doesn't work:

let purchaseButton = document.createElement('purchase-button', { is: 'button' })
document.body.appendChild(purchaseButton)

The resulting element does not extend HTMLButtonElemenet.


Solution

  • It's not possible, and it doesn't exist in the Custom Element specification.

    About this requirement, see issue 603 in the specification forum.

    Some are discussing similar needs in other issues (for example #509, good luck for the reading...).

    As a workaround, define a <purchase-button> custom element into which you insert a <button>. Then you can use a Shadow DOM with the <slot> element to reflect the original content of the <purchase-button>.

    customElements.define( 'purchase-button', class extends HTMLElement {
        constructor() {
            super()
            this.attachShadow( { mode: 'open' } )
                .innerHTML = '<button><slot></slot></button>'
        }
    } )
    
    PB.onclick = ev => 
      console.info( '[%s] clicked', ev.target.textContent )
    <purchase-button id=PB>Pay</purchase-button>