import React from 'react'; import { mount, ReactWrapper } from 'enzyme'; import Affix, { AffixProps, AffixState } from '..'; import { getObserverEntities } from '../utils'; import Button from '../../button'; import rtlTest from '../../../tests/shared/rtlTest'; import accessibilityTest from '../../../tests/shared/accessibilityTest'; import { sleep } from '../../../tests/utils'; const events: Partial) => void>> = {}; class AffixMounter extends React.Component<{ offsetBottom?: number; offsetTop?: number; onTestUpdatePosition?(): void; onChange?: () => void; }> { private container: HTMLDivElement; public affix: React.Component; componentDidMount() { this.container.addEventListener = jest .fn() .mockImplementation((event: keyof HTMLElementEventMap, cb: (ev: Partial) => void) => { events[event] = cb; }); } getTarget = () => this.container; render() { return (
{ this.container = node!; }} className="container" > { this.affix = ele!; }} {...this.props} >
); } } describe('Affix Render', () => { rtlTest(Affix); accessibilityTest(Affix); const domMock = jest.spyOn(HTMLElement.prototype, 'getBoundingClientRect'); let affixMounterWrapper: ReactWrapper; let affixWrapper: ReactWrapper>; const classRect: Record = { container: { top: 0, bottom: 100, } as DOMRect, }; beforeAll(() => { domMock.mockImplementation(function fn(this: HTMLElement) { return ( classRect[this.className] || { top: 0, bottom: 0, } ); }); }); afterAll(() => { domMock.mockRestore(); }); const movePlaceholder = async (top: number) => { classRect.fixed = { top, bottom: top, } as DOMRect; if (events.scroll == null) { throw new Error('scroll should be set'); } events.scroll({ type: 'scroll', }); await sleep(20); }; it('Anchor render perfectly', async () => { document.body.innerHTML = '
'; affixMounterWrapper = mount(, { attachTo: document.getElementById('mounter') }); await sleep(20); await movePlaceholder(0); expect(affixMounterWrapper.instance().affix.state.affixStyle).toBeFalsy(); await movePlaceholder(-100); expect(affixMounterWrapper.instance().affix.state.affixStyle).toBeTruthy(); await movePlaceholder(0); expect(affixMounterWrapper.instance().affix.state.affixStyle).toBeFalsy(); }); it('support offsetBottom', async () => { document.body.innerHTML = '
'; affixMounterWrapper = mount(, { attachTo: document.getElementById('mounter'), }); await sleep(20); await movePlaceholder(300); expect(affixMounterWrapper.instance().affix.state.affixStyle).toBeTruthy(); await movePlaceholder(0); expect(affixMounterWrapper.instance().affix.state.affixStyle).toBeFalsy(); await movePlaceholder(300); expect(affixMounterWrapper.instance().affix.state.affixStyle).toBeTruthy(); }); it('updatePosition when offsetTop changed', async () => { document.body.innerHTML = '
'; const onChange = jest.fn(); affixMounterWrapper = mount(, { attachTo: document.getElementById('mounter'), }); await sleep(20); await movePlaceholder(-100); expect(onChange).toHaveBeenLastCalledWith(true); expect(affixMounterWrapper.instance().affix.state.affixStyle?.top).toBe(0); affixMounterWrapper.setProps({ offsetTop: 10, }); await sleep(20); expect(affixMounterWrapper.instance().affix.state.affixStyle?.top).toBe(10); }); describe('updatePosition when target changed', () => { it('function change', () => { document.body.innerHTML = '
'; const container = document.querySelector('#id') as HTMLDivElement; const getTarget = () => container; affixWrapper = mount({null}); affixWrapper.setProps({ target: () => null }); expect(affixWrapper.find('Affix').last().state().status).toBe(0); expect(affixWrapper.find('Affix').last().state().affixStyle).toBe(undefined); expect(affixWrapper.find('Affix').last().state().placeholderStyle).toBe(undefined); }); it('instance change', async () => { const getObserverLength = () => Object.keys(getObserverEntities()).length; const container = document.createElement('div'); document.body.appendChild(container); let target: HTMLDivElement | null = container; const originLength = getObserverLength(); const getTarget = () => target; affixWrapper = mount({null}); await sleep(100); expect(getObserverLength()).toBe(originLength + 1); target = null; affixWrapper.setProps({}); affixWrapper.update(); await sleep(100); expect(getObserverLength()).toBe(originLength); }); }); describe('updatePosition when size changed', () => { it.each([ { name: 'inner', index: 0 }, { name: 'outer', index: 1 }, ])('inner or outer', async ({ index }) => { document.body.innerHTML = '
'; const updateCalled = jest.fn(); affixMounterWrapper = mount( , { attachTo: document.getElementById('mounter'), }, ); await sleep(20); await movePlaceholder(300); expect(affixMounterWrapper.instance().affix.state.affixStyle).toBeTruthy(); await sleep(20); affixMounterWrapper.update(); // Mock trigger resize updateCalled.mockReset(); (affixMounterWrapper as any).triggerResize(index); await sleep(20); expect(updateCalled).toHaveBeenCalled(); }); }); });