如何测试驱动开发 React 组件?

sxkk20081年前知识分享74

什么是 TDD

TDD(Test-driven development),就是测试驱动开发,是敏捷开发中的一项核心实践和技术,也是一种软件设计方法论。

它的原理就是在编写代码之前先编写测试用例,由测试来决定我们的代码。而且 TDD 更多地需要编写独立的测试用例,比如只测试一个组件的某个功能点,某个工具函数等。

TDD 的过程

  • 编写测试用例
  • 运行测试,测试失败
  • 修改代码
  • 测试通过
  • 重构/优化代码
  • 新增功能,重复上述步骤

image.png

在某种程度上,它可能在初学者看来是单调乏味或者不切实际的,但是严格按照这个步骤来做这件事,让你自己决定测试用例是否对你的组件有帮助,会让测试用例变得有意义。

本文将以创建一个 Confirmation 组件来说明,如何在 React 中如何实现测试驱动开发。

Confirmation 组件的特点:

  • Confirmation 标题
  • 确认描述 —— 接收外部程序想要确认的问题
  • 一个确认的按钮,支持外部回调函数
  • 一个取消的按钮,支持外部回调函数

这两个按钮都不知道点击时接下来要做什么事,因为它超出了组件的职责范围,但是组件应该接收这些点击按钮的回调事件。先找个设计图:

image.png

那么,让我们开始吧。

测试组件

首先使用 create-react-app 初始化一个 react 项目。目前 cra 已经内置了 @testing-library/react 作为测试框架。

npx create-react-app my-react-app

我们先从测试文件开始。先创建了组件的目录“Confirmation” 并在其中添加一个“index.test.js”文件。

确保渲染测试

第一个测试相当抽象。仅仅需要检查组件是否展现(任何东西) ,以确保这个组件是存在。但是实际上,我将要测试的组件还不存在。

首先通过 getByRole 方法 查找 role属性等于dialog能否文档中找到。

role 属性可能不太常用, 当现有的 HTML 标签不能充分表达语义性的时候,就可以借助 role 来说明. 例如点击的按钮,就是 role="button" ;会让这个元素可点击;也可以使用 role 属性告诉辅助设备(如屏幕阅读器)这个元素所扮演的角色。

import React from 'react'
import { render } from '@testing-library/react'

describe('Confirmation component', () => {
  it('should render', () => {
    const { getByRole } = render(<Confirmation />)
    expect(getByRole('dialog')).toBeInTheDocument()
  })
})

运行测试并且监听

yarn test --watch

image.png

用 “脚趾头” 思考都知道这肯定是不能通过测试的。接下来,让我们创建一个足够满足这个测试的组件:

import React from 'react'

const Confirmation = () => {
  return <div role="dialog">div>
}

export default Confirmation

然后把这个组件导入到测试中,它现在通过了。

image.png

接下来,组件应该有一个动态标题。

动态标题测试

创建一个测试用例:

it('should have a dynamic title', () => {
  const title = '标题'
  const { getByText } = render(<Confirmation title={title} />)
  expect(getByText(title)).toBeInTheDocument()
})

测试失败了,修改代码使它通过:

import React from 'react'

const Confirmation = ({ title }) => {
  return (
    <div role="dialog">
      <h1>{title}h1>
    div>
  )
}

export default Confirmation

下一个特性,这个组件中存在一个确认问题提示

动态问题测试

这个问题也是动态的,这样它就可以从组件外部传入。

it('should have a dynamic confirmation question', () => {
  const question = 'Do you confirm?'
  const { getByText } = render(<Confirmation question={question} />)
  expect(getByText(question)).toBeInTheDocument()
})

测试再次失败,修改代码让它通过:

import React from 'react'

const Confirmation = ({ title, question }) => {
  return (
    <div role="dialog">
      <h1>{title}h1>
      <div>{question}div>
    div>
  )
}

export default Confirmation

确认按钮测试

接下来是确认按钮测试。我们首先要检查组件上是否有一个按钮,上面写着“确认”。

编写测试用例代码:

it('should have an "OK" button', () => {
  const { getByRole } = render(<Confirmation />)
  expect(getByRole('button', { name: '确认' })).toBeInTheDocument()
})

