mirror of https://github.com/vuejs/vitepress
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
287 lines
8.3 KiB
287 lines
8.3 KiB
import { isBooleanAttr } from '@vue/shared'
|
|
import fs from 'fs-extra'
|
|
import path from 'path'
|
|
import { pathToFileURL } from 'url'
|
|
import { normalizePath, transformWithEsbuild, type Rollup } from 'vite'
|
|
import { version } from '../../../package.json'
|
|
import type { SiteConfig } from '../config'
|
|
import {
|
|
EXTERNAL_URL_RE,
|
|
createTitle,
|
|
escapeHtml,
|
|
mergeHead,
|
|
notFoundPageData,
|
|
resolveSiteDataByRoute,
|
|
sanitizeFileName,
|
|
slash,
|
|
type HeadConfig,
|
|
type PageData,
|
|
type SSGContext
|
|
} from '../shared'
|
|
|
|
export async function renderPage(
|
|
render: (path: string) => Promise<SSGContext>,
|
|
config: SiteConfig,
|
|
page: string, // foo.md
|
|
result: Rollup.RollupOutput | null,
|
|
appChunk: Rollup.OutputChunk | null,
|
|
cssChunk: Rollup.OutputAsset | null,
|
|
assets: string[],
|
|
pageToHashMap: Record<string, string>,
|
|
metadataScript: { html: string; inHead: boolean },
|
|
additionalHeadTags: HeadConfig[],
|
|
usedIcons: Set<string>
|
|
) {
|
|
const routePath = `/${page.replace(/\.md$/, '')}`
|
|
const siteData = resolveSiteDataByRoute(config.site, routePath)
|
|
|
|
// render page
|
|
const context = await render(routePath)
|
|
const { content, teleports, vpSocialIcons } =
|
|
(await config.postRender?.(context)) ?? context
|
|
|
|
// add used social icons to the set
|
|
vpSocialIcons.forEach((icon) => usedIcons.add(icon))
|
|
|
|
const pageName = sanitizeFileName(page.replace(/\//g, '_'))
|
|
// server build doesn't need hash
|
|
const pageServerJsFileName = pageName + '.js'
|
|
// for any initial page load, we only need the lean version of the page js
|
|
// since the static content is already on the page!
|
|
const pageHash = pageToHashMap[pageName.toLowerCase()]
|
|
const pageClientJsFileName = `${config.assetsDir}/${pageName}.${pageHash}.lean.js`
|
|
|
|
let pageData: PageData
|
|
let hasCustom404 = true
|
|
|
|
try {
|
|
// resolve page data so we can render head tags
|
|
const { __pageData } = await import(
|
|
pathToFileURL(path.join(config.tempDir, pageServerJsFileName)).href
|
|
)
|
|
pageData = __pageData
|
|
} catch (e) {
|
|
if (page === '404.md') {
|
|
hasCustom404 = false
|
|
pageData = notFoundPageData
|
|
} else {
|
|
throw e
|
|
}
|
|
}
|
|
|
|
const title: string = createTitle(siteData, pageData)
|
|
const description: string = pageData.description || siteData.description
|
|
const stylesheetLink = cssChunk
|
|
? `<link rel="preload stylesheet" href="${siteData.base}${cssChunk.fileName}" as="style">`
|
|
: ''
|
|
|
|
let preloadLinks =
|
|
config.mpa || (!hasCustom404 && page === '404.md')
|
|
? []
|
|
: result && appChunk
|
|
? [
|
|
...new Set([
|
|
// resolve imports for index.js + page.md.js and inject script tags
|
|
// for them as well so we fetch everything as early as possible
|
|
// without having to wait for entry chunks to parse
|
|
...resolvePageImports(config, page, result, appChunk),
|
|
pageClientJsFileName
|
|
])
|
|
]
|
|
: []
|
|
|
|
let prefetchLinks: string[] = []
|
|
|
|
const { shouldPreload } = config
|
|
if (shouldPreload) {
|
|
prefetchLinks = preloadLinks.filter((link) => !shouldPreload(link, page))
|
|
preloadLinks = preloadLinks.filter((link) => shouldPreload(link, page))
|
|
}
|
|
|
|
const toHeadTags = (files: string[], rel: string): HeadConfig[] =>
|
|
files.map((file) => [
|
|
'link',
|
|
{
|
|
rel,
|
|
// don't add base to external urls
|
|
href: (EXTERNAL_URL_RE.test(file) ? '' : siteData.base) + file
|
|
}
|
|
])
|
|
|
|
const preloadHeadTags = toHeadTags(preloadLinks, 'modulepreload')
|
|
const prefetchHeadTags = toHeadTags(prefetchLinks, 'prefetch')
|
|
|
|
const headBeforeTransform = [
|
|
...additionalHeadTags,
|
|
...preloadHeadTags,
|
|
...prefetchHeadTags,
|
|
...mergeHead(
|
|
siteData.head,
|
|
filterOutHeadDescription(pageData.frontmatter.head)
|
|
)
|
|
]
|
|
|
|
const head = mergeHead(
|
|
headBeforeTransform,
|
|
(await config.transformHead?.({
|
|
page,
|
|
siteConfig: config,
|
|
siteData,
|
|
pageData,
|
|
title,
|
|
description,
|
|
head: headBeforeTransform,
|
|
content,
|
|
assets
|
|
})) || []
|
|
)
|
|
|
|
let inlinedScript = ''
|
|
if (config.mpa && result) {
|
|
const matchingChunk = result.output.find(
|
|
(chunk) =>
|
|
chunk.type === 'chunk' &&
|
|
chunk.facadeModuleId === slash(path.join(config.srcDir, page))
|
|
) as Rollup.OutputChunk
|
|
if (matchingChunk) {
|
|
if (!matchingChunk.code.includes('import')) {
|
|
inlinedScript = `<script type="module">${matchingChunk.code}</script>`
|
|
fs.removeSync(path.resolve(config.outDir, matchingChunk.fileName))
|
|
} else {
|
|
inlinedScript = `<script type="module" src="${siteData.base}${matchingChunk.fileName}"></script>`
|
|
}
|
|
}
|
|
}
|
|
|
|
const dir = pageData.frontmatter.dir || siteData.dir || 'ltr'
|
|
|
|
const html = `<!DOCTYPE html>
|
|
<html lang="${siteData.lang}" dir="${dir}">
|
|
<head>
|
|
<meta charset="utf-8">
|
|
${
|
|
isMetaViewportOverridden(head)
|
|
? ''
|
|
: '<meta name="viewport" content="width=device-width,initial-scale=1">'
|
|
}
|
|
<title>${escapeHtml(title)}</title>
|
|
${
|
|
isDescriptionOverridden(head)
|
|
? ''
|
|
: `<meta name="description" content="${description}">`
|
|
}
|
|
<meta name="generator" content="VitePress v${version}">
|
|
${stylesheetLink}
|
|
<link rel="preload stylesheet" href="${siteData.base}vp-icons.css" as="style">
|
|
${metadataScript.inHead ? metadataScript.html : ''}
|
|
${
|
|
appChunk
|
|
? `<script type="module" src="${siteData.base}${appChunk.fileName}"></script>`
|
|
: ''
|
|
}
|
|
${await renderHead(head)}
|
|
</head>
|
|
<body>${teleports?.body || ''}
|
|
<div id="app">${page === '404.md' ? '' : content}</div>
|
|
${metadataScript.inHead ? '' : metadataScript.html}
|
|
${inlinedScript}
|
|
</body>
|
|
</html>`
|
|
|
|
const htmlFileName = path.join(config.outDir, page.replace(/\.md$/, '.html'))
|
|
await fs.ensureDir(path.dirname(htmlFileName))
|
|
const transformedHtml = await config.transformHtml?.(html, htmlFileName, {
|
|
page,
|
|
siteConfig: config,
|
|
siteData,
|
|
pageData,
|
|
title,
|
|
description,
|
|
head,
|
|
content,
|
|
assets
|
|
})
|
|
await fs.writeFile(htmlFileName, transformedHtml || html)
|
|
}
|
|
|
|
function resolvePageImports(
|
|
config: SiteConfig,
|
|
page: string,
|
|
result: Rollup.RollupOutput,
|
|
appChunk: Rollup.OutputChunk
|
|
) {
|
|
page = config.rewrites.inv[page] || page
|
|
// find the page's js chunk and inject script tags for its imports so that
|
|
// they start fetching as early as possible
|
|
let srcPath = path.resolve(config.srcDir, page)
|
|
try {
|
|
if (!config.vite?.resolve?.preserveSymlinks) {
|
|
srcPath = fs.realpathSync(srcPath)
|
|
}
|
|
} catch (e) {
|
|
// if the page is a virtual page generated by a dynamic route this would
|
|
// fail, which is expected
|
|
}
|
|
srcPath = normalizePath(srcPath)
|
|
const pageChunk = result.output.find(
|
|
(chunk) => chunk.type === 'chunk' && chunk.facadeModuleId === srcPath
|
|
) as Rollup.OutputChunk
|
|
return [
|
|
...appChunk.imports,
|
|
...appChunk.dynamicImports,
|
|
...pageChunk.imports,
|
|
...pageChunk.dynamicImports
|
|
]
|
|
}
|
|
|
|
async function renderHead(head: HeadConfig[]): Promise<string> {
|
|
const tags = await Promise.all(
|
|
head.map(async ([tag, attrs = {}, innerHTML = '']) => {
|
|
const openTag = `<${tag}${renderAttrs(attrs)}>`
|
|
if (tag !== 'link' && tag !== 'meta') {
|
|
if (
|
|
tag === 'script' &&
|
|
(attrs.type === undefined || attrs.type.includes('javascript'))
|
|
) {
|
|
innerHTML = (
|
|
await transformWithEsbuild(innerHTML, 'inline-script.js', {
|
|
minify: true
|
|
})
|
|
).code.trim()
|
|
}
|
|
return `${openTag}${innerHTML}</${tag}>`
|
|
} else {
|
|
return openTag
|
|
}
|
|
})
|
|
)
|
|
return tags.join('\n ')
|
|
}
|
|
|
|
function renderAttrs(attrs: Record<string, string>): string {
|
|
return Object.keys(attrs)
|
|
.map((key) => {
|
|
if (isBooleanAttr(key)) return ` ${key}`
|
|
return ` ${key}="${escapeHtml(attrs[key] as string)}"`
|
|
})
|
|
.join('')
|
|
}
|
|
|
|
function filterOutHeadDescription(head: HeadConfig[] = []) {
|
|
return head.filter(([type, attrs]) => {
|
|
return !(type === 'meta' && attrs?.name === 'description')
|
|
})
|
|
}
|
|
|
|
function isDescriptionOverridden(head: HeadConfig[] = []) {
|
|
return head.some(([type, attrs]) => {
|
|
return type === 'meta' && attrs?.name === 'description'
|
|
})
|
|
}
|
|
|
|
function isMetaViewportOverridden(head: HeadConfig[] = []) {
|
|
return head.some(([type, attrs]) => {
|
|
return type === 'meta' && attrs?.name === 'viewport'
|
|
})
|
|
}
|