用Jsx写Vue组件

前言

我们平常写vue的组件时,一般都是用的是模版,这种方式看起来比较简洁,而且vue作者也推荐使用这个方式,但是这种方式也有一些它的弊端,例如模版调试麻烦,或者在一些场景下模版描述可能没那么简单和方便。

下面我们要讲的是如何在vue里面写jsx,知道react的人应该都知道jsx,jsx的一个特性就是非常灵活,虽然有的人觉得jsx很丑陋,把逻辑都写到模版的感觉,但萝卜青菜各有所爱,适合自己适合团队的就是最好的。

在使用jsx之前我们需要安装一个babel插件(babel-plugin-transform-vue-jsx )

安装方式:


  1. npm install\ 
  2.  
  3.   babel-plugin-syntax-jsx\ 
  4.  
  5.   babel-plugin-transform-vue-jsx\ 
  6.  
  7.   babel-helper-vue-jsx-merge-props\ 
  8.  
  9.   babel-preset-es2015\ 
  10.  
  11.   --save-dev  

然后再.babelrc里面添加:


  1.  
  2. "presets": ["es2015"], 
  3.  
  4. "plugins": ["transform-vue-jsx"] 
  5.  
  6. }  

接着我们就可以愉快地在vue里面编写jsx了。

Test.vue


  1. <script> 
  2.  
  3. export default { 
  4.  
  5.     props: ['onClick', 'isShow'], 
  6.  
  7.     data() { 
  8.  
  9.         return { 
  10.  
  11.             test: 123 
  12.  
  13.         }; 
  14.  
  15.     }, 
  16.  
  17.     render() { 
  18.  
  19.         return ( 
  20.  
  21.             <div class="test" onClick={ this.onClick }> 
  22.  
  23.                 { this.test } 
  24.  
  25.                 { this.isShow + '' } 
  26.  
  27.             </div> 
  28.  
  29.         ); 
  30.  
  31.     } 
  32.  
  33.  
  34. </script>  

可以看到我们把jsx写在了render方法里面,render方法是vue2.0才支持的,用来提供对虚拟DOM的支持,也就是说只有vue2.0才支持jsx语法转换。

这里要注意的一点是vue里面编写jsx和在react里面的jsx语法还是有一点不一样的。

一下是一段覆盖大部分语法的vue jsx代码:


  1. render (h) { 
  2.  
  3.   return ( 
  4.  
  5.     <div 
  6.  
  7.       // normal attributes or component props. 
  8.  
  9.       id="foo" 
  10.  
  11.       // DOM properties are prefixed with `domProps` 
  12.  
  13.       domPropsInnerHTML="bar" 
  14.  
  15.       // event listeners are prefixed with `on` or `nativeOn` 
  16.  
  17.       onClick={this.clickHandler} 
  18.  
  19.       nativeOnClick={this.nativeClickHandler} 
  20.  
  21.       // other special top-level properties 
  22.  
  23.       class={{ foo: true, bar: false }} 
  24.  
  25.       style={{ color: 'red', fontSize: '14px' }} 
  26.  
  27.       key="key" 
  28.  
  29.       ref="ref" 
  30.  
  31.       // assign the `ref` is used on elements/components with v-for 
  32.  
  33.       refInFor 
  34.  
  35.       slot="slot"> 
  36.  
  37.     </div> 
  38.  
  39.   ) 
  40.  
  41. }  

可以看到DOM属性要加domProps前缀,但这里lass和style却不需要,因为这两个是特殊的模块,而且react的class用的是className,vue却用的class。事件监听是以“on”或者“nativeOn”为开始。

实际上vue2.0的模版最后都会被编译为render方法,所以模版声明的组件和jsx声明的组件最后都是一样的。

上面的jsx最后会被编译成下面这样:


  1. render (h) { 
  2.  
  3.   return h('div', { 
  4.  
  5.     // Component props 
  6.  
  7.     props: { 
  8.  
  9.       msg: 'hi' 
  10.  
  11.     }, 
  12.  
  13.     // normal HTML attributes 
  14.  
  15.     attrs: { 
  16.  
  17.       id: 'foo' 
  18.  
  19.     }, 
  20.  
  21.     // DOM props 
  22.  
  23.     domProps: { 
  24.  
  25.       innerHTML: 'bar' 
  26.  
  27.     }, 
  28.  
  29.     // Event handlers are nested under "on", though 
  30.  
  31.     // modifiers such as in v-on:keyup.enter are not 
  32.  
  33.     // supported. You'll have to manually check the 
  34.  
  35.     // keyCode in the handler instead. 
  36.  
  37.     on: { 
  38.  
  39.       click: this.clickHandler 
  40.  
  41.     }, 
  42.  
  43.     // For components only. Allows you to listen to 
  44.  
  45.     // native events, rather than events emitted from 
  46.  
  47.     // the component using vm.$emit. 
  48.  
  49.     nativeOn: { 
  50.  
  51.       click: this.nativeClickHandler 
  52.  
  53.     }, 
  54.  
  55.     // class is a special module, same API as `v-bind:class` 
  56.  
  57.     class: { 
  58.  
  59.       foo: true, 
  60.  
  61.       bar: false 
  62.  
  63.     }, 
  64.  
  65.     // style is also same as `v-bind:style` 
  66.  
  67.     style: { 
  68.  
  69.       color: 'red', 
  70.  
  71.       fontSize: '14px' 
  72.  
  73.     }, 
  74.  
  75.     // other special top-level properties 
  76.  
  77.     key: 'key', 
  78.  
  79.     ref: 'ref', 
  80.  
  81.     // assign the `ref` is used on elements/components with v-for 
  82.  
  83.     refInFor: true, 
  84.  
  85.     slot: 'slot' 
  86.  
  87.   }) 
  88.  
  89. }  

这也意味着两种形式的组件是可以相互引用的。

有时候我们难免会在模版里引入jsx编写的vue组件或者在jsx编写的vue组件里引入模版组件,这里还是有些需要注意的事项:

1.在模版里面引入jsx的组件,可以通过components引用,另外props的编写从驼峰式改为连接符:


  1. <template> 
  2.  
  3.   <div class="wrapper"> 
  4.  
  5.     <Test :on-click="clickHandler" :is-show="show"></Test> 
  6.  
  7.   </div> 
  8.  
  9. </template> 
  10.  
  11. <script> 
  12.  
  13. import Test from './Test.vue'; 
  14.  
  15. export default { 
  16.  
  17.   name: 'hello', 
  18.  
  19.   components: { 
  20.  
  21.     Test 
  22.  
  23.   }, 
  24.  
  25.   data() { 
  26.  
  27.     return { 
  28.  
  29.       msg: 'Welcome to Your Vue.js App', 
  30.  
  31.       show: true 
  32.  
  33.     }; 
  34.  
  35.   }, 
  36.  
  37.   methods: { 
  38.  
  39.     clickHandler(){ 
  40.  
  41.       this.show = !this.show; 
  42.  
  43.     } 
  44.  
  45.   } 
  46.  
  47. }; 
  48.  
  49. </script>  

2.在jsx里面引入vue模版组件,这里没有什么要注意的,除了连接符式的属性要转换成驼峰式,还有一个需要注意的是指令,如果用了jsx,那么内置的指令都不会生效(除了v-show),好在内置指令大部分都可以用jsx描述。那么自定义指令要怎么用呢?

自定义指令可以使用“v-name={value}”语法,如果要支持指令参数和modifier可以用“v-name={{ value, modifier: true }}”语法:


  1. <script> 
  2.  
  3. import Vue from 'vue'; 
  4.  
  5. Vue.directive('my-bold', { 
  6.  
  7.   inserted: function (el) { 
  8.  
  9.     el.style.fontWeight = 900; 
  10.  
  11.   } 
  12.  
  13. }) 
  14.  
  15. export default { 
  16.  
  17.     props: ['onClick', 'isShow'], 
  18.  
  19.     data() { 
  20.  
  21.         return { 
  22.  
  23.             test: 123 
  24.  
  25.         }; 
  26.  
  27.     }, 
  28.  
  29.     methods: { 
  30.  
  31.         afterLeave() { 
  32.  
  33.             console.log('afterLeave') 
  34.  
  35.         } 
  36.  
  37.     }, 
  38.  
  39.     render() { 
  40.  
  41.         const directives = [ 
  42.  
  43.             { name: 'my-bold', value: 666, modifiers: { abc: true } } 
  44.  
  45.         ]; 
  46.  
  47.         return ( 
  48.  
  49.             <transition onAfterLeave={this.afterLeave} name="fade"> 
  50.  
  51.                 <div class="test" onClick={this.onClick} v-show={ this.isShow } v-my-bold> 
  52.  
  53.                     {this.test} 
  54.  
  55.                     {this.isShow + ''} 
  56.  
  57.                 </div> 
  58.  
  59.             </transition> 
  60.  
  61.         ); 
  62.  
  63.     } 
  64.  
  65.  
  66. </script> 
  67.  
  68. <style> 
  69.  
  70. .fade-enter-active, .fade-leave-active { 
  71.  
  72.   transition: opacity .5s 
  73.  
  74.  
  75. .fade-enter, .fade-leave-to { 
  76.  
  77.   opacity: 0 
  78.  
  79.  
  80. </style>  

我们还可以用原生vnode的数据格式使用自定义指令:


  1. const directives = [ 
  2.  
  3. { name: 'my-dir', value: 123, modifiers: { abc: true } } 
  4.  
  5.  
  6. return <div {...{ directives }}/>  

扩展

如果有人觉得在vue组件里面要写data,props,computed和methods不够优雅,可以参考下这个插件vue-class-component,它能让你使用ES6的class和ES7的装饰器编写vue组件。

作者:佚名

来源:51CTO

时间: 2024-11-22 22:31:21

用Jsx写Vue组件的相关文章

React JSX vs Vue 模板:前端界的一次对决

本文讲的是React JSX vs Vue 模板:前端界的一次对决, React.js 与 Vue.js 是地球上最受欢迎的两个 JavaScript 库.他们都非常强大并且相对容易上手和使用. React 和 Vue 的共同点: 使用虚拟 DOM 提供响应式的视图组件 保持对视图的关注 伴随着如此多的相同点,你可能会怀疑它们就是同一个库的不同版本. 不过这两个库有一个主要的区别:就是他们如何授权给你(开发者)创建你自己的视图组件,甚至是你的应用. React 使用 JSX(由 React 小组

Vue组件BootPage实现简单的分页功能_javascript技巧

有一段时间没更新文章了,主要是因为自己一直在忙着学习新的东西而忘记分享了,实在惭愧.  这不,大半夜发文更一篇文章,分享一个自己编写的一个Vue的小组件,名叫BootPage.  不了解Vue.js的童鞋可以移步我的上一篇文章<浅谈Vue.js>了解一下.  BootPage组件简介  其实也不是啥高大上的组件了,相反确实一个简单的表格分页组件而已,主要是自己最近项目中需要一个表格分页组件,而Vue官方组件库里分页组件都功能太强大或者没有适合我的,所以就自己写了一个凑合着用,或许有人和我一样需

require.js 加载 vue组件 r.js 合并压缩的实例_javascript技巧

准备: vue.js 原本是学习vue组件 require.js 然后想到用require 加载 r.js 文件太多 合并 文件目录 忽略部分文件及文件夹 一.先说vue 组件 先引入vue 再引入vue组件 Vue.extend({}) 定义组件 template data methods Vue.component(),注册组件的标签,标签在html中是一个挂载点 new Vue() 进行实例化 index.html // css引入 略 <div id="header"&g

如何用VB6写COM组件(转)

所幸的是,我们可以用COM组件来在最大程度上扩展ASP的功能.你可知道,任何可以用诸如VB,VC和VJ以及其它的一些程序语言所写出来的东西,都可以被应用到你的网站的开发中去,这就是COM(组件对象模型).COM能够用来编写能够被ASP所调用的对象.当在VB等编程环境中完成了组件的编制后,就可以放在ASP页面中被ASP所调用了.在这篇文章中,我们将看到如何使用VB6来编写一个COM对象,如何将其注册在服务器上,并在ASP页面中对其进行调用.但在开始之前,让我先来简单的介绍一写关于COM的小知识.

c# com组件-求助c#写com组件有毒 看了无数文章也无法公开一个类中的函数

问题描述 求助c#写com组件有毒 看了无数文章也无法公开一个类中的函数 看了无数文章和博客尝试了无数方法依然无法解决这个问题 编译的com组件永远只有 _object 下的的几个函数自定义的函数一个也没有 下面是看着无数文章写的代码 using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; using System.Run

06Vue.js快速入门-Vue组件化开发

组件其实就是一个拥有样式.动画.js逻辑.HTML结构的综合块.前端组件化确实让大的前端团队更高效的开发前端项目.而作为前端比较流行的框架之一,Vue的组件和也做的非常彻底,而且有自己的特色.尤其是她单文件组件开发的方式更是非常方便,而且第三方工具支持也非常丰富,社区也非常活跃,第三方组件也呈井喷之势.当然学习和使用Vue的组件也是我们的最重要的目标. 6.1. 全局扩展方法Vue.extend Vue提供了一个全局的API,Vue.extend可以帮助我们对Vue实例进行扩展,扩展完了之后,就

LARAVEL 中使用 VUE 组件化开发例子

现今前端组件化开发.MVVM 模式,给开发带来了很多的便利,可读性.可维护性更高.然而自 Laravel 5.3 开始,VueJS 成为框架默认的标配. 本文基于 Laravel 5.1 LTS 版本引入 Vue 2.0 进行配置. 我已在 Github 配置好,Clone 下来后按照 README 安装依赖后即可用: https://github.com/jcc/vue-laravel-example 步骤一:配置 PACKAGE.JSON 1. 在根目录下修改 package.json, 可

Vue插件写、用详解(附demo)

Vue插件 1.概述 简单来说,插件就是指对Vue的功能的增强或补充. 比如说,让你在每个单页面的组件里,都可以调用某个方法,或者共享使用某个变量,或者在某个方法之前执行一段代码等 2.使用方法 总体流程应该是: [声明插件]--[写插件]--[注册插件]--[使用插件] 写插件和声明插件是同步的,然后注册到Vue对象中(不用担心重复注册),最后在写Vue组件的时候使用写的插件 声明插件 先写一个js文件,这个js文件就是插件文件,里面的基本内容如下: /* 说明: * 插件文件:service

require.js+vue开发微信上传图片组件_javascript技巧

由于项目是thinkPHP做后端框架,一直以来都是多页面的后端路由,想使用火热的webpack有点无从下手(原谅我太菜,而且推广vue只有我一个人--),没办法,想把vue用起来,唯有在原来的基础上改进.使用webpack的巨大好处就是可以使用 .vue 这样的单文件来写vue组件,这样每一个组件就是一个 .vue 文件,哪里用上这个组件就引入进来,维护起来确实很爽.然而一直以来项目用的都是require.js,那又想以这样的形式来组织vue组件,还要加上vue-router和vue-resou