ant-design/components/splitter/__tests__/index.test.tsx
二货爱吃白萝卜 e12a76b3ab
enhance: Resolve Splitter ssr render view (#51378)
* chore: ssr fix

* test: add test case

* test: fix test logic

* test: update snapshot
2024-10-25 17:23:12 +08:00

552 lines
16 KiB
TypeScript

import React from 'react';
import type { GetProps, SplitterProps } from 'antd';
import { ConfigProvider, Splitter } from 'antd';
import { spyElementPrototypes } from 'rc-util/lib/test/domHook';
import { resetWarned } from '../../_util/warning';
import {
act,
createEvent,
fireEvent,
render,
triggerResize,
waitFakeTimer,
} from '../../../tests/utils';
type PanelProps = GetProps<typeof Splitter.Panel>;
const resizeSplitter = async () => {
triggerResize(document.body.querySelector('.ant-splitter')!);
await waitFakeTimer();
};
const SplitterDemo = ({ items = [{}, {}], ...props }: { items?: PanelProps[] } & SplitterProps) => (
<Splitter {...props}>
{items?.map((item, idx) => {
const key = `panel-${idx}`;
return <Splitter.Panel key={key} {...item} />;
})}
</Splitter>
);
describe('Splitter', () => {
const errSpy = jest.spyOn(console, 'error').mockImplementation(() => {});
let containerSize = 100;
beforeAll(() => {
spyElementPrototypes(HTMLElement, {
offsetWidth: {
get: () => containerSize,
},
offsetHeight: {
get: () => containerSize,
},
});
});
beforeEach(() => {
containerSize = 100;
errSpy.mockReset();
resetWarned();
});
afterEach(() => {
jest.clearAllTimers();
jest.useRealTimers();
});
it('should correct render', () => {
const { container } = render(<SplitterDemo />);
expect(container.querySelector('.ant-splitter')).toBeTruthy();
expect(container.querySelectorAll('.ant-splitter-panel')).toHaveLength(2);
expect(container.querySelector('.ant-splitter-bar')).toBeTruthy();
});
it('should correct render panel size', async () => {
const { container } = render(<SplitterDemo items={[{ size: 20 }, { size: '45%' }, {}]} />);
await resizeSplitter();
const panels = container.querySelectorAll('.ant-splitter-panel');
expect(panels?.[0]).toHaveStyle('flex-basis: 20px');
expect(panels?.[1]).toHaveStyle('flex-basis: 45px');
expect(panels?.[2]).toHaveStyle('flex-basis: 35px');
});
it('The layout should work fine', () => {
const { container, rerender } = render(<SplitterDemo />);
expect(container.querySelector('.ant-splitter-horizontal')).toBeTruthy();
rerender(<SplitterDemo items={[{}, {}, {}]} layout="vertical" />);
expect(container.querySelector('.ant-splitter-vertical')).toBeTruthy();
});
it('The resizable should work fine', () => {
const { container, rerender } = render(
<SplitterDemo items={[{ size: 20 }, { resizable: false }, {}]} />,
);
expect(container.querySelectorAll('.ant-splitter-bar-dragger')).toHaveLength(2);
expect(container.querySelectorAll('.ant-splitter-bar-dragger-disabled')).toHaveLength(2);
rerender(<SplitterDemo items={[{ size: 20 }, {}, { resizable: false }]} />);
expect(container.querySelectorAll('.ant-splitter-bar-dragger')).toHaveLength(2);
expect(container.querySelectorAll('.ant-splitter-bar-dragger-disabled')).toHaveLength(1);
});
it('Splitter.Panel is syntactic sugar', () => {
const { container } = render(<Splitter.Panel />);
expect(container.innerHTML).toEqual('');
});
// ============================== Resizable ==============================
describe('drag', () => {
function mockDrag(draggerEle: HTMLElement, offset: number) {
// Down
const downEvent = createEvent.mouseDown(draggerEle);
(downEvent as any).pageX = 0;
(downEvent as any).pageY = 0;
fireEvent(draggerEle, downEvent);
// Move
const moveEvent = createEvent.mouseMove(draggerEle);
(moveEvent as any).pageX = offset;
(moveEvent as any).pageY = offset;
fireEvent(draggerEle, moveEvent);
// Up
fireEvent.mouseUp(draggerEle);
}
function mockTouchDrag(draggerEle: HTMLElement, offset: number) {
// Down
const touchStart = createEvent.touchStart(draggerEle, {
touches: [{}],
});
(touchStart as any).touches[0].pageX = 0;
(touchStart as any).touches[0].pageY = 0;
fireEvent(draggerEle, touchStart);
// Move
const touchMove = createEvent.touchMove(draggerEle, {
touches: [{}],
});
(touchMove as any).touches[0].pageX = offset;
(touchMove as any).touches[0].pageY = offset;
fireEvent(draggerEle, touchMove);
// Up
fireEvent.touchEnd(draggerEle);
}
it('The mousemove should work fine', async () => {
const onResize = jest.fn();
const onResizeEnd = jest.fn();
const { container } = render(
<SplitterDemo items={[{}, {}]} onResize={onResize} onResizeEnd={onResizeEnd} />,
);
await resizeSplitter();
// Right
mockDrag(container.querySelector('.ant-splitter-bar-dragger')!, 40);
expect(onResize).toHaveBeenCalledWith([90, 10]);
expect(onResizeEnd).toHaveBeenCalledWith([90, 10]);
// Left
mockDrag(container.querySelector('.ant-splitter-bar-dragger')!, -200);
expect(onResize).toHaveBeenCalledWith([0, 100]);
expect(onResizeEnd).toHaveBeenCalledWith([0, 100]);
});
it('The touchMove should work fine', async () => {
const onResize = jest.fn();
const onResizeEnd = jest.fn();
const { container } = render(
<SplitterDemo items={[{}, {}]} onResize={onResize} onResizeEnd={onResizeEnd} />,
);
await resizeSplitter();
// Right
mockTouchDrag(container.querySelector('.ant-splitter-bar-dragger')!, 40);
expect(onResize).toHaveBeenCalledWith([90, 10]);
expect(onResizeEnd).toHaveBeenCalledWith([90, 10]);
// Left
mockTouchDrag(container.querySelector('.ant-splitter-bar-dragger')!, -200);
expect(onResize).toHaveBeenCalledWith([0, 100]);
expect(onResizeEnd).toHaveBeenCalledWith([0, 100]);
});
it('with min', async () => {
const onResize = jest.fn();
const onResizeEnd = jest.fn();
const { container } = render(
<SplitterDemo items={[{ min: 10 }, {}]} onResize={onResize} onResizeEnd={onResizeEnd} />,
);
await resizeSplitter();
mockDrag(container.querySelector('.ant-splitter-bar-dragger')!, -100);
expect(onResize).toHaveBeenCalledWith([10, 90]);
expect(onResizeEnd).toHaveBeenCalledWith([10, 90]);
});
it('with max', async () => {
const onResize = jest.fn();
const onResizeEnd = jest.fn();
const { container } = render(
<SplitterDemo items={[{ max: 90 }, {}]} onResize={onResize} onResizeEnd={onResizeEnd} />,
);
await resizeSplitter();
mockDrag(container.querySelector('.ant-splitter-bar-dragger')!, 100);
expect(onResize).toHaveBeenCalledWith([90, 10]);
expect(onResizeEnd).toHaveBeenCalledWith([90, 10]);
});
it('both panel has min and max', async () => {
const onResize = jest.fn();
const onResizeEnd = jest.fn();
const { container } = render(
<SplitterDemo
items={[
{ min: 10, max: 80 },
{ min: 10, max: 80 },
]}
onResize={onResize}
onResizeEnd={onResizeEnd}
/>,
);
await resizeSplitter();
mockDrag(container.querySelector('.ant-splitter-bar-dragger')!, -100);
expect(onResize).toHaveBeenCalledWith([20, 80]);
expect(onResizeEnd).toHaveBeenCalledWith([20, 80]);
mockDrag(container.querySelector('.ant-splitter-bar-dragger')!, 100);
expect(onResize).toHaveBeenCalledWith([80, 20]);
expect(onResizeEnd).toHaveBeenCalledWith([80, 20]);
});
it('rtl', async () => {
const onResize = jest.fn();
const onResizeEnd = jest.fn();
const { container } = render(
<ConfigProvider direction="rtl">
<SplitterDemo items={[{}, {}]} onResize={onResize} onResizeEnd={onResizeEnd} />
</ConfigProvider>,
);
await resizeSplitter();
mockDrag(container.querySelector('.ant-splitter-bar-dragger')!, -40);
expect(onResize).toHaveBeenCalledWith([90, 10]);
expect(onResizeEnd).toHaveBeenCalledWith([90, 10]);
});
it('[true, 0, true] can be move left', async () => {
const onResize = jest.fn();
const onResizeEnd = jest.fn();
const { container } = render(
<SplitterDemo
items={[{}, { defaultSize: 0 }, {}]}
onResize={onResize}
onResizeEnd={onResizeEnd}
/>,
);
await resizeSplitter();
mockDrag(container.querySelectorAll<HTMLDivElement>('.ant-splitter-bar-dragger')[1], -100);
expect(onResize).toHaveBeenCalledWith([0, 50, 50]);
expect(onResizeEnd).toHaveBeenCalledWith([0, 50, 50]);
});
it('[false, 0, true] can not be move left', async () => {
const onResize = jest.fn();
const onResizeEnd = jest.fn();
const { container } = render(
<SplitterDemo
items={[{ resizable: false }, { defaultSize: 0 }, {}]}
onResize={onResize}
onResizeEnd={onResizeEnd}
/>,
);
await resizeSplitter();
mockDrag(container.querySelectorAll<HTMLDivElement>('.ant-splitter-bar-dragger')[1], -100);
expect(onResize).toHaveBeenCalledWith([50, 0, 50]);
expect(onResizeEnd).toHaveBeenCalledWith([50, 0, 50]);
});
it("aria-valuemin/aria-valuemax should not set NaN When container's width be setting zero", async () => {
containerSize = 0;
const App: React.FC = () => {
return <SplitterDemo items={[{}, {}, {}]} />;
};
const { container } = render(<App />);
await resizeSplitter();
mockDrag(container.querySelectorAll<HTMLDivElement>('.ant-splitter-bar-dragger')[1], -100);
triggerResize(container.querySelector('.ant-splitter')!);
await act(async () => {
await waitFakeTimer();
});
expect(errSpy).not.toHaveBeenCalled();
expect(container.querySelector('[aria-valuemin]')?.getAttribute('aria-valuemin')).not.toBe(
'NaN',
);
expect(container.querySelector('[aria-valuemax]')?.getAttribute('aria-valuemax')).not.toBe(
'NaN',
);
});
});
// ============================= Collapsible =============================
describe('collapsible', () => {
it('Basic', async () => {
const { container, rerender } = render(
<SplitterDemo items={[{ size: 20, collapsible: true }, { collapsible: true }]} />,
);
await resizeSplitter();
expect(container.querySelectorAll('.ant-splitter-bar-collapse-icon')).toHaveLength(2);
expect(container.querySelector('.ant-splitter-bar-collapse-start')).toBeTruthy();
expect(container.querySelector('.ant-splitter-bar-collapse-end')).toBeTruthy();
// support collapsible is object
rerender(
<SplitterDemo
items={[
{
size: 20,
collapsible: true,
},
{
collapsible: true,
},
{},
]}
/>,
);
expect(container.querySelectorAll('.ant-splitter-bar-collapse-start')).toHaveLength(2);
expect(container.querySelectorAll('.ant-splitter-bar-collapse-end')).toHaveLength(1);
});
it('collapsible - true', async () => {
const onResize = jest.fn();
const onResizeEnd = jest.fn();
const { container } = render(
<SplitterDemo
items={[
{
size: 20,
collapsible: true,
},
{},
]}
onResize={onResize}
onResizeEnd={onResizeEnd}
/>,
);
await resizeSplitter();
fireEvent.click(container.querySelector('.ant-splitter-bar-collapse-start')!);
expect(onResize).toHaveBeenCalledWith([0, 100]);
expect(onResizeEnd).toHaveBeenCalledWith([0, 100]);
});
it('collapsible - start:true', async () => {
const onResize = jest.fn();
const onResizeEnd = jest.fn();
const { container } = render(
<SplitterDemo
items={[
{},
{
size: 20,
collapsible: {
start: true,
},
},
{},
]}
onResize={onResize}
onResizeEnd={onResizeEnd}
/>,
);
await resizeSplitter();
expect(container.querySelector('.ant-splitter-bar-collapse-start')).toBeFalsy();
expect(container.querySelector('.ant-splitter-bar-collapse-end')).toBeTruthy();
fireEvent.click(container.querySelector('.ant-splitter-bar-collapse-end')!);
expect(onResize).toHaveBeenCalledWith([60, 0, 40]);
expect(onResizeEnd).toHaveBeenCalledWith([60, 0, 40]);
});
it('collapsible - end:true', async () => {
const onResize = jest.fn();
const onResizeEnd = jest.fn();
const { container } = render(
<SplitterDemo
items={[
{},
{
size: 20,
collapsible: {
end: true,
},
},
{},
]}
onResize={onResize}
onResizeEnd={onResizeEnd}
/>,
);
await resizeSplitter();
expect(container.querySelector('.ant-splitter-bar-collapse-start')).toBeTruthy();
expect(container.querySelector('.ant-splitter-bar-collapse-end')).toBeFalsy();
fireEvent.click(container.querySelector('.ant-splitter-bar-collapse-start')!);
expect(onResize).toHaveBeenCalledWith([40, 0, 60]);
expect(onResizeEnd).toHaveBeenCalledWith([40, 0, 60]);
});
it('both collapsible', async () => {
const onResize = jest.fn();
const onResizeEnd = jest.fn();
const { container } = render(
<SplitterDemo
items={[
{
collapsible: true,
},
{
collapsible: true,
},
]}
onResize={onResize}
onResizeEnd={onResizeEnd}
/>,
);
await resizeSplitter();
function expectClick(element: HTMLElement, size: number[]) {
onResize.mockReset();
onResizeEnd.mockReset();
fireEvent.click(element);
expect(onResize).toHaveBeenCalledWith(size);
expect(onResizeEnd).toHaveBeenCalledWith(size);
}
expectClick(container.querySelector('.ant-splitter-bar-collapse-start')!, [0, 100]);
expectClick(container.querySelector('.ant-splitter-bar-collapse-end')!, [50, 50]);
expectClick(container.querySelector('.ant-splitter-bar-collapse-end')!, [100, 0]);
expectClick(container.querySelector('.ant-splitter-bar-collapse-start')!, [50, 50]);
});
it('collapsible with min', async () => {
const onResize = jest.fn();
const onResizeEnd = jest.fn();
const { container } = render(
<SplitterDemo
items={[
{
defaultSize: 20,
collapsible: true,
min: 10,
},
{
collapsible: true,
min: '80%',
},
]}
onResize={onResize}
onResizeEnd={onResizeEnd}
/>,
);
await resizeSplitter();
// Collapse left
fireEvent.click(container.querySelector('.ant-splitter-bar-collapse-start')!);
expect(onResize).toHaveBeenCalledWith([0, 100]);
expect(onResizeEnd).toHaveBeenCalledWith([0, 100]);
expect(container.querySelector('.ant-splitter-bar-dragger-disabled')).toBeTruthy();
// Collapse back
onResize.mockReset();
onResizeEnd.mockReset();
fireEvent.click(container.querySelector('.ant-splitter-bar-collapse-end')!);
expect(onResize).toHaveBeenCalledWith([5, 95]);
expect(onResizeEnd).toHaveBeenCalledWith([5, 95]);
expect(container.querySelector('.ant-splitter-bar-dragger-disabled')).toBeFalsy();
// Collapse right
onResize.mockReset();
onResizeEnd.mockReset();
fireEvent.click(container.querySelector('.ant-splitter-bar-collapse-end')!);
expect(onResize).toHaveBeenCalledWith([100, 0]);
expect(onResizeEnd).toHaveBeenCalledWith([100, 0]);
expect(container.querySelector('.ant-splitter-bar-dragger-disabled')).toBeTruthy();
});
});
it('auto resize', async () => {
containerSize = 200;
const onResize = jest.fn();
const { container } = render(
<SplitterDemo
items={[
{
collapsible: true,
},
{},
]}
onResize={onResize}
/>,
);
triggerResize(container.querySelector('.ant-splitter')!);
await act(async () => {
await waitFakeTimer();
});
fireEvent.click(container.querySelector('.ant-splitter-bar-collapse-start')!);
expect(onResize).toHaveBeenCalledWith([0, 200]);
});
});