|
|
@ -1,14 +1,11 @@
|
|
|
|
import { shallowReactive, provide, inject, nextTick } from 'vue'
|
|
|
|
import { reactive, provide, inject, nextTick, markRaw } from 'vue'
|
|
|
|
import Theme from '/@theme/index'
|
|
|
|
|
|
|
|
import { hot } from '@hmr'
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const NotFound = Theme.NotFound || (() => '404 Not Found')
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
|
|
|
|
* @typedef {import('vue').Component} Component
|
|
|
|
|
|
|
|
*
|
|
|
|
* @typedef {{
|
|
|
|
* @typedef {{
|
|
|
|
* path: string
|
|
|
|
* path: string
|
|
|
|
* contentComponent: import('vue').Component | null
|
|
|
|
* contentComponent: Component | null
|
|
|
|
* pageData: { path: string } | null
|
|
|
|
|
|
|
|
* }} Route
|
|
|
|
* }} Route
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* @typedef {{
|
|
|
|
* @typedef {{
|
|
|
@ -27,29 +24,18 @@ const RouterSymbol = Symbol()
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
const getDefaultRoute = () => ({
|
|
|
|
const getDefaultRoute = () => ({
|
|
|
|
path: '/',
|
|
|
|
path: '/',
|
|
|
|
contentComponent: null,
|
|
|
|
contentComponent: null
|
|
|
|
pageData: null
|
|
|
|
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
|
|
|
|
* @param {(route: Route) => Component | Promise<Component>} loadComponent
|
|
|
|
|
|
|
|
* @param {Component} [fallbackComponent]
|
|
|
|
* @returns {Router}
|
|
|
|
* @returns {Router}
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
export function initRouter() {
|
|
|
|
export function initRouter(loadComponent, fallbackComponent) {
|
|
|
|
const route = shallowReactive(getDefaultRoute())
|
|
|
|
const route = reactive(getDefaultRoute())
|
|
|
|
const inBrowser = typeof window !== 'undefined'
|
|
|
|
const inBrowser = typeof window !== 'undefined'
|
|
|
|
|
|
|
|
|
|
|
|
if (__DEV__ && inBrowser) {
|
|
|
|
|
|
|
|
// hot reload pageData
|
|
|
|
|
|
|
|
hot.on('vitepress:pageData', (data) => {
|
|
|
|
|
|
|
|
if (
|
|
|
|
|
|
|
|
data.path.replace(/\.md$/, '') ===
|
|
|
|
|
|
|
|
location.pathname.replace(/\.html$/, '')
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
route.pageData = data.pageData
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* @param {string} href
|
|
|
|
* @param {string} href
|
|
|
|
* @returns {Promise<void>}
|
|
|
|
* @returns {Promise<void>}
|
|
|
@ -68,26 +54,22 @@ export function initRouter() {
|
|
|
|
* @param {number} scrollPosition
|
|
|
|
* @param {number} scrollPosition
|
|
|
|
* @returns {Promise<void>}
|
|
|
|
* @returns {Promise<void>}
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
function loadPage(href, scrollPosition = 0) {
|
|
|
|
async function loadPage(href, scrollPosition = 0) {
|
|
|
|
const targetLoc = new URL(href)
|
|
|
|
// we are just using URL to parse the pathname and hash - the base doesn't
|
|
|
|
|
|
|
|
// matter and is only passed to support same-host hrefs.
|
|
|
|
|
|
|
|
const targetLoc = new URL(href, `http://vuejs.org`)
|
|
|
|
const pendingPath = (route.path = targetLoc.pathname)
|
|
|
|
const pendingPath = (route.path = targetLoc.pathname)
|
|
|
|
let pagePath = pendingPath.replace(/\.html$/, '')
|
|
|
|
|
|
|
|
if (pagePath.endsWith('/')) {
|
|
|
|
|
|
|
|
pagePath += 'index'
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (__DEV__) {
|
|
|
|
try {
|
|
|
|
// awlays force re-fetch content in dev
|
|
|
|
let comp = loadComponent(route)
|
|
|
|
pagePath += `.md?t=${Date.now()}`
|
|
|
|
// only await if it returns a Promise - this allows sync resolution
|
|
|
|
} else {
|
|
|
|
// on initial render in SSR.
|
|
|
|
pagePath += `.md.js`
|
|
|
|
if ('then' in comp && typeof comp.then === 'function') {
|
|
|
|
}
|
|
|
|
comp = await comp
|
|
|
|
|
|
|
|
}
|
|
|
|
return import(pagePath)
|
|
|
|
if (route.path === pendingPath) {
|
|
|
|
.then(async (m) => {
|
|
|
|
route.contentComponent = markRaw(comp)
|
|
|
|
if (route.path === pendingPath) {
|
|
|
|
if (inBrowser) {
|
|
|
|
route.contentComponent = m.default
|
|
|
|
|
|
|
|
route.pageData = m.__pageData
|
|
|
|
|
|
|
|
await nextTick()
|
|
|
|
await nextTick()
|
|
|
|
|
|
|
|
|
|
|
|
if (targetLoc.hash && !scrollPosition) {
|
|
|
|
if (targetLoc.hash && !scrollPosition) {
|
|
|
@ -106,14 +88,16 @@ export function initRouter() {
|
|
|
|
behavior: 'auto'
|
|
|
|
behavior: 'auto'
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
.catch((err) => {
|
|
|
|
} catch (err) {
|
|
|
|
if (!err.message.match(/fetch/)) {
|
|
|
|
if (!err.message.match(/fetch/)) {
|
|
|
|
throw err
|
|
|
|
throw err
|
|
|
|
} else if (route.path === pendingPath) {
|
|
|
|
} else if (route.path === pendingPath) {
|
|
|
|
route.contentComponent = NotFound
|
|
|
|
route.contentComponent = fallbackComponent
|
|
|
|
}
|
|
|
|
? markRaw(fallbackComponent)
|
|
|
|
})
|
|
|
|
: null
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (inBrowser) {
|
|
|
|
if (inBrowser) {
|
|
|
@ -158,7 +142,7 @@ export function initRouter() {
|
|
|
|
* @param {*} e
|
|
|
|
* @param {*} e
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
(e) => {
|
|
|
|
(e) => {
|
|
|
|
loadPage((e.state && e.state.scrollPosition) || 0)
|
|
|
|
loadPage(location.href, (e.state && e.state.scrollPosition) || 0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|