在这里使用 name 选项,因为我们知道这个组件中至少还有一个按钮,需要更具体地说明查找断言的是哪个按钮

组件代码:

import React from 'react'

const Confirmation = ({ title, question }) => {
  return (
    <div role="dialog">
      <h1>{title}h1>
      <div>{question}div>
      <button>确认button>
    div>
  )
}

export default Confirmation

取消按钮测试

同样对“取消”按钮做同样的事情:

测试:

it('should have an "取消" button', () => {
  const { getByRole } = render(<Confirmation />)
  expect(getByRole('button', { name: '取消' })).toBeInTheDocument()
})

组件代码:

import React from 'react'

const Confirmation = ({ title, question }) => {
  return (
    <div role="dialog">
      <h1>{title}h1>
      <div>{question}div>
      <button>确认button>
      <button>取消button>
    div>
  )
}

export default Confirmation

好了。现在我们得到了我们想要的组件渲染的 HTML ,现在我想要确保我可以从外部传递这个组件的按钮的回调函数,并确保它们在单击按钮时被调用。

那么我将从“确认”按钮的测试开始:

it('should be able to receive a handler for the "确认" button and execute it upon click', () => {
  const onOk = jest.fn()
  const { getByRole } = render(<Confirmation onOk={onOk} />)
  const okButton = getByRole('button', { name: '确认' })

  fireEvent.click(okButton)

  expect(onOk).toHaveBeenCalled()
})

先用 jest.fn 创建一个模拟函数,将其作为“onOk”处理函数传递给组件,模拟单击“确认”按钮,并断言函数已被调用。

image.png

这个测试显然失败了,下面是补充代码:

import React from 'react'

const Confirmation = ({ title, question, onOk }) => {
  return (
    <div role="dialog">
      <h1>{title}h1>
      <div>{question}div>
      <button onClick={onOk}>确认button>
      <button>取消button>
    div>
  )
}

export default Confirmation

最后,让我们对“取消”按钮做同样的事情:

测试:

it('should be able to receive a handler for the "取消" button and execute it upon click', () => {
  const onCancel = jest.fn()
  const { getByRole } = render(<Confirmation onCancel={onCancel} />)
  const okButton = getByRole('button', { name: '取消' })

  fireEvent.click(okButton)

  expect(onCancel).toHaveBeenCalled()
})

组件:

import React from 'react'

const Confirmation = ({ title, question, onOk, onCancel }) => {
  return (
    <div role="dialog">
      <h1>{title}h1>
      <div>{question}div>
      <button onClick={onOk}>确认button>
      <button onClick={onCancel}>取消button>
    div>
  )
}

export default Confirmation

下面是完整的测试文件:

import React from 'react'
import { render, fireEvent } from '@testing-library/react'
import Confirmation from './index'

describe('Confirmation component', () => {
  it('should render', () => {
    const { getByRole } = render(<Confirmation />)
    expect(getByRole('dialog')).toBeInTheDocument()
  })

  it('should have a dynamic title', () => {
    const title = '标题'
    const { getByText } = render(<Confirmation title={title} />)
    expect(getByText(title)).toBeInTheDocument()
  })

  it('should have a dynamic confirmation question', () => {
    const question = 'Do you confirm?'
    const { getByText } = render(<Confirmation question={question} />)
    expect(getByText(question)).toBeInTheDocument()
  })

  it('should have an "确认" button', () => {
    const { getByRole } = render(<Confirmation />)
    expect(getByRole('button', { name: '确认' })).toBeInTheDocument()
  })

  it('should have an "取消" button', () => {
    const { getByRole } = render(<Confirmation />)
    expect(getByRole('button', { name: '取消' })).toBeInTheDocument()
  })

  it('should be able to receive a handler for the "确认" button and execute it upon click', () => {
    const onOk = jest.fn()
    const { getByRole } = render(<Confirmation onOk={onOk} />)
    const okButton = getByRole('button', { name: '确认' })

    fireEvent.click(okButton)

    expect(onOk).toHaveBeenCalled()
  })

  it('should be able to receive a handler for the "取消" button and execute it upon click', () => {
    const onCancel = jest.fn()
    const { getByRole } = render(<Confirmation onCancel={onCancel} />)
    const okButton = getByRole('button', { name: '取消' })

    fireEvent.click(okButton)

    expect(onCancel).toHaveBeenCalled()
  })
})

