Skip to content

Commit 5ff2cd0

Browse files
committed
feat(loaders): internal mods in route records
1 parent 4861467 commit 5ff2cd0

File tree

5 files changed

+97
-4
lines changed

5 files changed

+97
-4
lines changed

packages/router/__tests__/guards/extractComponentsGuards.spec.ts

+26
Original file line numberDiff line numberDiff line change
@@ -104,4 +104,30 @@ describe('extractComponentsGuards', () => {
104104
'custom'
105105
)
106106
})
107+
108+
it('preserves resolved modules in mods', async () => {
109+
const mod = {
110+
default: components.Home,
111+
__esModule: true,
112+
custom: true,
113+
}
114+
const mod2 = {
115+
default: components.Bar,
116+
__esModule: true,
117+
custom: true,
118+
}
119+
const record = normalizeRouteRecord({
120+
path: '/',
121+
components: { default: async () => mod, other: async () => mod2 },
122+
})
123+
expect(record.mods).toEqual({})
124+
const guards = extractComponentsGuards(
125+
[record],
126+
'beforeRouteEnter',
127+
to,
128+
from
129+
)
130+
await Promise.all(guards.map(guard => guard()))
131+
expect(record.mods).toEqual({ default: mod, other: mod2 })
132+
})
107133
})

packages/router/__tests__/guards/loadRouteLocation.spec.ts

+55
Original file line numberDiff line numberDiff line change
@@ -86,6 +86,61 @@ describe('loadRouteLocation', () => {
8686
])
8787
})
8888

89+
describe('mods', () => {
90+
const mod = {
91+
default: components.Home,
92+
__esModule: true,
93+
custom: true,
94+
}
95+
const mod2 = {
96+
default: FunctionalHome,
97+
__esModule: true,
98+
custom: true,
99+
}
100+
101+
it('preserves resolved modules', async () => {
102+
const router = createRouter({
103+
history: createMemoryHistory(),
104+
routes: [
105+
{
106+
path: '/',
107+
component: async () => mod,
108+
},
109+
],
110+
})
111+
112+
const loaded = await loadRouteLocation(router.resolve('/'))
113+
// mods follow the same structure as components
114+
expect(loaded.matched[0]?.mods).toEqual({
115+
default: expect.anything(),
116+
})
117+
expect(loaded.matched[0]?.mods?.default).toBe(mod)
118+
})
119+
120+
it('preserves resolved modules for named components', async () => {
121+
const router = createRouter({
122+
history: createMemoryHistory(),
123+
routes: [
124+
{
125+
path: '/',
126+
components: {
127+
default: async () => mod2,
128+
name: async () => mod,
129+
},
130+
},
131+
],
132+
})
133+
134+
const loaded = await loadRouteLocation(router.resolve('/'))
135+
expect(loaded.matched[0]?.mods).toEqual({
136+
default: expect.anything(),
137+
name: expect.anything(),
138+
})
139+
expect(loaded.matched[0]?.mods?.name).toBe(mod)
140+
expect(loaded.matched[0]?.mods?.default).toBe(mod2)
141+
})
142+
})
143+
89144
it('throws with non loadable routes', async () => {
90145
expect.assertions(1)
91146
await expect(

packages/router/src/matcher/index.ts

+1
Original file line numberDiff line numberDiff line change
@@ -394,6 +394,7 @@ export function normalizeRouteRecord(
394394
leaveGuards: new Set(),
395395
updateGuards: new Set(),
396396
enterCallbacks: {},
397+
mods: {},
397398
components:
398399
'components' in record
399400
? record.components || null

packages/router/src/matcher/types.ts

+7
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,13 @@ export interface RouteRecordNormalized {
3131
* {@inheritDoc RouteRecordMultipleViews.components}
3232
*/
3333
components: RouteRecordMultipleViews['components'] | null | undefined
34+
35+
/**
36+
* Contains the original modules for lazy loaded components.
37+
* @internal
38+
*/
39+
mods: Record<string, unknown>
40+
3441
/**
3542
* Nested route records.
3643
*/

packages/router/src/navigationGuards.ts

+8-4
Original file line numberDiff line numberDiff line change
@@ -316,21 +316,22 @@ export function extractComponentsGuards(
316316
guards.push(() =>
317317
componentPromise.then(resolved => {
318318
if (!resolved)
319-
return Promise.reject(
320-
new Error(
321-
`Couldn't resolve component "${name}" at "${record.path}"`
322-
)
319+
throw new Error(
320+
`Couldn't resolve component "${name}" at "${record.path}"`
323321
)
324322
const resolvedComponent = isESModule(resolved)
325323
? resolved.default
326324
: resolved
325+
// keep the resolved module for plugins like data loaders
326+
record.mods[name] = resolved
327327
// replace the function with the resolved component
328328
// cannot be null or undefined because we went into the for loop
329329
record.components![name] = resolvedComponent
330330
// __vccOpts is added by vue-class-component and contain the regular options
331331
const options: ComponentOptions =
332332
(resolvedComponent as any).__vccOpts || resolvedComponent
333333
const guard = options[guardType]
334+
334335
return (
335336
guard &&
336337
guardToPromiseFn(guard, to, from, record, name, runWithContext)()
@@ -373,9 +374,12 @@ export function loadRouteLocation(
373374
`Couldn't resolve component "${name}" at "${record.path}". Ensure you passed a function that returns a promise.`
374375
)
375376
)
377+
376378
const resolvedComponent = isESModule(resolved)
377379
? resolved.default
378380
: resolved
381+
// keep the resolved module for plugins like data loaders
382+
record.mods[name] = resolved
379383
// replace the function with the resolved component
380384
// cannot be null or undefined because we went into the for loop
381385
record.components![name] = resolvedComponent

0 commit comments

Comments
 (0)