2

反応フックを使用しているコンポーネントのテストに問題があります。ボタンをクリックした後、ant design range picker 内の日付範囲が更新されているかどうかをテストする必要があります。テストには Jest と react-testing-library を使用しています。

この時点で、次のようになります: (これらのネストされたコンポーネントの一部は、単にスタイル付きコンポーネントです:))

親.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;

サイドバー.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 関数を prop としてテスト内の DateSelect コンポーネントに渡していないため、範囲値は変更されません。次のように、テスト内にフックを追加しようとしました。

const [dateRange, setDateRange] = [moment(), moment()];

しかし、私は得ています: フックは、関数コンポーネントの本体内でのみ呼び出すことができます。 なのでそこでは使えないと思います。これに対する解決策はありますか?setDateRangeフックのように機能するテスト内の関数を指定する方法は?

私はこれを見つけました: https://github.com/mpeyper/react-hooks-testing-library しかし、これはカスタムフック用のようですか?

このテストケースには、react-hooks を使用しない解決策があるのではないでしょうか? どんなアイデアでも大歓迎です。:)

4

1 に答える 1