- Start Date: 2020-04-20
- Target Major Version: Vue Router 4.x
- Reference Issues:
- Implementation PR:
Given the changes to functional components in Vue 3, KeepAlive
and Transition
usage combined with RouterView
is no longer possible by simply wrapping RouterView
with KeepAlive
/Transition
. Instead we need a way to directly provide the component rendered by RouterView
to those components:
<router-view v-slot="{ Component }">
<transition :name="transitionName" mode="out-in">
<component :is="Component"></component>
</transition>
</router-view>
As described at vuejs/core#906 (comment), we need a new API to allow the usage of KeepAlive
and other components accepting slots with RouterView
. In order to implement such behavior we need to be able to directly wrap the component rendered by RouterView
. The only way to do so by having access to the component rendered by RouterView
and the props passed to it.
We can achieve this level of control through a slot:
<router-view v-slot="{ Component, route }">
<component :is="Component" v-bind="route.params"></component>
</router-view>
In the example, Component
is the component definition that can be passed to the function h
or to the is
prop of component
.
When defining a props: true
option in the route definition:
createRouter({
routes: [{ path: '/users/:id', component: User, props: true }],
})
The router-view
will automatically add the id
prop to the rendered component with the value of the param named id
. Note you can also do v-bind="route.params"
like shown in the example above instead of using props: true
.
When the current location isn't matched by any record registered by the Router, the matched
array of a RouteLocation
is empty and, by default, when no slot is provided, it renders nothing. When we provide a slot, we can decide of what to display, whether we want to display a not found page or we want the default behavior, we are able to do it. Component
will be falsy if there is no component to render:
<router-view v-slot="{ Component, route }">
<component v-if="route.matched.length > 0" :is="Component"/>
<div v-else>Not Found</div>
</router-view>
Note that this behavior is redundant with the catch all route (path: '/:pathMatch(.*)
) when it comes to displaying a not found page.
Component
: Component that can be passed to the functionh
or to theis
prop ofcomponent
.route
: Normalized Route location rendered byRouterView
. Same as$route
but allows easy and typed access in JSX.
If the user accidentally wraps RouterView
with Transition
or is migrating their application to Vue 3, we could issue a warning pointing to the documentation (this RFC in the meantime) and hinting them to use the v-slot
api.
When using KeepAlive
and Transition
at the same time, we need to do Transition
then KeepAlive
:
<router-view v-slot="{ Component }">
<transition name="fade" mode="out-in">
<keep-alive>
<component :is="Component"></component>
</keep-alive>
</transition>
</router-view>
- Using a function like
useView
that would returnComponent
andattrs
removing the use ofv-slot
.
- A codemod could rewrite v3 to v4.