cypress-ct-lit
Version:
Cypress Component Testing for Lit and Web Components
82 lines (78 loc) • 3.01 kB
JavaScript
import { html, render, nothing } from 'lit';
import { unsafeHTML } from 'lit/directives/unsafe-html.js';
const ROOT_SELECTOR = '[data-cy-root]';
/**
* Gets the root element used to mount the component.
* @returns {HTMLElement} The root element
* @throws {Error} If the root element is not found
*/
const getContainerEl = () => {
const el = document.querySelector(ROOT_SELECTOR);
if (el) {
return el;
}
throw Error(`No element found that matches selector ${ROOT_SELECTOR}. Please add a root element with data-cy-root attribute to your "component-index.html" file so that Cypress can attach your component to the DOM.`);
};
/**
* Utility function to register CT side effects and run cleanup code during the "test:before:run" Cypress hook
* @param optionalCallback Callback to be called before the next test runs
*/
function setupHooks(optionalCallback) {
// We don't want CT side effects to run when e2e
// testing so we early return.
// System test to verify CT side effects do not pollute e2e: system-tests/test/e2e_with_mount_import_spec.ts
if (Cypress.testingType !== 'component') {
return;
}
// When running component specs, we cannot allow "cy.visit"
// because it will wipe out our preparation work, and does not make much sense
// thus we overwrite "cy.visit" to throw an error
Cypress.Commands.overwrite('visit', () => {
throw new Error('cy.visit from a component spec is not allowed');
});
Cypress.Commands.overwrite('session', () => {
throw new Error('cy.session from a component spec is not allowed');
});
Cypress.Commands.overwrite('origin', () => {
throw new Error('cy.origin from a component spec is not allowed');
});
// @ts-ignore
Cypress.on('test:before:after:run:async', () => {
optionalCallback === null || optionalCallback === void 0 ? void 0 : optionalCallback();
});
}
let dispose;
function cleanup() {
dispose === null || dispose === void 0 ? void 0 : dispose();
}
function mount(template, options = {}) {
cleanup();
const root = getContainerEl();
const value = typeof template === 'string' ? html `${unsafeHTML(template)}` : template;
render(value, root, options.render);
dispose = () => {
render(nothing, root);
};
return cy
.wrap(root, { log: false })
.wait(0, { log: false })
.children({ log: false })
.first({ log: false })
.then((element) => {
const name = element.prop("tagName").toLowerCase();
// safe cast for current html element
const el = document.getElementsByTagName(name)[0];
if (options.log !== false) {
Cypress.log({
name: 'mount',
displayName: 'mount',
message: `<${name} ... />`
})
.snapshot('mounted')
.end();
}
return cy.wrap(el, { log: false });
});
}
setupHooks(cleanup);
export { mount };