Skip to content

Commit e08a0d0

Browse files
authored
fix(hash): force navigation restore on manual navigation (#921)
Fix #916
1 parent 9ef1b10 commit e08a0d0

File tree

2 files changed

+118
-4
lines changed

2 files changed

+118
-4
lines changed
+83
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,83 @@
1+
import { createMemoryHistory, createRouter, RouterHistory } from '../src'
2+
import { tick } from './utils'
3+
4+
const component = {}
5+
6+
interface RouterHistory_Test extends RouterHistory {
7+
changeURL(url: string): void
8+
}
9+
10+
describe('hash history edge cases', () => {
11+
it('correctly sets the url when it is manually changed but aborted with a redirect in guard', async () => {
12+
const history = createMemoryHistory() as RouterHistory_Test
13+
const router = createRouter({
14+
history,
15+
routes: [
16+
{ path: '/', component },
17+
{ path: '/foo', component },
18+
],
19+
})
20+
21+
await router.push('/foo?step=1')
22+
await router.push('/foo?step=2')
23+
await router.push('/foo?step=3')
24+
router.back()
25+
await tick() // wait for router listener on history
26+
expect(router.currentRoute.value.fullPath).toBe('/foo?step=2')
27+
28+
// force a redirect next time
29+
const removeListener = router.beforeEach(to => {
30+
if (to.path === '/') {
31+
removeListener()
32+
return '/foo?step=2'
33+
}
34+
return
35+
})
36+
37+
// const spy = jest.spyOn(history, 'go')
38+
39+
history.changeURL('/')
40+
await tick()
41+
expect(router.currentRoute.value.fullPath).toBe('/foo?step=2')
42+
expect(history.location).toBe('/foo?step=2')
43+
// expect(spy).toHaveBeenCalledTimes(1)
44+
// expect(spy).toHaveBeenCalledWith(-1)
45+
})
46+
47+
it('correctly sets the url when it is manually changed but aborted with guard', async () => {
48+
const history = createMemoryHistory() as RouterHistory_Test
49+
const router = createRouter({
50+
history,
51+
routes: [
52+
{ path: '/', component },
53+
{ path: '/foo', component },
54+
],
55+
})
56+
57+
await router.push('/foo?step=1')
58+
await router.push('/foo?step=2')
59+
await router.push('/foo?step=3')
60+
router.back()
61+
await tick() // wait for router listener on history
62+
expect(router.currentRoute.value.fullPath).toBe('/foo?step=2')
63+
64+
// force a redirect next time
65+
const removeListener = router.beforeEach(to => {
66+
if (to.path === '/') {
67+
removeListener()
68+
return false
69+
}
70+
71+
return
72+
})
73+
74+
// const spy = jest.spyOn(history, 'go')
75+
76+
history.changeURL('/')
77+
await tick()
78+
expect(router.currentRoute.value.fullPath).toBe('/foo?step=2')
79+
expect(history.location).toBe('/foo?step=2')
80+
// expect(spy).toHaveBeenCalledTimes(1)
81+
// expect(spy).toHaveBeenCalledWith(-1)
82+
})
83+
})

src/router.ts

+35-4
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ import {
1313
RouteLocationOptions,
1414
MatcherLocationRaw,
1515
} from './types'
16-
import { RouterHistory, HistoryState } from './history/common'
16+
import { RouterHistory, HistoryState, NavigationType } from './history/common'
1717
import {
1818
ScrollPosition,
1919
getSavedScrollPosition,
@@ -689,7 +689,7 @@ export function createRouter(options: RouterOptions): Router {
689689
) &&
690690
// and we have done it a couple of times
691691
redirectedFrom &&
692-
// @ts-expect-error
692+
// @ts-expect-error: added only in dev
693693
(redirectedFrom._count = redirectedFrom._count
694694
? // @ts-expect-error
695695
redirectedFrom._count + 1
@@ -980,7 +980,24 @@ export function createRouter(options: RouterOptions): Router {
980980
(error as NavigationRedirectError).to,
981981
toLocation
982982
// avoid an uncaught rejection, let push call triggerError
983-
).catch(noop)
983+
)
984+
.then(failure => {
985+
// manual change in hash history #916 ending up in the URL not
986+
// changing but it was changed by the manual url change, so we
987+
// need to manually change it ourselves
988+
if (
989+
isNavigationFailure(
990+
failure,
991+
ErrorTypes.NAVIGATION_ABORTED |
992+
ErrorTypes.NAVIGATION_DUPLICATED
993+
) &&
994+
!info.delta &&
995+
info.type === NavigationType.pop
996+
) {
997+
routerHistory.go(-1, false)
998+
}
999+
})
1000+
.catch(noop)
9841001
// avoid the then branch
9851002
return Promise.reject()
9861003
}
@@ -1000,7 +1017,21 @@ export function createRouter(options: RouterOptions): Router {
10001017
)
10011018

10021019
// revert the navigation
1003-
if (failure && info.delta) routerHistory.go(-info.delta, false)
1020+
if (failure) {
1021+
if (info.delta) {
1022+
routerHistory.go(-info.delta, false)
1023+
} else if (
1024+
info.type === NavigationType.pop &&
1025+
isNavigationFailure(
1026+
failure,
1027+
ErrorTypes.NAVIGATION_ABORTED | ErrorTypes.NAVIGATION_DUPLICATED
1028+
)
1029+
) {
1030+
// manual change in hash history #916
1031+
// it's like a push but lacks the information of the direction
1032+
routerHistory.go(-1, false)
1033+
}
1034+
}
10041035

10051036
triggerAfterEach(
10061037
toLocation as RouteLocationNormalizedLoaded,

0 commit comments

Comments
 (0)