tsx开发vue3:从零到全面覆盖

本篇介绍在vue3中使用tsx的使用方法,之前博主有一篇根据路由生成菜单的文章,里面也介绍了jsx语法的基本使用:vue3+jsx使用递归组件实现无限级菜单

本篇聚焦于vue3中使用tsx,从基础语法到复杂使用,再到一些特殊情况的处理方法,并且对照传统template写法,目的是覆盖日常开发的全部使用。本篇主要是总结tsx的使用,至于跟template写法的优劣以及原理,博主不会深入。

1.在项目中使用

安装与配置

首先要安装插件:

npm insatll @vue/babel-plugin-jsx --save

这是这个插件的github:babel-plugin-jsx
在github可以学到一些基础用法

安装完后在项目的 babel.config.js 文件的plugins中添加配置:

module.exports = {
  presets: [
    '@vue/cli-plugin-babel/preset'
  ],
  plugins: [
    '@vue/babel-plugin-jsx'
  ]
}

vue3的tsx文件基本结构

到这一步就配置好了,接下来看在组件中怎么使用:
tsx文件就相当一个ts文件,里面都是ts代码,不能像vue文件一样出现html和css,我们在tsx文件中返回一个组件

import { defineComponent } from 'vue'

export default defineComponent({
  setup() {
    return () => {
      return <div>
        hello tsx
      </div>
    }
  }
})

setup如果返回一个函数,那么这个函数就是render函数,然后在render函数中返回我们的组件模板也就是html,然后setup里面的其他东西与使用vue文件一模一样,所以tsx文件在用法上可以理解就是将template转移到render函数中。

2.基本语法对照vue文件写法

defineComponent与setup基本结构

上面说到tsx写法上就是把template转移到render函数中,defineComponent的其他配置,与setup其他参数与用法无区别:

import { defineComponent, ref, reactive } from 'vue'

export default defineComponent({
  props: {
  },
  setup(props) {
    const msg = ref('hello tsx')
    const state = reactive({
      count: 1
    })

    return () => {
      return <div>
        {msg.value} <span>{state.count}</span>
      </div>
    }
  }
})

render函数中的模版可以直接使用整个文件的变量,通过上面的代码可以看到,tsx使用变量是使用一个 {} ,只要在tsx想使用任何非字符串的代码,都需要用 {} 包裹,包括数字、布尔、函数表达式等

指令

bind:

vue文件:
<com :data="data"></com>

tsx文件:
<com data={data}></com>

这里表示传递给子组件的数据,顺便说一下引入组件写法的区别
vue文件, 需要注册,且在template中可以将驼峰换成中划线:

import TestCom from './test-com.vue'
export default defineComponent({
  components: {
    TestCom
  }
})

在template中:
<template>
  <test-com></test-com>
</template>

在tsx文件,不需要注册,且不能修改名称:

import TestCom from './test-com.vue'
export default defineComponent({
  setup(){
    return () => {
      return <TestCom></TestCom>
    }
  }
})

v-if

vue文件:
<div v-if="flag"></div>

tsx文件,js逻辑代码必须用大括号包裹:
{
  flag ? <div></div> : null
}

v-show

vue文件:
<div v-show="flag"></div>

tsx文件,插件已处理,可以直接使用:
<div v-show={flag}></div>

v-for

vue文件:
<ul>
  <li v-for="item in list" :key="item">{{item}}</li>
</ul>

tsx文件:
<ul>
  {
    list.map((item) => {
      return <li key={item}>{item}</li>
    })
  }
</ul>

v-model

v-model普通用法
vue文件:
<input v-model="keyword" />

tsx文件:
<input v-model={keyword} />
v-model传递参数

vue2.0可以用v-bind.sync来做组件的数据的双向绑定,vue3移除了这个语法,改用了v-model的写法,先来看看在vue文件中2.0和3.0的区别:

2.0
<ChildComponent :title.sync="pageTitle" />

然后在子组件里面使用:
this.$emit('update:title', newValue)
就可以更新父组件传递的值

3.0
<ChildComponent v-model="pageTitle" />
在子组件里面会接收到一个modelValue(默认名称)的变量
同样:
this.$emit('update:modelValue', newValue)
就可以更新父组件pageTitle的值

如果不想使用默认名称modelValue,就可以传递参数:
<ChildComponent v-model:pageTitle="pageTitle" />
子组件接收到的props就有一个pageTitle的变量

tsx文件写法:

<ChildComponent v-model={[pageTitle, 'pageTitle']} />
传递一个数组,第一项为传递的值,第二项为子组件接收的名称

