仲灏小栈 仲灏小栈
首页
大前端
后端&运维
其他技术
生活
关于我
收藏
  • 分类
  • 标签
  • 归档
GitHub (opens new window)

仲灏

诚意, 正心, 格物, 致知
首页
大前端
后端&运维
其他技术
生活
关于我
收藏
  • 分类
  • 标签
  • 归档
GitHub (opens new window)
  • 《前端项目基础建设》
  • HTML&CSS

  • JavaScript&TypeScript

  • Node

  • 构建

  • Vue

    • 持久化存储实时读取
    • vue自适应布局
    • Vue 中后台表格的增删改查统一解决方案(mixin版)
    • Vue 中后台表格的增删改查同一解决方案(组件封装版)
    • Vue Render自定义table单元格内容
    • 项目初始化编码规范(eslint,prettier等)
    • vue element-ui 换肤功能开发
    • vue element 定义多种主题实现换肤
    • 大屏自适应容器
    • el-upload 自定义上传进度
    • vue3初学注意点
    • vue3 hooks 自适应可视化大屏
    • vue3 element-plus ant- design 自定义主题
    • 博客开发
    • vue2中使用jsx
    • Vue 放弃繁琐的render函数,拥抱JSX
      • 背景
      • 安装配置
        • babel7 (vue cli 默认是使用了babel7的)
        • 安装babel插件
        • babel 6
      • 使用
        • slot
        • Table-column Scoped Slot
    • 组件设计步骤
    • 倒计时验证
    • Vue中computed和watch的区别
    • vue3面试聚焦
    • Vue3 原理 Proxy实现响应式
  • React

  • 小程序

  • 跨端

  • Electron

  • WebGL&GIS

  • 浏览器

  • 面经

  • 其他

  • 大前端
  • Vue
仲灏
2023-09-17
目录

Vue 放弃繁琐的render函数,拥抱JSX

# 背景

如果你写了很多 render 函数,可能会觉得下面这样的代码写起来很痛苦:

createElement(
  'anchored-heading', {
    props: {
      level: 1
    }
  }, [
    createElement('span', 'Hello'),
    ' world!'
  ]
)
[
	{ label: '最近更新时间', prop: 'update_time', align: 'center', visible: false },
  {
    label: '操作', align: 'center', 'min-width': '180px', render: (h, { row }) => {
      return h('div',
        [h('el-popconfirm', {
          props: { title: '确定删除吗?', 'confirm-button-type': 'danger' },
          class: 'mr-10',
          on: {
            confirm: () => { this.delItem(row) }
          }
        },
        [h('el-button', { slot: 'reference', props: { size: 'mini', type: 'danger' }}, '删除')]),
        h('el-button', { props: { size: 'mini', type: 'primary' }, nativeOn: { click: () => { this.editItem(row) } }}, '编辑')]
      )
    }
  }
]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28

特别是对应的模板如此简单的情况下:

<anchored-heading :level='1'>
  <span>Hello</span> world!
</anchored-heading>
<el-popconfirm
	title='确定删除吗?'
	'confirm-button-type'='danger'  
  onConfirm='() => this.delItem(row)'
>
   <el-button slot='reference' size='mini' type='danger'>删除</el-button>
</el-popconfirm>

<el-button size='mini' type='danger' onClick={() => this.editItem(row)}>编辑</el-button>
1
2
3
4
5
6
7
8
9
10
11
12

# 安装配置

# babel7 (vue cli 默认是使用了babel7的)

# 安装babel插件

npm install @vue/babel-preset-jsx @vue/babel-helper-vue-jsx-merge-props
1
  • 配置babel.config.js
module.exports = {
  presets: ['@vue/babel-preset-jsx'],
}
1
2
3

如果你出现这个错误: https://github.com/vuejs/jsx/issues/34

Duplicate declaration "h" (This is an error on an internal node. Probably an internal error.)
1

试试这个(需要说明H函数传递)

module.exports = {
  presets: [
    [
      '@vue/babel-preset-jsx',
      {
        injectH: false
      }
    ]
  ]
}
1
2
3
4
5
6
7
8
9
10

# babel 6

npm install babel-plugin-syntax-jsx babel-plugin-transform-vue-jsx babel-helper-vue-jsx-merge-props babel-preset-env --save-dev
1
  • 配置.babelrc
{
  "presets": ["env"],
  "plugins": ["transform-vue-jsx"]
}
1
2
3
4

# 使用

我这里举例比较复杂的例子 也是我遇到的

# slot

正常的code是这样的, element-ui举例

<el-table-column
  label="姓名"
  width="180">
  <template slot-scope="scope">
    <el-popover trigger="hover" placement="top">
      <p>姓名: {{ scope.row.name }}</p>
      <p>住址: {{ scope.row.address }}</p>
      <div slot="reference" class="name-wrapper">
        <el-tag size="medium">{{ scope.row.name }}</el-tag>
      </div>
    </el-popover>
  </template>
</el-table-column>
1
2
3
4
5
6
7
8
9
10
11
12
13

# Table-column Scoped Slot

name 说明
— 自定义列的内容,参数为 { row, column, $index }
header 自定义表头的内容. 参数为 { column, $index }

🍖转换成jsx就是

<el-table-column
  label='姓名'
  width='180' {
    ...{
      scopedSlots: {
        default: scope => {
          return <el-popover trigger='hover' placement='top'>
            <p>姓名: { scope.row.name }</p>
            <p>住址: { scope.row.address }</p>
            <div slot='reference' class='name-wrapper'>
              <el-tag size='medium'>{ scope.row.name }</el-tag>
            </div>
          </el-popover>
        }
      }
    }
  }>
</el-table-column>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
上次更新: 2023/09/17, 20:06:58
vue2中使用jsx
组件设计步骤

← vue2中使用jsx 组件设计步骤→

最近更新
01
vim日常使用记录
04-02
02
滑动窗口最大值
04-02
03
有效的字母异位词
04-02
更多文章>
Theme by Vdoing | Copyright © 2021-2025 izhaong | github | 蜀ICP备2021031194号
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式