Skip to content

Commit 96c9503

Browse files
committed
feat: add onBeforeRouteUpdate
1 parent feae2dc commit 96c9503

File tree

7 files changed

+99
-5
lines changed

7 files changed

+99
-5
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,55 @@
1+
/**
2+
* @jest-environment jsdom
3+
*/
4+
import {
5+
createRouter,
6+
createMemoryHistory,
7+
onBeforeRouteUpdate,
8+
} from '../../src'
9+
import { createApp, defineComponent } from 'vue'
10+
11+
const component = {
12+
template: '<div>Generic</div>',
13+
}
14+
15+
describe('onBeforeRouteUpdate', () => {
16+
it('invokes with the component context', async () => {
17+
expect.assertions(2)
18+
const spy = jest
19+
.fn()
20+
.mockImplementationOnce(function (this: any, to, from, next) {
21+
expect(typeof this.counter).toBe('number')
22+
next()
23+
})
24+
const WithLeave = defineComponent({
25+
template: `text`,
26+
// we use data to check if the context is the right one because saving `this` in a variable logs a few warnings
27+
data: () => ({ counter: 0 }),
28+
setup() {
29+
onBeforeRouteUpdate(spy)
30+
},
31+
})
32+
33+
const router = createRouter({
34+
history: createMemoryHistory(),
35+
routes: [
36+
{ path: '/', component },
37+
{ path: '/foo', component: WithLeave as any },
38+
],
39+
})
40+
const app = createApp({
41+
template: `
42+
<router-view />
43+
`,
44+
})
45+
app.use(router)
46+
const rootEl = document.createElement('div')
47+
document.body.appendChild(rootEl)
48+
app.mount(rootEl)
49+
50+
await router.isReady()
51+
await router.push('/foo')
52+
await router.push('/foo?q')
53+
expect(spy).toHaveBeenCalledTimes(1)
54+
})
55+
})

__tests__/matcher/records.spec.ts

+3
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ describe('normalizeRouteRecord', () => {
1212
aliasOf: undefined,
1313
components: { default: {} },
1414
leaveGuards: [],
15+
updateGuards: [],
1516
instances: {},
1617
meta: {},
1718
name: undefined,
@@ -35,6 +36,7 @@ describe('normalizeRouteRecord', () => {
3536
children: [{ path: '/child' }],
3637
components: { default: {} },
3738
leaveGuards: [],
39+
updateGuards: [],
3840
instances: {},
3941
meta: { foo: true },
4042
name: 'name',
@@ -76,6 +78,7 @@ describe('normalizeRouteRecord', () => {
7678
children: [{ path: '/child' }],
7779
components: { one: {} },
7880
leaveGuards: [],
81+
updateGuards: [],
7982
instances: {},
8083
meta: { foo: true },
8184
name: 'name',

src/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,7 @@ export {
3737

3838
export { NavigationFailureType, NavigationFailure } from './errors'
3939

40-
export { onBeforeRouteLeave } from './navigationGuards'
40+
export { onBeforeRouteLeave, onBeforeRouteUpdate } from './navigationGuards'
4141
export { RouterLink, useLink } from './RouterLink'
4242
export { RouterView } from './RouterView'
4343

src/matcher/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -304,6 +304,7 @@ export function normalizeRouteRecord(
304304
children: record.children || [],
305305
instances: {},
306306
leaveGuards: [],
307+
updateGuards: [],
307308
components:
308309
'components' in record
309310
? record.components

src/matcher/types.ts

+1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ export interface RouteRecordNormalized {
1616
props: Exclude<_RouteRecordBase['props'], void>
1717
beforeEnter: RouteRecordMultipleViews['beforeEnter']
1818
leaveGuards: NavigationGuard[]
19+
updateGuards: NavigationGuard[]
1920
instances: Record<string, ComponentPublicInstance | undefined | null>
2021
// can only be of of the same type as this record
2122
aliasOf: RouteRecordNormalized | undefined

src/navigationGuards.ts

+31-3
Original file line numberDiff line numberDiff line change
@@ -26,15 +26,18 @@ export function onBeforeRouteLeave(leaveGuard: NavigationGuard) {
2626
const instance = getCurrentInstance()
2727
if (!instance) {
2828
__DEV__ &&
29-
warn('onRouteLeave must be called at the top of a setup function')
29+
warn('onBeforeRouteLeave must be called at the top of a setup function')
3030
return
3131
}
3232

33-
const activeRecord = inject(matchedRouteKey, {} as any).value
33+
const activeRecord: RouteRecordNormalized | undefined = inject(
34+
matchedRouteKey,
35+
{} as any
36+
).value
3437

3538
if (!activeRecord) {
3639
__DEV__ &&
37-
warn('onRouteLeave must be called at the top of a setup function')
40+
warn('onBeforeRouteLeave must be called at the top of a setup function')
3841
return
3942
}
4043

@@ -44,6 +47,31 @@ export function onBeforeRouteLeave(leaveGuard: NavigationGuard) {
4447
)
4548
}
4649

50+
export function onBeforeRouteUpdate(updateGuard: NavigationGuard) {
51+
const instance = getCurrentInstance()
52+
if (!instance) {
53+
__DEV__ &&
54+
warn('onBeforeRouteUpdate must be called at the top of a setup function')
55+
return
56+
}
57+
58+
const activeRecord: RouteRecordNormalized | undefined = inject(
59+
matchedRouteKey,
60+
{} as any
61+
).value
62+
63+
if (!activeRecord) {
64+
__DEV__ &&
65+
warn('onBeforeRouteUpdate must be called at the top of a setup function')
66+
return
67+
}
68+
69+
activeRecord.updateGuards.push(
70+
// @ts-ignore do we even want to allow that? Passing the context in a composition api hook doesn't make sense
71+
updateGuard.bind(instance.proxy)
72+
)
73+
}
74+
4775
export function guardToPromiseFn(
4876
guard: NavigationGuard,
4977
to: RouteLocationNormalized,

src/router.ts

+7-1
Original file line numberDiff line numberDiff line change
@@ -395,7 +395,7 @@ export function createRouter({
395395

396396
const [
397397
leavingRecords,
398-
// updatingRecords,
398+
updatingRecords,
399399
// enteringRecords,
400400
] = extractChangingRecords(to, from)
401401

@@ -425,6 +425,12 @@ export function createRouter({
425425
from
426426
)
427427

428+
for (const record of updatingRecords) {
429+
for (const guard of record.updateGuards) {
430+
guards.push(guardToPromiseFn(guard, to, from))
431+
}
432+
}
433+
428434
// run the queue of per route beforeEnter guards
429435
return runGuardQueue(guards)
430436
})

0 commit comments

Comments
 (0)