hyperactiv π + lit-html βοΈ + extensions π = perlite π.
Perlite is a simple and declarative way to create rich client-side widgets designed with server-side apps in mind. Completely based on native/vanilla Javascript standards and doesn't require additional build steps or compilation. Plays well with server-side rendered apps and micro-frontends. For more details read the description.
- Description & Features.
- Installation
- Examples
- Basic usage
- Widget declaration
- Widget creation
- Widget multiple instantiation
- Widget container
- Widget state
- Widget template
- Widget lifecycle events
- Widget API
- $widget.target: HTMLElement | Node;
- $widget.state: ProxyConstructor;
- $widget.model: object;
- $widget.effect(fn: () => void, opts?: object): () => void;
- $widget.on(type: string, fn: (e: CustomEvent) => void, opts?: object | boolean): () => void;
- $widget.render(): void;
- $widget.destroy(): void;
- $widget.ctx(fn: (...ctx: any[]) => any): any;
- Widget container API
- $$widgets[index: number]: Widget;
- $$widgets.target: NodeList | Node[];
- $$widgets.state(fn: (state: ProxyConstructor) => void): void;
- $$widgets.effect(fn: (state: ProxyConstructor) => () => void, opts?: object): (() => void)[];
- $$widgets.on(type: string, fn: (e: CustomEvent) => void, opts?: object | boolean): (() => void)[];
- $$widgets.render(): void;
- $$widgets.destroy(): void;
- $$widgets.ctx(fn: (...ctx: any[]) => any): any;
- $$widgets.forEach(fn: (widget: Widget, index: number, widgets: Widget[]) => void): any;
- Advanced usage
- Structuring your project
- Tooling
- Typescript support
- Browsers support
- License
Unlike the other frontend frameworks, eg. React, Vue, Angular or Svelte, which are mostly created for building SPA/RIA applications, Perlite's main goal is to make the life of developers of classical server-side applications a little bit easier and the modern front-end development techniques more accessible. Without extra knowledge of building tools and other dark sides of the frontend ecosystem . πΎ
Perlite gives you a combination of the best ideas from the most popular SPA frameworks, like UI is a function of a state (React), reactive state driven development (Vue), observables (Angular) and lack of Virtual DOM (Svelte).
Perlite focuses on building standalone UI widgets placed across different parts of the server-generated page and provides handy tools to manage these widgets and interact between them.
Built on top lit-html - an efficient, expressive, extensible HTML templating library and hyperactiv - a super tiny reactive library. This means that your widgets will have a reactive state with direct object mutations, super-fast DOM updates, and low memory consumption.
The full bundle size of Perlite library is just 8.8Kb (min+gz). In addition, it's optimized for tree-shaking, so you can reduce the final size if not all features are used. At the same time, Perlite is full-featured enough to fulfill its purposes and no additional tools you needed in most of the cases.
npm i --save perlite
or
yarn add perlite
and use it
import { html } from 'perlite';
If you are not using NPM, in modern browsers, you can import bundled ES module from CDN:
import { html } from 'https://unpkg.com/perlite@latest/dist/perlite.min.mjs';
or
just add a regular script
-tag to your html for legacy browsers:
<script src="https://unpkg.com/perlite@latest/dist/perlite.min.js"></script>
and use it via global namespace:
const { html } = window.perlite;
dist/index.js
- UMD outputdist/index.mjs
- ESM outputdist/index.min.js
- UMD output (minified)dist/index.min.mjs
- ESM output (minified)dist/perlite.js
- IIFE bundledist/perlite.mjs
- ESM bundledist/perlite.min.js
- IIFE bundle (minified)dist/perlite.min.mjs
- ESM bundle (minified)
- single widget app;
- multiple fragments;
- store with actions for todos list with side-effect to the
sessionStorage
; - based on ES dev server and NPM.
- server-generated page (actually a static file);
- multiple client-side widgets and basic interactions between them;
- based on static web server and CDN.
- 7 standalone widgets with different ways to made UI stuff;
- implementation of The 7 GUIs tasks;
- based on ES dev server and NPM.
Basically, the widget consists of two main parts: state
(object or function) and render
function. Use ES6 Template literals to describe your templates and tag them by special html
function imported from perlite
.
import { html } from 'perlite';
export const state = {
name: 'world'
};
export function render(state, emit) {
return html`
<h1>Hello ${state.name}</h1>
`
}
To create a new widget and append it to the page, import and call $
constructor function and pass the config object with several properties:
target
- DOM element where the widget will be rendered;state
- object or function representing the state of the widget;render
- a function representing a declarative template of the widget;- any hyperactiv options for reactivity system (you don't need to change the defaults in most cases)
For example, you can use ES6 Spread syntax to pass widget declaration exports to the constructor.
import { $ } from 'perlite';
import * as HelloWorld from './widgets/HelloWorld.js';
export const $helloWorld = $({
target: document.getElementById('helloWorld-widget'),
...HelloWorld
});
The constructor function will return an object which allows you to manage a widget. To distinguish widgets from regular JS objects, it's recommended to follow the naming convention by prefixing widget names with the β$β sign.
Actually, a widget object is just a namespace without any overall context. So, you can use ES6 Destructuring assignment and use things separately:
const {
destroy,
render,
target,
state,
effect,
ctx,
on,
} = $helloWorld;
Most often widget is a singleton, but in many cases, you need to use multiple widgets with the same declaration, but an isolated state. First of all, you need to use state
function, instead of an object, in widget declaration. This function should return a new state object, otherwise, state
will be shared between all widgets with the same declaration.
export function state() {
return {
name: 'world'
};
}
After that, you can call $
function multiple times with the different targets
:
import { $ } from 'perlite';
import * as HelloWorld from './widgets/HelloWorld.js';
export const $helloWorld1 = $({
target: document.getElementById('helloWorld-widget-1'),
...HelloWorld
});
export const $helloWorld2 = $({
target: document.getElementById('helloWorld-widget-2'),
...HelloWorld
});
When you deal with multiple widget instantiations, sometimes you want to work with them in the same manner. To do that, you can use the handy $$
container function to work with a bunch of widgets at once:
import { $$ } from 'perlite';
import * as HelloWorld from './widgets/HelloWorld.js';
export const $$helloWorlds = $$({
target: document.querySelectorAll('.helloWorld-widget'),
...HelloWorld
});
Please, use $$
(double β$β sign) prefix to visually distinguish widget containers from single widgets and regular JS objects.
Widgets and widget containers have mostly the same APIs, but having specifics at some points. For example, these functions will work the same for the end-user:
$widget.on('eventName', () => { ... }); // add event listener for the widget
$$widgets.on('eventName', () => { ... }); // add event listener for every widget in container
$widget.render(); // re-render the widget
$$widgets.render(); // re-render all widgets in container
$widget.destroy(); // destroy widget
$$widgets.destroy(); // destroy all widgets in container
The other APIs looks the same, but should be used differently:
$widget.state.foo = 1; // directly change the state of the widget
$$widgets.state(state$ => { // use it as a function with callback
state$.foo = 1;
});
$widget.effect(() => { ... }); // add effect for the widget
$$widgets.effect(state$ => () => { ... }); // add effect for every widget in container
Also, you can iterate through the container using forEach
:
$$widgets.forEach(widget => {
// do your custom logic with each widget
});
WIP
WIP
WIP
WIP
WIP
html`
<h1>Title: ${title}</h1>
`;
html`
<h1>${title}</h1>
<h2>${a + b}</h2>
<h3>${user.name}</h3>
<h4>${description.substring(50)}</h4>
<h5>${formatDate(user.birthDay)}</h5>
`;
html`
<input value="${title}">
<div class="default-class ${class}"></div>
`;
html`
<button ?disabled=${isDisabled}>Click me</button>
`;
html`
<input .value=${title}>
`;
html`
<input @input=${handleInput}>
<button @click=${e => alert('Clicked!')}>Click me</button>
`;
function render(state, emit) {
const welcomeMessage = html`<h1>Welcome ${state.user.name}</h1>`;
return html`
${welcomeMessage}
<a href="/logout">Logout</a>
`;
}
function userInfo(user) {
return html`
<dl>
<dt>User name:</dt>
<dd>${user.name}</dd>
<dt>Email address:</dt>
<dd>${user.email}</dd>
<dt>Birthday:</dt>
<dd>${formatDate(user.birthDay)}</dd>
</dl>
`;
}
function render(state, emit) {
return html`
<h1>${state.title}</h1>
${userInfo(state.user)}
`;
}
in template
html`
${state.user ?
html`
<h1>Welcome ${state.user.name}</h1>
<a href="/logout">Logout</a>
` :
html`
<a href="/login">Login</a>
`
}
`;
or in code
function userMessage(user) {
if (user) {
return html`
<h1>Welcome ${user.name}</h1>
<a href="/logout">Logout</a>
`;
} else {
return html`
<a href="/login">Login</a>
`;
}
}
function render(state, emit) {
return html`
${userMessage(state.user)}
`;
}
in template
html`
<ul>
${state.items.map((item) => html`
<li>${item.title}</li>
`)}
</ul>
`;
or in code
function itemsList(items) {
return items.map((item) => html`
<li>${item.title}</li>
`);
}
function render(state, emit) {
return html`
<ul>
${itemsList(state.items)}
</ul>
`;
}
WIP
These events are pre-defined and emitted on target
node as the other widget custom events. In most cases, you should use the built-in on()
function, but you also can do-it-yourself and use the regular target.addEventListener()
function, but don't forget to remove when you don't need it.
mount
- fires once when the component has been first time rendered to the DOM;state
- fires on every state change, before DOM update;update
- fires on every DOM updated, afterstate
event;destroy
- fires once when the component is removed from the DOM;error
- fires on exception occur during the rendering cycle;
Each life-cycle event, except error
, receives a model
of the widget in event.detail
. This state is not reactive and its changes won't trigger widget re-rendering. If you really need to start new DOM update cycle from a life-cycle event handler (basically, you shouldn't do that), you can use reactive state
or manual call render()
function via widget object.
In difference with the other life-cycle events, error
event receives an exception in event.detail
.
$widget.on('update', e => {
console.log('Widget DOM updated. The current model is: ', e.detail);
});
$widget.on('destroy', e => {
console.log('Widget is destroyed.');
});
$widget.on('error', e => {
console.error(e.detail.message);
});
Just a reference to target node of a widget.
Reactive state of a widget based on initial state object (called model). Changing this state will perform a re-render and DOM updates.
$widget.state.foo = 1; // widget scheduled for update
$widget.state.bar = true;
$widget.state.baz = 'horse'; // updates will be bunched
It's just a reference to plain state object, which is a model for reactive state (proxy target). You can changing this model, but because it's not reactive, re-rendering won't be performed. To apply these changes to the DOM you can use render()
function.
The effect is a function which executed each time its dependencies changed. Dependencies are tracked automatically and don't need to be explicitly specified.
const cancel = $widget.effect(() => {
console.log('Foo is changed:', $widget.state.foo);
});
...
// somewhere latter
cancel();
effect()
function is just a wrapper ontop of hyperactiv's computed()
with automatic dispose on widget destroy. So, you can use all things described in hyperactiv guide. This function return cancel()
function, so you can dispose of an effect when you actually don't need it.
This function lets you add an event listener to the widget to catch custom events dispatched by emit()
function and automatically removes the handler on widget destroy. Also, you can remove the handler manually using off()
function:
const off = $widget.on('my-custom-event', (event) => {
console.log('Event payload', event.detail);
});
...
// somewhere latter
off();
Call this function to manually re-render a widget. Usually, it's not necessary, because you need just use a state-driven approach and change the reactive state to automatically perform a re-render. But sometimes you may want to force the DOM update. This function is idempotent and safe to re-call. If actual state wasn't changed, no changes in DOM will performed.
Completelly destroy a widget, removes all event listeners and effects, and clean up the markup. It also fires a destroy
life-cycle event.
Calling this function is the most proper way to destroy the widget, but if, for some reason, the target
node will be removed from the DOM by external code, it will be tracked on the next render cycle, and destroy operations will be performed automatically.
This function receives callback function to get context values passed to the widget during creation.
$widget.ctx((foo, bar, baz) => {
console.log('widget context values', foo, bar, baz);
});
This function is fully synchronous and just returns the result of the callback. So you can return values you need directly or chain it with the other methods.
const bar = $widget.ctx((foo, bar, baz) => bar);
$widget.ctx((...ctx) => ctx).forEach((val) => ...);
More details about context.
WIP
Gets any widget by index in the order of target
list provided on the creation of the container.
The original list of targets
of the container widgets.
Works almost the same as on()
function of a single widget but add events listener to every widget inside the container.
Works almost the same as render()
function of a single widget but apply re-render to every widget inside the container.
Works almost the same as render()
function of a single widget but destroy all widgets inside the container.
Equal to ctx()
function of a single widget. All widgets inside the container share the same context values.
Use it for looping through all the widgets inside the container:
$$widgets.forEach($widget => {
// do something with each widget
});
Most often, you will need some initial widget state to be set by the server during page rendering. Just use data-attributes
on widget target
element and render necessary values there. For example, using PHP templating:
<div
id="myWidget"
data-string="<?=$strVal?>"
data-number="<?=$numVal?>"
data-boolean="<?=$boolVal?>"
data-null="<?=$nullVal?>"
data-json="<?=$jsonVal?>"
></div>
All data-attributes
that matched declared widget state (in widget declaration) will be picked up and applied to the widget during creation. Note: to be properly matched, attributes names should be in kebab-case, and widget state properties names should be in camelCase. For more info, check how kebabCase()
and camelCase()
functions works.
Regardless, that attribute values are always strings, some types will be automatically converted to the corresponding JS types (eg. boolean, number, null/undefined and even json. For more info, check how attrToVal()
function works.
Moreover, your external client-side code is also able to change data-attributes
of widget target node directly like this:
const myWidgetTarget = document.getElementByID('myWidget');
myWidgetTarget.setAttribute('data-string', 'hello world');
and these changes will also be applied to the widget state at any moment it occurs and DOM will be triggered to update as well. It's strongly not recommended, but can be useful in cases when some part of your code doesn't have direct access to the widget object and its reactive state.
WIP
Perlite re-exports all lit-html built-in directives:
- repeat
- cache
- until
- live
- guard
- class-map
- style-map
- if-defined
- async-append
- async-replace
- template-content
- unsafe-html
- unsafe-svg
Follow the lit-html guide to lean how to use them.
WIP
import {
capture,
passive,
once,
self,
stop, // stopPropagation()
prevent, // preventDefault()
} from 'perlite';
...
html`
<form @submit=${prevent(e => { ... })}>
...
<button @click=${self(once(e => { ... }))}>
Submit
</button>
</form>
`;
import { ref } from 'perlite';
...
html`<div @=${ref(el => state.el = el)}></div>`;
import { decorator } from 'perlite';
...
function myDecorator(node, foo, bar, baz) {
// do something
return {
update(foo, bar, baz) { ... },
destroy() { ... }
};
}
...
html`<div @=${decorator(myDecorator, foo, bar, baz)}></div>`;
Directives are fully provided by lit-html without any specifics or limitations. So, you can use Creating directives section in lit-html guide to learn more about custom directive creation.
Basically, context
is just any additional arguments that can be passed to the widget's state
and render
functions on widget creation. These arguments can have any type and order you needed. The main thing you should know, context
values are static. They are passed through when the widget is created, their count and order can't be changed on all widget life-cycle. Of course, if some context
value is a reference to the object/array, its mutations could be applied in the next DOM update cycle. But, unlike the state
mutations, context mutations will never trigger a new DOM update cycle by itself.
const context = { ... };
const $widget = $({
target,
render,
state,
},
context
);
const context1 = { ... };
const context2 = true;
const $widget = $({
target,
render,
state,
},
context1,
context2,
...
);
export function state(context1, context2) {
// change initial state model depending on context values
return {
...
};
}
export function render(state, emit, context1, context2) {
const something = Object.entries(context1).reduce(() => { ... });
return html`
<div>Context2: ${context2}</div>
`;
}
Basically, Perlite widgets designed without a focus on their composition. But sometimes you still need to insert one widget into a DOM tree of another widget and communicate with a nested widget on the rendering cycle of its "parent".
To do that, at first, you can create a target element of the nested widget in memory and use the context to pass this widget to render function of the parent.
import * as Nested from './widgets/Nested.js';
import * as Wrapper from './widgets/Wrapper.js';
const $nested = $({
target: document.createElement('div'),
...Nested
},
);
const $wrapper = $({
target: document.getElementById('wrapper'),
...Wrapper
},
$nested
);
After that, you can just use target
of the nested widget in template expression and it will be rendered as a fragment of the current widget's DOM tree. Very simple!
export function render(state, emit, $nested) {
return html`
${$nested.target}
<button @click=${e => $nested.state.count += 1}>
Increment nested value
</button>
`;
}
The number of nested widgets, as well as the way they are passed through the context, is not limited in any way. You can choose the most appropriate way you like.
const $wrapper = $({
target: document.getElementById('wrapper'),
...Wrapper
}, {
$nested1,
$nested2
}
);
export function render(state, emit, { $nested1, $nested2 }) {
return html`
${$nested1.target}
<div>
${$nested2.target}
</div>
`;
}
Also known as store(s)
- a very simple observables
. Whenever a property of an observed object is changed, every function that depends on this property is called.
First of all, need to create a new observable
store:
import { observe } from 'perlite';
export const store$ = observe({
products: [],
config: {},
user: {},
});
As it's often in Angular, use observables named with a trailing β$β sign to distinguish it from the other objects.
import { computed, dispose } from 'perlite';
import { store$ } from './store.js';
const user$$ = computed(() => {
console.log('User data has changed', store$.user);
});
...
dispose(user$$);
To subscribe to the store properties, pass the callback function as a first argument of computed
function and just perform any operations or side-effects with needful properties. It also returns subscription
handler to dispose of a subscription if you no longer need it. It recommended to name subscriptions
with a trailing "$$" sign (double "$") for short and distinguish it from the other objects.
Dependencies are automatically tracked so you don't need to explicitly declare anything - just use the properties you need. Read more about these things in hyperactiv guide.
Just import the store to use it in a widget:
import store$ from './store.js'
...
function userInfo(user) { ... }
...
function render(state, emit) {
return html`
<h1>${state.title}</h1>
${userInfo(store$.user)}
`;
}
The widget will be automatically updated when store values have changed.
WIP
WIP
Defer the code to be executed after the next DOM update cycle. Use it immediately after youβve changed some data to wait for the DOM update:
import { tick } from 'perlite';
$widget.state.foo += 1;
await tick();
console.log('now DOM updated');
Or if it needed to perform some operation inside of render function/fragments that can trigger a rendering cycle again (which is most often not safe):
import { tick, html } from 'perlite';
export function render(state, emit) {
// WRONG WAY - will trigger a new DOM update cycle
// before the current one is completed
state.result = ...;
// RIGHT WAY - defer state change to the next DOM update cycle
tick(() => {
state.result = ...;
});
return html`...`;
}
Creates and returns a new memoized version of the passed function that will cache the result based on its arguments.
import { memo } from 'perlite';
function heavyFunc(foo, bar, baz) { ... }
const funcMemoized = memo(heavyFunc);
...
funcMemoized(1, 2, 3); // executes a function, caches the result, and returns it
...
funcMemoized(1, 2, 3); // just a returns the result from cache
funcMemoized(4, 5, 6); // new arguments - new execution
funcMemoized(1, 2, 3); // still returns the result from cache
or use the second argument, function which provides custom cache invalidation logic:
import { memo } from 'perlite';
function heavyFunc() { ... }
const funcMemoized = memo(heavyFunc, (foo, bar, baz) => {
// decide whether to use the cached value or re-calculate the function
return true;
});
Invalidation function should return true
to keep the cached result or false
to drop the cache, and re-execute the function. Also, it can return any unique value (string, number, even a reference) which will be used instead of arguments list to cache and retrieve the result.
const funcMemoized = memo(heavyFunc, (foo, bar, baz) => {
return `${foo}-${bar}-${baz.quux}`; // this will be used as a cache identifier
});
import { attrToVal } from 'perlite';
attrToVal('false'); // false
attrToVal('undefined'); // undefined
attrToVal('null'); // null
attrToVal('2'); // 2
attrToVal('{"foo":1}'); // { foo: 1 }
import { camelCase } from 'perlite';
camelCase('kebab-to-camel-case'); // kebabToCamelCase
camelCase('kebab-to-pascal-case', true); // KebabToPascalCase
import { kebabCase } from 'perlite';
kebabCase('camelToKebabCase'); // camel-to-kebab-case
Basically, Perlite is not really opinionated about how you should structure your projects. But to not leave you alone with this question, let's describe a possible project structure you may use.
So, the main project unit is a widget, and the main part of the widget is its declaration. That's why we suppose to create a widgets
folder to contain declarations of the project widgets. Any widget declaration can be a single file or subfolder for more complex widgets.
./widgets/
./Widget1.js
./Widget2/
./styles.css
./index.js
Any widget can have fragments that are just re-usable pieces of the templates. You can keep fragments in the widget file if their number and size are not so big. Otherwise, you can take them out to a separate file or even a folder.
./widgets/
./Widget1/
./fragments.js
./index.js
./Widget2/
./fragments/
./fragment1.js
./index.js
Next thing that we have a widget creation process. In most cases, you'll need to create widgets right after DOM is ready and be able to get a widget object in any place of your code.
We suppose to add an index.js
file in the widgets
folder and create the widgets there. To get access to created objects you can just export it from this file. ES modules approach based on single instance pattern, so you'll be able to import widget objects in other files.
./widgets/
./Widget1/
./Widget2/
./index.js
Your index.js
can look like this:
import { $, $$ } from 'perlite';
// importing widget declarations
import * as Widget1 from './Widget1.js';
import * as Widget2 from './Widget2.js';
// creating and exporting the widgets or widget containers
export const $widget1 = $({
target: document.getElementById('widget1Container'),
...Widget1
});
export const $$widget2 = $$({
target: document.querySelectorAll('.widget2Container'),
...Widget2
});
After that, you'll be able to import any widget in any file of your project.
import { $widget1 } from './widgets/';
$widget1.effect(() => {
// do something
});
Regarding the stores, you may use almost the same approach - create a stores
folder with subfolders if needed, and hold stores in different files, optionally, re-exports them from a single entry point (index.js
).
./stores/
./store1/
./store1-1.js
./store1-2.js
./index.js
./store2.js
./index.js
import { html, bind, computed } from 'perlite';
import { user$ } from './stores/';
export const userName$$ = computed(() => {
console.log('user name is ', user$.name);
});
export function fragment() {
return html`
<input value=${user$.name} @change=${bind(name => user$.name = name)}>
`;
}
This software is licensed under the MIT Β© Pavel Malyshev.