Babel plugin for Vue 2.0 JSX
Requirements
-
Assumes you are using Babel with a module bundler e.g. Webpack, because the spread merge helper is imported as a module to avoid duplication.
-
This is mutually exclusive with
babel-plugin-transform-react-jsx
.
Usage
npm install\babel-plugin-syntax-jsx\babel-plugin-transform-vue-jsx\babel-helper-vue-jsx-merge-props\babel-preset-env\--save-dev
In your .babelrc
:
The plugin transpiles the following JSX:
<div ="foo">thistext</div>
To the following JavaScript:
Note the h
function, which is a shorthand for a Vue instance's $createElement
method, must be in the scope where the JSX is. Since this method is passed to component render functions as the first argument, in most cases you'd do this:
Vue
h
auto-injection
Starting with version 3.4.0 we automatically inject const h = this.$createElement
in any method and getter (not functions or arrow functions) declared in ES2015 syntax that has JSX so you can drop the (h)
parameter.
Vue@Component{ // h will be injectedreturn <div id="foo">bar</div>}
Difference from React JSX
First, Vue 2.0's vnode format is different from React's. The second argument to the createElement
call is a "data object" that accepts nested objects. Each nested object will be then processed by corresponding modules:
{return}
The equivalent of the above in Vue 2.0 JSX is:
{return<div// .="foo"// ``="bar"// `` ``==//==="key"="ref"// `` /="slot"></div>}
Component Tip
If a custom element starts with lowercase, it will be treated as a string id and used to lookup a registered component. If it starts with uppercase, it will be treated as an identifier, which allows you to do:
{return <Todo/> // no need to register Todo via components option}
JSX Spread
JSX spread is supported, and this plugin will intelligently merge nested data properties. For example:
const data =class: 'b' 'c'const vnode = <div ="a" />
The merged data will be:
class: 'a' 'b' 'c'
Vue directives
Note that almost all built-in Vue directives are not supported when using JSX, the sole exception being v-show
, which can be used with the v-show={value}
syntax. In most cases there are obvious programmatic equivalents, for example v-if
is just a ternary expression, and v-for
is just an array.map()
expression, etc.
For custom directives, you can use the v-name={value}
syntax. However, note that directive arguments and modifiers are not supported using this syntax. There are two workarounds:
-
Pass everything as an object via
value
, e.g.v-name={{ value, modifier: true }}
-
Use the raw vnode directive data format:
const directives =name: 'my-dir' value: 123 modifiers: abc: truereturn <div ... directives />