被迫开始学习Typescript —— vue3的 props 与 interface

vue3 的 props

Vue3 的 props ,分为 composition API 的方式以及 option API 的方式,可以实现运行时判断类型,验证属性值是否符合要求,以及提供默认值等功能。

props 可以不依赖TS,自己有一套运行时的验证方式,如果加上TS的话,还可以实现在编写代码的时候提供约束、判断和提示等功能。

Prop 的校验

官网:https://staging-cn.vuejs.org/guide/components/props.html#prop-validation

Vue 提供了一种对 props 的属性进行验证的方法,有点像 Schema。不知道Vue内部有没有提供interface,目前没有找到,所以我们先自己定义一个:

/**
 * vue 的 props 的验证的类型约束
 */
export interface IPropsValidation {
  /**
   * 属性的类型,比较灵活,可以是 String、Number 等,也可以是数组、class等
   */
  type: Array | any,
  /**
   * 是否必须传递属性
   */
  required?: boolean,
  /**
   * 自定义类型校验函数(箭头函数),value:属性值
   */
  validator?: (value: any) => boolean,
  /**
   * 默认值,可以是值,也可以是函数(箭头函数)
   */
  default?: any
}

后面会用到。

composition API

官网:https://staging-cn.vuejs.org/guide/typescript/composition-api.html

准确的说是在 script setup 的情况下,如何设置 props,具体方法看官网,这里不搬运。

探讨一下优缺点。

interface Props {
  foo: string
  bar?: number
}

// 对 defineProps() 的响应性解构
// 默认值会被编译为等价的运行时选项
const { foo, bar = 100 } = defineProps()

// 引入 接口定义
import { Props } from './other-file'

// 不支持!
defineProps()

虽然可以单独定义 interface ,而且可以给整体 props 设置类型约束,但是只能在组件内部定义,目前暂时不支持从单独的文件里面读取。而且不能”扩充”属性。

也就是说,基本无法实现复用。

这个缺点恰恰和我的目的冲突,等待新版本可以解决吧。

option API

官网:https://staging-cn.vuejs.org/guide/typescript/options-api.html

这种方式支持Option API,也支持 setup 的方式,可以从外部引入 接口定义,但是似乎不能给props定义整体的接口。

import { defineComponent } from 'vue'
import type { PropType } from 'vue'

interface Book {
  title: string
  year?: number
}

