Skip to content

Commit 34d7390

Browse files
committed
feat(guards): wip context support in multi apps
1 parent f2846ff commit 34d7390

File tree

3 files changed

+100
-6
lines changed

3 files changed

+100
-6
lines changed

e2e/multi-app/index.ts

+63-6
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,60 @@
1-
import { createRouter, createWebHistory } from '../../src'
1+
import { createRouter, createWebHistory, onBeforeRouteUpdate } from '../../src'
22
import { RouteComponent } from '../../src/types'
3-
import { createApp, ref, watchEffect, App } from 'vue'
3+
import { createApp, ref, watchEffect, App, inject } from 'vue'
44

55
const Home: RouteComponent = {
66
template: `<div class="home">Home</div>`,
77
}
88

99
const User: RouteComponent = {
10-
template: `<div class="user">User {{ $route.params.id }}</div>`,
10+
template: `<div class="user">User {{ $route.params.id }}. Updated <span class="count">{{ count }}</span></div>`,
11+
data: () => ({ count: 0 }),
12+
13+
beforeRouteEnter(to, from, next) {
14+
next(vm => {
15+
// @ts-ignore
16+
console.log('enter from ', vm.id)
17+
// @ts-ignore
18+
})
19+
},
20+
21+
beforeRouteUpdate(to, from, next) {
22+
// this.count++
23+
next()
24+
},
25+
26+
setup() {
27+
const id = inject('id')!
28+
29+
if (id !== 1)
30+
onBeforeRouteUpdate(function (to, from, next) {
31+
// @ts-ignore
32+
console.log('update from ', id, this.id)
33+
// @ts-ignore
34+
// this.count++
35+
next()
36+
})
37+
38+
return { id }
39+
},
1140
}
1241

42+
let looper = [1, 2, 3]
43+
44+
const NamedViews: RouteComponent[] = looper.map(i => ({
45+
name: 'part-' + i,
46+
47+
template: `<div class="named part-${i}">Part ${i}. Updated <span class="count">{{ count }}</span></div>`,
48+
49+
data: () => ({ count: 0 }),
50+
51+
beforeRouteUpdate(to, from, next) {
52+
// @ts-ignore
53+
// this.count++
54+
next()
55+
},
56+
}))
57+
1358
// path popstate listeners to track the call count
1459
let activePopStateListeners = ref(0)
1560
let guardCallCount = ref(0)
@@ -43,7 +88,19 @@ const router = createRouter({
4388
history: createWebHistory('/' + __dirname),
4489
routes: [
4590
{ path: '/', component: Home },
46-
{ path: '/users/:id', component: User },
91+
{
92+
path: '/users/:id',
93+
components: {
94+
default: User,
95+
...NamedViews.reduce(
96+
(routeComponents, component) => ({
97+
...routeComponents,
98+
[component.name!]: component,
99+
}),
100+
{} as Record<string, RouteComponent>
101+
),
102+
},
103+
},
47104
],
48105
})
49106

@@ -52,8 +109,6 @@ router.beforeEach((to, from, next) => {
52109
next()
53110
})
54111

55-
let looper = [1, 2, 3]
56-
57112
let apps: Array<App | null> = [null, null, null]
58113

59114
looper.forEach((n, i) => {
@@ -71,10 +126,12 @@ looper.forEach((n, i) => {
71126
</ul>
72127
73128
<router-view></router-view>
129+
<router-view name="part-${n}"></router-view>
74130
</div>
75131
`,
76132
}))
77133
app.use(router)
134+
app.provide('id', n)
78135
app.mount('#app-' + n)
79136
})
80137

e2e/specs/multi-app.js

+24
Original file line numberDiff line numberDiff line change
@@ -97,4 +97,28 @@ module.exports = {
9797

9898
.end()
9999
},
100+
101+
/** @type {import('nightwatch').NightwatchTest} */
102+
'supports navigation guards context with multiple apps': function (browser) {
103+
browser
104+
.url(baseURL)
105+
.assert.urlEquals(baseURL + '/')
106+
107+
// mount multiple apps and expect to have one listener only
108+
.click('#mount1')
109+
.assert.containsText('#app-1 .home', 'Home')
110+
// toggle multiple times
111+
.click('#app-1 li:nth-child(2) a')
112+
.assert.containsText('#app-1 .count', '0')
113+
.click('#app-1 li:nth-child(3) a')
114+
.assert.containsText('#app-1 .count', '1')
115+
.click('#mount2')
116+
.assert.containsText('#app-2 .user', 'User 2')
117+
.click('#app-1 li:nth-child(2) a')
118+
// first one keeps updating
119+
.assert.containsText('#app-1 .count', '2')
120+
// second app only updated once
121+
.assert.containsText('#app-2 .count', '1')
122+
.click('#mount3')
123+
},
100124
}

src/navigationGuards.ts

+13
Original file line numberDiff line numberDiff line change
@@ -260,3 +260,16 @@ function isRouteComponent(
260260
'__vccOpts' in component
261261
)
262262
}
263+
264+
/**
265+
* 1. beforeRouteEnter callbacks
266+
* 2. Dictionary of instances per view name
267+
*/
268+
export type GuardManagerEntry = [
269+
NavigationGuardNextCallback[],
270+
Record<string, ComponentPublicInstance | undefined | null>
271+
]
272+
273+
export function createGuardManager() {
274+
return new Map<RouteRecordNormalized, GuardManagerEntry>()
275+
}

0 commit comments

Comments
 (0)