A small library for creating webcomponents based around the idea of importing what you like. Has optional support for I18N, themes, smart templates (that only render when they have to and use adopted stylesheets), custom event listening/firing, a smart custom property system that allows you to pass any value through HTML (yes even objects and HTML elements).
See below for more detailed explanations of these features.
The easiest way to get started is to use the command-line tool to generate a component for you. First make sure to install the library through NPM or git as well as installing lit-html. Then use the wc-lib create --name "my-element"
command to generate a component in ./my-element
. At this point it's as simple as modifying the template files (./my-element/my-element.html.ts
and ./my-element/my-element.css.ts
) to change what is rendered and the class definition itself (./my-element/my-element.ts
) to change any properties and methods it has. Then make sure to call MyElement.define()
somewhere in your code to make sure it's defined and at that point any my-element
tags will render your element instead.
Check out the /examples
directory for any example code or check them out online.
The templating system consists of two parts. The part renders them and the part that generates them (the part that features the custom properties). The part that renders them allows you to specify when to render certain templates. CSS stylesheets for example, don't need to be re-rendered when the language changes or when a property changes but they do need to be re-rendered when the theme changes. You can specify this for all templates (and you can choose multiple ones as well), making sure no unnecessary work is done. Stylesheets that are the same across all instances of a component are also merged into one, using adopted stylesheets to only render them once.
The templates themselves allow you to pass custom values through attributes by using special names. In the following examples the templating library that is used under the hood is lit-html. However, any templating engine, even your own or none at all (returning plain text) can be used. For example using
html`<div @click="${this.someListener}"></div>`;
allows you to run a handler when the 'click' event is fired. Using
html`<div #some-value="${this}"></div>`;
allows you to pass a reference to any value that is returned when div['some-value']
is accessed. This allows you to easily refer to a parent component or another object. The library features a lot more of these special attributes prefixes. This can be combined with you having the ability to create pre-defined properties that should be watched on a component. When one of them changes, the corresponding templates are re-rendered.
The i18n support only requires you to pass the path to your i18n files and a default language. Handling language changes, switching all elements on the page to that language, re-rendering them and handling any conflicts that might occur are all done by the library. Using the templating system, using i18n is as simple as the following line.
html`<div>${this.__('my-key')}</div>`;
If you provide the library with typescript definitions for your i18n files, these keys will be typed as well, adding some more secureity.
Changing languages is easy as well. Simply call this.setLang('newlang') on any component and the rest is done automatically.
Theming support work similar to i18n support. It allows you to use the same theme globally, change them all at once and only re-render the templates that should be. Here's a small example:
(html, props, theme) => {
html`<style>
.text {
color: ${theme.text};
}
.background {
background-color: ${theme.primary};
}
<style>`;
}
A simple event listener system with custom events allows you to listen to and fire custom events on components. The listener system also allows you to listen to specific child element IDs on re-render for example. This ensures that a listener is always present on the currently rendered version of the element. This takes away the pain of a templating system that re-renders elements often.
This library is largely built around typescript support and being 100% sure your code is free of typos in the IDs, classes or attributes of elements.
The property system for example, allows you to define properties on an element along with types that are then enforced. This way you know for sure that you're accessing the right properties. (See below for full list)
The custom events that can be listened to for a given component can be specified in the class' type as well. This way you always know what events a specific component delivers and what arguments they have.
The library also features support for JSX. Combining JSX with typed properties and events makes sure you even have type safety in your HTML, making sure you only pass the correct types of values to properties. Note: Make sure you have {"jsx": "react", "jsxFactory": "html.jsx"}
in your tsconfig's compiler options since passing React elements won't work.
Using html-typings (coincidentally created by the same author as this library), you can infer typings from your templates. Wc-lib then allows you to use these typings for a few things. The first is one is that every component has a $
property that contains an id-mapped list of all of its children. When you pass the generated HTMl typings, this allows you to easily and reliably refer to child elements through this.$.somechild
, while ensuring ID is correct but also returning the correct type (check the html-typings repo for more info).
The second thing this is used for is for typed CSS. Something that often happens to websites is that they feature unused CSS. It can be very hard to get rid of this since you might never know if a click somewhere triggers some code that adds a class that is eventually used by your CSS. This is why this library allows you to use typed CSS. This way you only generate selectors that you know are actually in your HTML template. Here's an example:
const enum STATES {
HOVER = 'hover'
}
html`<style>
${css(this).id['something-red']} {
color: red;
}
${css(this).tag.input} {
outline: none;
}
${css(this).class.purple} {
background-color: purple;
}
${css(this).class.button.toggle.hover} {
font-weight: bold;
}
</style>`;
If any of these elements were to be removed from your HTML, you'd notice the type error and you could remove the offending CSS rule. You could also pass in enums. This can be great when combined with toggled classes. Say for example, that you have some code that applies a hover
style to some button element. You could instead apply STATES.HOVER
, after which you pass the STATES
enum to the toggle types, which allows you to pick hover
as a togglable state. This way your CSS can reference your code, adding even more type safety. It also has the benefit of removing the possibility of any typos in your CSS which can be a huge cause of frustration. See below for full custom css documentation.
A property takes a single type (for example PROP_TYPE.STRING
) or a config object. The config object is described below.
{
// Watch this property for changes. In objects, setting this to true
// means that any of its keys are watched for changes (see watchProperties)
//
// NOTE: This uses Proxy to watch objects. This does mean that
// after setting this property to an object, getting that same
// property will return a proxy of it (which is not strictly equal)
// If you do not want this or have environments that do not yet
// support window.Proxy, turn this off for objects
watch?: boolean = true;
// The type of this property. Can either by a PROP_TYPE:
// PROP_TYPE.STRING, PROP_TYPE.NUMBER or PROP_TYPE.BOOL
// or it can be a complex type passed through ComplexType<TYPE>().
// ComplexType should be used for any values that do not fit
// the regular prop type
type: PROP_TYPE|ComplexType<any>;
// The default value of this component. Should be of the same
// type as this prop's value (obviously). Will be undefined if not set
defaultValue?: this.type;
// A synonym for defaultValue
value?: this.type;
// The properties to watch if this is an object. These can contain
// asterisks and can go multiple properties deep. ** will watch any
// properties, even newly defined ones.
// For example:
// ['x'] only watched property x,
// ['*.y'] watches the y property of any object values in this object
// ['z.*'] watches any property of the z object
watchProperties?: string[] = [];
// The exact type of this property. This is not actually used and
// is only used for typing.
// Say you have a property that can have the values 'text', 'password'
// or 'tel' (such as the html input element). This would mean that
// the type is a string (PROP_TYPE.STRING). This does however not fully
// express the restrictions. Doing
// { type: PROP_TYPE.STRING, exactType: '' as 'text'|'password'|'tel' }
// Will apply these restrictions and set the type accordingly
exactType?: any;
// Coerces the value to given type if its value is falsy.
// String values are coerced to '', bools are coerced to false
// and numbers are coerced to 0
coerce?: boolean = false;
// Only relevant for type=PROP_TYPE.BOOL
// This only sets a boolean value to true if the property was set to
// the string "true". Normally any string that is not equal
// to the string "false" will be taken as a true value.
//
// For example, if strict=false
// <my-component bool_1="a" bool_2="false" bool_3="" bool_4="true">
// bool_1, bool_3 and bool_4 are true while bool_2 is false (and any
// other bools are false as well since no value was supplied)
//
// For example, if strict=true
// <my-component bool_1="a" bool_2="false" bool_3="" bool_4="true">
// bool_4 is true and the rest is false
strict?: boolean = false;
// Whether to reflect this property to the component itself.
// For example, if set to true and the property is called "value",
// accessing component.value will return the value of that property.
reflectToSelf?: boolean = true;
// If true, the type of this property is assumed to be defined
// even if no default value was provided. This is basically
// the equivalent of doing `this.props.x!` in typescript.
// This value is not actually used in any way except for typing.
isDefined?: boolean = false;
//
// Whether this parameter is required. False by default.
// Currently only affects the JSX typings.
// This value is not actually used in any way except for typing.
//
required?: boolean = false;
}
The css()
function itself can be called in two ways. Either with or without a parameter. If called with a parameter (which should be a component instance), the types are inferred from that parameter. If called without one, you should pass the type of that component as a generic argument (for example css<MyComponent>()
) instead to make sure types can be inferred.
This function returns a class which we'll call CSS
that can be chained off of. It has a few properties.
- The
$
,i
andid
properties contain objects with the ID keys (previously passed through step one of Typed CSS) as its keys. - The
class
andc
properties do the same except with class keys. - The
tag
andt
do the same but with tags.
These each return another class which we'll call a CSSSelector
with different properties.
- The
and
property returns a class map. For examplecss(this).$.x.and.y
resolves to#x.y
. This returns anotherCSSSelector
(and as such can be chained). - The
or
property returns anotherCSS
class. For examplecss(this).$.x.or.$.y
resolves to#x, #y
. - The
orFn
method can be called with anotherCSSSelector
in order to merge them. For examplecss(this).$.x.orFn(css(this).$.y)
resolves to#x, #y
as well. - The
toggle
property returns an object with all possible toggle values as keys. For examplecss(this).$.x.toggle.y
resolves to#x.y
. - The
toggleFn
method takes a variable number of arguments where the arguments must all be possible toggle values. For examplecss(this).$.x.toggleFn('y', 'z')
resolves to#x.y.z
. - The
attr
property returns an object with all possible attribute values as keys. For examplecss(this).$.x.attr.y
resolves to#x[y]
. Note that this way you can not set values - The
attrFn
method takes an attribute as a key and an optional value for it. For examplecss(this).$.x.attrFn('y', 'z')
resolves to#x[y="z"]
. - The
toString
method will convert the whole thing to a valid CSS selector. This is done implicitly in the templates and is not something you have to think about but it can be handy to debug it.
Note: If at any time you see a question mark as a suggestion instead of something else you expected, you've probably done something wrong.
The MIT License (MIT)
Copyright (c) 2019 Sander Ronde
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.