Skip to content

Commit eb87757

Browse files
committed
feat(router): go, back and forward can be awaited
1 parent 7ea0b63 commit eb87757

File tree

2 files changed

+48
-14
lines changed

2 files changed

+48
-14
lines changed

__tests__/router.spec.ts

+17
Original file line numberDiff line numberDiff line change
@@ -176,6 +176,23 @@ describe('Router', () => {
176176
expect(router.currentRoute.value).not.toBe(START_LOCATION_NORMALIZED)
177177
})
178178

179+
it('can await router.go', async () => {
180+
const { router } = await newRouter()
181+
await router.push('/foo')
182+
let currentRoute = router.currentRoute.value
183+
const [p1, r1] = fakePromise()
184+
router.beforeEach(async (to, from, next) => {
185+
await p1
186+
next()
187+
})
188+
let p = router.go(-1)
189+
expect(router.currentRoute.value).toBe(currentRoute)
190+
r1()
191+
// resolves to undefined as a working navigation
192+
await expect(p).resolves.toBe(undefined)
193+
expect(router.currentRoute.value).not.toBe(currentRoute)
194+
})
195+
179196
it('can pass replace option to push', async () => {
180197
const { router, history } = await newRouter()
181198
jest.spyOn(history, 'replace')

src/router.ts

+31-14
Original file line numberDiff line numberDiff line change
@@ -139,13 +139,11 @@ export interface Router {
139139

140140
resolve(to: RouteLocationRaw): RouteLocation & { href: string }
141141

142-
push(to: RouteLocationRaw): Promise<NavigationFailure | void>
143-
replace(to: RouteLocationRaw): Promise<NavigationFailure | void>
144-
// TODO: return a promise when https://github.com/vuejs/rfcs/pull/150 is
145-
// merged
146-
back(): void
147-
forward(): void
148-
go(delta: number): void
142+
push(to: RouteLocationRaw): Promise<NavigationFailure | void | undefined>
143+
replace(to: RouteLocationRaw): Promise<NavigationFailure | void | undefined>
144+
back(): Promise<NavigationFailure | void | undefined>
145+
forward(): Promise<NavigationFailure | void | undefined>
146+
go(delta: number): Promise<NavigationFailure | void | undefined>
149147

150148
beforeEach(guard: NavigationGuardWithThis<undefined>): () => void
151149
beforeResolve(guard: NavigationGuardWithThis<undefined>): () => void
@@ -313,7 +311,7 @@ export function createRouter(options: RouterOptions): Router {
313311
function pushWithRedirect(
314312
to: RouteLocationRaw | RouteLocation,
315313
redirectedFrom?: RouteLocation
316-
): Promise<NavigationFailure | void> {
314+
): Promise<NavigationFailure | void | undefined> {
317315
const targetLocation: RouteLocation = (pendingLocation = resolve(to))
318316
const from = currentRoute.value
319317
const data: HistoryState | undefined = (to as RouteLocationOptions).state
@@ -346,7 +344,7 @@ export function createRouter(options: RouterOptions): Router {
346344
const toLocation = targetLocation as RouteLocationNormalized
347345

348346
toLocation.redirectedFrom = redirectedFrom
349-
let failure: NavigationFailure | void
347+
let failure: NavigationFailure | void | undefined
350348

351349
if (!force && isSameRouteLocation(from, targetLocation))
352350
failure = createRouterError<NavigationFailure>(
@@ -629,7 +627,7 @@ export function createRouter(options: RouterOptions): Router {
629627
(error as NavigationRedirectError).to,
630628
toLocation
631629
).catch(() => {
632-
// TODO: in dev show warning, in prod noop, same as initial navigation
630+
// TODO: in dev show warning, in prod triggerError, same as initial navigation
633631
})
634632
// avoid the then branch
635633
return Promise.reject()
@@ -659,7 +657,7 @@ export function createRouter(options: RouterOptions): Router {
659657
)
660658
})
661659
.catch(() => {
662-
// TODO: same as above: in dev show warning, in prod noop, same as initial navigation
660+
// TODO: same as above
663661
})
664662
})
665663

@@ -724,6 +722,25 @@ export function createRouter(options: RouterOptions): Router {
724722
.then(position => position && scrollToPosition(position))
725723
}
726724

725+
function go(delta: number) {
726+
return new Promise<NavigationFailure | void | undefined>(
727+
(resolve, reject) => {
728+
let removeError = errorHandlers.add(err => {
729+
removeError()
730+
removeAfterEach()
731+
reject(err)
732+
})
733+
let removeAfterEach = afterGuards.add((_to, _from, failure) => {
734+
removeError()
735+
removeAfterEach()
736+
resolve(failure)
737+
})
738+
739+
routerHistory.go(delta)
740+
}
741+
)
742+
}
743+
727744
const router: Router = {
728745
currentRoute,
729746

@@ -736,9 +753,9 @@ export function createRouter(options: RouterOptions): Router {
736753

737754
push,
738755
replace,
739-
go: routerHistory.go,
740-
back: () => routerHistory.go(-1),
741-
forward: () => routerHistory.go(1),
756+
go,
757+
back: () => go(-1),
758+
forward: () => go(1),
742759

743760
beforeEach: beforeGuards.add,
744761
beforeResolve: beforeResolveGuards.add,

0 commit comments

Comments
 (0)