Skip to content

Commit 6ed6eee

Browse files
committed
feat(router): remove partial Promise from router.go
BREAKING CHANGE: The `router.go()` methods doesn't return anything (like in Vue 3) anymore. The existing implementation was wrong as it would resolve the promise for the following navigation if `router.go()` was called with something that wasn't possible e.g. `router.go(-20)` right after entering the application would not do anything. Even worse, the promise returned by that call would resolve **after the next navigation**. There is no proper native API to implement this promise-based api properly, but one can write a version that should work in most scenarios by setting up multiple hooks right before calling `router.go()`: ```js export function go(delta) { return new Promise((resolve, reject) => { function popStateListener() { clearTimeout(timeout) } window.addEventListener('popstate', popStateListener) function clearHooks() { removeAfterEach() removeOnError() window.removeEventListener('popstate', popStateListener) } // if the popstate event is not called, consider this a failure const timeout = setTimeout(() => { clearHooks() reject(new Error('Failed to use router.go()')) // It's unclear of what value would always work here }, 10) setImmediate const removeAfterEach = router.afterEach((_to, _from, failure) => { clearHooks() resolve(failure) }) const removeOnError = router.onError(err => { clearHooks() reject(err) }) router.go(delta) }) } ```
1 parent b58e0aa commit 6ed6eee

File tree

2 files changed

+8
-45
lines changed

2 files changed

+8
-45
lines changed

__tests__/router.spec.ts

-17
Original file line numberDiff line numberDiff line change
@@ -175,23 +175,6 @@ describe('Router', () => {
175175
})
176176
})
177177

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

src/router.ts

+8-28
Original file line numberDiff line numberDiff line change
@@ -233,25 +233,22 @@ export interface Router {
233233
replace(to: RouteLocationRaw): Promise<NavigationFailure | void | undefined>
234234
/**
235235
* Go back in history if possible by calling `history.back()`. Equivalent to
236-
* `router.go(-1)`. Returns a Promise. See the limitations at
237-
* {@link Router.go}.
236+
* `router.go(-1)`.
238237
*/
239-
back(): Promise<NavigationFailure | void | undefined>
238+
back(): ReturnType<Router['go']>
240239
/**
241240
* Go forward in history if possible by calling `history.forward()`.
242-
* Equivalent to `router.go(1)`. Returns a Promise. See the limitations at
243-
* {@link Router.go}.
241+
* Equivalent to `router.go(1)`.
244242
*/
245-
forward(): Promise<NavigationFailure | void | undefined>
243+
forward(): ReturnType<Router['go']>
246244
/**
247-
* Allows you to move forward or backward through the history. Returns a
248-
* Promise that resolves when the navigation finishes. If it wasn't possible
249-
* to go back, the promise never resolves or rejects
245+
* Allows you to move forward or backward through the history. Calls
246+
* `history.go()`.
250247
*
251248
* @param delta - The position in the history to which you want to move,
252249
* relative to the current page
253250
*/
254-
go(delta: number): Promise<NavigationFailure | void | undefined>
251+
go(delta: number): void
255252

256253
/**
257254
* Add a navigation guard that executes before any navigation. Returns a
@@ -1025,24 +1022,7 @@ export function createRouter(options: RouterOptions): Router {
10251022
.catch(triggerError)
10261023
}
10271024

1028-
function go(delta: number) {
1029-
return new Promise<NavigationFailure | void | undefined>(
1030-
(resolve, reject) => {
1031-
let removeError = errorHandlers.add(err => {
1032-
removeError()
1033-
removeAfterEach()
1034-
reject(err)
1035-
})
1036-
let removeAfterEach = afterGuards.add((_to, _from, failure) => {
1037-
removeError()
1038-
removeAfterEach()
1039-
resolve(failure)
1040-
})
1041-
1042-
routerHistory.go(delta)
1043-
}
1044-
)
1045-
}
1025+
const go = (delta: number) => routerHistory.go(delta)
10461026

10471027
let started: boolean | undefined
10481028
const installedApps = new Set<App>()

0 commit comments

Comments
 (0)