Ответ 1
Попробуйте это
link.simulate('click', {
preventDefault: () => {
}
});
Я действительно не знаю, как смоделировать встроенную функцию в дочернем компоненте реагировать
Мой стек: sinon
, chai
, enzyme
;
Использование компонента:
<ListItem onClick={() => someFn()} />
Компонент рендеринга:
render() {
return (
<li>
<a href="#" onClick={e => {
e.preventDefault();
this.props.onClick();
}}
> whatever </a>
</li>
);
}
Здесь у нас есть функция onClick
, которая вызывает e.preventDefault()
. Как сказать <a href>
(link
) не звонить e.preventDefault()
? Как я могу издеваться над onClick
?
Вот что я попробовал в тестах:
Настройка мелкой копии
function setup() {
const someFn = sinon.stub();
const component = shallow(
<ListItem
onClick={() => {
someFn();
}}
/>
);
return {
component: component,
actions: someFn,
link: component.find('a'),
listItem: component.find('li'),
}
}
И тест
it('simulates click events', () => {
const { link, actions } = setup();
link.simulate('click'); //Click on <a href>
expect(actions).to.have.property('callCount', 1); //would be good if we'll remove e.preventDefault()
});
Ошибка вывода теста:
TypeError: Cannot read property 'preventDefault' of undefined
Попробуйте это
link.simulate('click', {
preventDefault: () => {
}
});
test('simulates click events', () => {
const e = { stopPropagation: jest.fn() };
const component = shallow(<ListItem{...props} />);
const li = component.find('li').at(0).childAt(0)
li.props().onClick(e)
expect();
});
Просто отметим, что это проблема только при использовании shallow
ферментатора. В случае полного mount
DOM-рендеринга объект события содержит метод preventDefault
, поэтому вам не нужно его издеваться.
Для тех, кто использует Jest и @testing-library
или fireEvent
react-testing-library
fireEvent
, вам необходимо предоставить инициализированный объект события, иначе событие не может быть отправлено через ваш элемент.
Затем можно утверждать, что e.preventDefault
, назначив свойство этому инициализированному событию:
test('prevents default on click', () => {
const {getByText} = render(<MyComponent />);
const button = getByText(/click me/);
// initialise an event, and assign your own preventDefault
const clickEvent = new MouseEvent('click');
Object.assign(clickEvent, {preventDefault: jest.fn()});
fireEvent(button, clickEvent);
expect(event.preventDefault).toHaveBeenCalledTimes(1);
});
Точно так же и для stopPropagation
.
Антон Карпенко ответ для Jest оказался полезным.
Я бы предложил создать новый объект на основе jest.fn() с помощью
const event = Object.assign(jest.fn(), {preventDefault: () => {}})
затем используйте его:
element.simulate('click', event);
Вы можете определить объект с помощью относящейся к нему функции, которую вы будете издеваться, с помощью некоторого инструмента тестирования, например, посмотрите на Jest и Enzyme
describe('Form component', () => {
test('deos not reload page after submition', () => {
const wrapper = shallow(<TodosForm />)
// an object with some function
const event = { preventDefault: () => {} }
// mocks for this function
jest.spyOn(event, 'preventDefault')
wrapper.find('form').simulate('submit', event)
// how would you know that function is called
expect(event.preventDefault).toBeCalled()
})
})