Render Functions & JSX
Vue recommends using templates to build applications in the vast majority of cases. However, there are situations where we need the full programmatic power of JavaScript. That's where we can use the render function .
If you are new to the concept of virtual DOM and render functions, make sure to read the Rendering Mechanism chapter first.
Basic Usage
Creating Vnodes
Vue provides an
h()
function for creating vnodes:
h()
is short for
hyperscript
- which means "JavaScript that produces HTML (hypertext markup language)". This name is inherited from conventions shared by many virtual DOM implementations. A more descriptive name could be
createVnode()
, but a shorter name helps when you have to call this function many times in a render function.
The
h()
function is designed to be very flexible:
The resulting vnode has the following shape:
Note
The full
VNode
interface contains many other internal properties, but it is strongly recommended to avoid relying on any properties other than the ones listed here. This avoids unintended breakage in case the internal properties are changed.
Declaring Render Functions
When using templates with Composition API, the return value of the
setup()
hook is used to expose data to the template. When using render functions, however, we can directly return the render function instead:
The render function is declared inside
setup()
so it naturally has access to the props and any reactive state declared in the same scope.
In addition to returning a single vnode, you can also return strings or arrays:
TIP
Make sure to return a function instead of directly returning values! The
setup()
function is called only once per component, while the returned render function will be called multiple times.
If a render function component doesn't need any instance state, they can also be declared directly as a function for brevity:
That's right, this is a valid Vue component! See Functional Components for more details on this syntax.
Vnodes Must Be Unique
All vnodes in the component tree must be unique. That means the following render function is invalid:
If you really want to duplicate the same element/component many times, you can do so with a factory function. For example, the following render function is a perfectly valid way of rendering 20 identical paragraphs:
JSX / TSX
JSX is an XML-like extension to JavaScript that allows us to write code like this:
Inside JSX expressions, use curly braces to embed dynamic values:
create-vue
and Vue CLI both have options for scaffolding projects with pre-configured JSX support. If you are configuring JSX manually, please refer to the documentation of
@vue/babel-plugin-jsx
for details.
Although first introduced by React, JSX actually has no defined runtime semantics and can be compiled into various different outputs. If you have worked with JSX before, do note that Vue JSX transform is different from React's JSX transform , so you can't use React's JSX transform in Vue applications. Some notable differences from React JSX include:
-
You can use HTML attributes such as
class
andfor
as props - no need to useclassName
orhtmlFor
. - Passing children to components (i.e. slots) works differently .
Vue's type definition also provides type inference for TSX usage. When using TSX, make sure to specify
"jsx": "preserve"
in
tsconfig.json
so that TypeScript leaves the JSX syntax intact for Vue JSX transform to process.
JSX Type Inference
Similar to the transform, Vue's JSX also needs different type definitions. Currently, Vue's types automatically registers Vue's JSX types globally. This means TSX will work out of the box when Vue's type is available.
The global JSX types may cause conflict with used together with other libraries that also needs JSX type inference, in particular React. Starting in 3.3, Vue supports specifying JSX namespace via TypeScript's jsxImportSource option. We plan to remove the default global JSX namespace registration in 3.4.
For TSX users, it is suggested to set
jsxImportSource
to
'vue'
in
tsconfig.json
after upgrading to 3.3, or opt-in per file with
/* @jsxImportSource vue */
. This will allow you to opt-in to the new behavior now and upgrade seamlessly when 3.4 releases.
If there is code that depends on the presence of the global
JSX
namespace, you can retain the exact pre-3.4 global behavior by explicitly referencing
vue/jsx
, which registers the global
JSX
namespace.
Render Function Recipes
Below we will provide some common recipes for implementing template features as their equivalent render functions / JSX.
v-if
Template:
Equivalent render function / JSX:
v-for
Template:
Equivalent render function / JSX:
v-on
Props with names that start with
on
followed by an uppercase letter are treated as event listeners. For example,
onClick
is the equivalent of
@click
in templates.
Event Modifiers
For the
.passive
,
.capture
, and
.once
event modifiers, they can be concatenated after the event name using camelCase.
For example:
For other event and key modifiers, the
withModifiers
helper can be used:
Components
To create a vnode for a component, the first argument passed to
h()
should be the component definition. This means when using render functions, it is unnecessary to register components - you can just use the imported components directly:
As we can see,
h
can work with components imported from any file format as long as it's a valid Vue component.
Dynamic components are straightforward with render functions:
If a component is registered by name and cannot be imported directly (for example, globally registered by a library), it can be programmatically resolved by using the
resolveComponent()
helper.
Rendering Slots
In render functions, slots can be accessed from the
setup()
context. Each slot on the
slots
object is a
function that returns an array of vnodes
:
JSX equivalent:
Passing Slots
Passing children to components works a bit differently from passing children to elements. Instead of an array, we need to pass either a slot function, or an object of slot functions. Slot functions can return anything a normal render function can return - which will always be normalized to arrays of vnodes when accessed in the child component.
JSX equivalent:
Passing slots as functions allows them to be invoked lazily by the child component. This leads to the slot's dependencies being tracked by the child instead of the parent, which results in more accurate and efficient updates.
Built-in Components
Built-in components
such as
<KeepAlive>
,
<Transition>
,
<TransitionGroup>
,
<Teleport>
and
<Suspense>
must be imported for use in render functions:
v-model
The
v-model
directive is expanded to
modelValue
and
onUpdate:modelValue
props during template compilation—we will have to provide these props ourselves:
Custom Directives
Custom directives can be applied to a vnode using
withDirectives
:
If the directive is registered by name and cannot be imported directly, it can be resolved using the
resolveDirective
helper.
Template Refs
With the Composition API, template refs are created by passing the
ref()
itself as a prop to the vnode:
Functional Components
Functional components are an alternative form of component that don't have any state of their own. They act like pure functions: props in, vnodes out. They are rendered without creating a component instance (i.e. no
this
), and without the usual component lifecycle hooks.
To create a functional component we use a plain function, rather than an options object. The function is effectively the
render
function for the component.
The signature of a functional component is the same as the
setup()
hook:
Most of the usual configuration options for components are not available for functional components. However, it is possible to define
props
and
emits
by adding them as properties:
If the
props
option is not specified, then the
props
object passed to the function will contain all attributes, the same as
attrs
. The prop names will not be normalized to camelCase unless the
props
option is specified.
For functional components with explicit
props
,
attribute fallthrough
works much the same as with normal components. However, for functional components that don't explicitly specify their
props
, only the
class
,
style
, and
onXxx
event listeners will be inherited from the
attrs
by default. In either case,
inheritAttrs
can be set to
false
to disable attribute inheritance:
Functional components can be registered and consumed just like normal components. If you pass a function as the first argument to
h()
, it will be treated as a functional component.
Typing Functional Components
Functional Components can be typed based on whether they are named or anonymous. Volar also supports type checking properly typed functional components when consuming them in SFC templates.
Named Functional Component
Anonymous Functional Component