Skip to content

Commit 181efe9

Browse files
committed
feat(scroll): handle scroll on popstate
1 parent 539c742 commit 181efe9

File tree

2 files changed

+32
-3
lines changed

2 files changed

+32
-3
lines changed

src/router.ts

+11-3
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,9 @@ import {
1818
ScrollToPosition,
1919
ScrollPosition,
2020
scrollToPosition,
21+
saveScrollOnLeave,
22+
getScrollKey,
23+
getSavedScroll,
2124
} from './utils/scroll'
2225
import { createRouterMatcher } from './matcher'
2326
import { createRouterError, ErrorTypes, NavigationError } from './errors'
@@ -420,9 +423,12 @@ export function createRouter({
420423
// TODO: this doesn't work on first load. Moving it to RouterView could allow automatically handling transitions too maybe
421424
// TODO: refactor with a state getter
422425
const state = isPush || !isClient ? {} : window.history.state
423-
handleScroll(toLocation, from, state && state.scroll).catch(err =>
424-
triggerError(err, false)
425-
)
426+
const savedScroll = getSavedScroll(getScrollKey(toLocation.fullPath, 0))
427+
handleScroll(
428+
toLocation,
429+
from,
430+
savedScroll || (state && state.scroll)
431+
).catch(err => triggerError(err, false))
426432

427433
// navigation is confirmed, call afterGuards
428434
for (const guard of afterGuards.list()) guard(toLocation, from)
@@ -439,6 +445,8 @@ export function createRouter({
439445
pendingLocation = toLocation
440446
const from = currentRoute.value
441447

448+
saveScrollOnLeave(getScrollKey(from.fullPath, info.distance))
449+
442450
try {
443451
await navigate(toLocation, from)
444452
finalizeNavigation(

src/utils/scroll.ts

+21
Original file line numberDiff line numberDiff line change
@@ -65,3 +65,24 @@ export function scrollToPosition(position: ScrollPosition): void {
6565
window.scrollTo(normalizedPosition.x, normalizedPosition.y)
6666
}
6767
}
68+
69+
/**
70+
* TODO: refactor the scroll behavior so it can be tree shaken
71+
*/
72+
73+
export const scrollPositions = new Map<string, ScrollToPosition>()
74+
75+
export function getScrollKey(path: string, distance: number): string {
76+
const position: number = history.state
77+
? history.state.position - distance
78+
: -1
79+
return position + path
80+
}
81+
82+
export function saveScrollOnLeave(key: string) {
83+
scrollPositions.set(key, computeScrollPosition())
84+
}
85+
86+
export function getSavedScroll(key: string) {
87+
return scrollPositions.get(key)
88+
}

0 commit comments

Comments
 (0)