在子组件里面想更新就:
emit('update:pageTitle', newValue)
这个vue文件tsx文件无区别
v-model修饰符
vue文件
<input v-model.trim="keyword" />

tsx文件
<input v-model={[keyword, ['trim']]} />
传递一个数组,第一项为传递的值,第二项为修饰器名称

vue3可以利用这个修饰符结合上面的传递参数实现一些功能,下面是官网链接:
处理v-model修饰符
一般是与子组件数据双向绑定时配合使用,具体功能看上面的官方文档,下面介绍一下在tsx中怎么使用:

vue文件:
<ChildComponent v-model.custom:pageTitle="pageTitle" />

tsx文件:
<ChildComponent v-model={[pageTitle, ['custom'], 'pageTitle']} />
传递一个数组,数组第一项为传递的数据,第二项也是一个数组,传入修饰符名称,第三项是子组件接收的名称

事件监听

基本对照

vue文件:
<div @click="handleClick"></div>

tsx文件:
<div onClick={handleClick}></div>

由v-on变成on+事件类型,首字母大写

传递参数

vue文件:
<div @click="handleClick(1,2)"></div>

tsx文件:
<div onClick={() => { handleClick(1,2) }}></div>

需要声明一个匿名函数,只能接收函数定义

监听自定义事件ts报错处理

在子组件中emit一个事件,父组件用v-on来接收,vue文件:

子组件:
emit('custom')

父组件:
<ChildComponent @custom="handleCustom" />

tsx文件:

子组件:
emit('custom')

父组件:
<ChildComponent onCustom={handleCustom} />

但是此时tsx会将 onCustom 当成一个prop传入,会报 “与子组件props类型不一致” 的错误
处理方法就是在子组件的props中定义emit的函数名称:

子组件:
props: {
  onCustom: {
    type: Function
  }
}

处理事件冒泡

vue文件:
<div @click.stop="handleClick"></div>

tsx中没有事件修饰符,只能通过原生写法来处理
<div onClick={handleClick}></div>

const handleClick = (e: MouseEvent) => {
  e.stopPropagation()
}

处理回车事件

vue文件:
<input @keyup.enter="search" />

tsx文件,通过监听键盘事件来实现:
<input onKeypress={search} />

const search = (e: any) => {
  if (e.keyCode === 13) {
    //
  }
}

样式相关

文件引入

tsx文件直接在文件里面引入样式文件

import './style.css'

但这样没有vue文件的scoped,容易造成样式冲突,如果项目是中小型的,可以通过将顶部类型写复杂来规避,通常为:模块名+文件名+组件名来命名顶部元素的claas,例如:

import { defineComponent } from 'vue'

export default defineComponent({
  setup() {
    return () => {
      return <div class="moudle-file-component-wrapper">
      </div>
    }
  }
})

目前博主是使用这种方式来处理,在迭代了一年的系统里面并没感觉不便

如果要保险规避,达到vue文件scoped的效果,可以参考博主之前的一篇文章:
vue3+ts jsx写法css module处理方案

动态class写法

vue文件:
<div class="box" :class="{active: count === 1}"></div>

tsx文件:
<div class={['box', count===1 ? 'active' : '']}></div>
class名称集合换成一个数组

顺便说下动态style的:

<div style={{width: count + 'px'}}></div>
style是一个对象,处理js代码要用大括号,所以有两层大括号

调用组件方法

ref 引用组件

vue文件:
<ChildComponent  ref="com" />

... js
setup() {
  const com = ref<any>(null)
  
  onMount(() => {
    console.log(com.value)
  })
  
  return {
    com
  }
}

tsx文件:
setup() {
  const com = ref<any>(null)
  onMount(() => {
    console.log(com.value)
  })

  return () => {
    return <ChildComponent ref={com} />
  }
}

注意引用的时候没有 .value,引用dom普通元素也是一样的写法

render配置写法暴露组件方法

上面已经引用了组件,这种场景无非就是调用子组件里面的方法,那要成功调用自组件就得暴露方法,在vue文件里面非常简单,就是在setup里面return就行:

setup(){
  return {
    fn1,
    fn2
  }
}
这样父组件用ref引用后就可以直接调用

那在tsx文件中怎么暴露呢,setup已经返回了一个render函数,里面返回我们的组件模版,处理方法就是将render函数和setup拆开,让setup可以正常返回方法。

那我们的模版改写在哪呢,其实setup返回一个函数就是render函数只是vue提供了一种便捷的方式,让我们可以在模版中快速使用setup中定义的变量,真正的render函数是和setup同级的,同属于 defineComponent 配置的一个属性,下面我分别写出setup返回render函数与单独编写render函数的写法:
setup返回函数写法:

