Dynamic Routing
Adding routes to your router is usually done via the routes
option but in some situations, you might want to add or remove routes while the application is already running. Application with extensible interfaces like Vue CLI UI can use this to make the application grow.
Adding Routes
Dynamic routing is achieved mainly via two functions: router.addRoute()
and router.removeRoute()
. They only register a new route, meaning that if the newly added route matches the current location, it would require you to manually navigate with router.push()
or router.replace()
to display that new route. Let's take a look at an example:
Imagine having the following router with one single route:
const router = createRouter({
history: createWebHistory(),
routes: [{ path: '/:articleName', component: Article }],
})
const router = createRouter({
history: createWebHistory(),
routes: [{ path: '/:articleName', component: Article }],
})
Going to any page, /about
, /store
, or /3-tricks-to-improve-your-routing-code
ends up rendering the Article
component. If we are on /about
and we add a new route:
router.addRoute({ path: '/about', component: About })
router.addRoute({ path: '/about', component: About })
The page will still show the Article
component, we need to manually call router.replace()
to change the current location and overwrite where we were (instead of pushing a new entry, ending up in the same location twice in our history):
router.addRoute({ path: '/about', component: About })
// we could also use this.$route or route = useRoute() (inside a setup)
router.replace(router.currentRoute.value.fullPath)
router.addRoute({ path: '/about', component: About })
// we could also use this.$route or route = useRoute() (inside a setup)
router.replace(router.currentRoute.value.fullPath)
Remember you can await router.replace()
if you need to wait for the new route to be displayed.
Adding Routes inside navigation guards
If you decide to add or remove routes inside of a navigation guard, you should not call router.replace()
but trigger a redirection by returning the new location:
router.beforeEach(to => {
if (!hasNecessaryRoute(to)) {
router.addRoute(generateRoute(to))
// trigger a redirection
return to.fullPath
}
})
router.beforeEach(to => {
if (!hasNecessaryRoute(to)) {
router.addRoute(generateRoute(to))
// trigger a redirection
return to.fullPath
}
})
The example above assumes two things: first, the newly added route record will match the to
location, effectively resulting in a different location from the one we were trying to access. Second, hasNecessaryRoute()
returns false
after adding the new route to avoid an infinite redirection.
Because we are redirecting, we are replacing the ongoing navigation, effectively behaving like the example shown before. In real world scenarios, adding is more likely to happen outside of navigation guards, e.g. when a view component mounts, it register new routes.
Removing routes
There are few different ways to remove existing routes:
By adding a route with a conflicting name. If you add a route that has the same name as an existing route, it will remove the route first and then add the route:
jsrouter.addRoute({ path: '/about', name: 'about', component: About }) // this will remove the previously added route because they have // the same name and names are unique across all routes router.addRoute({ path: '/other', name: 'about', component: Other })
router.addRoute({ path: '/about', name: 'about', component: About }) // this will remove the previously added route because they have // the same name and names are unique across all routes router.addRoute({ path: '/other', name: 'about', component: Other })
By calling the callback returned by
router.addRoute()
:jsconst removeRoute = router.addRoute(routeRecord) removeRoute() // removes the route if it exists
const removeRoute = router.addRoute(routeRecord) removeRoute() // removes the route if it exists
This is useful when the routes do not have a name
By using
router.removeRoute()
to remove a route by its name:jsrouter.addRoute({ path: '/about', name: 'about', component: About }) // remove the route router.removeRoute('about')
router.addRoute({ path: '/about', name: 'about', component: About }) // remove the route router.removeRoute('about')
Note you can use
Symbol
s for names in routes if you wish to use this function but want to avoid conflicts in names.
Whenever a route is removed, all of its aliases and children are removed with it.
Adding nested routes
To add nested routes to an existing route, you can pass the name of the route as its first parameter to router.addRoute()
, this will effectively add the route as if it was added through children
:
router.addRoute({ name: 'admin', path: '/admin', component: Admin })
router.addRoute('admin', { path: 'settings', component: AdminSettings })
router.addRoute({ name: 'admin', path: '/admin', component: Admin })
router.addRoute('admin', { path: 'settings', component: AdminSettings })
This is equivalent to:
router.addRoute({
name: 'admin',
path: '/admin',
component: Admin,
children: [{ path: 'settings', component: AdminSettings }],
})
router.addRoute({
name: 'admin',
path: '/admin',
component: Admin,
children: [{ path: 'settings', component: AdminSettings }],
})
Looking at existing routes
Vue Router gives you two functions to look at existing routes:
router.hasRoute()
: check if a route existsrouter.getRoutes()
: get an array with all the route records.