UNPKG

@material-ui/core

Version:

React components that implement Google's Material Design.

232 lines (210 loc) 7.11 kB
import { assert } from 'chai'; import React from 'react'; import findOutermostIntrinsic from './findOutermostIntrinsic'; import ReactTestRenderer from 'react-test-renderer'; import testRef from './testRef'; /** * Glossary * - root component: * - renders the outermost host component * - has the `root` class if the component has one * - excess props are spread to this component * - has the type of `inheritComponent` */ /** * Returns the component with the same constructor as `component` that renders * the outermost host * * @param {import('enzyme').ReactWrapper} wrapper * @param {object} options * @param {import('react').ElementType} component */ function findRootComponent(wrapper, { component }) { const outermostHostElement = findOutermostIntrinsic(wrapper).getElement(); return wrapper.find(component).filterWhere(componentWrapper => { return componentWrapper.contains(outermostHostElement); }); } function randomStringValue() { return Math.random().toString(36).slice(2); } /** * Material-UI components have a `className` prop. The `className` is applied to * the root component. * * @param {React.ReactElement} element * @param {() => ConformanceOptions} getOptions */ function testClassName(element, getOptions) { it('applies the className to the root component', () => { const { mount } = getOptions(); const className = randomStringValue(); const wrapper = mount(React.cloneElement(element, { className })); assert.strictEqual(findOutermostIntrinsic(wrapper).hasClass(className), true, 'does have a custom `className`'); }); } /** * Material-UI components have a `component` prop that allows rendering a different * Component from @inheritComponent * * @param {React.ReactElement} element * @param {() => ConformanceOptions} getOptions */ function testComponentProp(element, getOptions) { describe('prop: component', () => { it('can render another root component with the `component` prop', () => { const { classes, mount, testComponentPropWith: component = 'em' } = getOptions(); const wrapper = mount(React.cloneElement(element, { component })); assert.strictEqual(findRootComponent(wrapper, { classes, component }).exists(), true); }); }); } /** * Material-UI components can spread additional props to a documented component. * It's set via @inheritComponent in the source. * * @param {React.ReactElement} element * @param {() => ConformanceOptions} getOptions */ function testPropsSpread(element, getOptions) { it(`does spread props to the root component`, () => { // type def in ConformanceOptions const { classes, inheritComponent, mount } = getOptions(); const testProp = 'data-test-props-spread'; const value = randomStringValue(); const wrapper = mount(React.cloneElement(element, { [testProp]: value })); const root = findRootComponent(wrapper, { classes, component: inheritComponent }); assert.strictEqual(root.props()[testProp], value); }); } /** * Tests that the `ref` of a component will return the correct instance * * This is determined by a given constructor i.e. a React.Component or HTMLElement for * components that forward their ref and attach it to a host component. * * @param {React.ReactElement} element * @param {() => ConformanceOptions} getOptions */ function describeRef(element, getOptions) { describe('ref', () => { it(`attaches the ref`, () => { // type def in ConformanceOptions const { inheritComponent, mount, refInstanceof } = getOptions(); testRef(element, mount, (instance, wrapper) => { assert.instanceOf(instance, refInstanceof); if (inheritComponent && instance instanceof window.Element) { const rootHost = findOutermostIntrinsic(wrapper); assert.strictEqual(instance, rootHost.instance()); } }); }); }); } /** * Tests that the root component has the root class * * @param {React.ReactElement} element * @param {() => ConformanceOptions} getOptions */ function testRootClass(element, getOptions) { it('applies to root class to the root component if it has this class', () => { const { classes, mount } = getOptions(); if (classes.root == null) { return; } const wrapper = mount(element); // we established that the root component renders the outermost host previously. We immediately // jump to the host component because some components pass the `root` class // to the `classes` prop of the root component. // https://github.com/mui-org/material-ui/blob/f9896bcd129a1209153106296b3d2487547ba205/packages/material-ui/src/OutlinedInput/OutlinedInput.js#L101 assert.strictEqual(findOutermostIntrinsic(wrapper).hasClass(classes.root), true); }); } /** * Tests that the component can be rendered with react-test-renderer. * This is important for snapshot testing with Jest (even if we don't encourage it). * * @param {React.ReactElement} element */ function testReactTestRenderer(element) { it('should render without errors in ReactTestRenderer', () => { ReactTestRenderer.act(() => { ReactTestRenderer.create(element, { createNodeMock: node => { return document.createElement(node.type); } }); }); }); } const fullSuite = { componentProp: testComponentProp, mergeClassName: testClassName, propsSpread: testPropsSpread, refForwarding: describeRef, rootClass: testRootClass, reactTestRenderer: testReactTestRenderer }; /** * @typedef {Object} ConformanceOptions * @property {Record<string, string>} classes - `classes` of the component provided by `@material-ui/styles` * @property {string} inheritComponent - The element type that receives spread props. * @property {function} mount - Should be a return value from createMount * @property {(keyof typeof fullSuite)[]} [only] - If specified only run the tests listed * @property {any} refInstanceof - `ref` will be an instanceof this constructor. * @property {keyof typeof fullSuite[]} [skip] - Skip the specified tests * @property {string} [testComponentPropWith] - The host component that should be rendered instead. */ /** * Tests various aspects of a component that should be equal across Material-UI * components. * * @param {React.ReactElement} minimalElement - the component with it's minimal required props * @param {() => ConformanceOptions} getOptions * */ export default function describeConformance(minimalElement, getOptions) { const { after: runAfterHook = () => {}, only = Object.keys(fullSuite), skip = [] } = getOptions(); describe('Material-UI component API', () => { after(runAfterHook); Object.keys(fullSuite).filter(testKey => only.indexOf(testKey) !== -1 && skip.indexOf(testKey) === -1).forEach(testKey => { const test = fullSuite[testKey]; test(minimalElement, getOptions); }); }); }