export default defineComponent({
  props: {
    bookA: {
      type: Object as PropType,
      // 确保使用箭头函数
      default: () => ({
        title: 'Arrow Function Expression'
      }),
      validator: (book: Book) => !!book.title
    }
  },
  setup(props) {
    props.message //

想了半天,可以用”二段定义”方式的方式来解决:

  • 定义一个 interface,规定一个组件必须有哪些属性。
  • 定义 props 的 “描述对象”,作为共用的 props。

我的想法

为啥要给 props 设置一个 整体的 interface,而且还要从外部文件引入呢?

因为我理解的 interface 可以拥有”约束”的功能,即:可以通过 interface 约束多个(相关)组件的 props 里面必须有一些相同的属性。

所以需要在一个单独的文件里面定义接口,然后在组件里面引入,设置给组件的props。

Vue不倡导组件使用继承,那么如果想要约束多个组件,拥有相同的 props?似乎应该可以用 interface ,但是看官方文档,好像思考角度不是这样的。

应对方式

  • 先定义组件需要哪些属性的 interface:
/**
 * 表单子控件的共用属性。约束必须有的属性
 */
export interface ItemProps {
  /**
   * 字段ID、控件ID,sting | number
   */
  columnId: IPropsValidation,
  /**
   * 表单的 model,含义多个属性,any
   */
  model: IPropsValidation,
  /**
   * 字段名称,string
   */
  colName: IPropsValidation,
  /**
   * 控件类型,number
   */
  controlType: IPropsValidation,
  /**
   * 控件备选项,一级或者多级,Array
   */
  optionList: IPropsValidation,
  /**
   * 访问后端API的配置,IWebAPI
   */
  webapi: IPropsValidation,
  /**
   * 防抖延迟时间,0:不延迟,number
   */
  delay: IPropsValidation,
  /**
   * 防抖相关的事件() => void
   */
  events: IPropsValidation,
  /**
   * 控件的大小,string
   */
  size: IPropsValidation,
  /**
   * 是否显示清空的按钮,boolean
   */
  clearable: IPropsValidation,
  /**
   * 控件的扩展属性,any
   */
  extend: IPropsValidation,
}

ItemProps:目的是约束一个组件需要设置哪些属性,限制属性名称。

  • 然后定义 共用 的 props 的描述对象:
import type { PropType } from 'vue'

import type {
  ItemProps,
  IOptionItem,
  IOptionItemTree,
  IWebAPI
} from '../types/type'

/**
 * 基础控件的共用属性,即表单子控件的基础属性
 */
const itemProps: ItemProps = {
  /**
   * 字段ID、控件ID
   */
  columnId: {
    type: [Number, String],
    default: () => Math.floor((Math.random() * 1000000) + 1) // new Date().valueOf()
  },
  /**
   * // 表单的 model,可以整体传入,便于子控件维护字段值。
   */
  model: {
    type: Object
  },
  /**
   * 字段名称,控件使用 model 的哪个属性,多个字段名称用 "_" 分割
   */
  colName: {
    type: String,
    default: ''
  },
  /**
   * 控件类型,表单控件据此加载对应的子控件
   */
  controlType: {
    type: Number,
    default: 101
  },
  /**
   * 控件的备选项,单选、多选、等控件需要
   */
  optionList: {
    type: Object as PropType>,
    default: () =>  {return []}
  },
  /**
   * 访问后端API的参数,IWebAPI
   */
  webAPI: {
    type: Object as PropType,
    default: () => {
      return {
        serviceId: '',
        actionId: '',
        dataId: '',
        body: null,
        cascader: {
          lazy: false, // 是否需要动态加载
          actions: ['',''] // 按照level的顺序设置后端 API 的 action
        }
      }
    }
  },
  /**
   * 防抖的时间间隔,0:不用防抖。
   */
  delay: {
    type: Number,
    default: 0
  },
  /**
   * 事件集合,主要用于防抖
   */
  events: {
    type: Object,
    default: () => {
      return {
        input: () => {}, // input 事件
        enter: () => {}, // 按了回车
        keydown: () => {} // 正在输入
      }
    }
  },
  /**
   * 子控件的规格,默认设置。
   * * 【element-plus】large / default / small 三选一
   */
  size: { //
    type: String,
    default: 'small',
    validator: (value) => {
      // 这个值必须匹配下列字符串中的一个
      return ['large', 'default ', 'small'].indexOf(value) !== -1
    }
  },
  /**
   * 是否显示可清空的按钮,默认显示
   */
  clearable: {
    type: Boolean,
    default: true
  },
  /**
   * 扩展属性,对象形式,存放组件的扩展属性
   */
  extend: {
    type: Object,
    default: () => {return {}}
  }
}

export { itemProps }

定义 props 的属性的具体类型、默认值等。

  • 最后在组件里面引入

  import { itemProps } from '../../../lib/base/props-item'

  export default defineComponent({
    name: 'ui-core-form-item',
    props: {
      aa: String,
      ...itemProps
    },
    setup(props) {
      console.log('表单子控件的 props:', props)

      return {
        props
      }
    }
  })

使用解构的方式设置组件的 props,还可以有提示,还可以扩展自己的属性。

被迫开始学习Typescript —— vue3的 props 与 interface

好像哪里不对,不过先这样了。

vue3 的 props 到底是啥结构?

说起来比较复杂:

被迫开始学习Typescript —— vue3的 props 与 interface
  • 外层是 shallowReadonly。(第一层属性不能直接改,但是第二层(通过引用类型)可以直接改。)
  • 里面是 shallowReactive。(解构时不会强制把普通对象变成reactive,为了效率吧。)

基本就是这样。

Original: https://www.cnblogs.com/jyk/p/16288682.html
Author: 金色海洋(jyk)
Title: 被迫开始学习Typescript —— vue3的 props 与 interface

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/549192/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • 准备工作(二)—— 安装图形化软件

    可能就会有人在问:安装MySQL为什么还要图形化软件? 实际上MySQL有两种方式来执行请求,一是通过手打命令的方式,二是通过图形化界面来进行操作,后者本质上也是通过输入命令来执行…

    技术杂谈 2023年6月22日
    081
  • 搬砖_kafka的一些命令

    有关kafka的一些命令 kafka查看当前消费组消费到哪个offset cd /opt/kafka/bin #即进&#x516…

    技术杂谈 2023年7月25日
    058
  • 麦克风采集与播放 (源码)

    在网络聊天系统中,采集麦克风的声音并将其播放出来,是最基础的模块之一。本文我们就介绍如何快速地实现这个基础模块。 有几个与声音采集和播放相关的专业术语必须要先了解一下,否则,后面的…

    技术杂谈 2023年6月1日
    0113
  • 运用Spring Aop,一个注解实现日志记录

    运用Spring Aop,一个注解实现日志记录 1. 介绍 我们都知道Spring框架的两大特性分别是 IOC (控制反转)和 AOP (面向切面),这个是每一个Spring学习视…

    技术杂谈 2023年6月21日
    0113
  • 130-SLM130(NB-IOT)C-SDK(OpenCPU)学习开发-低功耗(PSM,DRX,eDRX)

    先看PSM (Power Save Mode)节能模式 1,NB的节能模式是有两个时间参数控制 TAU时间 和 active时间 PSM模式是模组断开了网络连接,只能人为或者程序上…

    技术杂谈 2023年6月1日
    092
  • Rust:axum学习笔记(1) hello world

    Rust中的Tokio几乎是同类框架的性能天花板了,而axum在Tokio基础上构建,起点就站在巨人的肩膀上。 先来一个Hello World的入门示例: bash;gutter:…

    技术杂谈 2023年5月31日
    091
  • pidstat命令详解

    pidstat命令详解 pidstat 命令是 sysstat 工具的一个命令,用来监控全部或者指定 进程的CPU、内存、线程、设备IO等系统资源的占用情况。 pidstat 首次…

    技术杂谈 2023年5月31日
    0114
  • SGU 311. Ice-cream Tycoon(线段树)

    Time limit per test: 0.5 second(s)Memory limit: 65536 kilobytes input: standardoutput: sta…

    技术杂谈 2023年5月31日
    066
  • 冒泡排序

    404. 抱歉,您访问的资源不存在。 可能是网址有误,或者对应的内容被删除,或者处于私有状态。 代码改变世界,联系邮箱 contact@cnblogs.com 园子的商业化努力-困…

    技术杂谈 2023年7月23日
    076
  • Maxima 使用教程

    说起数学软件,我们很多人脑子里浮现出的第一个就是 matlab,不可否认,matlab 确实是一个优秀的数学软件,但是它需要付费啊(这里不讨论盗版问题)。那么有没有一个同样强大但免…

    技术杂谈 2023年7月11日
    0101
  • python3GUI–实用!B站视频下载工具(附源码)

    一.准备工作 二.预览 1.启动 2.解析 3.下载中 4.下载完成 5.结果 三.设计流程 1.bilibili_video_spider 2.视频json的查找 四.源代码 1…

    技术杂谈 2023年6月21日
    0115
  • 剑指offer计划21( 位运算简单)—java

    1.1、题目1 剑指 Offer 15. 二进制中1的个数 1.2、解法 通过判断每一位的与来识别1的数量。 1.3、代码 public class Solution { // y…

    技术杂谈 2023年7月25日
    087
  • 字节跳动-财经-校招内推

    【团队介绍】 我们为字节跳动生态的抖音电商、直播、教育、健康、内容付费等场景服务,依托字节跳动的科技能力和产品,为用户提供更好的支付、消费金融、保险等金融服务体验。科技创新,普惠大…

    技术杂谈 2023年5月31日
    093
  • 什么是CLI?

    命令行界面(英语:command-line interface,缩写]:CLI)是在图形用户界面得到普及之前使用最为广泛的用户界面,它通常不支持鼠标,用户通过键盘输入指令,计算机接…

    技术杂谈 2023年5月31日
    092
  • quartz框架(四)-Job相关内容

    本篇博文,博主主要介绍job相关的内容。 job是业务类需要实现的接口,代表需要被调度框架进行调度的任务。job源码如下所示,从源码中我们可以看到,job接口只有一个excute方…

    技术杂谈 2023年7月24日
    072
  • Vue学习笔记(五):计算属性

    1 模板与方法的不足 ¶ 有两个输入款,一个输入姓(firstname),一个输入名(secondname),fullname由转换为大写字母的姓、横线、转换为大写字母的名拼接组成…

    技术杂谈 2023年7月24日
    087
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球