A tiny helper function for creating Web Components
- Zero dependencies, only 2kb ๐๏ธ
- Template literals as templating engine ๐ก
- Write standards-compliant Web Components ๐
- Use with Vue, React, Angular and friends ๐ค
- Small and easy API ๐ด
CDN
Create an index.html file and include Customel with the CDN:
<script src="https://unpkg.com/customel"></script>
If you are using native ES Modules, you can include it in the index.html like this:
<script type="module">
import customel from "//unpkg.com/customel?module";
</script>
NPM
npm install customel
import customel from "customel";
Custom elements are awesome. They make it possible for everybody to use your components, no matter which new javascript framework they are using. As a part of the native web platform they are a great way to build maintainable and long lasting UI-libraries.
Still, writing Web Components from scratch can require quite a lot of boiler plate. Customel is a small wrapper around the HTMLElement
class that reduces a lot of the boiler plate, enforces some best practises, and makes it easier to make new custom elements.
Customel is not meant to be a way to build software or large applications. It is built as a way to create small, reusable Web Components.
The easiest way to try out Customel is using the JSFiddle Hello World example. You can follow the Quick Start from here.
First create a file called my-element.js
. Import Customel, create your component, and define it.
// my-element.js
import customel from "//unpkg.com/customel?module";
const Element = {
template() {
return "<div>Hello my new element!</div>";
}
};
customElements.define("my-element", customel(Element));
Create a index.html
file, and import my-element.js
.
Your custom element is now ready to be used!
<!-- index.html --->
<html>
<head>
<script type="module" src="my-element.js"></script>
</head>
<body>
<my-element></my-element>
</body>
</html>
You probably want to render some html inside your custom element. The easiest way to do this is to just return a string with some HTML:
const Element = {
template() {
return "<div>HTML as a string</div>";
}
};
The template
function also provides a template literal called html
that we can use when we want to add event handlers to the component, or we want to loop through a list.
const items = ["Item 1", "Item 2"];
const Element = {
template: function(html) {
return html`
<button onclick=${() => alert("Clicked")}>Button</button>
<ul>
${items.map(
item =>
html`
<li>${item}</li>
`
)}
</ul>
`;
}
};
To apply styles to your custom element, you can return CSS as a string:
const Element = {
styles() {
return `
button {
background-color: red;
color: white;
}
`;
},
template: function(html) {
return html`
<button>My button</button>
`;
}
};
You can also use props or state to render styles conditionally.
const Element = {
props: {
active: false
},
styles() {
return `
button {
background-color: ${this.props.active ? "red" : "blue"};
color: white;
}
`;
},
template: function(html) {
return html`
<button>My button</button>
`;
}
};
This will however cause a rerender of the whole internal stylesheet of the component. So it is adviced to toggle classes based on props or state instead:
const Element = {
props: {
active: false
},
styles() {
return `
button {
background-color: blue;
color: white;
}
button.active {
background-color: red;
}
`;
},
template: function(html) {
const buttonClass = this.props.active ? "active" : null;
return html`
<button class=${buttonClass}>My button</button>
`;
}
};
Your state is contained in a state object.
You can modify the state with the this.setSate
function.
this.setState
will cause a rerender and you will get a fresh DOM with updated data.
const Element = {
state: {
active: false
},
template: function(html) {
return html`
<button onclick=${() => this.setState({ active: true })}>
Active state is ${this.state.active}
</button>
`;
}
};
Use props to define data that your custom element can recieve from the outside. Always provide a default value to your props.
If the default value is a string
, number
or a boolean
then your prop will be available as an attribute on your custom element:
const Accordion = {
props: {
open: false
},
template: function(html) {
return html`
<div>
<h1>Accordion title</h1>
${this.props.open
? html`
<p>Accordion content</p>
`
: null}
</div>
`;
}
};
customElement.define("my-accordion", customel(Accordion));
This makes attributes available on your component.
<my-accordion title="My special list" open></my-accordion>
<script>
const accordion = document.querySelector('my-accordion');
accordion.addEventListener("click" => {
if (accordion.hasAttribute("open")) {
accordion.removeAttribute("open");
} else {
accordion.setAttribute("open", "");
}
})
</script>
They will also be reflected as properties, so you could just do this as well:
<my-accordion title="My special list" open></my-accordion>
<script>
const accordion = document.querySelector("my-accordion");
accordion.open = !accordion.open;
</script>
If the default value is a function
, object
, array
or any other type of data, it will be available as a property on the element instead of an attribute.
const List = {
props: {
todos: ["Buy milk", "Learn stuff"]
},
template: function(html) {
return html`
<ul>
${this.props.todos.map(
todo => html`
<li>${todo}</li>
`
)}
</ul>
`;
}
};
customElement.define("my-list", customel(List));
<my-list></my-list>
<script>
// "todos" is an array and can only be updated to properties on the element
// setting the property will triggger a rerender of the component, just like setting a new attribute
document.querySelector("my-list").todos = ["Other things", "More things"];
</script>
const Element = {
props: {
title: "Title"
},
actions: {
showMessage() {
alert("Title: is" + this.title);
}
},
template: function(html) {
return html`
<button onclick=${() => this.actions.showMessage()}>
My button
</button>
`;
}
};
Custom Elements play nicely with Vue, but for rich data you will need to add a .prop
on the attribute to bind the value to the property and not the attribute of the custom element.
For now you will also need to replace the whole array every time you update it's value, as Customel does not watch for nested changes. This may come in the future via use of a Proxy.
<div id="vue">
<my-accordion :title="mainTitle" :items.prop="items"></my-accordion>
</div>
<script>
new Vue({
el: "#vue",
data: {
mainTitle: "My accordion",
items: [
{
title: "My title",
content: "Here is some content about this stuff"
},
{
title: "Another title",
content: "Here is some other content about this stuff"
}
]
}
});
</script>
Coming...
Coming...
Customel is MIT licensed.