2024-02-02 06:17:32 -05:00
|
|
|
import * as assert from 'node:assert/strict';
|
|
|
|
import { describe, it } from 'node:test';
|
2022-05-16 11:17:25 -05:00
|
|
|
import * as cheerio from 'cheerio';
|
|
|
|
import { LitElement, html } from 'lit';
|
2023-03-29 11:43:40 -05:00
|
|
|
// Must come after lit import because @lit/reactive-element defines
|
|
|
|
// globalThis.customElements which the server shim expects to be defined.
|
|
|
|
import server from '../server.js';
|
2022-05-16 11:16:30 -05:00
|
|
|
|
2022-05-16 11:17:25 -05:00
|
|
|
const { check, renderToStaticMarkup } = server;
|
2022-05-16 11:16:30 -05:00
|
|
|
|
|
|
|
describe('check', () => {
|
|
|
|
it('should be false with no component', async () => {
|
2024-02-02 06:17:32 -05:00
|
|
|
assert.equal(await check(), false);
|
2022-05-16 11:17:25 -05:00
|
|
|
});
|
2022-05-16 11:16:30 -05:00
|
|
|
|
|
|
|
it('should be false with a registered non-lit component', async () => {
|
2022-05-16 11:17:25 -05:00
|
|
|
const tagName = 'non-lit-component';
|
2023-03-29 11:43:40 -05:00
|
|
|
// Lit no longer shims HTMLElement globally, so we need to do it ourselves.
|
|
|
|
if (!globalThis.HTMLElement) {
|
|
|
|
globalThis.HTMLElement = class {};
|
|
|
|
}
|
2022-05-16 11:17:25 -05:00
|
|
|
customElements.define(tagName, class TestComponent extends HTMLElement {});
|
2024-02-02 06:17:32 -05:00
|
|
|
assert.equal(await check(tagName), false);
|
2022-05-16 11:17:25 -05:00
|
|
|
});
|
2022-05-16 11:16:30 -05:00
|
|
|
|
|
|
|
it('should be true with a registered lit component', async () => {
|
2022-05-16 11:17:25 -05:00
|
|
|
const tagName = 'lit-component';
|
|
|
|
customElements.define(tagName, class extends LitElement {});
|
2024-02-02 06:17:32 -05:00
|
|
|
assert.equal(await check(tagName), true);
|
2022-05-16 11:17:25 -05:00
|
|
|
});
|
|
|
|
});
|
2022-05-16 11:16:30 -05:00
|
|
|
|
|
|
|
describe('renderToStaticMarkup', () => {
|
|
|
|
it('should throw error if trying to render an unregistered component', async () => {
|
2022-05-16 11:17:25 -05:00
|
|
|
const tagName = 'non-registrered-component';
|
2022-05-16 11:16:30 -05:00
|
|
|
try {
|
2022-05-16 11:17:25 -05:00
|
|
|
await renderToStaticMarkup(tagName);
|
2022-05-16 11:16:30 -05:00
|
|
|
} catch (e) {
|
2024-02-02 06:17:32 -05:00
|
|
|
assert.equal(e instanceof TypeError, true);
|
2022-05-16 11:16:30 -05:00
|
|
|
}
|
2022-05-16 11:17:25 -05:00
|
|
|
});
|
2022-05-16 11:16:30 -05:00
|
|
|
|
2022-12-19 06:00:00 -05:00
|
|
|
it('should render empty component with default markup', async () => {
|
2022-05-16 11:17:25 -05:00
|
|
|
const tagName = 'nothing-component';
|
|
|
|
customElements.define(tagName, class extends LitElement {});
|
|
|
|
const render = await renderToStaticMarkup(tagName);
|
2024-02-02 06:17:32 -05:00
|
|
|
assert.deepEqual(render, {
|
2023-02-01 08:44:56 -05:00
|
|
|
html: `<${tagName}><template shadowroot="open" shadowrootmode="open"><!--lit-part--><!--/lit-part--></template></${tagName}>`,
|
2022-05-16 11:17:25 -05:00
|
|
|
});
|
|
|
|
});
|
2022-05-16 11:16:30 -05:00
|
|
|
|
|
|
|
it('should render component with default markup', async () => {
|
2022-05-16 11:17:25 -05:00
|
|
|
const tagName = 'simple-component';
|
|
|
|
customElements.define(
|
|
|
|
tagName,
|
|
|
|
class extends LitElement {
|
|
|
|
render() {
|
|
|
|
return html`<p>hola</p>`;
|
|
|
|
}
|
2022-05-16 11:16:30 -05:00
|
|
|
}
|
2022-05-16 11:17:25 -05:00
|
|
|
);
|
|
|
|
const render = await renderToStaticMarkup(tagName);
|
|
|
|
const $ = cheerio.load(render.html);
|
2024-02-02 06:17:32 -05:00
|
|
|
assert.equal($(`${tagName} template`).html().includes('<p>hola</p>'), true);
|
2022-05-16 11:17:25 -05:00
|
|
|
});
|
2022-05-16 11:16:30 -05:00
|
|
|
|
|
|
|
it('should render component with properties and attributes', async () => {
|
2022-05-16 11:17:25 -05:00
|
|
|
const tagName = 'props-and-attrs-component';
|
|
|
|
const attr1 = 'test';
|
|
|
|
const prop1 = 'Daniel';
|
|
|
|
customElements.define(
|
|
|
|
tagName,
|
|
|
|
class extends LitElement {
|
|
|
|
static properties = {
|
|
|
|
prop1: { type: String },
|
|
|
|
};
|
2022-05-16 11:16:30 -05:00
|
|
|
|
2022-05-16 11:17:25 -05:00
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
this.prop1 = 'someone';
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
return html`<p>Hello ${this.prop1}</p>`;
|
|
|
|
}
|
2022-05-16 11:16:30 -05:00
|
|
|
}
|
2022-05-16 11:17:25 -05:00
|
|
|
);
|
|
|
|
const render = await renderToStaticMarkup(tagName, { prop1, attr1 });
|
|
|
|
const $ = cheerio.load(render.html);
|
2024-02-02 06:17:32 -05:00
|
|
|
assert.equal($(tagName).attr('attr1'), attr1);
|
|
|
|
assert.equal($(`${tagName} template`).text().includes(`Hello ${prop1}`), true);
|
2022-05-16 11:17:25 -05:00
|
|
|
});
|
2023-02-24 13:10:56 -05:00
|
|
|
|
2023-04-05 08:05:45 -05:00
|
|
|
it('should render nested components', async () => {
|
|
|
|
const tagName = 'parent-component';
|
|
|
|
const childTagName = 'child-component';
|
|
|
|
customElements.define(
|
|
|
|
childTagName,
|
|
|
|
class extends LitElement {
|
|
|
|
render() {
|
|
|
|
return html`<p>child</p>`;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
customElements.define(
|
|
|
|
tagName,
|
|
|
|
class extends LitElement {
|
|
|
|
render() {
|
|
|
|
return html`<child-component></child-component>`;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
const render = await renderToStaticMarkup(tagName);
|
|
|
|
const $ = cheerio.load(render.html);
|
2024-02-02 06:17:32 -05:00
|
|
|
assert.equal($(`${tagName} template`).text().includes('child'), true);
|
2023-11-08 08:25:39 -05:00
|
|
|
// Child component should have `defer-hydration` attribute so it'll only
|
|
|
|
// hydrate after the parent hydrates
|
2024-02-02 06:17:32 -05:00
|
|
|
assert.equal($(childTagName).attr('defer-hydration'), '');
|
2023-04-05 08:05:45 -05:00
|
|
|
});
|
|
|
|
|
2023-02-24 13:10:56 -05:00
|
|
|
it('should render DSD attributes based on shadowRootOptions', async () => {
|
2023-03-29 11:43:40 -05:00
|
|
|
const tagName = 'shadow-root-options-component';
|
2023-02-24 13:13:07 -05:00
|
|
|
customElements.define(
|
|
|
|
tagName,
|
|
|
|
class extends LitElement {
|
|
|
|
static shadowRootOptions = { ...LitElement.shadowRootOptions, delegatesFocus: true };
|
|
|
|
}
|
|
|
|
);
|
2023-02-24 13:10:56 -05:00
|
|
|
const render = await renderToStaticMarkup(tagName);
|
2024-02-02 06:17:32 -05:00
|
|
|
assert.deepEqual(render, {
|
2023-02-24 13:10:56 -05:00
|
|
|
html: `<${tagName}><template shadowroot=\"open\" shadowrootmode=\"open\" shadowrootdelegatesfocus><!--lit-part--><!--/lit-part--></template></${tagName}>`,
|
|
|
|
});
|
|
|
|
});
|
2022-05-16 11:17:25 -05:00
|
|
|
});
|