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
|
|
|
|
})
|
|
|
|
|
2023-02-11 12:31:53 +01:00
|
|
|
const forceScrollToTop = () => {
|
2023-02-10 22:13:46 +01:00
|
|
|
storage.value[route.fullPath] = 0
|
|
|
|
window.scrollTo({ top: 0, left: 0, behavior: 'smooth' })
|
|
|
|
}
|
|
|
|
|
2023-02-11 00:15:57 +01:00
|
|
|
const restoreScrollCallback = (ignoreCustomRoutes: boolean) => {
|
2023-02-10 22:13:46 +01:00
|
|
|
const path = route.fullPath
|
|
|
|
return nextTick().then(() => {
|
2023-02-11 00:15:57 +01:00
|
|
|
if (route.meta?.noScrollTrack) {
|
2023-02-11 12:31:53 +01:00
|
|
|
forceScrollToTop()
|
2023-02-10 22:34:34 +01:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2023-02-11 00:23:58 +01:00
|
|
|
const r = ignoreCustomRoutes ? undefined : customRoutes.has(route.fullPath)
|
|
|
|
if (r) {
|
|
|
|
reject(new Error('custom routed detected'))
|
|
|
|
return
|
2023-02-10 22:13:46 +01:00
|
|
|
}
|
|
|
|
|
2023-02-11 00:23:58 +01:00
|
|
|
const scrollPosition = storage.value[route.fullPath]
|
|
|
|
if (scrollPosition)
|
|
|
|
window.scrollTo(0, scrollPosition)
|
|
|
|
|
2023-02-11 12:42:01 +01:00
|
|
|
// required for custom routes: first call will be rejected
|
|
|
|
// we need to enable scroll tracking again, it is disabled
|
2023-02-11 12:31:53 +01:00
|
|
|
if (!track.value) {
|
|
|
|
nextTick().then(() => {
|
|
|
|
track.value = true
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-02-10 22:13:46 +01:00
|
|
|
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: {
|
2023-02-11 12:42:01 +01:00
|
|
|
trackScroll: {
|
2023-02-11 12:31:53 +01:00
|
|
|
forceScrollToTop,
|
2023-02-10 23:37:09 +01:00
|
|
|
registerCustomRoute,
|
|
|
|
restoreCustomPageScroll,
|
2023-02-11 12:42:01 +01:00
|
|
|
},
|
2023-02-10 22:13:46 +01:00
|
|
|
},
|
|
|
|
}
|
|
|
|
})
|