Skip to content

Commit 9732758

Browse files
committed
fix(router): preserve navigation options with redirects
1 parent 0586394 commit 9732758

File tree

5 files changed

+58
-29
lines changed

5 files changed

+58
-29
lines changed

__tests__/initialNavigation.spec.ts

+2-17
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import { JSDOM } from 'jsdom'
2-
import { createRouter, createWebHistory, Router } from '../src'
3-
import { createDom, components } from './utils'
2+
import { createRouter, createWebHistory } from '../src'
3+
import { createDom, components, nextNavigation } from './utils'
44
import { RouteRecordRaw } from '../src/types'
55

66
// override the value of isBrowser because the variable is created before JSDOM
@@ -40,21 +40,6 @@ describe('Initial Navigation', () => {
4040
return { history, router }
4141
}
4242

43-
function nextNavigation(router: Router) {
44-
return new Promise((resolve, reject) => {
45-
let removeAfter = router.afterEach((_to, _from, failure) => {
46-
removeAfter()
47-
removeError()
48-
resolve(failure)
49-
})
50-
let removeError = router.onError(err => {
51-
removeAfter()
52-
removeError()
53-
reject(err)
54-
})
55-
})
56-
}
57-
5843
beforeAll(() => {
5944
dom = createDom()
6045
})

__tests__/router.spec.ts

+19-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import fakePromise from 'faked-promise'
22
import { createRouter, createMemoryHistory, createWebHistory } from '../src'
33
import { NavigationFailureType } from '../src/errors'
4-
import { createDom, components, tick } from './utils'
4+
import { createDom, components, tick, nextNavigation } from './utils'
55
import {
66
RouteRecordRaw,
77
RouteLocationRaw,
@@ -504,6 +504,24 @@ describe('Router', () => {
504504
})
505505
})
506506

507+
it('keeps original replace if redirect', async () => {
508+
const history = createMemoryHistory()
509+
const router = createRouter({ history, routes })
510+
await router.push('/search')
511+
512+
await expect(router.replace('/to-foo')).resolves.toEqual(undefined)
513+
expect(router.currentRoute.value).toMatchObject({
514+
path: '/foo',
515+
redirectedFrom: expect.objectContaining({ path: '/to-foo' }),
516+
})
517+
518+
history.go(-1)
519+
await nextNavigation(router)
520+
expect(router.currentRoute.value).not.toMatchObject({
521+
path: '/search',
522+
})
523+
})
524+
507525
it('can pass on query and hash when redirecting', async () => {
508526
const history = createMemoryHistory()
509527
const router = createRouter({ history, routes })

__tests__/utils.ts

+16-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ import {
1010
RouteRecordName,
1111
} from '../src/types'
1212
import { h, resolveComponent, ComponentOptions } from 'vue'
13-
import { RouterOptions, createWebHistory, createRouter } from '../src'
13+
import { RouterOptions, createWebHistory, createRouter, Router } from '../src'
1414

1515
export const tick = (time?: number) =>
1616
new Promise(resolve => {
@@ -24,6 +24,21 @@ export async function ticks(n: number) {
2424
}
2525
}
2626

27+
export function nextNavigation(router: Router) {
28+
return new Promise((resolve, reject) => {
29+
let removeAfter = router.afterEach((_to, _from, failure) => {
30+
removeAfter()
31+
removeError()
32+
resolve(failure)
33+
})
34+
let removeError = router.onError(err => {
35+
removeAfter()
36+
removeError()
37+
reject(err)
38+
})
39+
})
40+
}
41+
2742
export interface RouteRecordViewLoose
2843
extends Pick<
2944
RouteRecordMultipleViews,

src/router.ts

+17-9
Original file line numberDiff line numberDiff line change
@@ -229,33 +229,42 @@ export function createRouter({
229229
}
230230
}
231231

232+
function locationAsObject(
233+
to: RouteLocationRaw | RouteLocationNormalized
234+
): Exclude<RouteLocationRaw, string> | RouteLocationNormalized {
235+
return typeof to === 'string' ? { path: to } : to
236+
}
237+
232238
function push(to: RouteLocationRaw | RouteLocation) {
233-
return pushWithRedirect(to, undefined)
239+
return pushWithRedirect(to)
234240
}
235241

236242
function replace(to: RouteLocationRaw | RouteLocationNormalized) {
237-
const location = typeof to === 'string' ? { path: to } : to
238-
return push({ ...location, replace: true })
243+
return push({ ...locationAsObject(to), replace: true })
239244
}
240245

241246
async function pushWithRedirect(
242247
to: RouteLocationRaw | RouteLocation,
243-
redirectedFrom: RouteLocation | undefined
248+
redirectedFrom?: RouteLocation
244249
): Promise<NavigationFailure | void> {
245250
const targetLocation: RouteLocation = (pendingLocation = resolve(to))
246251
const from = currentRoute.value
247252
const data: HistoryState | undefined = (to as any).state
248-
// @ts-ignore: no need to check the string as force do not exist on a string
249-
const force: boolean | undefined = to.force
253+
const force: boolean | undefined = (to as any).force
254+
const replace: boolean | undefined = (to as any).replace === true
250255

251256
if (!force && isSameRouteLocation(from, targetLocation)) return
252257

253258
const lastMatched =
254259
targetLocation.matched[targetLocation.matched.length - 1]
255260
if (lastMatched && 'redirect' in lastMatched) {
256261
const { redirect } = lastMatched
262+
// transform it into an object to pass the original RouteLocaleOptions
263+
let newTargetLocation = locationAsObject(
264+
typeof redirect === 'function' ? redirect(targetLocation) : redirect
265+
)
257266
return pushWithRedirect(
258-
typeof redirect === 'function' ? redirect(targetLocation) : redirect,
267+
{ ...newTargetLocation, state: data, force, replace },
259268
// keep original redirectedFrom if it exists
260269
redirectedFrom || targetLocation
261270
)
@@ -301,8 +310,7 @@ export function createRouter({
301310
toLocation as RouteLocationNormalizedLoaded,
302311
from,
303312
true,
304-
// RouteLocationNormalized will give undefined
305-
(to as RouteLocationRaw).replace === true,
313+
replace,
306314
data
307315
)
308316

src/types/index.ts

+4-1
Original file line numberDiff line numberDiff line change
@@ -160,7 +160,10 @@ export interface _RouteRecordBase {
160160
| boolean
161161
| Record<string, any>
162162
| ((to: RouteLocationNormalized) => Record<string, any>)
163-
// TODO: beforeEnter has no effect with redirect, move and test
163+
/**
164+
* Before Enter guard specific to this record. Note `beforeEnter` has no
165+
* effect if the record has a `redirect` property.
166+
*/
164167
beforeEnter?:
165168
| NavigationGuardWithThis<undefined>
166169
| NavigationGuardWithThis<undefined>[]

0 commit comments

Comments
 (0)