虽然这个组件没有样式,或者说我们还可以优化,添加跟多的功能,以上步骤已经充分展示了测试驱动开发的逻辑。

image.png

TDD 一步一步地引导完成组件特性的规范,确保我们在组件重构或者他人修改代码的时候能够遵循现有开发的逻辑。这这是 TDD 的优势。

调试

我们可以使用 debug 打印渲染的 html 结构

代码

it('should be able to receive a handler for the "取消" button and execute it upon click', () => {
  const onCancel = jest.fn()
  const { getByRole, debug } = render(<Confirmation onCancel={onCancel} />)

  debug()
})

image.png

这样可以方便我们查找 dom。

小结

当然 @testing-library/react 还有很多方便的 api。大家可以自行查阅。

image.png

未来可能会出一些文章关于测试的文章。例如:

如何出测试 react hooks ?

如何测试 react 路由?

如何测试接口?

希望这篇文章对大家有所帮助,也可以参考我往期的文章或者在评论区交流你的想法和心得,欢迎一起探索前端。

相关文章

AI技术与个人隐私:如何平衡数据驱动和隐私保护

AI技术与个人隐私:如何平衡数据驱动和隐私保护

  人工智能技术通过收集和分析大量的个人信息和行为数据,为用户提供了更好的产品和服务,并且也帮助企业实现更高的商业价值。但是,随着AI技术的不断发展,隐私成为了越来越重要的话...

天翼智慧社区:领航智能生活,引领未来社区生态

天翼智慧社区:领航智能生活,引领未来社区生态

  随着科技的不断发展,智能生活已经成为当今社会的一大潮流。天翼智慧社区作为中国电信旗下的智慧社区品牌,以其先进的技术和全面的服务,成为了引领智能社区发展的领军者。从智能家居...

经过一晚上的配置和折腾,花钱不说,虽然可以实现在 iPad 上编程的需求,但体验远却比不上 PC,今天我就来推荐一种新的方式,让你的 iPad 变成真正的生产力工具。

Cloud Studio 简介

Cloud Studio 是基于浏览器的集成式开发环境(IDE),为开发者提供了一个永不间断的云端工作站。用户在使用 Cloud Studio 时无需安装,随时随地打开浏览器就能使用。

Cloud Studio iPad 编程 以上是我用 iPad 浏览器,在 1 分钟内初始化了一个 next 初始化模板,可以说速度比本地开发还快,当我在左侧修改代码时,右侧预览界面便会同步热更新。

Cloud Studio 安装 react snippts

Cloud Studio 可以说是提供了一台云服务器,并且把 VSCode 搬到了线上,我们可以同本地开发一样,在上面安装插件,比如可以在左侧扩展中搜索 react,安装这个 react snippts 扩展,便可以帮助我们提供常用代码片段,快速创建组件。

iPad 编程生产力

前言iPad 有个口号,就是“买前生产力,买后爱奇艺”,使用 iPad,配合 Procreate 来作画体验还可以, 如果你想让你的 iPad 可以编程,你却不得不为之花费时间和精力,我搜了网上的教程...

AI识图技术的进步对社会的影响

AI识图技术的进步对社会的影响

  随着人工智能(AI)技术的迅猛发展,AI识图应用正逐渐渗透到我们的日常生活。AI识图作为其中的重要应用之一,正在改变着我们的工作方式、生活习惯以及社会结构。  AI识图是...

绿米智能家居系统:AI技术引领未来智慧生活的创新巨擘

绿米智能家居系统:AI技术引领未来智慧生活的创新巨擘

  随着科技的飞速发展,智能家居已成为时代的新潮流。作为智能家居领域的重要参与者,绿米智能家居系统凭借其卓越的创新能力和出色的用户体验,已经成为业内的佼佼者。该系统以其全方位...

百度Unit平台开启创新之门:人工智能引领新媒体创新

百度Unit平台开启创新之门:人工智能引领新媒体创新

  随着人工智能技术的不断发展,新媒体领域也迎来了创新。近年来,百度Unit平台成为了新媒体领域引领潮流的先锋,不断推出创新的人工智能技术和应用,为用户带来全新的体验和价值。...

发表评论    

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。