7373d61cdc
* feat: pass all slots to renderers * refactor: pass `slots` as top-level props * test: add named slot test for frameworks * fix: nested hydration, slots that are not initially rendered * test: add nested-recursive e2e test * fix: render unmatched custom element children * chore: update lockfile * fix: unrendered slots for client:only * fix(lit): ensure lit integration uses new slots API * chore: add changeset * chore: add changesets * fix: lit slots * feat: convert dash-case or snake_case slots to camelCase for JSX * feat: remove tmpl special logic * test: add slot components-in-markdown test * refactor: prefer Object.entries.map() to for/of loop Co-authored-by: Nate Moore <nate@astro.build>
88 lines
2.2 KiB
JavaScript
88 lines
2.2 KiB
JavaScript
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;
|
|
} else if (typeof name === 'function') {
|
|
return name;
|
|
}
|
|
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(Component, attrs, slots) {
|
|
let tagName = Component;
|
|
if (typeof tagName !== 'string') {
|
|
tagName = Component[Symbol.for('tagName')];
|
|
}
|
|
const instance = new LitElementRenderer(tagName);
|
|
|
|
// LitElementRenderer creates a new element instance, so copy over.
|
|
const Ctr = getCustomElementConstructor(tagName);
|
|
if (attrs) {
|
|
for (let [name, value] of Object.entries(attrs)) {
|
|
// check if this is a reactive property
|
|
if (name in Ctr.prototype) {
|
|
instance.setProperty(name, value);
|
|
} else {
|
|
instance.setAttribute(name, value);
|
|
}
|
|
}
|
|
}
|
|
|
|
instance.connectedCallback();
|
|
|
|
yield `<${tagName}`;
|
|
yield* instance.renderAttributes();
|
|
yield `>`;
|
|
const shadowContents = instance.renderShadow({});
|
|
if (shadowContents !== undefined) {
|
|
yield '<template shadowroot="open">';
|
|
yield* shadowContents;
|
|
yield '</template>';
|
|
}
|
|
if (slots) {
|
|
for (const [slot, value] of Object.entries(slots)) {
|
|
if (slot === 'default') {
|
|
yield `<astro-slot>${value || ''}</astro-slot>`;
|
|
} else {
|
|
yield `<astro-slot slot="${slot}">${value || ''}</astro-slot>`;
|
|
}
|
|
}
|
|
}
|
|
yield `</${tagName}>`;
|
|
}
|
|
|
|
async function renderToStaticMarkup(Component, props, slots) {
|
|
let tagName = Component;
|
|
|
|
let out = '';
|
|
for (let chunk of render(tagName, props, slots)) {
|
|
out += chunk;
|
|
}
|
|
|
|
return {
|
|
html: out,
|
|
};
|
|
}
|
|
|
|
export default {
|
|
check,
|
|
renderToStaticMarkup,
|
|
};
|