import { defineComponent, ref, reactive } from 'vue'

export default defineComponent({
  props:{
    name: {
      type: String,
      default: '超人鸭'
    }
  },
  setup(props) {
    const msg = ref('hello tsx')
    const state = reactive({
      count: 1
    })
    
    const handleClick = () => {
      console.log('click')
    }

    return () => {
      return <div onClick={handleClick}>
        {msg.value} 
        <span>{state.count}</span>
        <span>{props.name}</span>
      </div>
    }
  }
})

单独编写render函数写法:

import { defineComponent, ref, reactive } from 'vue'

export default defineComponent({
  props:{
    name: {
      type: String,
      default: '超人鸭'
    }
  },
  setup(props) {
    const msg = ref('hello tsx')
    const state = reactive({
      count: 1
    })
    
    const handleClick = () => {
      console.log('click')
    }

    return {
      msg,
      state,
      handleClick
    }
  },
  render() {
    return <div onClick={this.handleClick}>
        {this.msg.value} 
        <span>{this.state.count}</span>
        <span>{this.name}</span>
    </div>
  }
})

setup中的变量要return,render中使用要使用this,props数据会和组件属性结合,所以直接使用this使用。
这样在setup中return后,父组件使用ref引用这个组件就可以调用setup中返回的方法,例如上面的handleClick

render写法使用ref引用组件

有这样一个场景,父组件有一个子组件,里面放着一个element-ui的table,此时父组件想要去触发element-ui的table的方法,比如清空筛选、清空排序等。
基于上面,我们在子组件里面要使用ref引用el-table组件,然后在setup里面暴露一个方法,所以要使用render函数写法。
在render函数中使用ref变量引用组件,写法会有点违背常规思路,这个问题是我使用tsx被坑得最厉害的问题

import { defineComponent, ref } from 'vue'

export default defineComponent({
  setup() {
    const elTableCom = ref<any>(null)

    const handle = () => {
      console.log('click')
    }

    return {
      handle,
      elTableCom
    }
  },
  render() {
    return <div>
       <el-table ref="elTableCom"></el-table>
    </div>
  }
})

不用 this , 不用大括号,直接字符串引用🧐🧐🧐

其他细节

占位标签

在vue文件里,template可以当作一个站位标签,不会渲染成什么,且vue3也不要求组件需要一个根标签。
但是tsx要求必须有一个根标签包裹,如果不想要这个根标签可以使用:

setup() {
  return () => {
    return <>
      <div></div>
      <div></div>
    </>
  }
}

递归组件

vue文件使用递归组件是通过name属性来引用自己:

<template>
  <test-com></test-com>
</template>

<script>
import {defineComponent} from 'vue'
export default defineComponent({
  name: 'TestCom'
})
</script>

注意使用v-if结束递归

tsx文件使用引用变量的方式

import {defineComponent} from 'vue'
const TestCom =  defineComponent({
  setup() {
    return () => {
      return <TestCom></TestCom>
    }
  }
})

export default TestCom

同样记得使用判断结束递归

3.插槽

插槽应该是tsx语法中最复杂的,所以单独提出来介绍

父组件中插入内容至子组件的插槽

先用vue文件写一个子组件,并且在这个组件中定义默认插槽、具名插槽、作用域插槽三种插槽:

<template>
  <div>
    <p>子组件</p>
    <!-- 这是默认插槽 -->
    <slot></slot>

    <!-- 这是具名插槽 -->
    <slot name="chaoren"></slot>

    <!-- 这是作用域插槽 -->
    <slot name="ya" :list="list"></slot>
  </div>
</template>

<script lang="ts">
import { defineComponent, reactive, toRefs } from 'vue'

export default defineComponent({
  setup() {
    const state = reactive({
      list: ['超人', '鸭']
    })

    return {
      ...toRefs(state)
    }
  }
})
</script>

vue文件中使用

父组件使用默认插槽:
<template>
  <div>
    <p>父组件</p>
    <children>
      <p>父组件插入内容至子组件:默认插槽</p>
    </children>
  </div>
</template>

<script lang="ts">
import { defineComponent, reactive, toRefs } from 'vue'
import children from './children.vue'

export default defineComponent({
  components: {
    children
  },
  setup() {
    const state = reactive({
    })

    return {
      ...toRefs(state)
    }
  }
})
</script>

任何没有使用具名插槽的元素都会被渲染至默认插槽中

效果:


image.png
父组件使用具名插槽:
<template>
  <div>
    <p>父组件</p>
    <children>
      <p>父组件插入内容至子组件:默认插槽</p>
      <template v-slot:chaoren>
        <p>父组件插入内容至子组件:具名插槽</p>
      </template>
    </children>
  </div>
