elk/plugins/track-scroll-position.client.ts

93 lines
2.4 KiB
TypeScript
Raw Normal View History

2023-02-10 22:13:46 +01:00
export default defineNuxtPlugin((nuxtApp) => {
const router = useRouter()
const route = useRoute()
const track = ref(false)
const { y } = useWindowScroll()
const storage = useLocalStorage<Record<string, number>>('elk-track-scroll', {})
2023-02-10 23:37:09 +01:00
const customRoutes = new Set<string>()
2023-02-10 22:13:46 +01:00
router.beforeEach(async () => {
track.value = false
})
router.onError(async () => {
track.value = true
})
const forceScroll = () => {
storage.value[route.fullPath] = 0
window.scrollTo({ top: 0, left: 0, behavior: 'smooth' })
}
2023-02-10 23:37:09 +01:00
const restoreScrollCallback = (ignoreHook: boolean) => {
2023-02-10 22:13:46 +01:00
const path = route.fullPath
return nextTick().then(() => {
if (route.meta && route.meta?.noScrollTrack) {
forceScroll()
return Promise.resolve()
}
2023-02-10 22:13:46 +01:00
return new Promise<void>((resolve, reject) => {
setTimeout(() => {
if (path !== route.fullPath) {
reject(new Error('navigation canceled'))
return
}
if (!route.meta || !route.meta?.noScrollTrack) {
2023-02-10 23:37:09 +01:00
const hook = ignoreHook ? undefined : customRoutes.has(route.fullPath)
if (hook) {
reject(new Error('hook detected'))
return
}
2023-02-10 22:13:46 +01:00
const scrollPosition = storage.value[route.fullPath]
if (scrollPosition)
window.scrollTo(0, scrollPosition)
}
else {
forceScroll()
}
resolve()
}, 600)
})
})
}
2023-02-10 23:37:09 +01:00
const restoreScroll = () => restoreScrollCallback(false)
2023-02-10 22:13:46 +01:00
2023-02-10 23:37:09 +01:00
const restoreScrollHook = () => {
2023-02-10 22:13:46 +01:00
if (isHydrated.value) {
restoreScroll().then(() => {
track.value = true
}).catch(noop)
}
2023-02-10 23:37:09 +01:00
}
const restoreCustomPageScroll = () => restoreScrollCallback(true)
nuxtApp.hooks.hook('app:suspense:resolve', restoreScrollHook)
nuxtApp.hooks.hook('page:finish', restoreScrollHook)
2023-02-10 22:13:46 +01:00
2023-02-10 22:28:53 +01:00
watch([track, y, () => route], ([trackEnabled, scrollPosition, r]) => {
2023-02-10 22:13:46 +01:00
if (trackEnabled && (!r.meta || !r.meta?.noScrollTrack))
storage.value[r.fullPath] = Math.floor(scrollPosition)
}, { immediate: true, flush: 'pre' })
2023-02-10 23:37:09 +01:00
const registerCustomRoute = (path: string) => {
customRoutes.add(path)
}
2023-02-10 22:13:46 +01:00
return {
provide: {
trackScroll: reactive({
forceScroll,
restoreScroll,
track,
2023-02-10 23:37:09 +01:00
registerCustomRoute,
restoreCustomPageScroll,
2023-02-10 22:13:46 +01:00
}),
},
}
})