Skip to content

Commit 6a8ccb6

Browse files
committed
fix(router): allow replace to have query and hash
fix #668
1 parent 9a6ae30 commit 6a8ccb6

File tree

2 files changed

+46
-8
lines changed

2 files changed

+46
-8
lines changed

__tests__/router.spec.ts

+29
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@ const routes: RouteRecordRaw[] = [
3131
{ path: '/to-foo', redirect: '/foo' },
3232
{ path: '/to-foo-named', redirect: { name: 'Foo' } },
3333
{ path: '/to-foo2', redirect: '/to-foo' },
34+
{ path: '/to-foo-query', redirect: '/foo?a=2#b' },
3435
{ path: '/to-p/:p', redirect: { name: 'Param' } },
3536
{ path: '/p/:p', name: 'Param', component: components.Bar },
3637
{ path: '/repeat/:r+', name: 'repeat', component: components.Bar },
@@ -115,6 +116,18 @@ describe('Router', () => {
115116
expect(history.replace).toHaveBeenCalledWith('/foo', expect.anything())
116117
})
117118

119+
it('parses query and hash with router.replace', async () => {
120+
const history = createMemoryHistory()
121+
const { router } = await newRouter({ history })
122+
jest.spyOn(history, 'replace')
123+
await router.replace('/foo?q=2#a')
124+
expect(history.replace).toHaveBeenCalledTimes(1)
125+
expect(history.replace).toHaveBeenCalledWith(
126+
'/foo?q=2#a',
127+
expect.anything()
128+
)
129+
})
130+
118131
it('replaces if a guard redirects', async () => {
119132
const history = createMemoryHistory()
120133
const { router } = await newRouter({ history })
@@ -537,6 +550,22 @@ describe('Router', () => {
537550
})
538551
})
539552

553+
it('handles query and hash passed in redirect string', async () => {
554+
const history = createMemoryHistory()
555+
const router = createRouter({ history, routes })
556+
await expect(router.push('/to-foo-query')).resolves.toEqual(undefined)
557+
expect(router.currentRoute.value).toMatchObject({
558+
name: 'Foo',
559+
path: '/foo',
560+
params: {},
561+
query: { a: '2' },
562+
hash: '#b',
563+
redirectedFrom: expect.objectContaining({
564+
fullPath: '/to-foo-query',
565+
}),
566+
})
567+
})
568+
540569
it('keeps query and hash when redirect is a string', async () => {
541570
const history = createMemoryHistory()
542571
const router = createRouter({ history, routes })

src/router.ts

+17-8
Original file line numberDiff line numberDiff line change
@@ -522,8 +522,9 @@ export function createRouter(options: RouterOptions): Router {
522522
function locationAsObject(
523523
to: RouteLocationRaw | RouteLocationNormalized
524524
): Exclude<RouteLocationRaw, string> | RouteLocationNormalized {
525-
// FIXME: does not take into account query params
526-
return typeof to === 'string' ? { path: to } : assign({}, to)
525+
return typeof to === 'string'
526+
? parseURL(parseQuery, to, currentRoute.value.path)
527+
: assign({}, to)
527528
}
528529

529530
function checkCanceledNavigation(
@@ -553,10 +554,16 @@ export function createRouter(options: RouterOptions): Router {
553554
const lastMatched = to.matched[to.matched.length - 1]
554555
if (lastMatched && lastMatched.redirect) {
555556
const { redirect } = lastMatched
556-
// transform it into an object to pass the original RouteLocaleOptions
557-
let newTargetLocation = locationAsObject(
557+
let newTargetLocation =
558558
typeof redirect === 'function' ? redirect(to) : redirect
559-
)
559+
560+
if (typeof newTargetLocation === 'string') {
561+
newTargetLocation =
562+
newTargetLocation.indexOf('?') > -1 ||
563+
newTargetLocation.indexOf('#') > -1
564+
? (newTargetLocation = locationAsObject(newTargetLocation))
565+
: { path: newTargetLocation }
566+
}
560567

561568
if (
562569
__DEV__ &&
@@ -587,9 +594,7 @@ export function createRouter(options: RouterOptions): Router {
587594
}
588595

589596
function pushWithRedirect(
590-
// TODO: should only be RouteLocation?
591597
to: RouteLocationRaw | RouteLocation,
592-
// TODO: add replace here
593598
redirectedFrom?: RouteLocation
594599
): Promise<NavigationFailure | void | undefined> {
595600
const targetLocation: RouteLocation = (pendingLocation = resolve(to))
@@ -603,7 +608,11 @@ export function createRouter(options: RouterOptions): Router {
603608

604609
if (shouldRedirect)
605610
return pushWithRedirect(
606-
assign(shouldRedirect, { state: data, force, replace }),
611+
assign(locationAsObject(shouldRedirect), {
612+
state: data,
613+
force,
614+
replace,
615+
}),
607616
// keep original redirectedFrom if it exists
608617
redirectedFrom || targetLocation
609618
)

0 commit comments

Comments
 (0)