Skip to content

Instantly share code, notes, and snippets.

@rodydavis
Last active March 5, 2025 15:29
Show Gist options
  • Save rodydavis/27c5c9644d6a9a517bc709d3579ae6ae to your computer and use it in GitHub Desktop.
Save rodydavis/27c5c9644d6a9a517bc709d3579ae6ae to your computer and use it in GitHub Desktop.
HTML Web Component with support for SSR
// @ts-check
import { WebComponent } from "./utils.js";
const tagName = "my-element";
const template = /*html*/ `
<span>
Hello, <span id="name">World</span>!
</span>
<br>
<input type="text" id="input" placeholder="Type your name">
<button id="button">Submit</button>
`;
const styles = /*css*/ `
:host {
display: block;
font-family: sans-serif;
padding: 1em;
border: 1px solid black;
border-radius: 5px;
}
#name {
color: blue;
}
`;
class MyElement extends WebComponent {
static get observedAttributes() {
return ["name"];
}
get _name() {
return this.getAttribute("name");
}
_updateName() {
const name =
/** @type HTMLDivElement */
(this.root.getElementById("name"));
name.textContent = this._name;
}
/**
* @param {string} name
* @param {string | null} oldValue
* @param {string | null} newValue
*/
attributeChangedCallback(name, oldValue, newValue) {
if (name === "name" && oldValue !== newValue) {
this._updateName();
}
}
constructor() {
super(template, styles);
const input =
/** @type HTMLInputElement */
(this.root.getElementById("input"));
const name =
/** @type HTMLDivElement */
(this.root.getElementById("name"));
const button =
/** @type HTMLButtonElement */
(this.root.getElementById("button"));
button.addEventListener("click", () => {
this.setAttribute("name", input.value);
});
}
}
if (!customElements.get(tagName)) {
customElements.define(tagName, MyElement);
}
export { MyElement, tagName, template, styles };
// @ts-check
export class WebComponent extends HTMLElement {
get internals() {
if (HTMLElement.prototype.hasOwnProperty("attachInternals")) {
return this.attachInternals();
} else {
return undefined;
}
}
/**
* @type {ShadowRootMode}
*/
get shadowRootMode() {
return "open";
}
/**
* @param {string} template
* @param {string} styles
*/
constructor(template, styles) {
super();
this.template = template;
this.styles = styles;
let shadow = this.internals?.shadowRoot;
if (!shadow) {
shadow = this.attachShadow({
mode: this.shadowRootMode,
});
shadow.innerHTML = template;
const sheet = new CSSStyleSheet();
sheet.replaceSync(styles);
shadow.adoptedStyleSheets = [sheet];
}
this.root = shadow;
}
prerender() {
const t = document.createElement("template");
t.setAttribute("shadowrootmode", this.shadowRootMode);
t.innerHTML = this.template;
const s = document.createElement("style");
s.innerHTML = this.styles;
t.appendChild(s);
if (this.firstChild) {
this.insertBefore(t, this.firstChild);
} else {
this.appendChild(t);
}
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment