@lit-labs/eleventy-plugin-lit
A plugin for Eleventy that pre-renders
Lit web components at build time, with optional hydration.
data:image/s3,"s3://crabby-images/5fc95/5fc95f3e1309d1add146617020d5951fc95d91a0" alt="Published on npm"
Contents
Status
🚧 @lit-labs/eleventy-plugin-lit
is part of the Lit
Labs set of packages - it is published in
order to get feedback on the design and not ready for production. Breaking
changes are likely to happen frequently. 🚧
Setup
Install
npm i @lit-labs/eleventy-plugin-lit
Register plugin
Edit your .eleventy.js
config file to register the Lit plugin:
const litPlugin = require('@lit-labs/eleventy-plugin-lit');
module.exports = function (eleventyConfig) {
eleventyConfig.addPlugin(litPlugin, {
mode: 'worker',
componentModules: [
'js/demo-greeter.js',
'js/other-component.js',
],
});
};
Configure mode
Use the mode
setting to tell the plugin which mode to use for rendering.
The plugin currently supports either 'worker'
or 'vm'
.
'worker'
mode (default) utilizes
worker threads
to render components in isolation.
'vm'
mode utilizes vm.Module
for context isolation and therefore eleventy must be executed with the
--experimental-vm-modules
Node flag enabled. This flag is available in
Node versions 12.16.0
and above.
NODE_OPTIONS=--experimental-vm-modules eleventy
Configure component modules
🚧 Note: Support for specifying component modules in Eleventy front matter is
on the roadmap. Follow
#2494 for progress and discussion. 🚧
Use the componentModules
setting to tell the plugin where to find the
definitions of your components.
Pass an array of paths to .js
files containing Lit component definitions.
Paths are interpreted relative to to the directory from which the eleventy
command is executed.
Each .js
file should be a JavaScript module (ESM) that imports lit
with a
bare module specifier and defines a component with customElements.define
.
Note that in 'worker'
mode, Node determines the module system
accordingly,
and as such care must be taken to ensure Node reads them as ESM files
while still reading the eleventy config file as CommonJS.
Some options are:
-
Add {"type": "module"}
to your base package.json
, make sure the
eleventy config file ends with the .cjs
extension, and supply it as
a command line argument to eleventy
.
eleventy --config=.eleventy.cjs
-
Put all component .js
files in a subdirectory with a nested package.json
with
{"type": "module"}
.
Watch mode
Use addWatchTarget
to tell Eleventy
to watch for changes in your JavaScript directory:
eleventyConfig.addWatchTarget('js/');
Usage
Whenever you use a custom element in your Eleventy Markdown and HTML files,
@lit-labs/eleventy-plugin-lit
will automatically render its template and
styles directly into your HTML.
For example, given a markdown file hello.md
:
# Greetings
<demo-greeter name="World"></demo-greeter>
And a component definition js/demo-greeter.js
:
import {LitElement, html, css} from 'lit';
class DemoGreeter extends LitElement {
static styles = css`
b { color: red; }
`;
static properties = {
name: {},
};
render() {
return html`Hello <b>${this.name}</b>!`;
}
}
customElements.define('demo-greeter', DemoGreeter);
Then the Eleventy will produce greeting/index.html
:
<h1>Greetings</h1>
<demo-greeter name="World">
<template shadowroot="open">
<style>
b { color: red; }
</style>
</template>
Hello <b>World</b>!
</demo-greeter>
The <template shadowroot="open">
element above is an HTML standard called
declarative shadow DOM. See the Declarative Shadow
DOM section below for more details.
Component compatibility
🚧 Note: Expanding this section with full details on component compatibility
is on the roadmap. Follow
#2494 for progress and discussion.
🚧
There are currently a number of restrictions that determine whether a component
will be compatible with Lit pre-rendering, because not all of the component
lifecycle methods are currently invoked, and the DOM APIs that can be used in
certain lifecycle methods are restricted.
The Lit team is working on finalizing and documenting the SSR lifecycle and
restrictions, follow #2494 for more
details.
Passing data to components
🚧 Note: Support for passing data as properties is on the roadmap.
Follow #2494 for progress and
discussion. 🚧
Data can be passed to your components by setting attributes (see the name
attribute in the example above), or passing child elements.
Declarative Shadow DOM
Lit SSR depends on Declarative Shadow
DOM, a browser feature that allows
Shadow DOM to be created and attached directly from HTML, without the use of
JavaScript.
Polyfill
As of February 2022, Chrome and Edge have native support for Declarative Shadow
DOM, but Firefox and Safari don't yet.
Therefore, unless you are developing for a very constrained environment, you
must use the Declarative Shadow DOM
Polyfill to emulate this
feature in browsers that don't yet support it.
Install the polyfill from NPM:
npm i @webcomponents/template-shadowroot
For usage, see the example bootup strategy which
demonstrates a recommended method for efficiently loading the polyfill alongside
Lit hydration support.
⏱️ The Declarative Shadow DOM polyfill must be applied after all
pre-rendered HTML has been parsed, because it is a one-shot operation. You
can guarantee this timing by importing the polyfill from a type=module
script, or by placing it at the end of your <body>
tag.
Note that even if you do not require hydration, you will still need to polyfill
Declarative Shadow DOM, otherwise your pre-rendered components will never be
displayed in some browsers.
Hydration
Hydration is the process where statically pre-rendered components are upgraded
to their JavaScript implementations, becoming responsive and interactive.
Lit components can automatically hydrate themselves when they detect that a
Shadow Root has already been attached, as long as Lit's experimental hydrate
support module has been installed by importing
lit/experimental-hydrate-support.js
.
⏱️ The Lit hydration support module must be loaded before Lit or any
components that depend on Lit are imported, because it modifies the initial
startup behavior of the lit-element.js
module and the LitElement
class.
Bootup
It is important to preserve some constraints when designing a boot-up strategy
for pages that use pre-rendered Lit components. In particular:
- The Declarative Shadow DOM polyfill must wait until all HTML has been parsed.
- Lit and Lit component definition modules must wait until the experimental Lit
hydration support module has loaded.
- Lit component definition modules must wait until the Declarative Shadow DOM
polyfill to have been invoked (if it was needed for the browser).
In the following diagram, each ->
edge represents a timing sequence
constraint:
parse load install lit
HTML polyfill hydration support
| | |
v v v
run polyfill load lit
| |
v v
load component
definitions
Example bootup strategy
🚧 Note: The pattern described here will only work in modern browsers such as
Firefox, Chrome, Edge, and Safari. IE11 is also supported, but will require a
different pattern that is not yet documented here. Documenting this pattern is
on the roadmap. Follow
#2494 for progress and discussion.
🚧
The following demonstrates an example strategy for booting up a page that
contains pre-rendered Lit components with Eleventy.
The Lit team is investigating ways to simplify this bootup strategy and help you
generate it. Follow #2487 and
#2490 for progress.
Typically in Eleventy your content is written in Markdown files which delegate
the outer HTML shell to a layout
. For example hello.md
could delegate to the
default.html
layout like this:
---
layout: default.html
---
# Greetings
<demo-greeter name="World"></demo-greeter>
The file _includes/default.html
would then contain the following:
<!DOCTYPE html>
<html>
<head>
<link
rel="modulepreload"
href="/node_modules/lit/experimental-hydrate-support.js"
/>
<link rel="modulepreload" href="/_js/component1.js" />
<link rel="modulepreload" href="/_js/component2.js" />
<style>
body[dsd-pending] {
display: none;
}
</style>
</head>
<body dsd-pending>
<script>
if (HTMLTemplateElement.prototype.hasOwnProperty('shadowRoot')) {
document.body.removeAttribute('dsd-pending');
}
</script>
{{ content }}
<script type="module">
(async () => {
const litHydrateSupportInstalled = import(
'/node_modules/lit/experimental-hydrate-support.js'
);
if (!HTMLTemplateElement.prototype.hasOwnProperty('shadowRoot')) {
const {hydrateShadowRoots} = await import(
'/node_modules/@webcomponents/template-shadowroot/template-shadowroot.js'
);
hydrateShadowRoots(document.body);
document.body.removeAttribute('dsd-pending');
}
await litHydrateSupportInstalled;
import('/_js/component1.js');
import('/_js/component2.js');
})();
</script>
</body>
</html>
Roadmap
The following features and fixes are on the roadmap for this plugin. See the
linked issues for more details, and feel free to comment on the issues if you
have any thoughts or questions.
-
[#2494] Document restrictions on SSR
compatible components.
-
[#2483] Allow specifying component
definition modules in front
matter instead of the
componentModules
setting.
-
[#2485] Provide a mechanism for
passing Eleventy data to components as
properties, instead of attributes.
-
[#2486] Patterns and documentation
for supporting IE11.
-
[#2487] Provide a mechanism for
automatically generating and inserting an appropriate hydration
configuration.
-
[#2490] Simplify and optimize the
polyfill + hydration bootup strategy.
Issues and comments
If you find any bugs in this package, please file an
issue. If you have any questions or
comments, start a discussion.
Contributing
Please see CONTRIBUTING.md.
Testing environment variables:
SHOW_TEST_OUTPUT
: Set to show all stdout
and stderr
from spawned eleventy invocations in test cases.