Skip to content

Commit 15345c7

Browse files
bennypowersLarsDenBakker
authored andcommitted
feat(testing-helpers): allow rendering non-TemplateResult (#910)
`render` from `lit-html` accepts more than just a `TemplateResult` nowadays. testing-helpers can support string, number, boolean, TemplateResult, Node, or Arrays or iterables of such. Note: in the case of Array or Iterable, only the first child will be taken. All others will be silently ignored. See #833 affects: @open-wc/testing-helpers
1 parent ca8623f commit 15345c7

File tree

4 files changed

+282
-9
lines changed

4 files changed

+282
-9
lines changed

packages/testing-helpers/src/fixture-no-side-effect.js

+8-6
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
1-
import { TemplateResult } from 'lit-html';
21
import { stringFixture, stringFixtureSync } from './stringFixture.js';
32
import { litFixture, litFixtureSync } from './litFixture.js';
3+
import { isValidRenderArg } from './lib.js';
44

55
/**
66
* Renders a string/TemplateResult and puts it in the DOM via a fixtureWrapper.
@@ -9,17 +9,19 @@ import { litFixture, litFixtureSync } from './litFixture.js';
99
* const el = fixtureSync('<my-el><span></span></my-el>');
1010
*
1111
* @template {Element} T
12-
* @param {string | TemplateResult} template Either a string or lit-html TemplateResult
12+
* @param {import('./litFixture').LitHTMLRenderable} template Either a string or lit-html TemplateResult
1313
* @returns {T} First child of the rendered DOM
1414
*/
1515
export function fixtureSync(template) {
1616
if (typeof template === 'string') {
1717
return stringFixtureSync(template);
1818
}
19-
if (template instanceof TemplateResult) {
19+
if (isValidRenderArg(template)) {
2020
return litFixtureSync(template);
2121
}
22-
throw new Error('Invalid template provided - string or lit-html TemplateResult is supported');
22+
throw new Error(
23+
'Invalid template provided - string, number, boolean, Node, TemplateResult, or array or iterable thereof are supported',
24+
);
2325
}
2426

2527
/**
@@ -39,14 +41,14 @@ export function fixtureSync(template) {
3941
* expect(el.fullyRendered).to.be.true;
4042
*
4143
* @template {Element} T
42-
* @param {string | TemplateResult} template Either a string or lit-html TemplateResult
44+
* @param {import('./litFixture').LitHTMLRenderable} template Either a string or lit-html TemplateResult
4345
* @returns {Promise<T>} A Promise that will resolve to the first child of the rendered DOM
4446
*/
4547
export async function fixture(template) {
4648
if (typeof template === 'string') {
4749
return stringFixture(template);
4850
}
49-
if (template instanceof TemplateResult) {
51+
if (isValidRenderArg(template)) {
5052
return litFixture(template);
5153
}
5254
throw new Error('Invalid template provided - string or lit-html TemplateResult is supported');

packages/testing-helpers/src/lib.js

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
import { TemplateResult } from 'lit-html';
2+
3+
export const isIterable = object => object != null && typeof object[Symbol.iterator] === 'function';
4+
5+
function isValidNonIterableRenderArg(x) {
6+
return (
7+
x instanceof TemplateResult ||
8+
x instanceof Node ||
9+
typeof x === 'number' ||
10+
typeof x === 'boolean' ||
11+
typeof x === 'string'
12+
);
13+
}
14+
15+
export function isValidRenderArg(x) {
16+
return isIterable(x) ? [...x].every(isValidNonIterableRenderArg) : isValidNonIterableRenderArg(x);
17+
}
18+
19+
/**
20+
* Node#nodeType enum
21+
* @readonly
22+
* @enum {number}
23+
*/
24+
export const NODE_TYPES = Object.freeze({
25+
ELEMENT_NODE: 1,
26+
TEXT_NODE: 3,
27+
COMMENT_NODE: 8,
28+
DOCUMENT_FRAGMENT_NODE: 11,
29+
});

packages/testing-helpers/src/litFixture.js

+31-3
Original file line numberDiff line numberDiff line change
@@ -1,25 +1,53 @@
1+
import { TemplateResult } from 'lit-html';
12
import { fixtureWrapper } from './fixtureWrapper.js';
23
import { render } from './lit-html.js';
34
import { elementUpdated } from './elementUpdated.js';
5+
import { NODE_TYPES } from './lib.js';
6+
7+
/**
8+
* @typedef {
9+
import('lit-html').TemplateResult | import('lit-html').TemplateResult[]
10+
| Node | Node[]
11+
| string | string[]
12+
| number | number[]
13+
| boolean | boolean[]
14+
} LitHTMLRenderable
15+
*/
16+
17+
const isUsefulNode = ({ nodeType, textContent }) => {
18+
switch (nodeType) {
19+
case NODE_TYPES.COMMENT_NODE:
20+
return false;
21+
case NODE_TYPES.TEXT_NODE:
22+
return textContent.trim();
23+
default:
24+
return true;
25+
}
26+
};
427

528
/**
629
* Setups an element synchronously from the provided lit-html template and puts it in the DOM.
730
*
831
* @template {Element} T - Is an element or a node
9-
* @param {import('lit-html').TemplateResult} template
32+
* @param {LitHTMLRenderable} template
1033
* @returns {T}
1134
*/
1235
export function litFixtureSync(template) {
1336
const wrapper = fixtureWrapper();
1437
render(template, wrapper);
15-
return /** @type {T} */ (wrapper.children[0]);
38+
if (template instanceof TemplateResult) {
39+
return /** @type {T} */ (wrapper.firstElementChild);
40+
}
41+
const [node] = Array.from(wrapper.childNodes).filter(isUsefulNode);
42+
43+
return /** @type {T} */ (node);
1644
}
1745

1846
/**
1947
* Setups an element asynchronously from the provided lit-html template and puts it in the DOM.
2048
*
2149
* @template {Element} T - Is an element or a node
22-
* @param {import('lit-html').TemplateResult} template
50+
* @param {LitHTMLRenderable} template
2351
* @returns {Promise<T>}
2452
*/
2553
export async function litFixture(template) {

packages/testing-helpers/test/fixture.test.js

+214
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ import { cachedWrappers } from '../src/fixtureWrapper.js';
66
import { defineCE } from '../src/helpers.js';
77
import { fixture, fixtureSync } from '../src/fixture.js';
88
import { html, unsafeStatic } from '../src/lit-html.js';
9+
import { NODE_TYPES } from '../src/lib.js';
910

1011
describe('fixtureSync & fixture', () => {
1112
it('supports strings', async () => {
@@ -44,6 +45,219 @@ describe('fixtureSync & fixture', () => {
4445
testElement(elementAsync);
4546
});
4647

48+
it('supports lit-html TemplateResult with whitespace', async () => {
49+
/**
50+
* @param {Element} element
51+
*/
52+
function testElement(element) {
53+
expect(element.localName).to.equal('div');
54+
}
55+
56+
const elementSync = fixtureSync(html`
57+
<div></div>
58+
`);
59+
// @ts-ignore
60+
testElement(elementSync);
61+
62+
const elementAsync = await fixture(html`
63+
<div></div>
64+
`);
65+
// @ts-ignore
66+
testElement(elementAsync);
67+
});
68+
69+
describe('Node', () => {
70+
it('supports Text Node', async () => {
71+
/**
72+
* @param {Node} node
73+
*/
74+
function doTest(node) {
75+
expect(node.textContent).to.equal('test');
76+
expect(node.nodeType).to.equal(NODE_TYPES.TEXT_NODE);
77+
}
78+
79+
const textNode = document.createTextNode('test');
80+
81+
const textNodeSync = fixtureSync(textNode);
82+
doTest(textNodeSync);
83+
84+
const textNodeAsync = await fixture(textNode);
85+
doTest(textNodeAsync);
86+
});
87+
88+
it('supports Text Node Array', async () => {
89+
/**
90+
* @param {Node} node
91+
*/
92+
function doTest(node) {
93+
expect(node.textContent).to.equal('test');
94+
expect(node.nodeType).to.equal(NODE_TYPES.TEXT_NODE);
95+
}
96+
97+
const textNodeArray = [
98+
document.createTextNode('test'),
99+
document.createTextNode('silently ignored'),
100+
];
101+
102+
const textNodeSync = fixtureSync(textNodeArray);
103+
doTest(textNodeSync);
104+
105+
const textNodeAsync = await fixture(textNodeArray);
106+
doTest(textNodeAsync);
107+
});
108+
109+
it('supports Element Node', async () => {
110+
/**
111+
* @param {Node} node
112+
*/
113+
function doTest(node) {
114+
expect(node.textContent).to.equal('test');
115+
expect(node.nodeType).to.equal(NODE_TYPES.ELEMENT_NODE);
116+
}
117+
118+
const elementNode = document.createElement('div');
119+
elementNode.innerHTML = 'test';
120+
121+
const textNodeSync = fixtureSync(elementNode);
122+
doTest(textNodeSync);
123+
124+
const textNodeAsync = await fixture(elementNode);
125+
doTest(textNodeAsync);
126+
});
127+
128+
it('supports Element Node Array', async () => {
129+
/**
130+
* @param {Node} node
131+
*/
132+
function doTest(node) {
133+
expect(node.textContent).to.equal('test');
134+
expect(node.nodeType).to.equal(NODE_TYPES.ELEMENT_NODE);
135+
}
136+
137+
const elementNodeArray = [document.createElement('div'), document.createElement('div')];
138+
139+
elementNodeArray[0].innerHTML = 'test';
140+
elementNodeArray[1].innerHTML = 'silently ignored';
141+
142+
const textNodeSync = fixtureSync(elementNodeArray);
143+
doTest(textNodeSync);
144+
145+
const textNodeAsync = await fixture(elementNodeArray);
146+
doTest(textNodeAsync);
147+
});
148+
149+
it('supports DOM tree', async () => {
150+
/**
151+
* @param {Node} node
152+
*/
153+
function doTest(node) {
154+
expect(node.textContent).to.equal('test the tree');
155+
expect(node.nodeType).to.equal(NODE_TYPES.ELEMENT_NODE);
156+
}
157+
158+
const parent = document.createElement('div');
159+
const child = document.createElement('div');
160+
const grandchild = document.createElement('div');
161+
162+
grandchild.appendChild(document.createTextNode('tree'));
163+
child.appendChild(document.createTextNode('the '));
164+
parent.appendChild(document.createTextNode('test '));
165+
166+
child.appendChild(grandchild);
167+
parent.appendChild(child);
168+
169+
/*
170+
<div>
171+
test
172+
<div>
173+
the
174+
<div>
175+
tree
176+
</div>
177+
</div>
178+
</div>
179+
*/
180+
181+
const textNodeSync = fixtureSync(parent);
182+
doTest(textNodeSync);
183+
184+
const textNodeAsync = await fixture(parent);
185+
doTest(textNodeAsync);
186+
});
187+
});
188+
189+
describe('primitives', () => {
190+
it('supports number', async () => {
191+
/**
192+
* @param {Node} node
193+
*/
194+
function doTest(node) {
195+
expect(node.textContent).to.equal('1');
196+
expect(node.nodeType).to.equal(NODE_TYPES.TEXT_NODE);
197+
}
198+
199+
const textNodeSync = fixtureSync(1);
200+
doTest(textNodeSync);
201+
202+
const textNodeAsync = await fixture(1);
203+
doTest(textNodeAsync);
204+
});
205+
206+
it('supports number array', async () => {
207+
/**
208+
* @param {Node} node
209+
*/
210+
function doTest(node) {
211+
expect(node.textContent).to.equal('0');
212+
expect(node.nodeType).to.equal(NODE_TYPES.TEXT_NODE);
213+
}
214+
215+
// NB: the 1 is silently ignored
216+
const numberArray = [0, 1];
217+
218+
const numberArraySync = fixtureSync(numberArray);
219+
doTest(numberArraySync);
220+
221+
const numberArrayAsync = await fixture(numberArray);
222+
doTest(numberArrayAsync);
223+
});
224+
225+
it('supports boolean', async () => {
226+
/**
227+
* @param {Node} node
228+
*/
229+
function doTest(node) {
230+
expect(node.textContent).to.equal('true');
231+
expect(node.nodeType).to.equal(NODE_TYPES.TEXT_NODE);
232+
}
233+
234+
const textNodeSync = fixtureSync(true);
235+
doTest(textNodeSync);
236+
237+
const textNodeAsync = await fixture(true);
238+
doTest(textNodeAsync);
239+
});
240+
241+
it('supports boolean array', async () => {
242+
/**
243+
* @param {Node} node
244+
*/
245+
function doTest(node) {
246+
expect(node.textContent).to.equal('true');
247+
expect(node.nodeType).to.equal(NODE_TYPES.TEXT_NODE);
248+
}
249+
250+
// NB: the `false` is silently ignored
251+
const booleanArray = [true, false];
252+
253+
const booleanArraySync = fixtureSync(booleanArray);
254+
doTest(booleanArraySync);
255+
256+
const booleanArrayAsync = await fixture(booleanArray);
257+
doTest(booleanArrayAsync);
258+
});
259+
});
260+
47261
it('will cleanup after each test', async () => {
48262
expect(cachedWrappers.length).to.equal(0);
49263
});

0 commit comments

Comments
 (0)