如何测试在点击按钮时是否执行了函数?

huangapple go评论76阅读模式
英文:

how to test if a function was executed when clicking on the button?

问题

Sure, here's the translated code portion:

我已经实现了下面的组件其中包含有关产品的信息以及一个按钮当按下该按钮时会将用户重定向到详细信息屏幕

export default function ProductItem(props: ProductItemProps) {
  const {navigate} = useNavigation<any>();

  const {title, id, rating, category, price, thumbnail} = props;

  function navigateToProduct() {
    navigate('Product', {id, title});
  }

  return (
    <TouchableOpacity
      testID="button-nav"
      style={styles.container}
      activeOpacity={0.75}
      onPress={navigateToProduct}>
      <Image source={{uri: thumbnail, width: 100, height: 100}} />

      <View style={styles.contentInfo}>
        <Text style={styles.title}>{title}</Text>
        <Text>{category}</Text>

        <StarRating rating={rating} />

        <Text style={styles.price}>R${price}</Text>
      </View>
    </TouchableOpacity>
  );
}

Regarding your question about testing if the function is executed, you can use Jest's jest.fn() to create a mock function and then spy on it using spyOn. Here's an example of how you can modify your test to check if navigateToProduct is executed:

import {fireEvent, render, screen} from '@testing-library/react-native';
import ProductItem from '..';

describe('<ProductItem/>', () => {
  const product = {
    title: 'title-test',
    id: 1,
    rating: 4.5,
    category: 'category-test',
    price: 499,
    thumbnail: 'thumbnail-test',
    description: 'description-test',
    stock: 1,
    brand: 'brand-test',
    images: ['image1-test', 'image2-test'],
  };

  it('Should render correctly product item component', () => {
    render(<ProductItem {...product} />);

    const title = screen.getByText(product.title);
    expect(title).toBeTruthy();
  });

  it('Should navigate to product screen when button pressed', async () => {
    // Create a mock function
    const navigateMock = jest.fn();

    // Mock the useNavigation hook to return the mock function
    jest.mock('@react-navigation/native', () => ({
      ...jest.requireActual('@react-navigation/native'),
      useNavigation: () => ({
        navigate: navigateMock,
      }),
    }));

    render(<ProductItem {...product} />);

    const toPress = screen.getByTestId('button-nav');

    fireEvent.press(toPress);

    // Check if the navigate function was called with the expected arguments
    expect(navigateMock).toHaveBeenCalledWith('Product', { id: product.id, title: product.title });
  });
});

This code creates a mock function for the navigate function and ensures that it's called with the expected arguments when the button is pressed in your component.

英文:

I have implemented the component below, which is a card containing information about a product and a button that, when pressed, redirects the user to the details screen.

export default function ProductItem(props: ProductItemProps) {
  const {navigate} = useNavigation&lt;any&gt;();

  const {title, id, rating, category, price, thumbnail} = props;

  function navigateToProduct() {
    navigate(&#39;Product&#39;, {id, title});
  }

  return (
    &lt;TouchableOpacity
      testID=&quot;button-nav&quot;
      style={styles.container}
      activeOpacity={0.75}
      onPress={navigateToProduct}&gt;
      &lt;Image source={{uri: thumbnail, width: 100, height: 100}} /&gt;

      &lt;View style={styles.contentInfo}&gt;
        &lt;Text style={styles.title}&gt;{title}&lt;/Text&gt;
        &lt;Text&gt;{category}&lt;/Text&gt;

        &lt;StarRating rating={rating} /&gt;

        &lt;Text style={styles.price}&gt;R${price}&lt;/Text&gt;
      &lt;/View&gt;
    &lt;/TouchableOpacity&gt;
  );
}

So far, I've already implemented the test to check the component's rendering, but I need to test if when the button is pressed, the "navigateToProduct" function will be executed.

import {fireEvent, render, screen} from &#39;@testing-library/react-native&#39;;
import ProductItem from &#39;..&#39;;

describe(&#39;&lt;ProductItem/&gt;&#39;, () =&gt; {
  const product = {
    title: &#39;title-test&#39;,
    id: 1,
    rating: 4.5,
    category: &#39;category-test&#39;,
    price: 499,
    thumbnail: &#39;thumbnail-test&#39;,
    description: &#39;description-test&#39;,
    stock: 1,
    brand: &#39;brand-test&#39;,
    images: [&#39;image1-test&#39;, &#39;image2-test&#39;],
  };

  it(&#39;Should render correctly product item component&#39;, () =&gt; {
    render(&lt;ProductItem {...product} /&gt;);

    const title = screen.getByText(product.title);
    expect(title).toBeTruthy();
  });

  it(&#39;Should navigate to product screen when button pressed&#39;, async () =&gt; {
    render(&lt;ProductItem {...product} /&gt;);

    const toPress = screen.getByTestId(&#39;button-nav&#39;);

    fireEvent.press(toPress);
  });
});

The whole mock of react navigation "useNavigation" already works. And until then, the tests are all passing. My question is just how can I test if the function is executed. I've tried using spyOn, but I don't know exactly how I can mock this function within my tests. Can anyone give me any suggestions?

答案1

得分: 0

你可以使用jest.fn()创建一个模拟函数,并在触发按键事件后使用expect(mockFn).toBeCalled()来测试该函数是否已被调用。以下是来自React Native Testing Library文档的示例(根据您的用例,可以交换toBeCalledWithtoBeCalled):

https://github.com/callstack/react-native-testing-library/blob/main/src/__tests__/questionsBoard.test.tsx

test('form submits two answers', () => {
  const allQuestions = ['q1', 'q2'];
  const mockFn = jest.fn();

  render(<QuestionsBoard questions={allQuestions} onSubmit={mockFn} />);

  const answerInputs = screen.getAllByLabelText('answer input');

  fireEvent.changeText(answerInputs[0], 'a1');
  fireEvent.changeText(answerInputs[1], 'a2');
  fireEvent.press(screen.getByText('Submit'));

  expect(mockFn).toBeCalledWith({
    1: { q: 'q1', a: 'a1' },
    2: { q: 'q2', a: 'a2' },
  });
});

请注意,上述代码是用于测试React Native组件的示例。

英文:

You can create a mock function with jest.fn() and use expect(mockFn).toBeCalled() after firing the key press even to test if the function has been called. Here is an example from the React Native Testing Library docs (swap toBeCalledWith and toBeCalled for your use case):

https://github.com/callstack/react-native-testing-library/blob/main/src/__tests__/questionsBoard.test.tsx


test(&#39;form submits two answers&#39;, () =&gt; {
  const allQuestions = [&#39;q1&#39;, &#39;q2&#39;];
  const mockFn = jest.fn();

  render(&lt;QuestionsBoard questions={allQuestions} onSubmit={mockFn} /&gt;);

  const answerInputs = screen.getAllByLabelText(&#39;answer input&#39;);

  fireEvent.changeText(answerInputs[0], &#39;a1&#39;);
  fireEvent.changeText(answerInputs[1], &#39;a2&#39;);
  fireEvent.press(screen.getByText(&#39;Submit&#39;));

  expect(mockFn).toBeCalledWith({
    1: { q: &#39;q1&#39;, a: &#39;a1&#39; },
    2: { q: &#39;q2&#39;, a: &#39;a2&#39; },
  });
});

huangapple
  • 本文由 发表于 2023年5月21日 23:13:23
  • 转载请务必保留本文链接:https://go.coder-hub.com/76300586.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定