diff options
author | 2021-07-13 08:27:40 -0400 | |
---|---|---|
committer | 2021-07-13 08:27:40 -0400 | |
commit | 48851c9d256b78c8e99e72d91ed98209a6a99e93 (patch) | |
tree | 19b4f4b293d4b569851a2d2531187415e13137d1 /packages/renderers/renderer-lit/server.js | |
parent | 1b13f5c158b5fbe157406308205c0939b415e1d7 (diff) | |
download | astro-48851c9d256b78c8e99e72d91ed98209a6a99e93.tar.gz astro-48851c9d256b78c8e99e72d91ed98209a6a99e93.tar.zst astro-48851c9d256b78c8e99e72d91ed98209a6a99e93.zip |
Adds support for a Lit renderer (#665)
* Adds support for a Lit renderer
This adds `@astrojs/renderer-lit`. An experimental Lit renderer.
* Removed cached submodule, mistake
* Prevent globals clobbering
* Add docs on globals
Diffstat (limited to 'packages/renderers/renderer-lit/server.js')
-rw-r--r-- | packages/renderers/renderer-lit/server.js | 64 |
1 files changed, 64 insertions, 0 deletions
diff --git a/packages/renderers/renderer-lit/server.js b/packages/renderers/renderer-lit/server.js new file mode 100644 index 000000000..83e1d51b4 --- /dev/null +++ b/packages/renderers/renderer-lit/server.js @@ -0,0 +1,64 @@ +import './server-shim.js'; +import '@lit-labs/ssr/lib/render-lit-html.js'; +import { LitElementRenderer } from '@lit-labs/ssr/lib/lit-element-renderer.js'; + +function isCustomElementTag(name) { + return typeof name === 'string' && /-/.test(name); +} + +function getCustomElementConstructor(name) { + if(typeof customElements !== 'undefined' && isCustomElementTag(name)) { + return customElements.get(name) || null; + } + return null; +} + +async function isLitElement(Component) { + const Ctr = getCustomElementConstructor(Component); + return !!(Ctr && Ctr._$litElement$); +} + +async function check(Component, _props, _children) { + // Lit doesn't support getting a tagName from a Constructor at this time. + // So this must be a string at the moment. + return !!(await isLitElement(Component)); +} + +function * render(tagName, attrs, children) { + const instance = new LitElementRenderer(tagName); + + // LitElementRenderer creates a new element instance, so copy over. + for(let [name, value] of Object.entries(attrs)) { + instance.setAttribute(name, value); + } + + yield `<${tagName}`; + yield* instance.renderAttributes(); + yield `>`; + const shadowContents = instance.renderShadow({}); + if (shadowContents !== undefined) { + yield '<template shadowroot="open">'; + yield* shadowContents; + yield '</template>'; + } + yield children; + yield `</${tagName}>`; +} + +async function renderToStaticMarkup(Component, props, children) { + let tagName = Component; + + let out = ''; + for(let chunk of render(tagName, props, children)) { + out += chunk; + } + + return { + html: out + }; +} + +export default { + check, + renderToStaticMarkup +}; |