3

有什么方法可以触发标签error事件来测试我的回调?例如,给定这个组件,imgonError

import React from 'react';

/**
 * Self removing <img> when the `src` or image
 * does not load or is unavailable.
 *
 * Usage: (tip: it's how you use the <img> tag, basically)
 * <Img src={} alt={} ../..>
 */
var Img = React.createClass({
  /**
   * Force update so `refs` will be available
   */
  componentDidMount() {
    this.forceUpdate();
  },

  render() {
    // Omit any passed `onError` prop so that
    // it is never overridden by mistake
    var { onError, ...other } = this.props;

    return (
      <span ref="container">
        <img {...other} onError={this._handleError} />
      </span>
    )
  },

  /**
   * Remove itself when image is not found
   */
  _handleError() {
    this.refs.container.getDOMNode().remove();
  }
});

export default Img;

这个测试套件:

it('should remove itself when an error occurs while loading the image', function() {
  // rendered = TestUtils.renderIntoDocument(<Img />);
});
4

2 回答 2

7

我发现 React 的测试实用程序 ( React.addons.TestUtils) 非常有用,提供了Simulate( React.addons.TestUtils.Simulate) 之类的实用程序。在这种情况下,Simulate会成功的。

来自React关于测试实用程序的文档:

使用可选的 eventData 事件数据模拟 DOM 节点上的事件调度。这可能是 ReactTestUtils 中最有用的一个实用程序。

it('should remove itself when an error occurs while loading the image', function() {
  var rendered = TestUtils.renderIntoDocument(<Img />);
  var img = TestUtils.findRenderedDOMComponentWithTag(rendered, 'img');
  TestUtils.Simulate.error(img);

  // In this case, we can try to find (again) the tag.
  // `TestUtils.findRenderedDOMComponentWithTag` throws an error
  // when the provided tag cannot be found.
 expect(function() {
   TestUtils.findRenderedDOMComponentWithTag(rendered, 'img');
 }).toThrow();
});

以下内容可能会尝试偏离主题。可以通过使用states 来改进示例组件。

import React from 'react';

/**
 * Self removing <img> when the `src` or image
 * does not load or is unavailable.
 *
 * Usage: (tip: it's how you use the <img> tag, basically)
 * <Img src={} alt={} ../..>
 */
var Img = React.createClass({
  // We'll set `error` to false to load and display the image,
  // only will it be true when an error occurs
  getInitialState() { 
    return { error: false }
  },

  render() {
    // Omit any passed `onError` prop so that
    // it is never overridden by mistake
    var { onError, ...other } = this.props;

    return !this.state.error ? (
      <span>
        <img {...other} onError={this._handleError} />
      </span>
    ) : null;
  },

  /**
   * Set `state` error to true to remove the
   * the dom nodes themselves
   */
  _handleError() {
    this.setState({ error: true });
  }
});

export default Img;

然后使用以下测试套件:

it('should remove itself when an the image loads with an error', function() {
  var rendered = TestUtils.renderIntoDocument(<Img />);
  var img = TestUtils.findRenderedDOMComponentWithTag(rendered, 'img');
  TestUtils.Simulate.error(img);

  // When no results come out of `findRenderedDOMComponentWithTag`,
  // it throws an error 'Did not find exactly one match for tag:img'
  expect(function() {
    TestUtils.findRenderedDOMComponentWithTag(rendered, 'img')
  }).toThrow('Did not find exactly one match for tag:img');
  expect(rendered.state.error).toBe(true);
});
于 2015-03-15T16:51:48.747 回答
1

使用 jest 和 react 测试库非常容易。

在实际代码中,我们正在监听 onError 事件

const handleImgError = (e) => {
  e.target.src = 'https://default-placehodler-image-path.svg';
};

<img
  src='http://4040.jpg'
  onError={handleImgError}
/>

在测试代​​码中

import { render, waitFor, fireEvent } from '@testing-library/react';

const { container } = render(<TestComponent />);

// get img element
const imgEl = container.querySelector('img');

// simulate error event on element
fireEvent.error(imgEl, {
  target: imgEl,
});

// expect updated change on img element
await waitFor(() => {
  expect(imgEl.src).toEqual(
    'https://default-placehodler-image-path.svg'
  );
});

于 2021-05-10T03:30:55.800 回答