</template>

效果:


image.png

其实默认插槽也有一个名字,叫做 default ,所以使用默认插槽也可以写成:

<template>
  <div>
    <p>父组件</p>
    <children>
      <template v-slot:default>
        <p>父组件插入内容至子组件:默认插槽</p>
      </template>
      <template v-slot:chaoren>
        <p>父组件插入内容至子组件:具名插槽</p>
      </template>
    </children>
  </div>
</template>
父组件作用域插槽:
<template>
  <div>
    <p>父组件</p>
    <children>
      <template v-slot:default>
        <p>父组件插入内容至子组件:默认插槽</p>
      </template>
      <template v-slot:chaoren>
        <p>父组件插入内容至子组件:具名插槽</p>
      </template>
      <template v-slot:ya="scope">
        <p>父组件插入内容至子组件:作用域插槽</p>
        <p v-for="item in scope.list" :key="item">{{item}}</p>
      </template>
    </children>
  </div>
</template>

效果:


image.png

作用域插槽其实就是一个具名插槽,然后传递数据给父组件,父组件可以用这些数据拿去做自定义渲染。
父组件在拿这个数据的时候,拿到的是包裹着传递数据的对象,因为子组件可以传递很多数据,也就是上面代码的:

<template v-slot:ya="scope">

这个 scope 就代表着包裹数据的对象,可以随便命名。

在tsx文件中使用

还是基于上面的children组件,同时包含默认插槽,具名插槽,作用域插槽,下面展示一下一起用这三种插槽的写法:

import { defineComponent } from 'vue'
import Children from './children.vue'

export default defineComponent({
  setup() {
    const childrenSlot = {
      default: () => {
        return <p>父组件插入内容至子组件:默认插槽</p>
      },
      chaoren: () => {
        return <p>父组件插入内容至子组件:具名插槽</p>
      },
      ya: (scope: any) => {
        return <>
          <p>父组件插入内容至子组件:作用域插槽</p>
          {
            scope.list.map((item: any) => {
              return <p key={item}>{item}</p>
            })
          }
        </>
      }
    }

    return () => {
      return <div>
        <p>父组件</p>
        <Children v-slots={childrenSlot}>
        </Children>
      </div>
    }
  }
})

tsx文件使用插槽是传入一个对象,每一个插槽都是一个方法,里面返回要渲染的dom,如果是作用域插槽,传递的数据会在方法的参数里面。效果:


image.png

tsx编写子组件定义插槽

上面的子组件使用vue文件写的,下面看一下在tsx文件中如果定义插槽,供父组件使用:

import { defineComponent, reactive } from 'vue'

export default defineComponent({
  setup(props, { slots }) {
    const state = reactive({
      list: ['超人', '鸭']
    })

    return () => {
      return <div>
        <p>子组件</p>
        {/* 这是默认插槽 */}
        {
          slots.default ? slots.default() : null
        }

        {/* 这是具名插槽 */}
        {
          slots.chaoren ? slots.chaoren() : null
        }

        {/* 这是作用域插槽 */}
        {
          slots.ya ? slots.ya({ list: state.list }) : null
        }
      </div>
    }
  }
})

setup的第二个参数是一个上下文对象,可能平时用的最多的就是emit,他其中还有 slot 这个参数,我觉得就是专门为jsx文件准备的,其实通过上面tsx文件父组件使用子组件插槽时就可以发现,vue在处理插槽时其实是传入一个函数,这个函数返回要渲染的dom,所以上面的子组件就是将外部传入的函数进行执行,渲染传进来的dom。

父组件引用:

import { defineComponent } from 'vue'
import Children from './children'

export default defineComponent({
  setup() {
    const childrenSlot = {
      default: () => {
        return <p>父组件插入内容至子组件:默认插槽</p>
      },
      chaoren: () => {
        return <p>父组件插入内容至子组件:具名插槽</p>
      },
      ya: (scope: any) => {
        return <>
          <p>父组件插入内容至子组件:作用域插槽</p>
          {
            scope.list.map((item: any) => {
              return <p key={item}>{item}</p>
            })
          }
        </>
      }
    }

    return () => {
      return <div>
        <p>父组件</p>
        <Children v-slots={childrenSlot}>
        </Children>
      </div>
    }
  }
})

效果:


image.png

一致。

4.结语

到这里就是博主介绍tsx用法的全部内容,当然后面如果发现有遗漏还会再来补充,欢迎大家指教。
博主微信:Promise_fulfilled

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容