diff --git a/change/@fluentui-react-ab8e2990-b17d-4210-b514-30a0cb04b452.json b/change/@fluentui-react-ab8e2990-b17d-4210-b514-30a0cb04b452.json new file mode 100644 index 00000000000000..30af8843d4c643 --- /dev/null +++ b/change/@fluentui-react-ab8e2990-b17d-4210-b514-30a0cb04b452.json @@ -0,0 +1,7 @@ +{ + "type": "none", + "comment": "Fix test issues", + "packageName": "@fluentui/react", + "email": "elcraig@microsoft.com", + "dependentChangeType": "none" +} diff --git a/change/@fluentui-react-experiments-0b77bb7d-5bbe-457b-95c9-6a5970609121.json b/change/@fluentui-react-experiments-0b77bb7d-5bbe-457b-95c9-6a5970609121.json new file mode 100644 index 00000000000000..5534356965c7b1 --- /dev/null +++ b/change/@fluentui-react-experiments-0b77bb7d-5bbe-457b-95c9-6a5970609121.json @@ -0,0 +1,7 @@ +{ + "type": "none", + "comment": "Fix test issues", + "packageName": "@fluentui/react-experiments", + "email": "elcraig@microsoft.com", + "dependentChangeType": "none" +} diff --git a/change/@fluentui-react-hooks-c07633ea-c71c-4e99-9a4a-cc10394bf71c.json b/change/@fluentui-react-hooks-c07633ea-c71c-4e99-9a4a-cc10394bf71c.json new file mode 100644 index 00000000000000..a3a3b3f6a87d48 --- /dev/null +++ b/change/@fluentui-react-hooks-c07633ea-c71c-4e99-9a4a-cc10394bf71c.json @@ -0,0 +1,7 @@ +{ + "type": "none", + "comment": "Fix test issues", + "packageName": "@fluentui/react-hooks", + "email": "elcraig@microsoft.com", + "dependentChangeType": "none" +} diff --git a/change/@fluentui-test-utilities-493d57c1-c975-4c23-aea6-fd2dff463193.json b/change/@fluentui-test-utilities-493d57c1-c975-4c23-aea6-fd2dff463193.json new file mode 100644 index 00000000000000..d9832218aeaaca --- /dev/null +++ b/change/@fluentui-test-utilities-493d57c1-c975-4c23-aea6-fd2dff463193.json @@ -0,0 +1,7 @@ +{ + "type": "minor", + "comment": "Simplify safeMount attach API", + "packageName": "@fluentui/test-utilities", + "email": "elcraig@microsoft.com", + "dependentChangeType": "none" +} diff --git a/change/@fluentui-utilities-2d40b849-0e8a-4a69-a942-c4454fdde763.json b/change/@fluentui-utilities-2d40b849-0e8a-4a69-a942-c4454fdde763.json new file mode 100644 index 00000000000000..be420015331457 --- /dev/null +++ b/change/@fluentui-utilities-2d40b849-0e8a-4a69-a942-c4454fdde763.json @@ -0,0 +1,7 @@ +{ + "type": "none", + "comment": "Fix test issues", + "packageName": "@fluentui/utilities", + "email": "elcraig@microsoft.com", + "dependentChangeType": "none" +} diff --git a/packages/react-experiments/src/components/FloatingSuggestionsComposite/FloatingSuggestionsComposite.test.tsx b/packages/react-experiments/src/components/FloatingSuggestionsComposite/FloatingSuggestionsComposite.test.tsx index 382c7928db4351..49e6bfa72dee19 100644 --- a/packages/react-experiments/src/components/FloatingSuggestionsComposite/FloatingSuggestionsComposite.test.tsx +++ b/packages/react-experiments/src/components/FloatingSuggestionsComposite/FloatingSuggestionsComposite.test.tsx @@ -1,5 +1,5 @@ import * as React from 'react'; -import { create, act } from 'react-test-renderer'; +import { create } from 'react-test-renderer'; import { BaseFloatingSuggestions } from './FloatingSuggestions'; import * as ReactDOM from 'react-dom'; import * as ReactTestUtils from 'react-dom/test-utils'; @@ -95,7 +95,7 @@ describe('FloatingSuggestions', () => { // since callout mount a new react root with ReactDOM. // // see https://github.com/facebook/react/pull/12895 - act(() => { + ReactTestUtils.act(() => { (ReactDOM.render( { }); it('renders FloatingSuggestions and updates when suggestions are removed', () => { - act(() => { + ReactTestUtils.act(() => { (ReactDOM.render( { it('shows no suggestions when no suggestions are provided', () => { _suggestions = []; - act(() => { + ReactTestUtils.act(() => { (ReactDOM.render( { expect(onUnmount).toBeCalledTimes(0); const wrapper = mount(); expect(onUnmount).toBeCalledTimes(0); - wrapper.unmount(); + ReactTestUtils.act(() => { + wrapper.unmount(); + }); expect(onUnmount).toBeCalledTimes(1); }); }); diff --git a/packages/react/__mocks__/@fluentui/utilities.ts b/packages/react/__mocks__/@fluentui/utilities.ts index 0643a079bbbf62..27b9bd03e4b2fc 100644 --- a/packages/react/__mocks__/@fluentui/utilities.ts +++ b/packages/react/__mocks__/@fluentui/utilities.ts @@ -36,7 +36,9 @@ class MockAsync extends Async { } public dispose() { - clearTimeout(this._timeoutId); + if (this._timeoutId) { + clearTimeout(this._timeoutId); + } this._timeoutId = undefined; super.dispose(); diff --git a/packages/react/src/common/testUtilities.ts b/packages/react/src/common/testUtilities.ts index 176b05853328a2..574c192c23189b 100644 --- a/packages/react/src/common/testUtilities.ts +++ b/packages/react/src/common/testUtilities.ts @@ -28,6 +28,7 @@ export function delay(millisecond: number): Promise { /** * Mounts the element attached to a child of document.body. This is primarily for tests involving * event handlers (which don't work right unless the element is attached). + * @deprecated Use `safeMount` from `@fluentui/test-utilities` instead */ export function mountAttached( element: React.ReactElement

, diff --git a/packages/react/src/components/ChoiceGroup/ChoiceGroup.test.tsx b/packages/react/src/components/ChoiceGroup/ChoiceGroup.test.tsx index c715859926b7d4..e879074607c679 100644 --- a/packages/react/src/components/ChoiceGroup/ChoiceGroup.test.tsx +++ b/packages/react/src/components/ChoiceGroup/ChoiceGroup.test.tsx @@ -5,7 +5,7 @@ import * as renderer from 'react-test-renderer'; import { ChoiceGroup } from './ChoiceGroup'; import { merge, resetIds } from '../../Utilities'; -import { mountAttached } from '../../common/testUtilities'; +import { safeMount } from '@fluentui/test-utilities'; import { isConformant } from '../../common/isConformant'; import type { IChoiceGroupOption, IChoiceGroup, IChoiceGroupProps } from './ChoiceGroup.types'; @@ -253,30 +253,34 @@ describe('ChoiceGroup', () => { it('can focus the checked option', () => { // This test has to mount the element to the document since ChoiceGroup.focus() uses document.getElementById() const choiceGroupRef = React.createRef(); - choiceGroup = mountAttached( + safeMount( , + choiceGroup2 => { + const option = choiceGroup2.getDOMNode().querySelector(CHOICE_QUERY_SELECTOR) as HTMLInputElement; + const focusSpy = jest.spyOn(option, 'focus'); + + choiceGroupRef.current!.focus(); + expect(focusSpy).toHaveBeenCalled(); + }, + true /* attach */, ); - - const option = choiceGroup.getDOMNode().querySelector(CHOICE_QUERY_SELECTOR) as HTMLInputElement; - const focusSpy = jest.spyOn(option, 'focus'); - - choiceGroupRef.current!.focus(); - expect(focusSpy).toHaveBeenCalled(); }); it('can focus the first enabled option', () => { const choiceGroupRef = React.createRef(); - choiceGroup = mountAttached( + safeMount( , + choiceGroup2 => { + const option = choiceGroup2.getDOMNode().querySelectorAll(CHOICE_QUERY_SELECTOR)![1] as HTMLInputElement; + const focusSpy = jest.spyOn(option, 'focus'); + + choiceGroupRef.current!.focus(); + expect(focusSpy).toHaveBeenCalled(); + }, + true /* attach */, ); - - const option = choiceGroup.getDOMNode().querySelectorAll(CHOICE_QUERY_SELECTOR)![1] as HTMLInputElement; - const focusSpy = jest.spyOn(option, 'focus'); - - choiceGroupRef.current!.focus(); - expect(focusSpy).toHaveBeenCalled(); }); }); diff --git a/packages/react/src/components/DetailsList/DetailsList.test.tsx b/packages/react/src/components/DetailsList/DetailsList.test.tsx index 24c1bbcf9a0a99..c990eaf773eadb 100644 --- a/packages/react/src/components/DetailsList/DetailsList.test.tsx +++ b/packages/react/src/components/DetailsList/DetailsList.test.tsx @@ -2,7 +2,7 @@ import * as React from 'react'; import * as ReactDOM from 'react-dom'; import * as renderer from 'react-test-renderer'; import { ReactWrapper } from 'enzyme'; -import { createTestContainer, safeMount } from '@fluentui/test-utilities'; +import { safeMount } from '@fluentui/test-utilities'; import { EventGroup, KeyCodes, resetIds } from '../../Utilities'; import { SelectionMode, Selection, SelectionZone } from '../../Selection'; import { getTheme } from '../../Styling'; @@ -78,11 +78,10 @@ describe('DetailsList', () => { } }); - it('renders List correctly with onRenderDivider props', () => { + it('renders List correctly', () => { const component = renderer.create( null} skipViewportMeasures={true} onShouldVirtualize={() => false} @@ -92,11 +91,11 @@ describe('DetailsList', () => { expect(tree).toMatchSnapshot(); }); - it('renders List with custom icon as column divider', () => { + it('renders List correctly with onRenderDivider props', () => { const component = renderer.create( null} skipViewportMeasures={true} onShouldVirtualize={() => false} @@ -106,10 +105,11 @@ describe('DetailsList', () => { expect(tree).toMatchSnapshot(); }); - it('renders List correctly', () => { + it('renders List with custom icon as column divider', () => { const component = renderer.create( null} skipViewportMeasures={true} onShouldVirtualize={() => false} @@ -337,8 +337,6 @@ describe('DetailsList', () => { it('focuses row by index', () => { jest.useFakeTimers(); - const testContainer = createTestContainer(); - let component: IDetailsList | null; safeMount( { () => { expect(component).toBeTruthy(); component!.focusIndex(2); - setTimeout(() => { - expect( - (document.activeElement as HTMLElement).querySelector('[data-automationid=DetailsRowCell]')!.textContent, - ).toEqual('2'); - expect((document.activeElement as HTMLElement).className.split(' ')).toContain('ms-DetailsRow'); - }, 0); - jest.runOnlyPendingTimers(); + jest.runAllTimers(); + expect( + (document.activeElement as HTMLElement).querySelector('[data-automationid=DetailsRowCell]')!.textContent, + ).toEqual('2'); + expect((document.activeElement as HTMLElement).className.split(' ')).toContain('ms-DetailsRow'); }, - { attachTo: testContainer }, + true /* attach */, ); }); @@ -512,8 +508,6 @@ describe('DetailsList', () => { return valueKey; }; - const testContainer = createTestContainer(); - let component: IDetailsList | null; safeMount( { expect((document.activeElement as HTMLElement).className.split(' ')).toContain('ms-DetailsRow'); // Set element visibility manually as a test workaround - (component as IDetailsList).focusIndex(4); + component!.focusIndex(4); jest.runOnlyPendingTimers(); ((document.activeElement as HTMLElement).children[1] as any).isVisible = true; ((document.activeElement as HTMLElement).children[1].children[0] as any).isVisible = true; ((document.activeElement as HTMLElement).children[1].children[0].children[0] as any).isVisible = true; - (component as IDetailsList).focusIndex(4, true); + component!.focusIndex(4, true); jest.runOnlyPendingTimers(); expect((document.activeElement as HTMLElement).textContent).toEqual('4'); expect((document.activeElement as HTMLElement).className.split(' ')).toContain('test-column'); }, - { attachTo: testContainer }, + true /* attach */, ); }); it('reset focusedItemIndex when setKey updates', () => { jest.useFakeTimers(); - const testContainer = createTestContainer(); - - let component: any; + let component: DetailsListBase | null; safeMount( (component = ref)} + componentRef={value => (component = value as any)} skipViewportMeasures={true} onShouldVirtualize={() => false} />, (wrapper: ReactWrapper) => { expect(component).toBeTruthy(); component!.focusIndex(3); - jest.runOnlyPendingTimers(); - expect( - (document.activeElement as HTMLElement).querySelector('[data-automationid=DetailsRowCell]')!.textContent, - ).toEqual('3'); + jest.runAllTimers(); + expect(component!.state.focusedItemIndex).toEqual(3); // update props to new setKey const newProps = { items: mockData(7), setKey: 'set2', initialFocusedIndex: 0 }; wrapper.setProps(newProps); wrapper.update(); - setTimeout(() => { - expect( - (document.activeElement as HTMLElement).querySelector('[data-automationid=DetailsRowCell]')!.textContent, - ).toEqual('0'); - expect((document.activeElement as HTMLElement).className.split(' ')).toContain('ms-DetailsRow'); - }, 0); - jest.runOnlyPendingTimers(); + // verify that focusedItemIndex is reset to 0 and 0th row is focused + jest.runAllTimers(); + expect(component!.state.focusedItemIndex).toEqual(0); + expect( + (document.activeElement as HTMLElement).querySelector('[data-automationid=DetailsRowCell]')!.textContent, + ).toEqual('0'); + expect((document.activeElement as HTMLElement).className.split(' ')).toContain('ms-DetailsRow'); }, - { attachTo: testContainer }, + true /* attach */, ); }); @@ -683,48 +673,16 @@ describe('DetailsList', () => { it('handles updates to items and groups', () => { const tableOneItems = [ - { - f1: 'A1', - f2: 'B1', - f3: 'C1', - }, - { - f1: 'A2', - f2: 'B2', - f3: 'C2', - }, - { - f1: 'A3', - f2: 'B3', - f3: 'C3', - }, - { - f1: 'A4', - f2: 'B4', - f3: 'C4', - }, + { f1: 'A1', f2: 'B1', f3: 'C1' }, + { f1: 'A2', f2: 'B2', f3: 'C2' }, + { f1: 'A3', f2: 'B3', f3: 'C3' }, + { f1: 'A4', f2: 'B4', f3: 'C4' }, ]; const tableTwoItems = [ - { - f1: 'D1', - f2: 'E1', - f3: 'F1', - }, - { - f1: 'D2', - f2: 'E2', - f3: 'F2', - }, - { - f1: 'D3', - f2: 'E3', - f3: 'F3', - }, - { - f1: 'D4', - f2: 'E4', - f3: 'F4', - }, + { f1: 'D1', f2: 'E1', f3: 'F1' }, + { f1: 'D2', f2: 'E2', f3: 'F2' }, + { f1: 'D3', f2: 'E3', f3: 'F3' }, + { f1: 'D4', f2: 'E4', f3: 'F4' }, ]; const groupOneGroups: IGroup[] = [ @@ -829,41 +787,12 @@ describe('DetailsList', () => { }); it('handles paged updates to items within groups', () => { - const roundOneItems = [ - { - f1: 'A1', - f2: 'B1', - f3: 'C1', - }, - undefined, - { - f1: 'A3', - f2: 'B3', - f3: 'C3', - }, - undefined, - ]; + const roundOneItems = [{ f1: 'A1', f2: 'B1', f3: 'C1' }, undefined, { f1: 'A3', f2: 'B3', f3: 'C3' }, undefined]; const roundTwoItems = [ - { - f1: 'A1', - f2: 'B1', - f3: 'C1', - }, - { - f1: 'A2', - f2: 'B2', - f3: 'C2', - }, - { - f1: 'A3', - f2: 'B3', - f3: 'C3', - }, - { - f1: 'A4', - f2: 'B4', - f3: 'C4', - }, + { f1: 'A1', f2: 'B1', f3: 'C1' }, + { f1: 'A2', f2: 'B2', f3: 'C2' }, + { f1: 'A3', f2: 'B3', f3: 'C3' }, + { f1: 'A4', f2: 'B4', f3: 'C4' }, ]; const groups: IGroup[] = [ @@ -984,35 +913,35 @@ describe('DetailsList', () => { expect(checkboxId).toEqual(detailsRowCheckSource[`aria-labelledby`].split(' ')[0]); }); -}); -it('names group header checkboxes based on checkButtonGroupAriaLabel', () => { - const container = document.createElement('div'); - ReactDOM.render( - , - container, - ); + it('names group header checkboxes based on checkButtonGroupAriaLabel', () => { + const container = document.createElement('div'); + ReactDOM.render( + , + container, + ); - const checkbox = container.querySelector('[role="checkbox"][aria-label="select section"]') as HTMLElement; - expect(checkbox).not.toBeNull(); + const checkbox = container.querySelector('[role="checkbox"][aria-label="select section"]') as HTMLElement; + expect(checkbox).not.toBeNull(); - const groupNameId = checkbox.getAttribute('aria-labelledby')?.split(' ')[1]; - expect(container.querySelector(`#${groupNameId} span`)!.textContent).toEqual('Group 0'); + const groupNameId = checkbox.getAttribute('aria-labelledby')?.split(' ')[1]; + expect(container.querySelector(`#${groupNameId} span`)!.textContent).toEqual('Group 0'); + }); }); diff --git a/packages/react/src/components/Pivot/Pivot.test.tsx b/packages/react/src/components/Pivot/Pivot.test.tsx index 0ac389c609dc60..6f6d9c37602bf6 100644 --- a/packages/react/src/components/Pivot/Pivot.test.tsx +++ b/packages/react/src/components/Pivot/Pivot.test.tsx @@ -1,26 +1,31 @@ import * as React from 'react'; -import { create } from '@fluentui/utilities/lib/test'; import { mount } from 'enzyme'; import { resetIds } from '@fluentui/utilities'; -import { Pivot, PivotItem } from './index'; +import { safeMount, safeCreate } from '@fluentui/test-utilities'; +import { Pivot, PivotItem, IPivot } from './index'; import { isConformant } from '../../common/isConformant'; -import { mountAttached } from '../../common/testUtilities'; -import type { IPivot } from './index'; describe('Pivot', () => { beforeEach(() => { // Resetting ids to create predictability in generated ids. resetIds(); }); + + afterEach(() => { + delete (HTMLElement.prototype as any).isVisible; + }); + it('renders link Pivot correctly', () => { - const component = create( + safeCreate( , + component => { + const tree = component.toJSON(); + expect(tree).toMatchSnapshot(); + }, ); - const tree = component.toJSON(); - expect(tree).toMatchSnapshot(); }); isConformant({ @@ -31,29 +36,31 @@ describe('Pivot', () => { it('can be focused', () => { const pivotRef = React.createRef(); - mountAttached( + // Instruct FocusZone to treat all elements as visible. + (HTMLElement.prototype as any).isVisible = true; + + safeMount( , - ); - - // Instruct FocusZone to treat all elements as visible. - (HTMLElement.prototype as any).isVisible = true; + () => { + try { + expect(pivotRef.current).toBeTruthy(); - try { - expect(pivotRef.current).toBeTruthy(); - - pivotRef.current!.focus(); - expect(document.activeElement).toBeTruthy(); - expect(document.activeElement!.textContent?.trim()).toEqual('Link 1'); - } finally { - delete (HTMLElement.prototype as any).isVisible; - } + pivotRef.current!.focus(); + expect(document.activeElement).toBeTruthy(); + expect(document.activeElement!.textContent?.trim()).toEqual('Link 1'); + } finally { + delete (HTMLElement.prototype as any).isVisible; + } + }, + true /* attach, for focus tests */, + ); }); it('supports JSX expressions', () => { - const component = create( + safeCreate(

This is item 1
@@ -63,81 +70,105 @@ describe('Pivot', () => {
This is Item 3
, + component => { + const tree = component.toJSON(); + expect(tree).toMatchSnapshot(); + }, ); - const tree = component.toJSON(); - expect(tree).toMatchSnapshot(); }); + it('renders large link Pivot correctly', () => { - const component = create( + safeCreate( , + component => { + const tree = component.toJSON(); + expect(tree).toMatchSnapshot(); + }, ); - const tree = component.toJSON(); - expect(tree).toMatchSnapshot(); }); + it('renders tabbed Pivot correctly', () => { - const component = create( + safeCreate( , + component => { + const tree = component.toJSON(); + expect(tree).toMatchSnapshot(); + }, ); - const tree = component.toJSON(); - expect(tree).toMatchSnapshot(); }); + it('renders large tabbed Pivot correctly', () => { - const component = create( + safeCreate( , + component => { + const tree = component.toJSON(); + expect(tree).toMatchSnapshot(); + }, ); - const tree = component.toJSON(); - expect(tree).toMatchSnapshot(); }); + it('renders Pivot correctly with custom className', () => { - const component = create( + safeCreate( , + component => { + const tree = component.toJSON(); + expect(tree).toMatchSnapshot(); + }, ); - const tree = component.toJSON(); - expect(tree).toMatchSnapshot(); }); + it('renders Pivot correctly with icon, text and count', () => { - const component = create( + safeCreate( , + component => { + const tree = component.toJSON(); + expect(tree).toMatchSnapshot(); + }, ); - const tree = component.toJSON(); - expect(tree).toMatchSnapshot(); }); + it('renders Pivot correctly when itemCount is a string', () => { - const component = create( + safeCreate( , + component => { + const tree = component.toJSON(); + expect(tree).toMatchSnapshot(); + }, ); - const tree = component.toJSON(); - expect(tree).toMatchSnapshot(); }); + it('renders Pivot with overflow', () => { - const component = create( + safeCreate( , + component => { + const tree = component.toJSON(); + expect(tree).toMatchSnapshot(); + }, ); - const tree = component.toJSON(); - expect(tree).toMatchSnapshot(); }); + it('passes aria-label and aria-labelledby to tablist', () => { const wrapper = mount( diff --git a/packages/react/src/components/TextField/TextField.test.tsx b/packages/react/src/components/TextField/TextField.test.tsx index 89751ce50e9ff6..1227f018dd868f 100644 --- a/packages/react/src/components/TextField/TextField.test.tsx +++ b/packages/react/src/components/TextField/TextField.test.tsx @@ -6,7 +6,8 @@ import { renderToStaticMarkup } from 'react-dom/server'; import * as path from 'path'; import { resetIds, setWarningCallback, resetControlledWarnings } from '../../Utilities'; -import { mountAttached, mockEvent, flushPromises } from '../../common/testUtilities'; +import { mockEvent, flushPromises } from '../../common/testUtilities'; +import { safeMount } from '@fluentui/test-utilities'; import { TextField } from './TextField'; import { TextFieldBase } from './TextField.base'; @@ -826,23 +827,33 @@ describe('TextField', () => { }); it('sets focus to the input via ITextField focus', () => { - wrapper = mountAttached(); - const inputEl = wrapper.find('input').getDOMNode(); + safeMount( + , + wrapper2 => { + const inputEl = wrapper2.find('input').getDOMNode(); - textField!.focus(); + textField!.focus(); - expect(document.activeElement).toBe(inputEl); + expect(document.activeElement).toBe(inputEl); + }, + true /* attach */, + ); }); it('blurs the input via ITextField blur', () => { - wrapper = mountAttached(); - const inputEl = wrapper.find('input').getDOMNode(); + safeMount( + , + wrapper2 => { + const inputEl = wrapper2.find('input').getDOMNode(); - textField!.focus(); - expect(document.activeElement).toBe(inputEl); + textField!.focus(); + expect(document.activeElement).toBe(inputEl); - textField!.blur(); - expect(document.activeElement).toBe(document.body); + textField!.blur(); + expect(document.activeElement).toBe(document.body); + }, + true /* attach */, + ); }); it('can switch from single to multi line and back', () => { @@ -866,45 +877,55 @@ describe('TextField', () => { }); it('maintains focus when switching single to multi line and back', () => { - wrapper = mountAttached(); - // focus input - textField!.focus(); - let input = wrapper.find('input').getDOMNode(); - expect(document.activeElement).toBe(input); - - // switch to multiline - wrapper.setProps({ multiline: true }); - // verify still focused - const textarea = wrapper.find('textarea').getDOMNode(); - expect(document.activeElement).toBe(textarea); - - // back to single line - wrapper.setProps({ multiline: false }); - // verify still focused - input = wrapper.find('input').getDOMNode(); - expect(document.activeElement).toBe(input); + safeMount( + , + wrapper2 => { + // focus input + textField!.focus(); + let input = wrapper2.find('input').getDOMNode(); + expect(document.activeElement).toBe(input); + + // switch to multiline + wrapper2.setProps({ multiline: true }); + // verify still focused + const textarea = wrapper2.find('textarea').getDOMNode(); + expect(document.activeElement).toBe(textarea); + + // back to single line + wrapper2.setProps({ multiline: false }); + // verify still focused + input = wrapper2.find('input').getDOMNode(); + expect(document.activeElement).toBe(input); + }, + true /* attach */, + ); }); it('maintains selection when switching single to multi line and back', () => { const start = 1; const end = 3; - wrapper = mountAttached(); - // select - textField!.focus(); - textField!.setSelectionRange(start, end); - expect(textField!.selectionStart).toBe(start); - expect(textField!.selectionEnd).toBe(end); - - // switch to multiline - wrapper.setProps({ multiline: true }); - // verify still selected - expect(textField!.selectionStart).toBe(start); - expect(textField!.selectionEnd).toBe(end); - - // back to single line - wrapper.setProps({ multiline: false }); - // verify still selected - expect(textField!.selectionStart).toBe(start); - expect(textField!.selectionEnd).toBe(end); + safeMount( + , + wrapper2 => { + // select + textField!.focus(); + textField!.setSelectionRange(start, end); + expect(textField!.selectionStart).toBe(start); + expect(textField!.selectionEnd).toBe(end); + + // switch to multiline + wrapper2.setProps({ multiline: true }); + // verify still selected + expect(textField!.selectionStart).toBe(start); + expect(textField!.selectionEnd).toBe(end); + + // back to single line + wrapper2.setProps({ multiline: false }); + // verify still selected + expect(textField!.selectionStart).toBe(start); + expect(textField!.selectionEnd).toBe(end); + }, + true /* attach */, + ); }); }); diff --git a/packages/react/src/utilities/ThemeProvider/ThemeProvider.test.tsx b/packages/react/src/utilities/ThemeProvider/ThemeProvider.test.tsx index de1cf6777c8b96..af4a8a38e96928 100644 --- a/packages/react/src/utilities/ThemeProvider/ThemeProvider.test.tsx +++ b/packages/react/src/utilities/ThemeProvider/ThemeProvider.test.tsx @@ -1,4 +1,5 @@ import * as React from 'react'; +import * as ReactTestUtils from 'react-dom/test-utils'; import { ThemeProvider } from './ThemeProvider'; import * as renderer from 'react-test-renderer'; import { useTheme } from './useTheme'; @@ -65,7 +66,9 @@ describe('ThemeProvider', () => { expect(themeProvider1.getAttribute('dir')).toBe('ltr'); expect(themeProvider2.getAttribute('dir')).toBe(null); - wrapper.unmount(); + ReactTestUtils.act(() => { + wrapper.unmount(); + }); }); it('renders a div with styling', () => { @@ -132,7 +135,9 @@ describe('ThemeProvider', () => { expect(document.body).toMatchSnapshot(); - wrapper.unmount(); + ReactTestUtils.act(() => { + wrapper.unmount(); + }); expect(document.body.className).toBe(''); diff --git a/packages/test-utilities/src/safeMount.ts b/packages/test-utilities/src/safeMount.ts index 57daf2db5c6783..97cdbc412edc5e 100644 --- a/packages/test-utilities/src/safeMount.ts +++ b/packages/test-utilities/src/safeMount.ts @@ -1,5 +1,6 @@ import * as React from 'react'; -import { mount, MountRendererProps, ReactWrapper } from 'enzyme'; +import { mount, ReactWrapper } from 'enzyme'; +import { createTestContainer } from './createTestContainer'; /** * Calls `mount` from enzyme, calls the callback, and unmounts. This prevents mounted components @@ -7,8 +8,8 @@ import { mount, MountRendererProps, ReactWrapper } from 'enzyme'; * * @param content - JSX content to test. * @param callback - Function callback which receives the component to use. - * @param mountOptions - Options for enzyme mount function. If `attachTo` is provided, the element - * will be removed during the cleanup. + * @param attach - Whether to use a container element which is attached to the document + * (sometimes needed for event handlers) */ export function safeMount< TComponent extends React.Component, @@ -17,9 +18,10 @@ export function safeMount< >( content: React.ReactElement, callback?: (wrapper: ReactWrapper) => void, - mountOptions?: MountRendererProps, + attach?: boolean, ): void { - const wrapper = mount(content, mountOptions); + const testContainer = attach ? createTestContainer() : undefined; + const wrapper = mount(content, { ...(testContainer && { attachTo: testContainer }) }); try { callback?.(wrapper); @@ -27,8 +29,6 @@ export function safeMount< if (wrapper.exists()) { wrapper.unmount(); } - if (mountOptions?.attachTo) { - mountOptions.attachTo.remove(); - } + testContainer?.remove(); } } diff --git a/packages/utilities/src/customizations/useCustomizationSettings.test.tsx b/packages/utilities/src/customizations/useCustomizationSettings.test.tsx index eed2dd38c64cde..a6ee2b7cbfd9e4 100644 --- a/packages/utilities/src/customizations/useCustomizationSettings.test.tsx +++ b/packages/utilities/src/customizations/useCustomizationSettings.test.tsx @@ -10,10 +10,10 @@ describe('useCustomizatioSettings', () => { let wrapper: ReactWrapper | undefined; afterEach(() => { - if (wrapper) { - wrapper.unmount(); - } - + ReactTestUtils.act(() => { + wrapper?.unmount(); + wrapper = undefined; + }); Customizations.reset(); }); diff --git a/packages/utilities/src/styled.test.tsx b/packages/utilities/src/styled.test.tsx index 08627a5effb58c..5c6464ba1c8f74 100644 --- a/packages/utilities/src/styled.test.tsx +++ b/packages/utilities/src/styled.test.tsx @@ -1,5 +1,6 @@ /* eslint-disable deprecation/deprecation */ import * as React from 'react'; +import * as ReactTestUtils from 'react-dom/test-utils'; import { styled } from './styled'; import * as renderer from 'react-test-renderer'; import { Customizer } from './customizations/Customizer'; @@ -74,10 +75,10 @@ describe('styled', () => { }); afterEach(() => { - if (component) { - component.unmount(); + ReactTestUtils.act(() => { + component?.unmount(); component = undefined; - } + }); lastStylesInBaseComponent = undefined; }); diff --git a/packages/utilities/src/useFocusRects.test.tsx b/packages/utilities/src/useFocusRects.test.tsx index db0e8e25563854..33b34d3de52c63 100644 --- a/packages/utilities/src/useFocusRects.test.tsx +++ b/packages/utilities/src/useFocusRects.test.tsx @@ -1,4 +1,5 @@ import * as React from 'react'; +import * as ReactTestUtils from 'react-dom/test-utils'; import { FocusRects } from './useFocusRects'; import { IsFocusHiddenClassName, IsFocusVisibleClassName } from './setFocusVisibility'; import { KeyCodes } from './KeyCodes'; @@ -136,9 +137,13 @@ describe('useFocusRects', () => { expect(mockWindow.addEventListenerCallCount).toBe(3); expect(mockWindow.removeEventListenerCallCount).toBe(0); - focusRects1.unmount(); + ReactTestUtils.act(() => { + focusRects1.unmount(); + }); expect(mockWindow.removeEventListenerCallCount).toBe(0); - focusRects2.unmount(); + ReactTestUtils.act(() => { + focusRects2.unmount(); + }); expect(mockWindow.removeEventListenerCallCount).toBe(3); }); @@ -169,12 +174,16 @@ describe('useFocusRects', () => { expect(mockWindow.addEventListenerCallCount).toBe(3); expect(mockWindow.removeEventListenerCallCount).toBe(0); - focusRects1.unmount(); + ReactTestUtils.act(() => { + focusRects1.unmount(); + }); expect(mockWindow.removeEventListenerCallCount).toBe(3); expect(mockWindow2.addEventListenerCallCount).toBe(3); expect(mockWindow2.removeEventListenerCallCount).toBe(0); - focusRects2.unmount(); + ReactTestUtils.act(() => { + focusRects2.unmount(); + }); expect(mockWindow2.removeEventListenerCallCount).toBe(3); }); @@ -191,9 +200,13 @@ describe('useFocusRects', () => { expect(mockWindow.addEventListenerCallCount).toBe(3); expect(mockWindow.removeEventListenerCallCount).toBe(0); - focusRects1.unmount(); + ReactTestUtils.act(() => { + focusRects1.unmount(); + }); expect(mockWindow.removeEventListenerCallCount).toBe(0); - focusRects2.unmount(); + ReactTestUtils.act(() => { + focusRects2.unmount(); + }); expect(mockWindow.removeEventListenerCallCount).toBe(3); }); @@ -214,7 +227,9 @@ describe('useFocusRects', () => { expect(mockWindow.addEventListenerCallCount).toBe(3); expect(mockWindow.removeEventListenerCallCount).toBe(0); - focusRects1.unmount(); + ReactTestUtils.act(() => { + focusRects1.unmount(); + }); expect(mockWindow.removeEventListenerCallCount).toBe(3); }); @@ -237,9 +252,13 @@ describe('useFocusRects', () => { expect(mockWindow.addEventListenerCallCount).toBe(3); expect(mockWindow.removeEventListenerCallCount).toBe(0); - focusRects1.unmount(); + ReactTestUtils.act(() => { + focusRects1.unmount(); + }); expect(mockWindow.removeEventListenerCallCount).toBe(0); - focusRects2.unmount(); + ReactTestUtils.act(() => { + focusRects2.unmount(); + }); expect(mockWindow.removeEventListenerCallCount).toBe(3); }); @@ -257,7 +276,9 @@ describe('useFocusRects', () => { expect(mockWindow.addEventListenerCallCount).toBe(0); expect(mockWindow.removeEventListenerCallCount).toBe(0); - focusRect.unmount(); + ReactTestUtils.act(() => { + focusRect.unmount(); + }); expect(mockWindow.removeEventListenerCallCount).toBe(0); expect(mockWindow.removeEventListenerCallCount).toBe(0); });