我在测试使用反应挂钩的组件时遇到了问题。我需要测试点击按钮后是否更新了蚂蚁设计范围选择器内的日期范围。为了测试我正在使用Jest和react-testing-library。
在这一点上它看起来像这样:(这些嵌套组件中的一些只是样式组件:))
Parent.jsx:
const Parent = () => {
const [dateRange, setDateRange] = useState([moment(), moment()]);
return (
<div>
<Sidebar
visible={sidebarVisible}
setSidebarVisible={setSidebarVisible}
setTableData={setTableData}
setStatementsLength={setStatementsLength}
setDateRange={setDateRange}
dateRange={dateRange}
/>
</div>
);
};
export default Parent;
Sidebar.jsx:
const Sidebar = props => {
return (
<div>
<Drawer
placement="left"
closable={false}
onClose={() => props.setSidebarVisible(false)}
visible={props.visible}
width={488}
bodyStyle={drawerStyle}
>
<DrawerContent data-test-id="statementsSidebar">
<DrawerHeader data-test-id="statementsSidebarHeader">
{t('settlements.statements.sidebar.createQuery')}
</DrawerHeader>
<DrawerBody>
<DateSelect
dateRange={props.dateRange}
setDateRange={props.setDateRange}
/>
</DrawerBody>
<DrawerFooter/>
</DrawerContent>
</Drawer>
</div>
);
};
export default Sidebar;
DateSelect.jsx:
const DateSelect = props => {
const RangePicker = DatePicker.RangePicker;
const [offset, setOffset] = useState(0);
const timeUnit = Object.freeze({ day: 'd', week: 'w', month: 'm' });
const [offsetUnit, setOffsetUnit] = useState(timeUnit.day);
const setRangeToLastWeek = () => {
if (offset > 0) {
props.setDateRange([
moment().subtract(1, 'w'),
moment().add(offset, 'd'),
]);
} else if (offset < 0) {
props.setDateRange([
moment()
.subtract(1, 'w')
.add(offset, 'd'),
moment(),
]);
} else {
props.setDateRange([moment().subtract(1, 'w'), moment()]);
}
setOffsetUnit('w');
};
const manuallySetDate = range => {
setOffset(0);
props.setDateRange(range);
};
return (
<div>
<HeaderRow/>
<DateSelectorRow>
<ButtonsColumn>
<Button
value="lastWeek"
onClick={setRangeToLastWeek}
style={styleButtonPadding}
data-test-id="setRangeToLastWeek"
data-testid="lastWeekButton"
>
Last 7 days
</Button>
</ButtonsColumn>
<div data-testid="range-picker-value">
<RangePicker
value={props.dateRange}
format={DateFormat}
onChange={v => manuallySetDate(v)}
/>
</div>
</DateSelectorRow>
</div>
);
};
export default DateSelect;
DateSelect.test.js:
it('After clicking last 7 days button date range is changed', () => {
// const setDateRange = ??????
const range = [moment(), moment()];
const { container } = render(<DateSelect setDateRange={setDateRange} dateRange={range}/>);
const startDate = getByPlaceholderText(container, 'Start date');
const endDate = getByPlaceholderText(container, 'End date');
const lastWeekButton = getByTestId(container,'lastWeekButton');
expect(startDate.value).toBe(moment().format(DateFormat));
expect(endDate.value).toBe(moment().format(DateFormat));
fireEvent.click(lastWeekButton);
expect(startDate).toBe(moment().subtract(1, 'w').format(DateFormat));
expect(endDate.value).toBe(moment().format(DateFormat));
});
正如预期的那样,因为我没有将setDateRange函数作为测试中的DateSelect组件的prop传递,所以范围值不会改变。我试图在测试中添加一个钩子,如下所示:
const [dateRange, setDateRange] = [moment(), moment()];
但是我得到了:Hook只能在函数组件的主体内部调用。所以我猜我不能在那里使用它。这有什么解决方案吗?如何在测试中指定setDateRange
函数,它会像钩子一样?
我发现了这个:https://github.com/mpeyper/react-hooks-testing-library但它看起来像是自定义钩子?
也许有一个解决方案,这个测试用例没有使用react-hooks?任何想法高度赞赏。 :)
你必须渲染Parent
,因为它包含更新DateSelect
的逻辑。这样可以获得更好的测试,因为您正在测试组件之间的集成是否有效。