[译] 用 Typescript + Composition API 重构 Vue 3 组件

原文:https://vuejs-course.com/blog/vuejs-3-typescript-options-composition-api

[译] 用 Typescript + Composition API 重构 Vue 3 组件_第1张图片

译注:原文作者著有 “Vue Testing Handbook”,其中文版已授权本公众号翻译为 ????《Vue 测试指南》中文版,欢迎参阅!


Options API、Composition API、JavaScript,以及 TypeScript -- 这些 API 和语言真能混在一起用?

本文会将使用 JavaScript 和 Options API 构建的传统结构 Vue 3 组件,重构为使用 TypeScript 和 Composition API 的版本。我们将看到一些不同之处,以及可能带来的益处。

同时因为这些既有组件拥有单元测试,我们也将观察这些测试在重构过程中是否仍有效、我们要不要改进它们。至少经验告诉我们,如果只是进行不改变组件对外行为的单纯重构,是不用改变测试的;而如果需要的话,说明你的测试并不理想,它们关注了实现细节。

1. 既有组件

我们将重构 FilterPosts 组件。鉴于 Vue Test Utils 和 Jest 尚无对 Vue.js 3 组件的官方支持,该组件使用了 render 函数编写。为照顾对其不太熟悉的读者,我将其对应的 HTML 写在了注释里。因为源码过长,先来看看其生成的基本模板结构:

  

Posts from {{ selectedFilter }}

     

这个片段用渲染 子组件来展示若干新闻。用户也可以通过 子组件来配置他们要以何种时间优先级来浏览新闻,如点击 “Today”、“This Week” 等按钮。

并且假设有如下 mock 数据:

const posts = [
  {
    id: 1,
    title: 'In the news today...',
    created: moment()
  },
  {
    id: 2,
    title: 'In the news this week...',
    created: moment().add(4 ,'days')
  }
]

在重构过程中,我将介绍每个组件。在此之前,先通过测试用例来了解一下用户的交互:

describe('FilterPosts', () => {
  it('renders today posts by default', async () => {
    const wrapper = mount(FilterPosts)

    expect(wrapper.find('.post').text()).toBe('In the news today...')
    expect(wrapper.findAll('.post')).toHaveLength(1)
  })

  it('toggles the filter', async () => {
    const wrapper = mount(FilterPosts)

    wrapper.findAll('button')[1].trigger('click')
    await nextTick()

    expect(wrapper.findAll('.post')).toHaveLength(2)
    expect(wrapper.find('h1').text()).toBe('Posts from this week')
    expect(wrapper.findAll('.post')[0].text()).toBe('In the news today...')
    expect(wrapper.findAll('.post')[1].text()).toBe('In the news this week...')
  })
})

对该组件,将讨论如下改变:

  • 使用 Composition API 的 refcomputed 代替 datacomputed

  • 使用 TypeScript 将 postsfilters 等改为强类型

  • JS 和 TS 的优缺点对比

2. 断言 filter 的类型并重构 Filter 组件

从最简单的组件开始并逐步推进,是很好的方式。Filter 组件如下:

const filters = ['today', 'this week']

export const Filter = defineComponent({
  props: {
    filter: {
      type: String,
      required: true
    }
  },

  render(h, ctx) {
    // {{ filter }}/

这里主要要做的就是声明 filter 属性的类型。可以使用 TS 中的 type (用 enum 也行) 来实现:

type FilterPeriod = 'today' | 'this week'
const filters: FilterPeriod[] = ['today', 'this week']

export const Filter = defineComponent({
  props: {
    filter: {
      type: String as () => FilterPeriod,
      required: true
    }
  },
  // ...
)

译注 - 关于 String as () => FilterPeriod 类型断言:

考察 Vue 2 中的相关类型定义:
type Prop = { (): T } | { new(...args: never[]): T & object } | { new(...args: string[]): Function }

以及 Vue 3 中类似的定义:
type PropConstructor = | { new (...args: any[]): T & object } | { (): T } | PropMethod

其实不难发现,在 Prop 的 TypeScript 静态检查阶段,对于 FilterPeriod 这类 type 或 interface,因为其并非包含构造函数(new)的完整类型,所以就要用符合类型签名 { (): T } 的形式。

而之所以不能直接写 String as FilterPeriod,因为这不符合 TS 定义, FilterPeriod 类型本身并非完整兼容 String 的,没有包含其所有方法,会报错;而用 () => FilterPeriod 得到的,会被 TS 认为是合法的、并限定在定义取值范围内的字符串类型实例。

同理,形如 interface User { name: string } 之于 Object,也是一样的。

相比于要代码的阅读者去清所谓的 String 实际仅限于合法的 filter 来说,这已经是个很大的改善了;并且结合利用 IDE 的提示功能,这也能在运行测试或运行应用之前就找到可能的输入错误。

下面把 render 函数的逻辑移入 setup 函数;通过这种方式,我们获得了对于 this.filterthis.$emit 更好的类型推断:

setup(props, context) {
  return () => h( // import { h } from 'vue'
    'button', 
    { onClick: () => context.emit('select', props.filter) }, 
    props.filter
  )
}

能够获得上述类型推断改善的主要原因,就在于摆脱了 JS 中高度动态化的 this

听说 VSCode 的 Vue 组件插件 “Vetur” 也为 Vue 3 进行了升级,在