From b4ff6a74f3de3c50cc9823d0d47c1f0d80448a5f Mon Sep 17 00:00:00 2001 From: meteorlxy <meteor.lxy@foxmail.com> Date: Wed, 11 Sep 2024 00:57:03 +0800 Subject: [PATCH 01/11] refactor: remove temp page files and load page component via bundler --- .../components/ComponentForMarkdownImport.vue | 5 - .../ComponentForMarkdownImportBar.vue | 5 + .../ComponentForMarkdownImportFoo.vue | 5 + .../markdowns/dangling-markdown-file.md | 5 + e2e/docs/README.md | 2 + e2e/docs/markdown/vue-components.md | 11 +- e2e/tests/markdown/vue-components.spec.ts | 7 +- packages/bundler-vite/src/plugins/index.ts | 1 + .../src/plugins/vuepressMarkdownPlugin.ts | 53 +++ .../src/plugins/vuepressVuePlugin.ts | 1 + .../bundler-vite/src/resolveViteConfig.ts | 2 + packages/bundler-webpack/package.json | 1 + .../src/build/createClientConfig.ts | 12 - .../src/build/createServerConfig.ts | 15 - .../src/config/createBaseConfig.ts | 2 +- .../src/config/handleModule.ts | 5 +- .../src/config/handleModuleVue.ts | 62 +++- .../src/loaders/vuepressMarkdownLoader.cts | 3 + .../src/loaders/vuepressMarkdownLoader.ts | 37 +++ packages/bundler-webpack/tsup.config.ts | 1 + packages/core/src/app/appInit.ts | 4 +- .../src/app/prepare/preparePageComponent.ts | 12 +- packages/core/src/app/resolveAppPages.ts | 11 +- packages/core/src/page/createPage.ts | 2 + .../core/src/page/resolvePageComponentInfo.ts | 13 +- packages/core/src/types/app/app.ts | 7 + .../core/tests/app/resolveAppPages.spec.ts | 8 +- .../page/resolvePageComponentInfo.spec.ts | 18 +- .../src/plugins/assetsPlugin/assetsPlugin.ts | 17 +- .../src/plugins/assetsPlugin/resolveLink.ts | 24 +- .../tests/plugins/assetsPlugin.spec.ts | 313 ++++-------------- 31 files changed, 319 insertions(+), 345 deletions(-) delete mode 100644 e2e/docs/.vuepress/components/ComponentForMarkdownImport.vue create mode 100644 e2e/docs/.vuepress/components/ComponentForMarkdownImportBar.vue create mode 100644 e2e/docs/.vuepress/components/ComponentForMarkdownImportFoo.vue create mode 100644 e2e/docs/.vuepress/markdowns/dangling-markdown-file.md create mode 100644 packages/bundler-vite/src/plugins/vuepressMarkdownPlugin.ts create mode 100644 packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.cts create mode 100644 packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.ts diff --git a/e2e/docs/.vuepress/components/ComponentForMarkdownImport.vue b/e2e/docs/.vuepress/components/ComponentForMarkdownImport.vue deleted file mode 100644 index 7108c4a426..0000000000 --- a/e2e/docs/.vuepress/components/ComponentForMarkdownImport.vue +++ /dev/null @@ -1,5 +0,0 @@ -<template> - <div class="component-for-markdown-import"> - <p>component for markdown import</p> - </div> -</template> diff --git a/e2e/docs/.vuepress/components/ComponentForMarkdownImportBar.vue b/e2e/docs/.vuepress/components/ComponentForMarkdownImportBar.vue new file mode 100644 index 0000000000..0d9fa2f2f6 --- /dev/null +++ b/e2e/docs/.vuepress/components/ComponentForMarkdownImportBar.vue @@ -0,0 +1,5 @@ +<template> + <div class="component-for-markdown-import-bar"> + <p>component for markdown import bar</p> + </div> +</template> diff --git a/e2e/docs/.vuepress/components/ComponentForMarkdownImportFoo.vue b/e2e/docs/.vuepress/components/ComponentForMarkdownImportFoo.vue new file mode 100644 index 0000000000..3905e34a83 --- /dev/null +++ b/e2e/docs/.vuepress/components/ComponentForMarkdownImportFoo.vue @@ -0,0 +1,5 @@ +<template> + <div class="component-for-markdown-import-foo"> + <p>component for markdown import foo</p> + </div> +</template> diff --git a/e2e/docs/.vuepress/markdowns/dangling-markdown-file.md b/e2e/docs/.vuepress/markdowns/dangling-markdown-file.md new file mode 100644 index 0000000000..5bf006ada9 --- /dev/null +++ b/e2e/docs/.vuepress/markdowns/dangling-markdown-file.md @@ -0,0 +1,5 @@ +<div class="dangling-markdown-file"> + +dangling markdown file + +</div> diff --git a/e2e/docs/README.md b/e2e/docs/README.md index eb63f0ccbf..b3586d97fe 100644 --- a/e2e/docs/README.md +++ b/e2e/docs/README.md @@ -1,3 +1,5 @@ foo ## Home H2 + +demo diff --git a/e2e/docs/markdown/vue-components.md b/e2e/docs/markdown/vue-components.md index c038c08795..f38b703d0b 100644 --- a/e2e/docs/markdown/vue-components.md +++ b/e2e/docs/markdown/vue-components.md @@ -1,8 +1,13 @@ <ComponentForMarkdownGlobal /> -<ComponentForMarkdownImport /> +<ComponentForMarkdownImportFoo /> + +<ComponentForMarkdownImportBar /> <script setup> -// TODO: relative path import? -import ComponentForMarkdownImport from '@source/.vuepress/components/ComponentForMarkdownImport.vue'; +// import via alias +import ComponentForMarkdownImportFoo from '@source/.vuepress/components/ComponentForMarkdownImportFoo.vue'; + +// import via relative path +import ComponentForMarkdownImportBar from '../.vuepress/components/ComponentForMarkdownImportBar.vue'; </script> diff --git a/e2e/tests/markdown/vue-components.spec.ts b/e2e/tests/markdown/vue-components.spec.ts index f33b1dbc68..1f3f1818d6 100644 --- a/e2e/tests/markdown/vue-components.spec.ts +++ b/e2e/tests/markdown/vue-components.spec.ts @@ -6,7 +6,10 @@ test('should render vue components correctly', async ({ page }) => { await expect(page.locator('.component-for-markdown-global p')).toHaveText( 'component for markdown global', ) - await expect(page.locator('.component-for-markdown-import p')).toHaveText( - 'component for markdown import', + await expect(page.locator('.component-for-markdown-import-foo p')).toHaveText( + 'component for markdown import foo', + ) + await expect(page.locator('.component-for-markdown-import-bar p')).toHaveText( + 'component for markdown import bar', ) }) diff --git a/packages/bundler-vite/src/plugins/index.ts b/packages/bundler-vite/src/plugins/index.ts index 0268dc0dba..9ff20c8972 100644 --- a/packages/bundler-vite/src/plugins/index.ts +++ b/packages/bundler-vite/src/plugins/index.ts @@ -1,5 +1,6 @@ export * from './vuepressBuildPlugin.js' export * from './vuepressConfigPlugin.js' export * from './vuepressDevPlugin.js' +export * from './vuepressMarkdownPlugin.js' export * from './vuepressUserConfigPlugin.js' export * from './vuepressVuePlugin.js' diff --git a/packages/bundler-vite/src/plugins/vuepressMarkdownPlugin.ts b/packages/bundler-vite/src/plugins/vuepressMarkdownPlugin.ts new file mode 100644 index 0000000000..eb072f840c --- /dev/null +++ b/packages/bundler-vite/src/plugins/vuepressMarkdownPlugin.ts @@ -0,0 +1,53 @@ +import type { App } from '@vuepress/core' +import { parsePageContent, renderPageSfcBlocksToVue } from '@vuepress/core' +import { path } from '@vuepress/utils' +import type { Plugin } from 'vite' + +/** + * Handle markdown transformation + */ +export const vuepressMarkdownPlugin = ({ app }: { app: App }): Plugin => ({ + name: 'vuepress:markdown', + + enforce: 'pre', + + transform(code, id) { + if (!id.endsWith('.md')) return + + // get the matched page by file path (id) + const page = app.pagesMap[id] + + // if the page content is not changed, render it to vue component directly + if (page?.content === code) { + return renderPageSfcBlocksToVue(page.sfcBlocks) + } + + // parse the markdown content to sfc blocks and render it to vue component + const { sfcBlocks } = parsePageContent({ + app, + content: code, + filePath: id, + filePathRelative: path.relative(app.dir.source(), id), + options: {}, + }) + return renderPageSfcBlocksToVue(sfcBlocks) + }, + + async handleHotUpdate(ctx) { + if (!ctx.file.endsWith('.md')) return + + // read the source code + const code = await ctx.read() + + // parse the content to sfc blocks + const { sfcBlocks } = parsePageContent({ + app, + content: code, + filePath: ctx.file, + filePathRelative: path.relative(app.dir.source(), ctx.file), + options: {}, + }) + + ctx.read = () => renderPageSfcBlocksToVue(sfcBlocks) + }, +}) diff --git a/packages/bundler-vite/src/plugins/vuepressVuePlugin.ts b/packages/bundler-vite/src/plugins/vuepressVuePlugin.ts index d85656c731..5688e4c2fb 100644 --- a/packages/bundler-vite/src/plugins/vuepressVuePlugin.ts +++ b/packages/bundler-vite/src/plugins/vuepressVuePlugin.ts @@ -11,5 +11,6 @@ export const vuepressVuePlugin = ({ options: ViteBundlerOptions }): Plugin => vuePlugin({ + include: [/\.vue$/, /\.md$/], ...options.vuePluginOptions, }) diff --git a/packages/bundler-vite/src/resolveViteConfig.ts b/packages/bundler-vite/src/resolveViteConfig.ts index 6fafcec449..f950836f92 100644 --- a/packages/bundler-vite/src/resolveViteConfig.ts +++ b/packages/bundler-vite/src/resolveViteConfig.ts @@ -5,6 +5,7 @@ import { vuepressBuildPlugin, vuepressConfigPlugin, vuepressDevPlugin, + vuepressMarkdownPlugin, vuepressUserConfigPlugin, vuepressVuePlugin, } from './plugins/index.js' @@ -31,6 +32,7 @@ export const resolveViteConfig = ({ }, plugins: [ vuepressConfigPlugin({ app, isBuild, isServer }), + vuepressMarkdownPlugin({ app }), vuepressDevPlugin({ app }), vuepressBuildPlugin({ isServer }), vuepressVuePlugin({ options }), diff --git a/packages/bundler-webpack/package.json b/packages/bundler-webpack/package.json index 28d32a8f12..893252e963 100644 --- a/packages/bundler-webpack/package.json +++ b/packages/bundler-webpack/package.json @@ -20,6 +20,7 @@ "author": "meteorlxy", "type": "module", "imports": { + "#vuepress-markdown-loader": "./dist/vuepress-markdown-loader.cjs", "#vuepress-ssr-loader": "./dist/vuepress-ssr-loader.cjs" }, "exports": { diff --git a/packages/bundler-webpack/src/build/createClientConfig.ts b/packages/bundler-webpack/src/build/createClientConfig.ts index 6266b9bda1..7f0ae415de 100644 --- a/packages/bundler-webpack/src/build/createClientConfig.ts +++ b/packages/bundler-webpack/src/build/createClientConfig.ts @@ -1,4 +1,3 @@ -import { createRequire } from 'node:module' import type { App } from '@vuepress/core' import { fs } from '@vuepress/utils' import CopyWebpackPlugin from 'copy-webpack-plugin' @@ -10,8 +9,6 @@ import { createClientBaseConfig } from '../config/index.js' import type { WebpackBundlerOptions } from '../types.js' import { createClientPlugin } from './createClientPlugin.js' -const require = createRequire(import.meta.url) - /** * Filename of the client manifest file that generated by client plugin */ @@ -27,15 +24,6 @@ export const createClientConfig = async ( isBuild: true, }) - // use internal vuepress-ssr-loader to handle SSR dependencies - config.module - .rule('vue') - .test(/\.vue$/) - .use('vuepress-ssr-loader') - .before('vue-loader') - .loader(require.resolve('#vuepress-ssr-loader')) - .end() - // vuepress client plugin, handle client assets info for ssr config .plugin('vuepress-client') diff --git a/packages/bundler-webpack/src/build/createServerConfig.ts b/packages/bundler-webpack/src/build/createServerConfig.ts index d21b75b034..600141751d 100644 --- a/packages/bundler-webpack/src/build/createServerConfig.ts +++ b/packages/bundler-webpack/src/build/createServerConfig.ts @@ -1,11 +1,8 @@ -import { createRequire } from 'node:module' import type { App } from '@vuepress/core' import type Config from 'webpack-5-chain' import { createBaseConfig } from '../config/index.js' import type { WebpackBundlerOptions } from '../types.js' -const require = createRequire(import.meta.url) - export const createServerConfig = async ( app: App, options: WebpackBundlerOptions, @@ -43,17 +40,5 @@ export const createServerConfig = async ( // do not need to minimize server bundle config.optimization.minimize(false) - // use internal vuepress-ssr-loader to handle SSR dependencies - config.module - .rule('vue') - .test(/\.vue$/) - .use('vuepress-ssr-loader') - .before('vue-loader') - .loader(require.resolve('#vuepress-ssr-loader')) - .options({ - app, - }) - .end() - return config } diff --git a/packages/bundler-webpack/src/config/createBaseConfig.ts b/packages/bundler-webpack/src/config/createBaseConfig.ts index 0b6e0accc3..b6d8bec3b0 100644 --- a/packages/bundler-webpack/src/config/createBaseConfig.ts +++ b/packages/bundler-webpack/src/config/createBaseConfig.ts @@ -52,7 +52,7 @@ export const createBaseConfig = async ({ /** * module */ - handleModule({ options, config, isBuild, isServer }) + handleModule({ app, options, config, isBuild, isServer }) /** * plugins diff --git a/packages/bundler-webpack/src/config/handleModule.ts b/packages/bundler-webpack/src/config/handleModule.ts index be33ca6a73..14d8a57b0c 100644 --- a/packages/bundler-webpack/src/config/handleModule.ts +++ b/packages/bundler-webpack/src/config/handleModule.ts @@ -1,3 +1,4 @@ +import type { App } from '@vuepress/core' import type Config from 'webpack-5-chain' import type { WebpackBundlerOptions } from '../types.js' import { handleModuleAssets } from './handleModuleAssets.js' @@ -11,11 +12,13 @@ import { handleModuleVue } from './handleModuleVue.js' * Set webpack module */ export const handleModule = ({ + app, options, config, isBuild, isServer, }: { + app: App options: WebpackBundlerOptions config: Config isBuild: boolean @@ -27,7 +30,7 @@ export const handleModule = ({ ) // vue files - handleModuleVue({ options, config, isServer }) + handleModuleVue({ app, options, config, isBuild, isServer }) // pug files, for templates handleModulePug({ config }) diff --git a/packages/bundler-webpack/src/config/handleModuleVue.ts b/packages/bundler-webpack/src/config/handleModuleVue.ts index 335159cc1b..4065ff16a7 100644 --- a/packages/bundler-webpack/src/config/handleModuleVue.ts +++ b/packages/bundler-webpack/src/config/handleModuleVue.ts @@ -1,7 +1,9 @@ import { createRequire } from 'node:module' +import type { App } from '@vuepress/core' import type { VueLoaderOptions } from 'vue-loader' import { VueLoaderPlugin } from 'vue-loader' import type Config from 'webpack-5-chain' +import type { VuepressMarkdownLoaderOptions } from '../loaders/vuepressMarkdownLoader' import type { WebpackBundlerOptions } from '../types.js' const require = createRequire(import.meta.url) @@ -10,26 +12,62 @@ const require = createRequire(import.meta.url) * Set webpack module to handle vue files */ export const handleModuleVue = ({ + app, options, config, + isBuild, isServer, }: { + app: App options: WebpackBundlerOptions config: Config + isBuild: boolean isServer: boolean }): void => { - // .vue files - config.module - .rule('vue') - .test(/\.vue$/) - // use vue-loader - .use('vue-loader') - .loader(require.resolve('vue-loader')) - .options({ - ...options.vue, - isServerBuild: isServer, - } as VueLoaderOptions) - .end() + const applyVuePipeline = ({ + rule, + isMd, + }: { + rule: Config.Rule + isMd: boolean + }): void => { + // use internal vuepress-ssr-loader to handle SSR dependencies + if (isBuild) { + rule + .use('vuepress-ssr-loader') + .loader(require.resolve('#vuepress-ssr-loader')) + .end() + } + + // use official vue-loader + rule + .use('vue-loader') + .loader(require.resolve('vue-loader')) + .options({ + ...options.vue, + isServerBuild: isServer, + } satisfies VueLoaderOptions) + .end() + + // use internal vuepress-markdown-loader to handle markdown files + if (isMd) { + rule + .use('vuepress-markdown-loader') + .loader(require.resolve('#vuepress-markdown-loader')) + .options({ app } satisfies VuepressMarkdownLoaderOptions) + .end() + } + } + + applyVuePipeline({ + rule: config.module.rule('md').test(/\.md$/), + isMd: true, + }) + + applyVuePipeline({ + rule: config.module.rule('vue').test(/\.vue$/), + isMd: false, + }) // use vue-loader plugin config.plugin('vue-loader').use(VueLoaderPlugin) diff --git a/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.cts b/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.cts new file mode 100644 index 0000000000..f4b3001431 --- /dev/null +++ b/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.cts @@ -0,0 +1,3 @@ +const loader = require('./vuepressMarkdownLoader.js') + +module.exports = loader.vuepressMarkdownLoader diff --git a/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.ts b/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.ts new file mode 100644 index 0000000000..d9ca435d14 --- /dev/null +++ b/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.ts @@ -0,0 +1,37 @@ +import type { App } from '@vuepress/core' +import type { LoaderDefinitionFunction } from 'webpack' + +export interface VuepressMarkdownLoaderOptions { + app: App +} + +/** + * A webpack loader to transform markdown content to vue component + */ +export const vuepressMarkdownLoader: LoaderDefinitionFunction<VuepressMarkdownLoaderOptions> = + async function vuepressMarkdownLoader(source) { + // import esm dependencies + const [{ parsePageContent, renderPageSfcBlocksToVue }, { path }] = + await Promise.all([import('@vuepress/core'), import('@vuepress/utils')]) + + // get app instance from loader options + const { app } = this.getOptions() + + // get the matched page by file path + const page = app.pagesMap[this.resourcePath] + + // if the page content is not changed, render it to vue component directly + if (page?.content === source) { + return renderPageSfcBlocksToVue(page.sfcBlocks) + } + + // parse the markdown content to sfc blocks and render it to vue component + const { sfcBlocks } = parsePageContent({ + app, + content: source, + filePath: this.resourcePath, + filePathRelative: path.relative(app.dir.source(), this.resourcePath), + options: {}, + }) + return renderPageSfcBlocksToVue(sfcBlocks) + } diff --git a/packages/bundler-webpack/tsup.config.ts b/packages/bundler-webpack/tsup.config.ts index 4899bc89aa..9f8379c060 100644 --- a/packages/bundler-webpack/tsup.config.ts +++ b/packages/bundler-webpack/tsup.config.ts @@ -18,6 +18,7 @@ export default defineConfig([ { ...shared, entry: { + 'vuepress-markdown-loader': './src/loaders/vuepressMarkdownLoader.cts', 'vuepress-ssr-loader': './src/loaders/vuepressSsrLoader.cts', }, format: ['cjs'], diff --git a/packages/core/src/app/appInit.ts b/packages/core/src/app/appInit.ts index 1db287fd28..ac485cfe65 100644 --- a/packages/core/src/app/appInit.ts +++ b/packages/core/src/app/appInit.ts @@ -22,7 +22,9 @@ export const appInit = async (app: App): Promise<void> => { app.markdown = await resolveAppMarkdown(app) // create pages - app.pages = await resolveAppPages(app) + const { pages, pagesMap } = await resolveAppPages(app) + app.pages = pages + app.pagesMap = pagesMap // plugin hook: onInitialized await app.pluginApi.hooks.onInitialized.process(app) diff --git a/packages/core/src/app/prepare/preparePageComponent.ts b/packages/core/src/app/prepare/preparePageComponent.ts index 569d63697a..e88d6123d5 100644 --- a/packages/core/src/app/prepare/preparePageComponent.ts +++ b/packages/core/src/app/prepare/preparePageComponent.ts @@ -2,14 +2,16 @@ import { renderPageSfcBlocksToVue } from '../../page/index.js' import type { App, Page } from '../../types/index.js' /** - * Generate page component temp file of a single page + * Generate page component temp file if the page does not have a source file */ export const preparePageComponent = async ( app: App, page: Page, ): Promise<void> => { - await app.writeTemp( - page.componentFilePathRelative, - renderPageSfcBlocksToVue(page.sfcBlocks), - ) + if (page.filePath === null) { + await app.writeTemp( + page.componentFilePathRelative, + renderPageSfcBlocksToVue(page.sfcBlocks), + ) + } } diff --git a/packages/core/src/app/resolveAppPages.ts b/packages/core/src/app/resolveAppPages.ts index d8cd00bd41..33c280867b 100644 --- a/packages/core/src/app/resolveAppPages.ts +++ b/packages/core/src/app/resolveAppPages.ts @@ -7,7 +7,9 @@ const log = debug('vuepress:core/app') /** * Resolve pages for vuepress app */ -export const resolveAppPages = async (app: App): Promise<Page[]> => { +export const resolveAppPages = async ( + app: App, +): Promise<Pick<App, 'pages' | 'pagesMap'>> => { log('resolveAppPages start') // resolve page absolute file paths according to the page patterns @@ -19,9 +21,14 @@ export const resolveAppPages = async (app: App): Promise<Page[]> => { let hasNotFoundPage = false as boolean // create pages from files + const pagesMap: Record<string, Page> = {} const pages = await Promise.all( pageFilePaths.map(async (filePath) => { const page = await createPage(app, { filePath }) + // if there is a source file for the page, set it to the pages map + if (page.filePath) { + pagesMap[page.filePath] = page + } // if there is a 404 page, set the default layout to NotFound if (page.path === '/404.html') { page.frontmatter.layout ??= 'NotFound' @@ -44,5 +51,5 @@ export const resolveAppPages = async (app: App): Promise<Page[]> => { log('resolveAppPages finish') - return pages + return { pages, pagesMap } } diff --git a/packages/core/src/page/createPage.ts b/packages/core/src/page/createPage.ts index c000873c84..0c48399162 100644 --- a/packages/core/src/page/createPage.ts +++ b/packages/core/src/page/createPage.ts @@ -85,6 +85,8 @@ export const createPage = async ( const { componentFilePath, componentFilePathRelative } = resolvePageComponentInfo({ app, + filePath, + filePathRelative, htmlFilePathRelative, }) diff --git a/packages/core/src/page/resolvePageComponentInfo.ts b/packages/core/src/page/resolvePageComponentInfo.ts index b1ee312cc4..2105bdb187 100644 --- a/packages/core/src/page/resolvePageComponentInfo.ts +++ b/packages/core/src/page/resolvePageComponentInfo.ts @@ -6,15 +6,26 @@ import type { App } from '../types/index.js' */ export const resolvePageComponentInfo = ({ app, + filePath, + filePathRelative, htmlFilePathRelative, }: { app: App + filePath: string | null + filePathRelative: string | null htmlFilePathRelative: string }): { componentFilePath: string componentFilePathRelative: string } => { - // resolve component file path + // if there is a source file for the page, use it as the component file + if (filePath && filePathRelative) { + return { + componentFilePath: filePath, + componentFilePathRelative: filePathRelative, + } + } + // otherwise, generate a component file for the page const componentFilePathRelative = path.join( 'pages', `${htmlFilePathRelative}.vue`, diff --git a/packages/core/src/types/app/app.ts b/packages/core/src/types/app/app.ts index e943950ec9..3bab071811 100644 --- a/packages/core/src/types/app/app.ts +++ b/packages/core/src/types/app/app.ts @@ -79,6 +79,13 @@ export interface App { * Only available after initialization */ pages: Page[] + + /** + * Page source filepath map. + * + * Only available after initialization + */ + pagesMap: Record<string, Page | undefined> } /** diff --git a/packages/core/tests/app/resolveAppPages.spec.ts b/packages/core/tests/app/resolveAppPages.spec.ts index f394888238..a0366c1ab3 100644 --- a/packages/core/tests/app/resolveAppPages.spec.ts +++ b/packages/core/tests/app/resolveAppPages.spec.ts @@ -13,7 +13,7 @@ describe('core > app > resolveAppPages', () => { }) app.markdown = createMarkdown() - const pages = await resolveAppPages(app) + const { pages } = await resolveAppPages(app) const fooPage = pages.find((page) => page.path === '/foo.html') const barPage = pages.find((page) => page.path === '/bar.html') const notFoundPage = pages.find((page) => page.path === '/404.html') @@ -33,7 +33,7 @@ describe('core > app > resolveAppPages', () => { }) app.markdown = createMarkdown() - const pages = await resolveAppPages(app) + const { pages } = await resolveAppPages(app) const fooPage = pages.find((page) => page.path === '/foo.html') const barPage = pages.find((page) => page.path === '/bar.html') const notFoundPage = pages.find((page) => page.path === '/404.html') @@ -61,7 +61,7 @@ describe('core > app > resolveAppPages', () => { app.pluginApi.registerHooks() app.markdown = createMarkdown() - const pages = await resolveAppPages(app) + const { pages } = await resolveAppPages(app) pages.forEach((page) => { expect(page.frontmatter.foo).toBe('bar') @@ -84,7 +84,7 @@ describe('core > app > resolveAppPages', () => { app.pluginApi.registerHooks() app.markdown = createMarkdown() - const pages = await resolveAppPages(app) + const { pages } = await resolveAppPages(app) pages.forEach((page) => { expect(page.frontmatter.foo).toBe('baz') diff --git a/packages/core/tests/page/resolvePageComponentInfo.spec.ts b/packages/core/tests/page/resolvePageComponentInfo.spec.ts index 7feabdd495..c8a009e1ea 100644 --- a/packages/core/tests/page/resolvePageComponentInfo.spec.ts +++ b/packages/core/tests/page/resolvePageComponentInfo.spec.ts @@ -9,9 +9,11 @@ const app = createBaseApp({ bundler: {} as Bundler, }) -it('should resolve page component info correctly', () => { +it('should resolve page component info correctly without source file path', () => { const resolved = resolvePageComponentInfo({ app, + filePath: null, + filePathRelative: null, htmlFilePathRelative: 'foo.html', }) @@ -20,3 +22,17 @@ it('should resolve page component info correctly', () => { componentFilePathRelative: 'pages/foo.html.vue', }) }) + +it('should resolve page component info correctly with source file path', () => { + const resolved = resolvePageComponentInfo({ + app, + filePath: app.dir.source('foo.md'), + filePathRelative: 'foo.md', + htmlFilePathRelative: 'foo.html', + }) + + expect(resolved).toEqual({ + componentFilePath: app.dir.source('foo.md'), + componentFilePathRelative: 'foo.md', + }) +}) diff --git a/packages/markdown/src/plugins/assetsPlugin/assetsPlugin.ts b/packages/markdown/src/plugins/assetsPlugin/assetsPlugin.ts index 1434888055..2524963372 100644 --- a/packages/markdown/src/plugins/assetsPlugin/assetsPlugin.ts +++ b/packages/markdown/src/plugins/assetsPlugin/assetsPlugin.ts @@ -8,11 +8,6 @@ export interface AssetsPluginOptions { * Whether to prepend base to absolute path */ absolutePathPrependBase?: boolean - - /** - * Prefix to add to relative assets links - */ - relativePathPrefix?: string } /** @@ -20,10 +15,7 @@ export interface AssetsPluginOptions { */ export const assetsPlugin: PluginWithOptions<AssetsPluginOptions> = ( md, - { - absolutePathPrependBase = false, - relativePathPrefix = '@source', - }: AssetsPluginOptions = {}, + { absolutePathPrependBase = false }: AssetsPluginOptions = {}, ) => { // wrap raw image renderer rule const rawImageRule = md.renderer.rules.image! @@ -35,10 +27,7 @@ export const assetsPlugin: PluginWithOptions<AssetsPluginOptions> = ( if (link) { // replace the original link with resolved link - token.attrSet( - 'src', - resolveLink(link, { env, absolutePathPrependBase, relativePathPrefix }), - ) + token.attrSet('src', resolveLink(link, { env, absolutePathPrependBase })) } return rawImageRule(tokens, idx, options, env, self) @@ -57,7 +46,6 @@ export const assetsPlugin: PluginWithOptions<AssetsPluginOptions> = ( `${prefix}${quote}${resolveLink(src.trim(), { env, absolutePathPrependBase, - relativePathPrefix, strict: true, })}${quote}`, ) @@ -74,7 +62,6 @@ export const assetsPlugin: PluginWithOptions<AssetsPluginOptions> = ( `${resolveLink(url.trim(), { env, absolutePathPrependBase, - relativePathPrefix, strict: true, })}${descriptor.replace(/[ \n]+/g, ' ').trimEnd()}`, ), diff --git a/packages/markdown/src/plugins/assetsPlugin/resolveLink.ts b/packages/markdown/src/plugins/assetsPlugin/resolveLink.ts index b9277ed855..a1081fe9ed 100644 --- a/packages/markdown/src/plugins/assetsPlugin/resolveLink.ts +++ b/packages/markdown/src/plugins/assetsPlugin/resolveLink.ts @@ -5,18 +5,12 @@ import type { MarkdownEnv } from '../../types.js' interface ResolveLinkOptions { env: MarkdownEnv absolutePathPrependBase?: boolean - relativePathPrefix: string strict?: boolean } export const resolveLink = ( link: string, - { - env, - absolutePathPrependBase = false, - relativePathPrefix, - strict = false, - }: ResolveLinkOptions, + { env, absolutePathPrependBase = false }: ResolveLinkOptions, ): string => { // do not resolve data uri if (link.startsWith('data:')) return link @@ -24,22 +18,6 @@ export const resolveLink = ( // decode link to ensure bundler can find the file correctly let resolvedLink = decode(link) - // check if the link is relative path - const isRelativePath = strict - ? // in strict mode, only link that starts with `./` or `../` is considered as relative path - /^\.{1,2}\//.test(link) - : // in non-strict mode, link that does not start with `/` and does not have protocol is considered as relative path - !link.startsWith('/') && !/[A-z]+:\/\//.test(link) - - // if the link is relative path, and the `env.filePathRelative` exists - // add `@source` alias to the link - if (isRelativePath && env.filePathRelative) { - resolvedLink = `${relativePathPrefix}/${path.join( - path.dirname(env.filePathRelative), - resolvedLink, - )}` - } - // prepend base to absolute path if needed if (absolutePathPrependBase && env.base && link.startsWith('/')) { resolvedLink = path.join(env.base, resolvedLink) diff --git a/packages/markdown/tests/plugins/assetsPlugin.spec.ts b/packages/markdown/tests/plugins/assetsPlugin.spec.ts index af149f84d9..34dca2e41d 100644 --- a/packages/markdown/tests/plugins/assetsPlugin.spec.ts +++ b/packages/markdown/tests/plugins/assetsPlugin.spec.ts @@ -49,37 +49,36 @@ describe('@vuepress/markdown > plugins > assetsPlugin', () => { md: MarkdownIt().use(assetsPlugin), env: { base: '/base/', - filePathRelative: 'sub/foo.md', }, expected: [ // relative paths - '<img src="@source/sub/foo.png" alt="foo">', - '<img src="@source/sub/foo.png" alt="foo2">', - '<img src="@source/sub/foo/bar.png" alt="foo-bar">', - '<img src="@source/sub/foo/bar.png" alt="foo-bar2">', - '<img src="@source/baz.png" alt="baz">', - '<img src="@source/../out.png" alt="out">', - '<img src="@source/sub/汉字.png" alt="汉字">', - '<img src="@source/sub/100%.png" alt="100%">', + '<img src="./foo.png" alt="foo">', + '<img src="../sub/foo.png" alt="foo2">', + '<img src="./foo/bar.png" alt="foo-bar">', + '<img src="../sub/foo/bar.png" alt="foo-bar2">', + '<img src="../baz.png" alt="baz">', + '<img src="../../out.png" alt="out">', + '<img src="./汉字.png" alt="汉字">', + '<img src="./100%.png" alt="100%">', // absolute paths '<img src="/absolute.png" alt="absolute">', '<img src="/foo/absolute.png" alt="absolute-foo">', // no-prefix paths - '<img src="@source/sub/no-prefix.png" alt="no-prefix">', - '<img src="@source/sub/foo/no-prefix.png" alt="no-prefix-foo">', - '<img src="@source/sub/@alias/foo.png" alt="alias">', - '<img src="@source/sub/@alias/汉字.png" alt="汉字">', - '<img src="@source/sub/@alias/100%.png" alt="100%">', - '<img src="@source/sub/~@alias/foo.png" alt="~alias">', - '<img src="@source/sub/~@alias/汉字.png" alt="~汉字">', - '<img src="@source/sub/~@alias/100%.png" alt="~100%">', + '<img src="no-prefix.png" alt="no-prefix">', + '<img src="foo/no-prefix.png" alt="no-prefix-foo">', + '<img src="@alias/foo.png" alt="alias">', + '<img src="@alias/汉字.png" alt="汉字">', + '<img src="@alias/100%.png" alt="100%">', + '<img src="~@alias/foo.png" alt="~alias">', + '<img src="~@alias/汉字.png" alt="~汉字">', + '<img src="~@alias/100%.png" alt="~100%">', // keep as is '<img src="http://foobar.com/icon.png" alt="url">', '<img src="" alt="empty">', // invalid paths - '<img src="@source/sub/.../invalid.png" alt="invalid">', - '<img src="@source/sub/.../汉字.png" alt="汉字">', - '<img src="@source/sub/.../100%.png" alt="100%">', + '<img src=".../invalid.png" alt="invalid">', + '<img src=".../汉字.png" alt="汉字">', + '<img src=".../100%.png" alt="100%">', // data uri '<img src="" alt="data-uri">', ], @@ -91,87 +90,7 @@ describe('@vuepress/markdown > plugins > assetsPlugin', () => { }), env: { base: '/base/', - filePathRelative: 'sub/foo.md', }, - expected: [ - // relative paths - '<img src="@source/sub/foo.png" alt="foo">', - '<img src="@source/sub/foo.png" alt="foo2">', - '<img src="@source/sub/foo/bar.png" alt="foo-bar">', - '<img src="@source/sub/foo/bar.png" alt="foo-bar2">', - '<img src="@source/baz.png" alt="baz">', - '<img src="@source/../out.png" alt="out">', - '<img src="@source/sub/汉字.png" alt="汉字">', - '<img src="@source/sub/100%.png" alt="100%">', - // absolute paths - '<img src="/base/absolute.png" alt="absolute">', - '<img src="/base/foo/absolute.png" alt="absolute-foo">', - // no-prefix paths - '<img src="@source/sub/no-prefix.png" alt="no-prefix">', - '<img src="@source/sub/foo/no-prefix.png" alt="no-prefix-foo">', - '<img src="@source/sub/@alias/foo.png" alt="alias">', - '<img src="@source/sub/@alias/汉字.png" alt="汉字">', - '<img src="@source/sub/@alias/100%.png" alt="100%">', - '<img src="@source/sub/~@alias/foo.png" alt="~alias">', - '<img src="@source/sub/~@alias/汉字.png" alt="~汉字">', - '<img src="@source/sub/~@alias/100%.png" alt="~100%">', - // keep as is - '<img src="http://foobar.com/icon.png" alt="url">', - '<img src="" alt="empty">', - // invalid paths - '<img src="@source/sub/.../invalid.png" alt="invalid">', - '<img src="@source/sub/.../汉字.png" alt="汉字">', - '<img src="@source/sub/.../100%.png" alt="100%">', - // data uri - '<img src="" alt="data-uri">', - ], - }, - { - description: 'should respect `relativePathPrefix` option', - md: MarkdownIt().use(assetsPlugin, { - relativePathPrefix: '@foo', - }), - env: { - filePathRelative: 'sub/foo.md', - }, - expected: [ - // relative paths - '<img src="@foo/sub/foo.png" alt="foo">', - '<img src="@foo/sub/foo.png" alt="foo2">', - '<img src="@foo/sub/foo/bar.png" alt="foo-bar">', - '<img src="@foo/sub/foo/bar.png" alt="foo-bar2">', - '<img src="@foo/baz.png" alt="baz">', - '<img src="@foo/../out.png" alt="out">', - '<img src="@foo/sub/汉字.png" alt="汉字">', - '<img src="@foo/sub/100%.png" alt="100%">', - // absolute paths - '<img src="/absolute.png" alt="absolute">', - '<img src="/foo/absolute.png" alt="absolute-foo">', - // no-prefix paths - '<img src="@foo/sub/no-prefix.png" alt="no-prefix">', - '<img src="@foo/sub/foo/no-prefix.png" alt="no-prefix-foo">', - '<img src="@foo/sub/@alias/foo.png" alt="alias">', - '<img src="@foo/sub/@alias/汉字.png" alt="汉字">', - '<img src="@foo/sub/@alias/100%.png" alt="100%">', - '<img src="@foo/sub/~@alias/foo.png" alt="~alias">', - '<img src="@foo/sub/~@alias/汉字.png" alt="~汉字">', - '<img src="@foo/sub/~@alias/100%.png" alt="~100%">', - // keep as is - '<img src="http://foobar.com/icon.png" alt="url">', - '<img src="" alt="empty">', - // invalid paths - '<img src="@foo/sub/.../invalid.png" alt="invalid">', - '<img src="@foo/sub/.../汉字.png" alt="汉字">', - '<img src="@foo/sub/.../100%.png" alt="100%">', - // data uri - '<img src="" alt="data-uri">', - ], - }, - { - description: - 'should not handle relative paths if `env.filePathRelative` is not provided', - md: MarkdownIt().use(assetsPlugin), - env: {}, expected: [ // relative paths '<img src="./foo.png" alt="foo">', @@ -183,8 +102,8 @@ describe('@vuepress/markdown > plugins > assetsPlugin', () => { '<img src="./汉字.png" alt="汉字">', '<img src="./100%.png" alt="100%">', // absolute paths - '<img src="/absolute.png" alt="absolute">', - '<img src="/foo/absolute.png" alt="absolute-foo">', + '<img src="/base/absolute.png" alt="absolute">', + '<img src="/base/foo/absolute.png" alt="absolute-foo">', // no-prefix paths '<img src="no-prefix.png" alt="no-prefix">', '<img src="foo/no-prefix.png" alt="no-prefix-foo">', @@ -302,20 +221,20 @@ describe('@vuepress/markdown > plugins > assetsPlugin', () => { description: 'should handle assets link with default options', md: MarkdownIt({ html: true }).use(assetsPlugin), env: { - filePathRelative: 'sub/foo.md', + base: '/base/', }, expected: [ /* src */ // relative paths - '<img src="@source/sub/foo.png">', - '<img src="@source/sub/foo.png">', - '<img src="@source/sub/foo/bar.png">', - '<img src="@source/sub/foo/bar.png">', - '<img src="@source/baz.png">', - '<img src="@source/../out.png">', - '<img src="@source/sub/汉字.png">', - '<img src="@source/sub/100%.png">', - '<img alt="attrs" src="@source/sub/attrs.png" width="100px">', + '<img src="./foo.png">', + '<img src="../sub/foo.png">', + '<img src="./foo/bar.png">', + '<img src="../sub/foo/bar.png">', + '<img src="../baz.png">', + '<img src="../../out.png">', + '<img src="./汉字.png">', + '<img src="./100%.png">', + '<img alt="attrs" src="./attrs.png" width="100px">', // aliases '<img src="@alias/foo.png">', '<img src="@alias/汉字.png">', @@ -344,34 +263,35 @@ describe('@vuepress/markdown > plugins > assetsPlugin', () => { /* srcset */ // relative paths - '<img srcset="@source/sub/foo.png 1x, @source/sub/foo.png 2x, @source/sub/foo/bar.png 1024w, @source/sub/foo/bar.png 2048w, @source/baz.png 4096w, @source/../out.png">', - '<img srcset="@source/sub/汉字.png 1x, @source/sub/100%.png">', - '<img alt="attrs" srcset="@source/sub/attrs.png" width="100px">', + '<img srcset="./foo.png 1x, ../sub/foo.png 2x, ./foo/bar.png 1024w, ../sub/foo/bar.png 2048w, ../baz.png 4096w, ../../out.png">', + '<img srcset="./汉字.png 1x, ./100%.png">', + '<img alt="attrs" srcset="./attrs.png" width="100px">', // aliases '<img srcset="@alias/foo.png 1x, @alias/汉字.png 2x, @alias/100%.png 3x">', '<img alt="attrs" srcset="@alias/attrs.png 1024w" width="100px">', // webpack legacy aliases '<img srcset="~@alias/foo.png 1x, ~@alias/汉字.png 2x, ~@alias/100%.png 3x">', '<img alt="attrs" srcset="~@alias/attrs.png 1024w" width="100px">', - // keep as is + // absolute paths and no-prefix paths '<img srcset="/absolute.png 1x, no-prefix.png 2x, http://foobar.com/icon.png">', + // keep as is '<img srcset="">', '<img alt="attrs" srcset="attrs.png 1x, default.png" width="100px">', // invalid paths '<img srcset=".../invalid.png 1x, .../汉字.png 2x, .../100%.png 3x">', '<img alt="attrs" srcset=".../attrs.png 1x, .../default.png" width="100px">', // invalid srcset - '<img srcset="@source/invalid.png, @source/汉字.png, .../100%.png 3x">', + '<img srcset="../invalid.png, ../汉字.png, .../100%.png 3x">', /* both */ // relative paths - '<img srcset="@source/sub/foo.png 1x, @source/sub/foo.png 2x, @source/sub/foo/bar.png 1024w, @source/sub/foo/bar.png 2048w, @source/baz.png 4096w, @source/../out.png 3x" src="@source/sub/default.png">', - '<img src="@source/sub/100%.png" srcset="@source/sub/汉字.png 1x">', - '<img src="@source/sub/default.png" srcset="@source/sub/attrs1.png 1x, @source/sub/attrs2.png 2x" alt="attrs" width="100px">', + '<img srcset="./foo.png 1x, ../sub/foo.png 2x, ./foo/bar.png 1024w, ../sub/foo/bar.png 2048w, ../baz.png 4096w, ../../out.png 3x" src="./default.png">', + '<img src="./100%.png" srcset="./汉字.png 1x">', + '<img src="./default.png" srcset="./attrs1.png 1x, ./attrs2.png 2x" alt="attrs" width="100px">', // aliases '<img srcset="@alias/foo.png 1x, @alias/汉字.png 2x, @alias/100%.png 3x" alt="attrs" src="@alias/attrs.png" width="100px">', '<img srcset="~@alias/foo.png 1x, ~@alias/汉字.png 2x, ~@alias/100%.png 3x" alt="attrs" src="~@alias/attrs.png" width="100px">', - // keep as is + // absolute paths and no-prefix paths '<img alt="attrs" src="" width="100px" srcset="/absolute.png 1x, no-prefix.png 2x, http://foobar.com/icon.png">', /* data uri */ @@ -379,92 +299,13 @@ describe('@vuepress/markdown > plugins > assetsPlugin', () => { ], }, { - description: 'should respect `relativePathPrefix` option', + description: 'should respect `absolutePathPrependBase` option', md: MarkdownIt({ html: true }).use(assetsPlugin, { - relativePathPrefix: '@foo', + absolutePathPrependBase: true, }), env: { - filePathRelative: 'sub/foo.md', + base: '/base/', }, - expected: [ - /* src */ - // relative paths - '<img src="@foo/sub/foo.png">', - '<img src="@foo/sub/foo.png">', - '<img src="@foo/sub/foo/bar.png">', - '<img src="@foo/sub/foo/bar.png">', - '<img src="@foo/baz.png">', - '<img src="@foo/../out.png">', - '<img src="@foo/sub/汉字.png">', - '<img src="@foo/sub/100%.png">', - '<img alt="attrs" src="@foo/sub/attrs.png" width="100px">', - // aliases - '<img src="@alias/foo.png">', - '<img src="@alias/汉字.png">', - '<img src="@alias/100%.png">', - '<img alt="attrs" src="@alias/attrs.png" width="100px">', - // webpack legacy aliases - '<img src="~@alias/foo.png">', - '<img src="~@alias/汉字.png">', - '<img src="~@alias/100%.png">', - '<img alt="attrs" src="~@alias/attrs.png" width="100px">', - // absolute paths - '<img src="/absolute.png">', - '<img src="/foo/absolute.png">', - // no-prefix paths - '<img src="no-prefix.png">', - '<img src="foo/no-prefix.png">', - '<img alt="attrs" src="attrs.png" width="100px">', - // keep as is - '<img src="http://foobar.com/icon.png">', - '<img src="">', - // invalid paths - '<img src=".../invalid.png">', - '<img src=".../汉字.png">', - '<img src=".../100%.png">', - '<img alt="attrs" src=".../attrs.png" width="100px">', - - /* srcset */ - // relative paths - '<img srcset="@foo/sub/foo.png 1x, @foo/sub/foo.png 2x, @foo/sub/foo/bar.png 1024w, @foo/sub/foo/bar.png 2048w, @foo/baz.png 4096w, @foo/../out.png">', - '<img srcset="@foo/sub/汉字.png 1x, @foo/sub/100%.png">', - '<img alt="attrs" srcset="@foo/sub/attrs.png" width="100px">', - // aliases - '<img srcset="@alias/foo.png 1x, @alias/汉字.png 2x, @alias/100%.png 3x">', - '<img alt="attrs" srcset="@alias/attrs.png 1024w" width="100px">', - // webpack legacy aliases - '<img srcset="~@alias/foo.png 1x, ~@alias/汉字.png 2x, ~@alias/100%.png 3x">', - '<img alt="attrs" srcset="~@alias/attrs.png 1024w" width="100px">', - // keep as is - '<img srcset="/absolute.png 1x, no-prefix.png 2x, http://foobar.com/icon.png">', - '<img srcset="">', - '<img alt="attrs" srcset="attrs.png 1x, default.png" width="100px">', - // invalid paths - '<img srcset=".../invalid.png 1x, .../汉字.png 2x, .../100%.png 3x">', - '<img alt="attrs" srcset=".../attrs.png 1x, .../default.png" width="100px">', - // invalid srcset - '<img srcset="@foo/invalid.png, @foo/汉字.png, .../100%.png 3x">', - - /* both */ - // relative paths - '<img srcset="@foo/sub/foo.png 1x, @foo/sub/foo.png 2x, @foo/sub/foo/bar.png 1024w, @foo/sub/foo/bar.png 2048w, @foo/baz.png 4096w, @foo/../out.png 3x" src="@foo/sub/default.png">', - '<img src="@foo/sub/100%.png" srcset="@foo/sub/汉字.png 1x">', - '<img src="@foo/sub/default.png" srcset="@foo/sub/attrs1.png 1x, @foo/sub/attrs2.png 2x" alt="attrs" width="100px">', - // aliases - '<img srcset="@alias/foo.png 1x, @alias/汉字.png 2x, @alias/100%.png 3x" alt="attrs" src="@alias/attrs.png" width="100px">', - '<img srcset="~@alias/foo.png 1x, ~@alias/汉字.png 2x, ~@alias/100%.png 3x" alt="attrs" src="~@alias/attrs.png" width="100px">', - // keep as is - '<img alt="attrs" src="" width="100px" srcset="/absolute.png 1x, no-prefix.png 2x, http://foobar.com/icon.png">', - - /* data uri */ - '<img src="">', - ], - }, - { - description: - 'should not handle relative paths if `env.filePathRelative` is not provided', - md: MarkdownIt({ html: true }).use(assetsPlugin), - env: {}, expected: [ /* src */ // relative paths @@ -488,8 +329,8 @@ describe('@vuepress/markdown > plugins > assetsPlugin', () => { '<img src="~@alias/100%.png">', '<img alt="attrs" src="~@alias/attrs.png" width="100px">', // absolute paths - '<img src="/absolute.png">', - '<img src="/foo/absolute.png">', + '<img src="/base/absolute.png">', + '<img src="/base/foo/absolute.png">', // no-prefix paths '<img src="no-prefix.png">', '<img src="foo/no-prefix.png">', @@ -514,8 +355,9 @@ describe('@vuepress/markdown > plugins > assetsPlugin', () => { // webpack legacy aliases '<img srcset="~@alias/foo.png 1x, ~@alias/汉字.png 2x, ~@alias/100%.png 3x">', '<img alt="attrs" srcset="~@alias/attrs.png 1024w" width="100px">', + // absolute paths and no-prefix paths + '<img srcset="/base/absolute.png 1x, no-prefix.png 2x, http://foobar.com/icon.png">', // keep as is - '<img srcset="/absolute.png 1x, no-prefix.png 2x, http://foobar.com/icon.png">', '<img srcset="">', '<img alt="attrs" srcset="attrs.png 1x, default.png" width="100px">', // invalid paths @@ -532,8 +374,8 @@ describe('@vuepress/markdown > plugins > assetsPlugin', () => { // aliases '<img srcset="@alias/foo.png 1x, @alias/汉字.png 2x, @alias/100%.png 3x" alt="attrs" src="@alias/attrs.png" width="100px">', '<img srcset="~@alias/foo.png 1x, ~@alias/汉字.png 2x, ~@alias/100%.png 3x" alt="attrs" src="~@alias/attrs.png" width="100px">', - // keep as is - '<img alt="attrs" src="" width="100px" srcset="/absolute.png 1x, no-prefix.png 2x, http://foobar.com/icon.png">', + // absolute paths and no-prefix paths + '<img alt="attrs" src="" width="100px" srcset="/base/absolute.png 1x, no-prefix.png 2x, http://foobar.com/icon.png">', /* data uri */ '<img src="">', @@ -630,22 +472,25 @@ describe('@vuepress/markdown > plugins > assetsPlugin', () => { `<img alt="attrs" src=" .../attrs.png " width="100px">`, + `<img alt="attrs" src=" + /absolute/attrs.png + " width="100px">`, /* srcset */ `<img srcset="./foo.png 1x , ../sub/foo.png 2x,./foo/bar.png - 1024w ,../../out.png">`, + 1024w ,../../out.png, /absolute/attrs.png">`, `<img alt="attrs" srcset=" ./attrs.png 1x - ,default.png " width="100px">`, + ,default.png ,/absolute/attrs.png" width="100px">`, /** both */ `<img src=" ./default.png " srcset="./foo.png 1x , ../sub/foo.png 2x,./foo/bar.png - 1024w ,../../out.png">`, + 1024w ,../../out.png, /absolute/attrs.png">`, `<img alt="attrs" src="./default.png" srcset=" ./attrs.png 1x - ,default.png " width="100px">`, + ,default.png ,/absolute/attrs.png" width="100px">`, ] const TEST_CASES: { @@ -658,58 +503,42 @@ describe('@vuepress/markdown > plugins > assetsPlugin', () => { description: 'should handle assets link with default options', md: MarkdownIt({ html: true }).use(assetsPlugin), env: { - filePathRelative: 'sub/foo.md', + base: '/base/', }, expected: [ /* src */ '<p><img alt="attrs" src=".../attrs.png" width="100px"></p>', + '<p><img alt="attrs" src="/absolute/attrs.png" width="100px"></p>', /* srcset */ - '<p><img srcset="@source/sub/foo.png 1x, @source/sub/foo.png 2x, @source/sub/foo/bar.png 1024w, @source/../out.png"></p>', - '<p><img alt="attrs" srcset="@source/sub/attrs.png 1x, default.png" width="100px"></p>', + '<p><img srcset="./foo.png 1x, ../sub/foo.png 2x, ./foo/bar.png 1024w, ../../out.png, /absolute/attrs.png"></p>', + '<p><img alt="attrs" srcset="./attrs.png 1x, default.png, /absolute/attrs.png" width="100px"></p>', /* both */ - '<p><img src="@source/sub/default.png" srcset="@source/sub/foo.png 1x, @source/sub/foo.png 2x, @source/sub/foo/bar.png 1024w, @source/../out.png"></p>', - '<p><img alt="attrs" src="@source/sub/default.png" srcset="@source/sub/attrs.png 1x, default.png" width="100px"></p>', + '<p><img src="./default.png" srcset="./foo.png 1x, ../sub/foo.png 2x, ./foo/bar.png 1024w, ../../out.png, /absolute/attrs.png"></p>', + '<p><img alt="attrs" src="./default.png" srcset="./attrs.png 1x, default.png, /absolute/attrs.png" width="100px"></p>', ], }, { - description: 'should respect `relativePathPrefix` option', + description: 'should respect `absolutePathPrependBase` option', md: MarkdownIt({ html: true }).use(assetsPlugin, { - relativePathPrefix: '@foo', + absolutePathPrependBase: true, }), env: { - filePathRelative: 'sub/foo.md', + base: '/base/', }, expected: [ /* src */ '<p><img alt="attrs" src=".../attrs.png" width="100px"></p>', + '<p><img alt="attrs" src="/base/absolute/attrs.png" width="100px"></p>', /* srcset */ - '<p><img srcset="@foo/sub/foo.png 1x, @foo/sub/foo.png 2x, @foo/sub/foo/bar.png 1024w, @foo/../out.png"></p>', - '<p><img alt="attrs" srcset="@foo/sub/attrs.png 1x, default.png" width="100px"></p>', - - /* both */ - '<p><img src="@foo/sub/default.png" srcset="@foo/sub/foo.png 1x, @foo/sub/foo.png 2x, @foo/sub/foo/bar.png 1024w, @foo/../out.png"></p>', - '<p><img alt="attrs" src="@foo/sub/default.png" srcset="@foo/sub/attrs.png 1x, default.png" width="100px"></p>', - ], - }, - { - description: - 'should not handle assets link if `filePathRelative` is not provided', - md: MarkdownIt({ html: true }).use(assetsPlugin), - env: {}, - expected: [ - /* src */ - '<p><img alt="attrs" src=".../attrs.png" width="100px"></p>', - - /* srcset */ - '<p><img srcset="./foo.png 1x, ../sub/foo.png 2x, ./foo/bar.png 1024w, ../../out.png"></p>', - '<p><img alt="attrs" srcset="./attrs.png 1x, default.png" width="100px"></p>', + '<p><img srcset="./foo.png 1x, ../sub/foo.png 2x, ./foo/bar.png 1024w, ../../out.png, /base/absolute/attrs.png"></p>', + '<p><img alt="attrs" srcset="./attrs.png 1x, default.png, /base/absolute/attrs.png" width="100px"></p>', /* both */ - '<p><img src="./default.png" srcset="./foo.png 1x, ../sub/foo.png 2x, ./foo/bar.png 1024w, ../../out.png"></p>', - '<p><img alt="attrs" src="./default.png" srcset="./attrs.png 1x, default.png" width="100px"></p>', + '<p><img src="./default.png" srcset="./foo.png 1x, ../sub/foo.png 2x, ./foo/bar.png 1024w, ../../out.png, /base/absolute/attrs.png"></p>', + '<p><img alt="attrs" src="./default.png" srcset="./attrs.png 1x, default.png, /base/absolute/attrs.png" width="100px"></p>', ], }, ] From e681fb3f5a73e93c72fb62961eabe3044e3f0f58 Mon Sep 17 00:00:00 2001 From: meteorlxy <meteor.lxy@foxmail.com> Date: Wed, 11 Sep 2024 12:33:15 +0800 Subject: [PATCH 02/11] refactor: updates --- eslint.config.ts | 1 + .../src/plugins/vuepressMarkdownPlugin.ts | 25 +++--- .../src/loaders/vuepressMarkdownLoader.ts | 14 ++-- .../cli/src/commands/dev/handlePageAdd.ts | 11 +-- .../cli/src/commands/dev/handlePageChange.ts | 9 +-- .../cli/src/commands/dev/handlePageUnlink.ts | 1 + packages/client/src/components/Content.ts | 2 +- packages/client/src/setupGlobalComputed.ts | 11 ++- packages/client/src/types/routes.ts | 4 +- packages/core/src/app/appPrepare.ts | 7 +- packages/core/src/app/prepare/index.ts | 1 - .../core/src/app/prepare/preparePageChunk.ts | 32 +------- .../src/app/prepare/preparePageComponent.ts | 17 ---- packages/core/src/page/createPage.ts | 15 +--- packages/core/src/page/index.ts | 5 +- .../core/src/page/renderPageSfcBlocksToVue.ts | 17 ---- packages/core/src/page/renderPageToVue.ts | 79 +++++++++++++++++++ .../core/src/page/resolvePageChunkInfo.ts | 24 +++++- .../core/src/page/resolvePageComponentInfo.ts | 39 --------- ...geFileContent.ts => resolvePageContent.ts} | 18 +++-- packages/core/src/types/page.ts | 17 +--- ...ent.spec.ts => resolvePageContent.spec.ts} | 30 +++++-- 22 files changed, 181 insertions(+), 198 deletions(-) delete mode 100644 packages/core/src/app/prepare/preparePageComponent.ts delete mode 100644 packages/core/src/page/renderPageSfcBlocksToVue.ts create mode 100644 packages/core/src/page/renderPageToVue.ts delete mode 100644 packages/core/src/page/resolvePageComponentInfo.ts rename packages/core/src/page/{resolvePageFileContent.ts => resolvePageContent.ts} (51%) rename packages/core/tests/page/{resolvePageFileContent.spec.ts => resolvePageContent.spec.ts} (51%) diff --git a/eslint.config.ts b/eslint.config.ts index fa0eed31d7..1cb1edfb7b 100644 --- a/eslint.config.ts +++ b/eslint.config.ts @@ -19,6 +19,7 @@ export default vuepress( allow: [ '__dirname', '_context', + '_pageData', '_pageChunk', '_registeredComponents', ], diff --git a/packages/bundler-vite/src/plugins/vuepressMarkdownPlugin.ts b/packages/bundler-vite/src/plugins/vuepressMarkdownPlugin.ts index eb072f840c..94da9828c8 100644 --- a/packages/bundler-vite/src/plugins/vuepressMarkdownPlugin.ts +++ b/packages/bundler-vite/src/plugins/vuepressMarkdownPlugin.ts @@ -1,6 +1,5 @@ import type { App } from '@vuepress/core' -import { parsePageContent, renderPageSfcBlocksToVue } from '@vuepress/core' -import { path } from '@vuepress/utils' +import { createPage, renderPageToVue } from '@vuepress/core' import type { Plugin } from 'vite' /** @@ -11,7 +10,7 @@ export const vuepressMarkdownPlugin = ({ app }: { app: App }): Plugin => ({ enforce: 'pre', - transform(code, id) { + async transform(code, id) { if (!id.endsWith('.md')) return // get the matched page by file path (id) @@ -19,18 +18,15 @@ export const vuepressMarkdownPlugin = ({ app }: { app: App }): Plugin => ({ // if the page content is not changed, render it to vue component directly if (page?.content === code) { - return renderPageSfcBlocksToVue(page.sfcBlocks) + return renderPageToVue(page) } - // parse the markdown content to sfc blocks and render it to vue component - const { sfcBlocks } = parsePageContent({ - app, + // create a new page with the new content + const newPage = await createPage(app, { content: code, filePath: id, - filePathRelative: path.relative(app.dir.source(), id), - options: {}, }) - return renderPageSfcBlocksToVue(sfcBlocks) + return renderPageToVue(newPage) }, async handleHotUpdate(ctx) { @@ -39,15 +35,12 @@ export const vuepressMarkdownPlugin = ({ app }: { app: App }): Plugin => ({ // read the source code const code = await ctx.read() - // parse the content to sfc blocks - const { sfcBlocks } = parsePageContent({ - app, + // create a new page with the new content + const newPage = await createPage(app, { content: code, filePath: ctx.file, - filePathRelative: path.relative(app.dir.source(), ctx.file), - options: {}, }) - ctx.read = () => renderPageSfcBlocksToVue(sfcBlocks) + ctx.read = () => renderPageToVue(newPage) }, }) diff --git a/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.ts b/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.ts index d9ca435d14..819dc29f66 100644 --- a/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.ts +++ b/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.ts @@ -11,8 +11,7 @@ export interface VuepressMarkdownLoaderOptions { export const vuepressMarkdownLoader: LoaderDefinitionFunction<VuepressMarkdownLoaderOptions> = async function vuepressMarkdownLoader(source) { // import esm dependencies - const [{ parsePageContent, renderPageSfcBlocksToVue }, { path }] = - await Promise.all([import('@vuepress/core'), import('@vuepress/utils')]) + const { createPage, renderPageToVue } = await import('@vuepress/core') // get app instance from loader options const { app } = this.getOptions() @@ -22,16 +21,13 @@ export const vuepressMarkdownLoader: LoaderDefinitionFunction<VuepressMarkdownLo // if the page content is not changed, render it to vue component directly if (page?.content === source) { - return renderPageSfcBlocksToVue(page.sfcBlocks) + return renderPageToVue(page) } - // parse the markdown content to sfc blocks and render it to vue component - const { sfcBlocks } = parsePageContent({ - app, + // create a new page with the new content + const newPage = await createPage(app, { content: source, filePath: this.resourcePath, - filePathRelative: path.relative(app.dir.source(), this.resourcePath), - options: {}, }) - return renderPageSfcBlocksToVue(sfcBlocks) + return renderPageToVue(newPage) } diff --git a/packages/cli/src/commands/dev/handlePageAdd.ts b/packages/cli/src/commands/dev/handlePageAdd.ts index bcf5415a97..4c8d7c2512 100644 --- a/packages/cli/src/commands/dev/handlePageAdd.ts +++ b/packages/cli/src/commands/dev/handlePageAdd.ts @@ -1,10 +1,5 @@ import type { App, Page } from '@vuepress/core' -import { - createPage, - preparePageChunk, - preparePageComponent, - prepareRoutes, -} from '@vuepress/core' +import { createPage, preparePageChunk, prepareRoutes } from '@vuepress/core' /** * Event handler for page add event @@ -28,9 +23,9 @@ export const handlePageAdd = async ( // add the new page app.pages.push(page) + app.pagesMap[filePath] = page - // prepare page files - await preparePageComponent(app, page) + // prepare page file await preparePageChunk(app, page) // prepare routes file diff --git a/packages/cli/src/commands/dev/handlePageChange.ts b/packages/cli/src/commands/dev/handlePageChange.ts index c2255096f0..c7e29debf6 100644 --- a/packages/cli/src/commands/dev/handlePageChange.ts +++ b/packages/cli/src/commands/dev/handlePageChange.ts @@ -1,10 +1,5 @@ import type { App, Page } from '@vuepress/core' -import { - createPage, - preparePageChunk, - preparePageComponent, - prepareRoutes, -} from '@vuepress/core' +import { createPage, preparePageChunk, prepareRoutes } from '@vuepress/core' /** * Event handler for page change event @@ -31,9 +26,9 @@ export const handlePageChange = async ( // replace the old page with the new page app.pages.splice(pageIndex, 1, pageNew) + app.pagesMap[filePath] = pageNew // prepare page files - await preparePageComponent(app, pageNew) await preparePageChunk(app, pageNew) const isPathChanged = pageOld.path !== pageNew.path diff --git a/packages/cli/src/commands/dev/handlePageUnlink.ts b/packages/cli/src/commands/dev/handlePageUnlink.ts index ad4d01a2c4..40dca36d4e 100644 --- a/packages/cli/src/commands/dev/handlePageUnlink.ts +++ b/packages/cli/src/commands/dev/handlePageUnlink.ts @@ -20,6 +20,7 @@ export const handlePageUnlink = async ( // remove the old page app.pages.splice(pageIndex, 1) + delete app.pagesMap[filePath] // re-prepare routes file await prepareRoutes(app) diff --git a/packages/client/src/components/Content.ts b/packages/client/src/components/Content.ts index e58b46214d..c17bc44a3a 100644 --- a/packages/client/src/components/Content.ts +++ b/packages/client/src/components/Content.ts @@ -22,7 +22,7 @@ export const Content = defineComponent({ if (!props.path) return pageComponent.value const route = resolveRoute(props.path) return defineAsyncComponent(async () => - route.loader().then(({ comp }) => comp), + route.loader().then((m) => m.default), ) }) diff --git a/packages/client/src/setupGlobalComputed.ts b/packages/client/src/setupGlobalComputed.ts index e258cb7673..79c8fe459a 100644 --- a/packages/client/src/setupGlobalComputed.ts +++ b/packages/client/src/setupGlobalComputed.ts @@ -47,12 +47,15 @@ export const setupGlobalComputed = ( if (__VUEPRESS_DEV__ && (import.meta.webpackHot || import.meta.hot)) { __VUE_HMR_RUNTIME__.updatePageData = async (newPageData: PageData) => { const oldPageChunk = await routes.value[newPageData.path].loader() - const newPageChunk = { comp: oldPageChunk.comp, data: newPageData } + const newPageChunk: PageChunk = { + default: oldPageChunk.default, + _pageData: newPageData, + } routes.value[newPageData.path].loader = async () => Promise.resolve(newPageChunk) if ( newPageData.path === - router.currentRoute.value.meta._pageChunk?.data.path + router.currentRoute.value.meta._pageChunk?._pageData.path ) { pageChunk.value = newPageChunk } @@ -67,8 +70,8 @@ export const setupGlobalComputed = ( const siteLocaleData = computed(() => resolvers.resolveSiteLocaleData(siteData.value, routeLocale.value), ) - const pageComponent = computed(() => pageChunk.value.comp) - const pageData = computed(() => pageChunk.value.data) + const pageComponent = computed(() => pageChunk.value.default) + const pageData = computed(() => pageChunk.value._pageData) const pageFrontmatter = computed(() => pageData.value.frontmatter) const pageHeadTitle = computed(() => resolvers.resolvePageHeadTitle(pageData.value, siteLocaleData.value), diff --git a/packages/client/src/types/routes.ts b/packages/client/src/types/routes.ts index 6aa5d10871..c102dee700 100644 --- a/packages/client/src/types/routes.ts +++ b/packages/client/src/types/routes.ts @@ -2,8 +2,8 @@ import type { Component } from 'vue' import type { PageData } from '../types/index.js' export interface PageChunk { - comp: Component - data: PageData + default: Component + _pageData: PageData } export type RouteMeta = Record<string, unknown> diff --git a/packages/core/src/app/appPrepare.ts b/packages/core/src/app/appPrepare.ts index f8d2585f50..a41f8d72be 100644 --- a/packages/core/src/app/appPrepare.ts +++ b/packages/core/src/app/appPrepare.ts @@ -3,7 +3,6 @@ import type { App } from '../types/index.js' import { prepareClientConfigs, preparePageChunk, - preparePageComponent, prepareRoutes, prepareSiteData, } from './prepare/index.js' @@ -13,7 +12,6 @@ const log = debug('vuepress:core/app') /** * Prepare files for development or build * - * - page components * - page chunks * - routes * - site data @@ -23,10 +21,7 @@ export const appPrepare = async (app: App): Promise<void> => { log('prepare start') await Promise.all([ - ...app.pages.flatMap((page) => [ - preparePageComponent(app, page), - preparePageChunk(app, page), - ]), + ...app.pages.map(async (page) => preparePageChunk(app, page)), prepareRoutes(app), prepareSiteData(app), prepareClientConfigs(app), diff --git a/packages/core/src/app/prepare/index.ts b/packages/core/src/app/prepare/index.ts index ea49aa7600..460f3b497f 100644 --- a/packages/core/src/app/prepare/index.ts +++ b/packages/core/src/app/prepare/index.ts @@ -1,5 +1,4 @@ export * from './prepareClientConfigs.js' export * from './preparePageChunk.js' -export * from './preparePageComponent.js' export * from './prepareRoutes.js' export * from './prepareSiteData.js' diff --git a/packages/core/src/app/prepare/preparePageChunk.ts b/packages/core/src/app/prepare/preparePageChunk.ts index 353a98eee1..2ade465418 100644 --- a/packages/core/src/app/prepare/preparePageChunk.ts +++ b/packages/core/src/app/prepare/preparePageChunk.ts @@ -1,35 +1,11 @@ +import { renderPageToVue } from '../../page/index.js' import type { App, Page } from '../../types/index.js' -const HMR_CODE = ` -if (import.meta.webpackHot) { - import.meta.webpackHot.accept() - if (__VUE_HMR_RUNTIME__.updatePageData) { - __VUE_HMR_RUNTIME__.updatePageData(data) - } -} - -if (import.meta.hot) { - import.meta.hot.accept(({ data }) => { - __VUE_HMR_RUNTIME__.updatePageData(data) - }) -} -` - /** - * Generate page chunk temp file of a single page + * Generate temp file the page does not have a source file */ export const preparePageChunk = async (app: App, page: Page): Promise<void> => { - // page chunk file content - let content = `\ -import comp from ${JSON.stringify(page.componentFilePath)} -const data = JSON.parse(${JSON.stringify(JSON.stringify(page.data))}) -export { comp, data } -` - - // inject HMR code - if (app.env.isDev) { - content += HMR_CODE + if (page.filePath === null) { + await app.writeTemp(page.chunkFilePathRelative, renderPageToVue(page)) } - - await app.writeTemp(page.chunkFilePathRelative, content) } diff --git a/packages/core/src/app/prepare/preparePageComponent.ts b/packages/core/src/app/prepare/preparePageComponent.ts deleted file mode 100644 index e88d6123d5..0000000000 --- a/packages/core/src/app/prepare/preparePageComponent.ts +++ /dev/null @@ -1,17 +0,0 @@ -import { renderPageSfcBlocksToVue } from '../../page/index.js' -import type { App, Page } from '../../types/index.js' - -/** - * Generate page component temp file if the page does not have a source file - */ -export const preparePageComponent = async ( - app: App, - page: Page, -): Promise<void> => { - if (page.filePath === null) { - await app.writeTemp( - page.componentFilePathRelative, - renderPageSfcBlocksToVue(page.sfcBlocks), - ) - } -} diff --git a/packages/core/src/page/createPage.ts b/packages/core/src/page/createPage.ts index 0c48399162..9965abf4a4 100644 --- a/packages/core/src/page/createPage.ts +++ b/packages/core/src/page/createPage.ts @@ -2,9 +2,8 @@ import type { App, Page, PageOptions } from '../types/index.js' import { inferPagePath } from './inferPagePath.js' import { parsePageContent } from './parsePageContent.js' import { resolvePageChunkInfo } from './resolvePageChunkInfo.js' -import { resolvePageComponentInfo } from './resolvePageComponentInfo.js' +import { resolvePageContent } from './resolvePageContent.js' import { resolvePageDate } from './resolvePageDate.js' -import { resolvePageFileContent } from './resolvePageFileContent.js' import { resolvePageFilePath } from './resolvePageFilePath.js' import { resolvePageHtmlInfo } from './resolvePageHtmlInfo.js' import { resolvePageLang } from './resolvePageLang.js' @@ -27,7 +26,7 @@ export const createPage = async ( }) // read the raw file content according to the absolute file path - const content = await resolvePageFileContent({ filePath, options }) + const content = await resolvePageContent({ filePath, options }) // render page content and extract information const { @@ -81,18 +80,14 @@ export const createPage = async ( path, }) - // resolve page component and extract headers & links - const { componentFilePath, componentFilePathRelative } = - resolvePageComponentInfo({ + const { chunkFilePath, chunkFilePathRelative, chunkName } = + resolvePageChunkInfo({ app, filePath, filePathRelative, htmlFilePathRelative, }) - const { chunkFilePath, chunkFilePathRelative, chunkName } = - resolvePageChunkInfo({ app, htmlFilePathRelative }) - const page: Page = { // page data data: { @@ -127,8 +122,6 @@ export const createPage = async ( // file info filePath, filePathRelative, - componentFilePath, - componentFilePathRelative, chunkFilePath, chunkFilePathRelative, chunkName, diff --git a/packages/core/src/page/index.ts b/packages/core/src/page/index.ts index 6e1bc3d592..e6bc9edd07 100644 --- a/packages/core/src/page/index.ts +++ b/packages/core/src/page/index.ts @@ -1,11 +1,10 @@ export * from './createPage.js' export * from './inferPagePath.js' export * from './parsePageContent.js' -export * from './renderPageSfcBlocksToVue.js' +export * from './renderPageToVue.js' export * from './resolvePageChunkInfo.js' -export * from './resolvePageComponentInfo.js' export * from './resolvePageDate.js' -export * from './resolvePageFileContent.js' +export * from './resolvePageContent.js' export * from './resolvePageFilePath.js' export * from './resolvePageHtmlInfo.js' export * from './resolvePageLang.js' diff --git a/packages/core/src/page/renderPageSfcBlocksToVue.ts b/packages/core/src/page/renderPageSfcBlocksToVue.ts deleted file mode 100644 index 7428643823..0000000000 --- a/packages/core/src/page/renderPageSfcBlocksToVue.ts +++ /dev/null @@ -1,17 +0,0 @@ -import type { MarkdownSfcBlocks } from '@vuepress/markdown' - -/** - * Render page sfc blocks to vue component - */ -export const renderPageSfcBlocksToVue = ( - sfcBlocks: MarkdownSfcBlocks, -): string => - [ - // #688: wrap the content of `<template>` with a `<div>` to avoid some potential issues of fragment component - `${sfcBlocks.template?.tagOpen}<div>${sfcBlocks.template?.contentStripped}</div>${sfcBlocks.template?.tagClose}\n`, - // hoist `<script>`, `<style>` and other custom blocks - sfcBlocks.script?.content, - sfcBlocks.scriptSetup?.content, - ...sfcBlocks.styles.map((item) => item.content), - ...sfcBlocks.customBlocks.map((item) => item.content), - ].join('\n') diff --git a/packages/core/src/page/renderPageToVue.ts b/packages/core/src/page/renderPageToVue.ts new file mode 100644 index 0000000000..849e647d4c --- /dev/null +++ b/packages/core/src/page/renderPageToVue.ts @@ -0,0 +1,79 @@ +import { isString } from '@vuepress/shared' +import type { Page } from '../types/index.js' + +const TEMPLATE_WRAPPER_TAG_OPEN = '<div>' +const TEMPLATE_WRAPPER_TAG_CLOSE = '</div>' + +const SCRIPT_TAG_OPEN = '<script>' +const SCRIPT_TAG_CLOSE = '</script>' + +const SCRIPT_TAG_OPEN_LANG_TS_REGEX = /<\s*script[^>]*\blang=['"]ts['"][^>]*/ +const SCRIPT_TAG_OPEN_LANG_TS = '<script lang="ts">' + +const HMR_CODE = ` +if (import.meta.webpackHot) { + import.meta.webpackHot.accept() + if (__VUE_HMR_RUNTIME__.updatePageData) { + __VUE_HMR_RUNTIME__.updatePageData(_pageData) + } +} + +if (import.meta.hot) { + import.meta.hot.accept(({ _pageData }) => { + __VUE_HMR_RUNTIME__.updatePageData(_pageData) + }) +} +` + +/** + * Util to resolve the open tag of script block + */ +const resolveScriptTagOpen = (sfcBlocks: Page['sfcBlocks']): string => { + // use existing script open tag + if (sfcBlocks.script?.tagOpen) { + return sfcBlocks.script.tagOpen + } + // if the setup script block is using typescript, we should use the same language for script block + const isUsingTs = sfcBlocks.scriptSetup?.tagOpen.match( + SCRIPT_TAG_OPEN_LANG_TS_REGEX, + ) + return isUsingTs ? SCRIPT_TAG_OPEN_LANG_TS : SCRIPT_TAG_OPEN +} + +/** + * Render page to vue component + */ +export const renderPageToVue = ({ data, sfcBlocks }: Page): string => { + // #688: wrap the content of `<template>` with a `<div>` to avoid some potential issues of fragment component + const templateContent = + sfcBlocks.template && + [ + sfcBlocks.template.tagOpen, + TEMPLATE_WRAPPER_TAG_OPEN, + sfcBlocks.template.contentStripped, + TEMPLATE_WRAPPER_TAG_CLOSE, + sfcBlocks.template.tagClose, + ].join('') + + // inject page data code and HMR code into the script content + const pageDataCode = `export const _pageData = JSON.parse(${JSON.stringify(JSON.stringify(data))})` + const scriptContent = [ + resolveScriptTagOpen(sfcBlocks), + sfcBlocks.script?.contentStripped, + pageDataCode, + HMR_CODE, + sfcBlocks.script?.tagClose ?? SCRIPT_TAG_CLOSE, + ] + .filter(isString) + .join('\n') + + return [ + templateContent, + scriptContent, + sfcBlocks.scriptSetup?.content, + ...sfcBlocks.styles.map((item) => item.content), + ...sfcBlocks.customBlocks.map((item) => item.content), + ] + .filter(isString) + .join('\n') +} diff --git a/packages/core/src/page/resolvePageChunkInfo.ts b/packages/core/src/page/resolvePageChunkInfo.ts index 037018c6cc..b297794c04 100644 --- a/packages/core/src/page/resolvePageChunkInfo.ts +++ b/packages/core/src/page/resolvePageChunkInfo.ts @@ -2,23 +2,41 @@ import { path, transformPathToFileName } from '@vuepress/utils' import type { App } from '../types/index.js' /** - * Resolve page data file path + * Resolve page chunk file relative info */ export const resolvePageChunkInfo = ({ app, + filePath, + filePathRelative, htmlFilePathRelative, }: { app: App + filePath: string | null + filePathRelative: string | null htmlFilePathRelative: string }): { chunkFilePath: string chunkFilePathRelative: string chunkName: string } => { - const chunkFilePathRelative = path.join('pages', `${htmlFilePathRelative}.js`) - const chunkFilePath = app.dir.temp(chunkFilePathRelative) const chunkName = transformPathToFileName(htmlFilePathRelative) + // if there is a source file for the page, use it directly as the page chunk + if (filePath && filePathRelative) { + return { + chunkFilePath: filePath, + chunkFilePathRelative: filePathRelative, + chunkName, + } + } + + // otherwise, generate a temp file for the page + const chunkFilePathRelative = path.join( + 'pages', + `${htmlFilePathRelative}.vue`, + ) + const chunkFilePath = app.dir.temp(chunkFilePathRelative) + return { chunkFilePath, chunkFilePathRelative, diff --git a/packages/core/src/page/resolvePageComponentInfo.ts b/packages/core/src/page/resolvePageComponentInfo.ts deleted file mode 100644 index 2105bdb187..0000000000 --- a/packages/core/src/page/resolvePageComponentInfo.ts +++ /dev/null @@ -1,39 +0,0 @@ -import { path } from '@vuepress/utils' -import type { App } from '../types/index.js' - -/** - * Resolve page component and related info - */ -export const resolvePageComponentInfo = ({ - app, - filePath, - filePathRelative, - htmlFilePathRelative, -}: { - app: App - filePath: string | null - filePathRelative: string | null - htmlFilePathRelative: string -}): { - componentFilePath: string - componentFilePathRelative: string -} => { - // if there is a source file for the page, use it as the component file - if (filePath && filePathRelative) { - return { - componentFilePath: filePath, - componentFilePathRelative: filePathRelative, - } - } - // otherwise, generate a component file for the page - const componentFilePathRelative = path.join( - 'pages', - `${htmlFilePathRelative}.vue`, - ) - const componentFilePath = app.dir.temp(componentFilePathRelative) - - return { - componentFilePath, - componentFilePathRelative, - } -} diff --git a/packages/core/src/page/resolvePageFileContent.ts b/packages/core/src/page/resolvePageContent.ts similarity index 51% rename from packages/core/src/page/resolvePageFileContent.ts rename to packages/core/src/page/resolvePageContent.ts index e05bf9a501..c4bda0732b 100644 --- a/packages/core/src/page/resolvePageFileContent.ts +++ b/packages/core/src/page/resolvePageContent.ts @@ -1,21 +1,30 @@ +import { isString } from '@vuepress/shared' import { debug, fs } from '@vuepress/utils' import type { PageOptions } from '../types/index.js' const log = debug('vuepress:core/page') +// fallback to empty string +const FALLBACK_CONTENT = '' + /** - * Resolve page file content according to filePath or options content + * Resolve page content according to `content` or `filePath` */ -export const resolvePageFileContent = async ({ +export const resolvePageContent = async ({ filePath, options, }: { filePath: string | null options: PageOptions }): Promise<string> => { + // if `content` is provided by options, use it directly + if (isString(options.content)) { + return options.content + } + + // if `filePath` is resolved, read content from file if (filePath) { try { - // read page content from file const content = await fs.readFile(filePath, 'utf-8') return content } catch (e) { @@ -23,6 +32,5 @@ export const resolvePageFileContent = async ({ } } - // load raw content from options - return options.content ?? '' + return FALLBACK_CONTENT } diff --git a/packages/core/src/types/page.ts b/packages/core/src/types/page.ts index 178fec9782..ade17a2fd8 100644 --- a/packages/core/src/types/page.ts +++ b/packages/core/src/types/page.ts @@ -104,16 +104,6 @@ export type Page< */ filePathRelative: string | null - /** - * Component file path - */ - componentFilePath: string - - /** - * Component file path relative to temp directory - */ - componentFilePathRelative: string - /** * Chunk file path */ @@ -147,11 +137,10 @@ export type Page< */ export interface PageOptions { /** - * If `filePath` is not set, this option will be used as the raw - * markdown content of the page. + * The raw markdown content of the page. * - * If `filePath` is set, this option will be ignored, while the - * content of the file will be used. + * If `content` is not provided, the file content of the `filePath` + * will be used. */ content?: string diff --git a/packages/core/tests/page/resolvePageFileContent.spec.ts b/packages/core/tests/page/resolvePageContent.spec.ts similarity index 51% rename from packages/core/tests/page/resolvePageFileContent.spec.ts rename to packages/core/tests/page/resolvePageContent.spec.ts index 41f92f9132..f7a22b50b4 100644 --- a/packages/core/tests/page/resolvePageFileContent.spec.ts +++ b/packages/core/tests/page/resolvePageContent.spec.ts @@ -1,10 +1,10 @@ import { fs, path } from '@vuepress/utils' import { expect, it } from 'vitest' -import { resolvePageFileContent } from '../../src/index.js' +import { resolvePageContent } from '../../src/index.js' it('should resolve file content correctly from file path', async () => { const filePath = path.resolve(__dirname, '../__fixtures__/pages/foo.md') - const resolved = await resolvePageFileContent({ filePath, options: {} }) + const resolved = await resolvePageContent({ filePath, options: {} }) const expected = (await fs.readFile(filePath)).toString() expect(resolved).toBe(expected) @@ -12,24 +12,40 @@ it('should resolve file content correctly from file path', async () => { it('should use content from page options', async () => { const content = 'foobar' - const resolved = await resolvePageFileContent({ + const resolved = await resolvePageContent({ filePath: null, options: { content }, }) - expect(resolved).toBe(resolved) + + const expected = content + expect(resolved).toBe(expected) }) it('should return empty string if nothing provided', async () => { - const resolved = await resolvePageFileContent({ + const resolved = await resolvePageContent({ filePath: null, options: {}, }) - expect(resolved).toBe('') + + const expected = '' + expect(resolved).toBe(expected) +}) + +it('should use content from page options and ignore file path', async () => { + const filePath = path.resolve(__dirname, '../__fixtures__/pages/foo.md') + const content = 'foobar' + const resolved = await resolvePageContent({ + filePath, + options: { content }, + }) + + const expected = content + expect(resolved).toBe(expected) }) it('should throw error if the file does not exist', async () => { try { - await resolvePageFileContent({ + await resolvePageContent({ filePath: '404', options: {}, }) From e7b4e27f9bedb50e70abd8a4e76d7c5ff53f817c Mon Sep 17 00:00:00 2001 From: meteorlxy <meteor.lxy@foxmail.com> Date: Wed, 11 Sep 2024 12:34:59 +0800 Subject: [PATCH 03/11] chore: tweaks --- eslint.config.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/eslint.config.ts b/eslint.config.ts index 1cb1edfb7b..77ca52bc6c 100644 --- a/eslint.config.ts +++ b/eslint.config.ts @@ -19,8 +19,8 @@ export default vuepress( allow: [ '__dirname', '_context', - '_pageData', '_pageChunk', + '_pageData', '_registeredComponents', ], }, From 3bb73dcc3f8148420e127af155abd6a0f773398f Mon Sep 17 00:00:00 2001 From: meteorlxy <meteor.lxy@foxmail.com> Date: Wed, 11 Sep 2024 12:47:12 +0800 Subject: [PATCH 04/11] refactor: improve hmr code injection --- .../core/src/app/prepare/prepareRoutes.ts | 21 ++++++----- .../core/src/app/prepare/prepareSiteData.ts | 12 +++---- packages/core/src/page/renderPageToVue.ts | 35 +++++++++++++------ 3 files changed, 41 insertions(+), 27 deletions(-) diff --git a/packages/core/src/app/prepare/prepareRoutes.ts b/packages/core/src/app/prepare/prepareRoutes.ts index b3556e8041..c60d840425 100644 --- a/packages/core/src/app/prepare/prepareRoutes.ts +++ b/packages/core/src/app/prepare/prepareRoutes.ts @@ -1,21 +1,20 @@ import { normalizeRoutePath } from '@vuepress/shared' import type { App, Page } from '../../types/index.js' +const ROUTES_VAR_NAME = 'routes' +const REDIRECTS_VAR_NAME = 'redirects' + const HMR_CODE = ` if (import.meta.webpackHot) { import.meta.webpackHot.accept() - if (__VUE_HMR_RUNTIME__.updateRoutes) { - __VUE_HMR_RUNTIME__.updateRoutes(routes) - } - if (__VUE_HMR_RUNTIME__.updateRedirects) { - __VUE_HMR_RUNTIME__.updateRedirects(redirects) - } + __VUE_HMR_RUNTIME__.updateRoutes?.(${ROUTES_VAR_NAME}) + __VUE_HMR_RUNTIME__.updateRedirects?.(${REDIRECTS_VAR_NAME}}) } if (import.meta.hot) { - import.meta.hot.accept(({ routes, redirects }) => { - __VUE_HMR_RUNTIME__.updateRoutes(routes) - __VUE_HMR_RUNTIME__.updateRedirects(redirects) + import.meta.hot.accept((m) => { + __VUE_HMR_RUNTIME__.updateRoutes?.(m.${ROUTES_VAR_NAME}) + __VUE_HMR_RUNTIME__.updateRedirects?.(m.${REDIRECTS_VAR_NAME}) }) } ` @@ -47,7 +46,7 @@ const resolvePageRedirects = ({ path, pathInferred }: Page): string[] => { export const prepareRoutes = async (app: App): Promise<void> => { // routes file content let content = `\ -export const redirects = JSON.parse(${JSON.stringify( +export const ${REDIRECTS_VAR_NAME} = JSON.parse(${JSON.stringify( JSON.stringify( Object.fromEntries( app.pages.flatMap((page) => @@ -57,7 +56,7 @@ export const redirects = JSON.parse(${JSON.stringify( ), )}) -export const routes = Object.fromEntries([ +export const ${ROUTES_VAR_NAME} = Object.fromEntries([ ${app.pages .map( ({ chunkFilePath, chunkName, path, routeMeta }) => diff --git a/packages/core/src/app/prepare/prepareSiteData.ts b/packages/core/src/app/prepare/prepareSiteData.ts index c2749c0dc2..af0027fdd9 100644 --- a/packages/core/src/app/prepare/prepareSiteData.ts +++ b/packages/core/src/app/prepare/prepareSiteData.ts @@ -1,16 +1,16 @@ import type { App } from '../../types/index.js' +const SITE_DATA_VAR_NAME = 'siteData' + const HMR_CODE = ` if (import.meta.webpackHot) { import.meta.webpackHot.accept() - if (__VUE_HMR_RUNTIME__.updateSiteData) { - __VUE_HMR_RUNTIME__.updateSiteData(siteData) - } + __VUE_HMR_RUNTIME__.updateSiteData?.(${SITE_DATA_VAR_NAME}) } if (import.meta.hot) { - import.meta.hot.accept(({ siteData }) => { - __VUE_HMR_RUNTIME__.updateSiteData(siteData) + import.meta.hot.accept((m) => { + __VUE_HMR_RUNTIME__.updateSiteData?.(m.${SITE_DATA_VAR_NAME}) }) } ` @@ -20,7 +20,7 @@ if (import.meta.hot) { */ export const prepareSiteData = async (app: App): Promise<void> => { let content = `\ -export const siteData = JSON.parse(${JSON.stringify( +export const ${SITE_DATA_VAR_NAME} = JSON.parse(${JSON.stringify( JSON.stringify(app.siteData), )}) ` diff --git a/packages/core/src/page/renderPageToVue.ts b/packages/core/src/page/renderPageToVue.ts index 849e647d4c..14f6fb8311 100644 --- a/packages/core/src/page/renderPageToVue.ts +++ b/packages/core/src/page/renderPageToVue.ts @@ -1,5 +1,5 @@ import { isString } from '@vuepress/shared' -import type { Page } from '../types/index.js' +import type { App, Page } from '../types/index.js' const TEMPLATE_WRAPPER_TAG_OPEN = '<div>' const TEMPLATE_WRAPPER_TAG_CLOSE = '</div>' @@ -10,21 +10,32 @@ const SCRIPT_TAG_CLOSE = '</script>' const SCRIPT_TAG_OPEN_LANG_TS_REGEX = /<\s*script[^>]*\blang=['"]ts['"][^>]*/ const SCRIPT_TAG_OPEN_LANG_TS = '<script lang="ts">' +const PAGE_DATA_CODE_VAR_NAME = '_pageData' +const PAGE_DATA_CODE_TEMPLATE_OUTLET = '__PAGE_DATA__' +const PAGE_DATA_CODE_TEMPLATE = `export const ${PAGE_DATA_CODE_VAR_NAME} = JSON.parse(${PAGE_DATA_CODE_TEMPLATE_OUTLET})` + const HMR_CODE = ` if (import.meta.webpackHot) { import.meta.webpackHot.accept() - if (__VUE_HMR_RUNTIME__.updatePageData) { - __VUE_HMR_RUNTIME__.updatePageData(_pageData) - } + __VUE_HMR_RUNTIME__.updatePageData?.(${PAGE_DATA_CODE_VAR_NAME}) } if (import.meta.hot) { - import.meta.hot.accept(({ _pageData }) => { - __VUE_HMR_RUNTIME__.updatePageData(_pageData) + import.meta.hot.accept((m) => { + __VUE_HMR_RUNTIME__.updatePageData?.(m.${PAGE_DATA_CODE_VAR_NAME}) }) } ` +/** + * Util to resolve the page data code + */ +const resolvePageDataCode = (data: Page['data']): string => + PAGE_DATA_CODE_TEMPLATE.replace( + PAGE_DATA_CODE_TEMPLATE_OUTLET, + JSON.stringify(JSON.stringify(data)), + ) + /** * Util to resolve the open tag of script block */ @@ -43,7 +54,10 @@ const resolveScriptTagOpen = (sfcBlocks: Page['sfcBlocks']): string => { /** * Render page to vue component */ -export const renderPageToVue = ({ data, sfcBlocks }: Page): string => { +export const renderPageToVue = ( + app: App, + { data, sfcBlocks }: Page, +): string => { // #688: wrap the content of `<template>` with a `<div>` to avoid some potential issues of fragment component const templateContent = sfcBlocks.template && @@ -56,12 +70,13 @@ export const renderPageToVue = ({ data, sfcBlocks }: Page): string => { ].join('') // inject page data code and HMR code into the script content - const pageDataCode = `export const _pageData = JSON.parse(${JSON.stringify(JSON.stringify(data))})` + const scriptTagOpen = resolveScriptTagOpen(sfcBlocks) + const pageDataCode = resolvePageDataCode(data) const scriptContent = [ - resolveScriptTagOpen(sfcBlocks), + scriptTagOpen, sfcBlocks.script?.contentStripped, pageDataCode, - HMR_CODE, + app.env.isDev && HMR_CODE, sfcBlocks.script?.tagClose ?? SCRIPT_TAG_CLOSE, ] .filter(isString) From cb80a9e637a67165c8f7252c1fcceb7d6947d63f Mon Sep 17 00:00:00 2001 From: meteorlxy <meteor.lxy@foxmail.com> Date: Wed, 11 Sep 2024 12:48:45 +0800 Subject: [PATCH 05/11] chore: fix params --- packages/bundler-vite/src/plugins/vuepressMarkdownPlugin.ts | 6 +++--- .../bundler-webpack/src/loaders/vuepressMarkdownLoader.ts | 4 ++-- packages/core/src/app/prepare/preparePageChunk.ts | 2 +- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/packages/bundler-vite/src/plugins/vuepressMarkdownPlugin.ts b/packages/bundler-vite/src/plugins/vuepressMarkdownPlugin.ts index 94da9828c8..9dc2ec29c9 100644 --- a/packages/bundler-vite/src/plugins/vuepressMarkdownPlugin.ts +++ b/packages/bundler-vite/src/plugins/vuepressMarkdownPlugin.ts @@ -18,7 +18,7 @@ export const vuepressMarkdownPlugin = ({ app }: { app: App }): Plugin => ({ // if the page content is not changed, render it to vue component directly if (page?.content === code) { - return renderPageToVue(page) + return renderPageToVue(app, page) } // create a new page with the new content @@ -26,7 +26,7 @@ export const vuepressMarkdownPlugin = ({ app }: { app: App }): Plugin => ({ content: code, filePath: id, }) - return renderPageToVue(newPage) + return renderPageToVue(app, newPage) }, async handleHotUpdate(ctx) { @@ -41,6 +41,6 @@ export const vuepressMarkdownPlugin = ({ app }: { app: App }): Plugin => ({ filePath: ctx.file, }) - ctx.read = () => renderPageToVue(newPage) + ctx.read = () => renderPageToVue(app, newPage) }, }) diff --git a/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.ts b/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.ts index 819dc29f66..df071dfc90 100644 --- a/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.ts +++ b/packages/bundler-webpack/src/loaders/vuepressMarkdownLoader.ts @@ -21,7 +21,7 @@ export const vuepressMarkdownLoader: LoaderDefinitionFunction<VuepressMarkdownLo // if the page content is not changed, render it to vue component directly if (page?.content === source) { - return renderPageToVue(page) + return renderPageToVue(app, page) } // create a new page with the new content @@ -29,5 +29,5 @@ export const vuepressMarkdownLoader: LoaderDefinitionFunction<VuepressMarkdownLo content: source, filePath: this.resourcePath, }) - return renderPageToVue(newPage) + return renderPageToVue(app, newPage) } diff --git a/packages/core/src/app/prepare/preparePageChunk.ts b/packages/core/src/app/prepare/preparePageChunk.ts index 2ade465418..0dd2475008 100644 --- a/packages/core/src/app/prepare/preparePageChunk.ts +++ b/packages/core/src/app/prepare/preparePageChunk.ts @@ -6,6 +6,6 @@ import type { App, Page } from '../../types/index.js' */ export const preparePageChunk = async (app: App, page: Page): Promise<void> => { if (page.filePath === null) { - await app.writeTemp(page.chunkFilePathRelative, renderPageToVue(page)) + await app.writeTemp(page.chunkFilePathRelative, renderPageToVue(app, page)) } } From 5f156ae1ef2bb68ff8c004c07551267e7aa05fc4 Mon Sep 17 00:00:00 2001 From: meteorlxy <meteor.lxy@foxmail.com> Date: Wed, 11 Sep 2024 12:58:17 +0800 Subject: [PATCH 06/11] refactor: normalize module config --- .../src/config/handleModuleVue.ts | 25 ++++++++----------- 1 file changed, 10 insertions(+), 15 deletions(-) diff --git a/packages/bundler-webpack/src/config/handleModuleVue.ts b/packages/bundler-webpack/src/config/handleModuleVue.ts index 4065ff16a7..cca19e8fab 100644 --- a/packages/bundler-webpack/src/config/handleModuleVue.ts +++ b/packages/bundler-webpack/src/config/handleModuleVue.ts @@ -24,13 +24,15 @@ export const handleModuleVue = ({ isBuild: boolean isServer: boolean }): void => { - const applyVuePipeline = ({ - rule, - isMd, + const handleVue = ({ + lang, + test, }: { - rule: Config.Rule - isMd: boolean + lang: 'md' | 'vue' + test: RegExp }): void => { + const rule = config.module.rule(lang).test(test) + // use internal vuepress-ssr-loader to handle SSR dependencies if (isBuild) { rule @@ -50,7 +52,7 @@ export const handleModuleVue = ({ .end() // use internal vuepress-markdown-loader to handle markdown files - if (isMd) { + if (lang === 'md') { rule .use('vuepress-markdown-loader') .loader(require.resolve('#vuepress-markdown-loader')) @@ -59,15 +61,8 @@ export const handleModuleVue = ({ } } - applyVuePipeline({ - rule: config.module.rule('md').test(/\.md$/), - isMd: true, - }) - - applyVuePipeline({ - rule: config.module.rule('vue').test(/\.vue$/), - isMd: false, - }) + handleVue({ lang: 'md', test: /\.md$/ }) + handleVue({ lang: 'vue', test: /\.vue$/ }) // use vue-loader plugin config.plugin('vue-loader').use(VueLoaderPlugin) From e597cab94f909885639f772183502a1378f9565e Mon Sep 17 00:00:00 2001 From: meteorlxy <meteor.lxy@foxmail.com> Date: Wed, 11 Sep 2024 13:07:56 +0800 Subject: [PATCH 07/11] test: fix unit tests --- .../core/src/app/prepare/prepareRoutes.ts | 2 +- packages/core/src/types/page.ts | 242 +++++++++--------- packages/core/tests/page/createPage.spec.ts | 12 +- .../tests/page/resolvePageChunkInfo.spec.ts | 23 +- .../page/resolvePageComponentInfo.spec.ts | 38 --- 5 files changed, 144 insertions(+), 173 deletions(-) delete mode 100644 packages/core/tests/page/resolvePageComponentInfo.spec.ts diff --git a/packages/core/src/app/prepare/prepareRoutes.ts b/packages/core/src/app/prepare/prepareRoutes.ts index c60d840425..d449d237d5 100644 --- a/packages/core/src/app/prepare/prepareRoutes.ts +++ b/packages/core/src/app/prepare/prepareRoutes.ts @@ -8,7 +8,7 @@ const HMR_CODE = ` if (import.meta.webpackHot) { import.meta.webpackHot.accept() __VUE_HMR_RUNTIME__.updateRoutes?.(${ROUTES_VAR_NAME}) - __VUE_HMR_RUNTIME__.updateRedirects?.(${REDIRECTS_VAR_NAME}}) + __VUE_HMR_RUNTIME__.updateRedirects?.(${REDIRECTS_VAR_NAME}) } if (import.meta.hot) { diff --git a/packages/core/src/types/page.ts b/packages/core/src/types/page.ts index ade17a2fd8..561b283af0 100644 --- a/packages/core/src/types/page.ts +++ b/packages/core/src/types/page.ts @@ -4,133 +4,131 @@ import type { PageBase, PageData, PageFrontmatter } from '@vuepress/shared' /** * Vuepress Page */ -export type Page< +export interface Page< ExtraPageData extends Record<string, unknown> = Record<string, unknown>, ExtraPageFrontmatter extends Record<string, unknown> = Record< string, unknown >, - ExtraPageFields extends Record<string, unknown> = Record<string, unknown>, -> = ExtraPageFields & - PageBase<ExtraPageFrontmatter> & { - /** - * Data of the page, which will be available in client code - */ - data: PageData<ExtraPageData, ExtraPageFrontmatter> - - /** - * Raw Content of the page - */ - content: string - - /** - * Rendered content of the page - */ - contentRendered: string - - /** - * Date of the page, in 'yyyy-MM-dd' format - * - * @example '2020-09-09' - */ - date: string - - /** - * Dependencies of the page - */ - deps: string[] - - /** - * Links of the page - */ - links: MarkdownLink[] - - /** - * Markdown env object of the page - */ - markdownEnv: Record<string, unknown> - - /** - * Path of the page that inferred from file path - * - * If the page does not come from a file, it would be `null` - * - * @example '/guide/index.html' - */ - pathInferred: string | null - - /** - * Locale path prefix of the page - * - * @example '/getting-started.html' -> '/' - * @example '/en/getting-started.html' -> '/en/' - * @example '/zh/getting-started.html' -> '/zh/' - */ - pathLocale: string - - /** - * Permalink of the page - * - * If the page does not have a permalink, it would be `null` - */ - permalink: string | null - - /** - * Custom data to be attached to route record - */ - routeMeta: Record<string, unknown> - - /** - * Extracted sfc blocks of the page - */ - sfcBlocks: MarkdownSfcBlocks - - /** - * Slug of the page - */ - slug: string - - /** - * Source file path - * - * If the page does not come from a file, it would be `null` - */ - filePath: string | null - - /** - * Source file path relative to source directory - * - * If the page does not come from a file, it would be `null` - */ - filePathRelative: string | null - - /** - * Chunk file path - */ - chunkFilePath: string - - /** - * Chunk file path relative to temp directory - */ - chunkFilePathRelative: string - - /** - * Chunk name - * - * This will only take effect in webpack - */ - chunkName: string - - /** - * Rendered html file path - */ - htmlFilePath: string - - /** - * Rendered html file path relative to dest directory - */ - htmlFilePathRelative: string - } +> extends PageBase<ExtraPageFrontmatter> { + /** + * Data of the page, which will be available in client code + */ + data: PageData<ExtraPageData, ExtraPageFrontmatter> + + /** + * Raw Content of the page + */ + content: string + + /** + * Rendered content of the page + */ + contentRendered: string + + /** + * Date of the page, in 'yyyy-MM-dd' format + * + * @example '2020-09-09' + */ + date: string + + /** + * Dependencies of the page + */ + deps: string[] + + /** + * Links of the page + */ + links: MarkdownLink[] + + /** + * Markdown env object of the page + */ + markdownEnv: Record<string, unknown> + + /** + * Path of the page that inferred from file path + * + * If the page does not come from a file, it would be `null` + * + * @example '/guide/index.html' + */ + pathInferred: string | null + + /** + * Locale path prefix of the page + * + * @example '/getting-started.html' -> '/' + * @example '/en/getting-started.html' -> '/en/' + * @example '/zh/getting-started.html' -> '/zh/' + */ + pathLocale: string + + /** + * Permalink of the page + * + * If the page does not have a permalink, it would be `null` + */ + permalink: string | null + + /** + * Custom data to be attached to route record + */ + routeMeta: Record<string, unknown> + + /** + * Extracted sfc blocks of the page + */ + sfcBlocks: MarkdownSfcBlocks + + /** + * Slug of the page + */ + slug: string + + /** + * Source file path + * + * If the page does not come from a file, it would be `null` + */ + filePath: string | null + + /** + * Source file path relative to source directory + * + * If the page does not come from a file, it would be `null` + */ + filePathRelative: string | null + + /** + * Chunk file path + */ + chunkFilePath: string + + /** + * Chunk file path relative to temp directory + */ + chunkFilePathRelative: string + + /** + * Chunk name + * + * This will only take effect in webpack + */ + chunkName: string + + /** + * Rendered html file path + */ + htmlFilePath: string + + /** + * Rendered html file path relative to dest directory + */ + htmlFilePathRelative: string +} /** * Options to create vuepress page diff --git a/packages/core/tests/page/createPage.spec.ts b/packages/core/tests/page/createPage.spec.ts index 313fed3876..f15fe825c6 100644 --- a/packages/core/tests/page/createPage.spec.ts +++ b/packages/core/tests/page/createPage.spec.ts @@ -73,19 +73,13 @@ describe('should work without plugins', () => { expect(page.filePathRelative).toBeNull() expect(page.htmlFilePath).toBe(app.dir.dest(`index.html`)) expect(page.htmlFilePathRelative).toBe(`index.html`) - expect(page.componentFilePath).toBe( - app.dir.temp(`pages/${page.htmlFilePathRelative}.vue`), - ) - expect(page.componentFilePathRelative).toBe( - `pages/${page.htmlFilePathRelative}.vue`, - ) expect(page.chunkFilePath).toBe( - app.dir.temp(`pages/${page.htmlFilePathRelative}.js`), + app.dir.temp(`pages/${page.htmlFilePathRelative}.vue`), ) expect(page.chunkFilePathRelative).toBe( - `pages/${page.htmlFilePathRelative}.js`, + `pages/${page.htmlFilePathRelative}.vue`, ) - expect(page.chunkName).toBeTruthy() + expect(page.chunkName).toBe(`index.html`) }) }) diff --git a/packages/core/tests/page/resolvePageChunkInfo.spec.ts b/packages/core/tests/page/resolvePageChunkInfo.spec.ts index 738d3c1f05..bf4cc88274 100644 --- a/packages/core/tests/page/resolvePageChunkInfo.spec.ts +++ b/packages/core/tests/page/resolvePageChunkInfo.spec.ts @@ -9,15 +9,32 @@ const app = createBaseApp({ bundler: {} as Bundler, }) -it('should resolve page chunk info correctly', () => { +it('should resolve page chunk info correctly without source file path', () => { const resolved = resolvePageChunkInfo({ app, + filePath: null, + filePathRelative: null, htmlFilePathRelative: 'foo.html', }) expect(resolved).toEqual({ - chunkFilePath: app.dir.temp('pages/foo.html.js'), - chunkFilePathRelative: 'pages/foo.html.js', + chunkFilePath: app.dir.temp('pages/foo.html.vue'), + chunkFilePathRelative: 'pages/foo.html.vue', + chunkName: sanitizeFileName('foo.html'), + }) +}) + +it('should resolve page chunk info correctly with source file path', () => { + const resolved = resolvePageChunkInfo({ + app, + filePath: app.dir.source('foo.md'), + filePathRelative: 'foo.md', + htmlFilePathRelative: 'foo.html', + }) + + expect(resolved).toEqual({ + chunkFilePath: app.dir.source('foo.md'), + chunkFilePathRelative: 'foo.md', chunkName: sanitizeFileName('foo.html'), }) }) diff --git a/packages/core/tests/page/resolvePageComponentInfo.spec.ts b/packages/core/tests/page/resolvePageComponentInfo.spec.ts deleted file mode 100644 index c8a009e1ea..0000000000 --- a/packages/core/tests/page/resolvePageComponentInfo.spec.ts +++ /dev/null @@ -1,38 +0,0 @@ -import { path } from '@vuepress/utils' -import { expect, it } from 'vitest' -import type { Bundler } from '../../src/index.js' -import { createBaseApp, resolvePageComponentInfo } from '../../src/index.js' - -const app = createBaseApp({ - source: path.resolve(__dirname, 'fake-source'), - theme: { name: 'test' }, - bundler: {} as Bundler, -}) - -it('should resolve page component info correctly without source file path', () => { - const resolved = resolvePageComponentInfo({ - app, - filePath: null, - filePathRelative: null, - htmlFilePathRelative: 'foo.html', - }) - - expect(resolved).toEqual({ - componentFilePath: app.dir.temp('pages/foo.html.vue'), - componentFilePathRelative: 'pages/foo.html.vue', - }) -}) - -it('should resolve page component info correctly with source file path', () => { - const resolved = resolvePageComponentInfo({ - app, - filePath: app.dir.source('foo.md'), - filePathRelative: 'foo.md', - htmlFilePathRelative: 'foo.html', - }) - - expect(resolved).toEqual({ - componentFilePath: app.dir.source('foo.md'), - componentFilePathRelative: 'foo.md', - }) -}) From 0e85a0036c88955c4faae7b5b06398eeb4042e7b Mon Sep 17 00:00:00 2001 From: meteorlxy <meteor.lxy@foxmail.com> Date: Wed, 11 Sep 2024 13:18:20 +0800 Subject: [PATCH 08/11] chore: revert page type update --- packages/core/src/types/page.ts | 242 ++++++++++++++++---------------- 1 file changed, 122 insertions(+), 120 deletions(-) diff --git a/packages/core/src/types/page.ts b/packages/core/src/types/page.ts index 561b283af0..ade17a2fd8 100644 --- a/packages/core/src/types/page.ts +++ b/packages/core/src/types/page.ts @@ -4,131 +4,133 @@ import type { PageBase, PageData, PageFrontmatter } from '@vuepress/shared' /** * Vuepress Page */ -export interface Page< +export type Page< ExtraPageData extends Record<string, unknown> = Record<string, unknown>, ExtraPageFrontmatter extends Record<string, unknown> = Record< string, unknown >, -> extends PageBase<ExtraPageFrontmatter> { - /** - * Data of the page, which will be available in client code - */ - data: PageData<ExtraPageData, ExtraPageFrontmatter> - - /** - * Raw Content of the page - */ - content: string - - /** - * Rendered content of the page - */ - contentRendered: string - - /** - * Date of the page, in 'yyyy-MM-dd' format - * - * @example '2020-09-09' - */ - date: string - - /** - * Dependencies of the page - */ - deps: string[] - - /** - * Links of the page - */ - links: MarkdownLink[] - - /** - * Markdown env object of the page - */ - markdownEnv: Record<string, unknown> - - /** - * Path of the page that inferred from file path - * - * If the page does not come from a file, it would be `null` - * - * @example '/guide/index.html' - */ - pathInferred: string | null - - /** - * Locale path prefix of the page - * - * @example '/getting-started.html' -> '/' - * @example '/en/getting-started.html' -> '/en/' - * @example '/zh/getting-started.html' -> '/zh/' - */ - pathLocale: string - - /** - * Permalink of the page - * - * If the page does not have a permalink, it would be `null` - */ - permalink: string | null - - /** - * Custom data to be attached to route record - */ - routeMeta: Record<string, unknown> - - /** - * Extracted sfc blocks of the page - */ - sfcBlocks: MarkdownSfcBlocks - - /** - * Slug of the page - */ - slug: string - - /** - * Source file path - * - * If the page does not come from a file, it would be `null` - */ - filePath: string | null - - /** - * Source file path relative to source directory - * - * If the page does not come from a file, it would be `null` - */ - filePathRelative: string | null - - /** - * Chunk file path - */ - chunkFilePath: string - - /** - * Chunk file path relative to temp directory - */ - chunkFilePathRelative: string - - /** - * Chunk name - * - * This will only take effect in webpack - */ - chunkName: string - - /** - * Rendered html file path - */ - htmlFilePath: string - - /** - * Rendered html file path relative to dest directory - */ - htmlFilePathRelative: string -} + ExtraPageFields extends Record<string, unknown> = Record<string, unknown>, +> = ExtraPageFields & + PageBase<ExtraPageFrontmatter> & { + /** + * Data of the page, which will be available in client code + */ + data: PageData<ExtraPageData, ExtraPageFrontmatter> + + /** + * Raw Content of the page + */ + content: string + + /** + * Rendered content of the page + */ + contentRendered: string + + /** + * Date of the page, in 'yyyy-MM-dd' format + * + * @example '2020-09-09' + */ + date: string + + /** + * Dependencies of the page + */ + deps: string[] + + /** + * Links of the page + */ + links: MarkdownLink[] + + /** + * Markdown env object of the page + */ + markdownEnv: Record<string, unknown> + + /** + * Path of the page that inferred from file path + * + * If the page does not come from a file, it would be `null` + * + * @example '/guide/index.html' + */ + pathInferred: string | null + + /** + * Locale path prefix of the page + * + * @example '/getting-started.html' -> '/' + * @example '/en/getting-started.html' -> '/en/' + * @example '/zh/getting-started.html' -> '/zh/' + */ + pathLocale: string + + /** + * Permalink of the page + * + * If the page does not have a permalink, it would be `null` + */ + permalink: string | null + + /** + * Custom data to be attached to route record + */ + routeMeta: Record<string, unknown> + + /** + * Extracted sfc blocks of the page + */ + sfcBlocks: MarkdownSfcBlocks + + /** + * Slug of the page + */ + slug: string + + /** + * Source file path + * + * If the page does not come from a file, it would be `null` + */ + filePath: string | null + + /** + * Source file path relative to source directory + * + * If the page does not come from a file, it would be `null` + */ + filePathRelative: string | null + + /** + * Chunk file path + */ + chunkFilePath: string + + /** + * Chunk file path relative to temp directory + */ + chunkFilePathRelative: string + + /** + * Chunk name + * + * This will only take effect in webpack + */ + chunkName: string + + /** + * Rendered html file path + */ + htmlFilePath: string + + /** + * Rendered html file path relative to dest directory + */ + htmlFilePathRelative: string + } /** * Options to create vuepress page From d45c6781da0049a5c48eb277f2ac7eb73609ecbb Mon Sep 17 00:00:00 2001 From: meteorlxy <meteor.lxy@foxmail.com> Date: Wed, 11 Sep 2024 21:22:56 +0800 Subject: [PATCH 09/11] feat: inject default export --- packages/core/src/page/renderPageToVue.ts | 80 ++++++++++++++++------- 1 file changed, 56 insertions(+), 24 deletions(-) diff --git a/packages/core/src/page/renderPageToVue.ts b/packages/core/src/page/renderPageToVue.ts index 14f6fb8311..d6a7c021a7 100644 --- a/packages/core/src/page/renderPageToVue.ts +++ b/packages/core/src/page/renderPageToVue.ts @@ -1,4 +1,5 @@ import { isString } from '@vuepress/shared' +import { path } from '@vuepress/utils' import type { App, Page } from '../types/index.js' const TEMPLATE_WRAPPER_TAG_OPEN = '<div>' @@ -10,6 +11,13 @@ const SCRIPT_TAG_CLOSE = '</script>' const SCRIPT_TAG_OPEN_LANG_TS_REGEX = /<\s*script[^>]*\blang=['"]ts['"][^>]*/ const SCRIPT_TAG_OPEN_LANG_TS = '<script lang="ts">' +const SCRIPT_DEFAULT_EXPORT_REGEX = /((?:^|\n|;)\s*)export(\s*)default/ +const SCRIPT_DEFAULT_NAMED_EXPORT_REGEX = + /((?:^|\n|;)\s*)export(.+)as(\s*)default/ + +const SCRIPT_DEFAULT_EXPORT_CODE_TEMPLATE_OUTLET = '__SCRIPT_DEFAULT_EXPORT__' +const SCRIPT_DEFAULT_EXPORT_CODE_TEMPLATE = `export default { name: ${SCRIPT_DEFAULT_EXPORT_CODE_TEMPLATE_OUTLET} }` + const PAGE_DATA_CODE_VAR_NAME = '_pageData' const PAGE_DATA_CODE_TEMPLATE_OUTLET = '__PAGE_DATA__' const PAGE_DATA_CODE_TEMPLATE = `export const ${PAGE_DATA_CODE_VAR_NAME} = JSON.parse(${PAGE_DATA_CODE_TEMPLATE_OUTLET})` @@ -27,37 +35,65 @@ if (import.meta.hot) { } ` -/** - * Util to resolve the page data code - */ -const resolvePageDataCode = (data: Page['data']): string => - PAGE_DATA_CODE_TEMPLATE.replace( - PAGE_DATA_CODE_TEMPLATE_OUTLET, - JSON.stringify(JSON.stringify(data)), - ) - /** * Util to resolve the open tag of script block */ -const resolveScriptTagOpen = (sfcBlocks: Page['sfcBlocks']): string => { +const resolveScriptTagOpen = (page: Page): string => { // use existing script open tag - if (sfcBlocks.script?.tagOpen) { - return sfcBlocks.script.tagOpen + if (page.sfcBlocks.script?.tagOpen) { + return page.sfcBlocks.script.tagOpen } // if the setup script block is using typescript, we should use the same language for script block - const isUsingTs = sfcBlocks.scriptSetup?.tagOpen.match( + const isUsingTs = page.sfcBlocks.scriptSetup?.tagOpen.match( SCRIPT_TAG_OPEN_LANG_TS_REGEX, ) return isUsingTs ? SCRIPT_TAG_OPEN_LANG_TS : SCRIPT_TAG_OPEN } +/** + * Util to resolve the default export code + */ +const resolveDefaultExportCode = (page: Page): string => + SCRIPT_DEFAULT_EXPORT_CODE_TEMPLATE.replace( + SCRIPT_DEFAULT_EXPORT_CODE_TEMPLATE_OUTLET, + JSON.stringify(path.basename(page.chunkFilePath)), + ) + +/** + * Util to resolve the page data code + */ +const resolvePageDataCode = (page: Page): string => + PAGE_DATA_CODE_TEMPLATE.replace( + PAGE_DATA_CODE_TEMPLATE_OUTLET, + JSON.stringify(JSON.stringify(page.data)), + ) + +/** + * Resolve the stripped content of script block + */ +const resolveScriptContentStripped = (app: App, page: Page): string => { + const rawContentStripped = page.sfcBlocks.script?.contentStripped + const hasDefaultExport = rawContentStripped + ? SCRIPT_DEFAULT_EXPORT_REGEX.test(rawContentStripped) || + SCRIPT_DEFAULT_NAMED_EXPORT_REGEX.test(rawContentStripped) + : false + return [ + rawContentStripped, + resolvePageDataCode(page), // inject page data code + !hasDefaultExport && resolveDefaultExportCode(page), // inject default export with component name + app.env.isDev && HMR_CODE, // inject HMR code in dev mode + ] + .filter(isString) + .join('\n') +} + /** * Render page to vue component */ -export const renderPageToVue = ( - app: App, - { data, sfcBlocks }: Page, -): string => { +export const renderPageToVue = (app: App, page: Page): string => { + const { sfcBlocks } = page + + // get the content of template block // #688: wrap the content of `<template>` with a `<div>` to avoid some potential issues of fragment component const templateContent = sfcBlocks.template && @@ -69,14 +105,10 @@ export const renderPageToVue = ( sfcBlocks.template.tagClose, ].join('') - // inject page data code and HMR code into the script content - const scriptTagOpen = resolveScriptTagOpen(sfcBlocks) - const pageDataCode = resolvePageDataCode(data) + // get the content of script block const scriptContent = [ - scriptTagOpen, - sfcBlocks.script?.contentStripped, - pageDataCode, - app.env.isDev && HMR_CODE, + resolveScriptTagOpen(page), + resolveScriptContentStripped(app, page), sfcBlocks.script?.tagClose ?? SCRIPT_TAG_CLOSE, ] .filter(isString) From e4211ba52a88be122d9601d35fcaea9dac4a8f72 Mon Sep 17 00:00:00 2001 From: meteorlxy <meteor.lxy@foxmail.com> Date: Wed, 11 Sep 2024 21:30:38 +0800 Subject: [PATCH 10/11] chore: tweaks --- packages/cli/src/commands/dev/handlePageAdd.ts | 6 ++---- packages/cli/src/commands/dev/handlePageChange.ts | 4 +--- 2 files changed, 3 insertions(+), 7 deletions(-) diff --git a/packages/cli/src/commands/dev/handlePageAdd.ts b/packages/cli/src/commands/dev/handlePageAdd.ts index 4c8d7c2512..0d91181927 100644 --- a/packages/cli/src/commands/dev/handlePageAdd.ts +++ b/packages/cli/src/commands/dev/handlePageAdd.ts @@ -17,9 +17,7 @@ export const handlePageAdd = async ( } // create page - const page = await createPage(app, { - filePath, - }) + const page = await createPage(app, { filePath }) // add the new page app.pages.push(page) @@ -28,7 +26,7 @@ export const handlePageAdd = async ( // prepare page file await preparePageChunk(app, page) - // prepare routes file + // re-prepare routes file await prepareRoutes(app) return page diff --git a/packages/cli/src/commands/dev/handlePageChange.ts b/packages/cli/src/commands/dev/handlePageChange.ts index c7e29debf6..84df9d5d4c 100644 --- a/packages/cli/src/commands/dev/handlePageChange.ts +++ b/packages/cli/src/commands/dev/handlePageChange.ts @@ -20,9 +20,7 @@ export const handlePageChange = async ( const pageOld = app.pages[pageIndex] // create a new page from the changed file - const pageNew = await createPage(app, { - filePath, - }) + const pageNew = await createPage(app, { filePath }) // replace the old page with the new page app.pages.splice(pageIndex, 1, pageNew) From a0da5333d46677a5319a980f16ebde6074a70213 Mon Sep 17 00:00:00 2001 From: meteorlxy <meteor.lxy@foxmail.com> Date: Fri, 13 Sep 2024 01:53:31 +0800 Subject: [PATCH 11/11] refactor(core): improve comments --- e2e/docs/.vuepress/plugins/foo/fooPlugin.ts | 3 +- packages/core/src/app/appInit.ts | 2 + packages/core/src/app/appPrepare.ts | 2 + packages/core/src/app/appUse.ts | 7 ++ packages/core/src/app/createBaseApp.ts | 6 +- packages/core/src/app/createBuildApp.ts | 2 +- packages/core/src/app/createDevApp.ts | 2 +- .../src/app/prepare/prepareClientConfigs.ts | 2 + .../core/src/app/prepare/prepareRoutes.ts | 2 + .../core/src/app/prepare/prepareSiteData.ts | 2 + packages/core/src/app/resolveAppDir.ts | 2 + packages/core/src/app/resolveAppEnv.ts | 2 + packages/core/src/app/resolveAppMarkdown.ts | 2 + packages/core/src/app/resolveAppOptions.ts | 2 + packages/core/src/app/resolveAppPages.ts | 2 + packages/core/src/app/resolveAppSiteData.ts | 2 + packages/core/src/app/resolveAppVersion.ts | 2 + packages/core/src/app/resolveAppWriteTemp.ts | 2 + packages/core/src/app/resolvePluginObject.ts | 2 + packages/core/src/app/resolveThemeInfo.ts | 2 + .../core/src/app/setupAppThemeAndPlugins.ts | 2 + packages/core/src/page/createPage.ts | 3 + packages/core/src/page/inferPagePath.ts | 2 + packages/core/src/page/parsePageContent.ts | 2 + .../core/src/page/renderPageSfcBlocksToVue.ts | 2 + .../core/src/page/resolvePageChunkInfo.ts | 2 + .../core/src/page/resolvePageComponentInfo.ts | 2 + packages/core/src/page/resolvePageContent.ts | 2 + packages/core/src/page/resolvePageDate.ts | 2 + packages/core/src/page/resolvePageFilePath.ts | 2 + packages/core/src/page/resolvePageHtmlInfo.ts | 2 + packages/core/src/page/resolvePageLang.ts | 2 + packages/core/src/page/resolvePagePath.ts | 2 + .../core/src/page/resolvePagePermalink.ts | 2 + .../core/src/page/resolvePageRouteMeta.ts | 2 + packages/core/src/page/resolvePageSlug.ts | 2 + .../core/src/pluginApi/createHookQueue.ts | 2 + .../core/src/pluginApi/createPluginApi.ts | 5 ++ .../src/pluginApi/createPluginApiHooks.ts | 5 ++ .../pluginApi/createPluginApiRegisterHooks.ts | 5 ++ .../src/pluginApi/normalizeAliasDefineHook.ts | 2 + .../normalizeClientConfigFileHook.ts | 2 + packages/core/src/types/app/options.ts | 90 ++++++++++++++++++- packages/core/src/types/app/utils.ts | 23 +++++ packages/core/src/types/bundler.ts | 11 +++ packages/core/src/types/plugin.ts | 8 +- 46 files changed, 227 insertions(+), 9 deletions(-) diff --git a/e2e/docs/.vuepress/plugins/foo/fooPlugin.ts b/e2e/docs/.vuepress/plugins/foo/fooPlugin.ts index 93ef0c8cff..a6031278ff 100644 --- a/e2e/docs/.vuepress/plugins/foo/fooPlugin.ts +++ b/e2e/docs/.vuepress/plugins/foo/fooPlugin.ts @@ -1,8 +1,9 @@ +import type { Plugin } from 'vuepress/core' import { getDirname, path } from 'vuepress/utils' const __dirname = getDirname(import.meta.url) -export const fooPlugin = { +export const fooPlugin: Plugin = { name: 'test-plugin', clientConfigFile: path.resolve( __dirname, diff --git a/packages/core/src/app/appInit.ts b/packages/core/src/app/appInit.ts index 1db287fd28..95ce110178 100644 --- a/packages/core/src/app/appInit.ts +++ b/packages/core/src/app/appInit.ts @@ -9,6 +9,8 @@ const log = debug('vuepress:core/app') * Initialize a vuepress app * * Plugins should be used before initialization. + * + * @internal */ export const appInit = async (app: App): Promise<void> => { log('init start') diff --git a/packages/core/src/app/appPrepare.ts b/packages/core/src/app/appPrepare.ts index f8d2585f50..15dd6986b7 100644 --- a/packages/core/src/app/appPrepare.ts +++ b/packages/core/src/app/appPrepare.ts @@ -18,6 +18,8 @@ const log = debug('vuepress:core/app') * - routes * - site data * - other files that generated by plugins + * + * @internal */ export const appPrepare = async (app: App): Promise<void> => { log('prepare start') diff --git a/packages/core/src/app/appUse.ts b/packages/core/src/app/appUse.ts index df53e43cab..9c98443a39 100644 --- a/packages/core/src/app/appUse.ts +++ b/packages/core/src/app/appUse.ts @@ -4,6 +4,13 @@ import { resolvePluginObject } from './resolvePluginObject.js' const log = debug('vuepress:core/app') +/** + * Use a plugin in vuepress app. + * + * Should be called before initialization. + * + * @internal + */ export const appUse = (app: App, rawPlugin: Plugin): App => { const pluginObject = resolvePluginObject(app, rawPlugin) diff --git a/packages/core/src/app/createBaseApp.ts b/packages/core/src/app/createBaseApp.ts index dc2b2ecaf4..dc8b18750d 100644 --- a/packages/core/src/app/createBaseApp.ts +++ b/packages/core/src/app/createBaseApp.ts @@ -17,7 +17,11 @@ import { resolveAppWriteTemp } from './resolveAppWriteTemp.js' import { setupAppThemeAndPlugins } from './setupAppThemeAndPlugins.js' /** - * Create vuepress app + * Create base vuepress app. + * + * Notice that the base app could not be used for dev nor build. + * + * It would be used for creating dev app or build app, or for testing. */ export const createBaseApp = (config: AppConfig): App => { const options = resolveAppOptions(config) diff --git a/packages/core/src/app/createBuildApp.ts b/packages/core/src/app/createBuildApp.ts index 2bc29eddff..13e1651b8e 100644 --- a/packages/core/src/app/createBuildApp.ts +++ b/packages/core/src/app/createBuildApp.ts @@ -2,7 +2,7 @@ import type { AppConfig, BuildApp } from '../types/index.js' import { createBaseApp } from './createBaseApp.js' /** - * Create vuepress build app + * Create vuepress build app. */ export const createBuildApp = (config: AppConfig): BuildApp => { const app = createBaseApp(config) as BuildApp diff --git a/packages/core/src/app/createDevApp.ts b/packages/core/src/app/createDevApp.ts index 3a0f4003e6..3d21ea35ed 100644 --- a/packages/core/src/app/createDevApp.ts +++ b/packages/core/src/app/createDevApp.ts @@ -2,7 +2,7 @@ import type { AppConfig, DevApp } from '../types/index.js' import { createBaseApp } from './createBaseApp.js' /** - * Create vuepress dev app + * Create vuepress dev app. */ export const createDevApp = (config: AppConfig): DevApp => { const app = createBaseApp(config) as DevApp diff --git a/packages/core/src/app/prepare/prepareClientConfigs.ts b/packages/core/src/app/prepare/prepareClientConfigs.ts index 59bb1f5624..4636bd5556 100644 --- a/packages/core/src/app/prepare/prepareClientConfigs.ts +++ b/packages/core/src/app/prepare/prepareClientConfigs.ts @@ -2,6 +2,8 @@ import type { App } from '../../types/index.js' /** * Generate client configs temp file + * + * @internal */ export const prepareClientConfigs = async (app: App): Promise<void> => { // plugin hook: clientConfigFile diff --git a/packages/core/src/app/prepare/prepareRoutes.ts b/packages/core/src/app/prepare/prepareRoutes.ts index b3556e8041..a4f8d6f549 100644 --- a/packages/core/src/app/prepare/prepareRoutes.ts +++ b/packages/core/src/app/prepare/prepareRoutes.ts @@ -22,6 +22,8 @@ if (import.meta.hot) { /** * Resolve page redirects + * + * @internal */ const resolvePageRedirects = ({ path, pathInferred }: Page): string[] => { // paths that should redirect to this page, use set to dedupe diff --git a/packages/core/src/app/prepare/prepareSiteData.ts b/packages/core/src/app/prepare/prepareSiteData.ts index c2749c0dc2..7121e72845 100644 --- a/packages/core/src/app/prepare/prepareSiteData.ts +++ b/packages/core/src/app/prepare/prepareSiteData.ts @@ -17,6 +17,8 @@ if (import.meta.hot) { /** * Generate site data temp file + * + * @internal */ export const prepareSiteData = async (app: App): Promise<void> => { let content = `\ diff --git a/packages/core/src/app/resolveAppDir.ts b/packages/core/src/app/resolveAppDir.ts index 5e78b9d344..18819dca9d 100644 --- a/packages/core/src/app/resolveAppDir.ts +++ b/packages/core/src/app/resolveAppDir.ts @@ -6,6 +6,8 @@ const require = createRequire(import.meta.url) /** * Create directory util function + * + * @internal */ export const createAppDirFunction = (baseDir: string): AppDirFunction => diff --git a/packages/core/src/app/resolveAppEnv.ts b/packages/core/src/app/resolveAppEnv.ts index 7ec4bf96c0..d0b069c6a4 100644 --- a/packages/core/src/app/resolveAppEnv.ts +++ b/packages/core/src/app/resolveAppEnv.ts @@ -2,6 +2,8 @@ import type { AppEnv, AppOptions } from '../types/index.js' /** * Resolve environment flags for vuepress app + * + * @internal */ export const resolveAppEnv = (options: AppOptions): AppEnv => ({ isBuild: false, diff --git a/packages/core/src/app/resolveAppMarkdown.ts b/packages/core/src/app/resolveAppMarkdown.ts index 84ce7d0a1a..d3b35b4ef2 100644 --- a/packages/core/src/app/resolveAppMarkdown.ts +++ b/packages/core/src/app/resolveAppMarkdown.ts @@ -4,6 +4,8 @@ import type { App } from '../types/index.js' /** * Resolve markdown-it instance for vuepress app + * + * @internal */ export const resolveAppMarkdown = async (app: App): Promise<Markdown> => { // plugin hook: extendsMarkdownOptions diff --git a/packages/core/src/app/resolveAppOptions.ts b/packages/core/src/app/resolveAppOptions.ts index 63b9379962..756227ffe5 100644 --- a/packages/core/src/app/resolveAppOptions.ts +++ b/packages/core/src/app/resolveAppOptions.ts @@ -6,6 +6,8 @@ const require = createRequire(import.meta.url) /** * Create app options with default values + * + * @internal */ export const resolveAppOptions = ({ // site config diff --git a/packages/core/src/app/resolveAppPages.ts b/packages/core/src/app/resolveAppPages.ts index d8cd00bd41..51ed100308 100644 --- a/packages/core/src/app/resolveAppPages.ts +++ b/packages/core/src/app/resolveAppPages.ts @@ -6,6 +6,8 @@ const log = debug('vuepress:core/app') /** * Resolve pages for vuepress app + * + * @internal */ export const resolveAppPages = async (app: App): Promise<Page[]> => { log('resolveAppPages start') diff --git a/packages/core/src/app/resolveAppSiteData.ts b/packages/core/src/app/resolveAppSiteData.ts index 9f2b654e3e..bc37720131 100644 --- a/packages/core/src/app/resolveAppSiteData.ts +++ b/packages/core/src/app/resolveAppSiteData.ts @@ -4,6 +4,8 @@ import type { AppOptions, SiteData } from '../types/index.js' * Resolve site data for vuepress app * * Site data will also be used in client + * + * @internal */ export const resolveAppSiteData = (options: AppOptions): SiteData => ({ base: options.base, diff --git a/packages/core/src/app/resolveAppVersion.ts b/packages/core/src/app/resolveAppVersion.ts index 590340ffc4..5f356432a3 100644 --- a/packages/core/src/app/resolveAppVersion.ts +++ b/packages/core/src/app/resolveAppVersion.ts @@ -5,6 +5,8 @@ const require = createRequire(import.meta.url) /** * Resolve version of vuepress app + * + * @internal */ export const resolveAppVersion = (): string => { const pkgJson = fs.readJsonSync( diff --git a/packages/core/src/app/resolveAppWriteTemp.ts b/packages/core/src/app/resolveAppWriteTemp.ts index f4e3b6c7d5..2a3560f072 100644 --- a/packages/core/src/app/resolveAppWriteTemp.ts +++ b/packages/core/src/app/resolveAppWriteTemp.ts @@ -3,6 +3,8 @@ import type { AppDir, AppWriteTemp } from '../types/index.js' /** * Resolve write temp file util for vuepress app + * + * @internal */ export const resolveAppWriteTemp = (dir: AppDir): AppWriteTemp => { const writeTemp: AppWriteTemp = async (file: string, content: string) => { diff --git a/packages/core/src/app/resolvePluginObject.ts b/packages/core/src/app/resolvePluginObject.ts index 8ccdd72aaa..4636f48ac0 100644 --- a/packages/core/src/app/resolvePluginObject.ts +++ b/packages/core/src/app/resolvePluginObject.ts @@ -3,6 +3,8 @@ import type { App, Plugin, PluginObject } from '../types/index.js' /** * Resolve a plugin object according to name / path / module and config + * + * @internal */ export const resolvePluginObject = <T extends PluginObject = PluginObject>( app: App, diff --git a/packages/core/src/app/resolveThemeInfo.ts b/packages/core/src/app/resolveThemeInfo.ts index 34e9ff3b0d..b16c371429 100644 --- a/packages/core/src/app/resolveThemeInfo.ts +++ b/packages/core/src/app/resolveThemeInfo.ts @@ -3,6 +3,8 @@ import { resolvePluginObject } from './resolvePluginObject.js' /** * Resolve theme info and its parent theme info + * + * @internal */ export const resolveThemeInfo = (app: App, theme: Theme): ThemeInfo => { // resolve current theme info diff --git a/packages/core/src/app/setupAppThemeAndPlugins.ts b/packages/core/src/app/setupAppThemeAndPlugins.ts index f28c9f04e7..39aceda92b 100644 --- a/packages/core/src/app/setupAppThemeAndPlugins.ts +++ b/packages/core/src/app/setupAppThemeAndPlugins.ts @@ -3,6 +3,8 @@ import { resolveThemeInfo } from './resolveThemeInfo.js' /** * Setup theme and plugins for vuepress app + * + * @internal */ export const setupAppThemeAndPlugins = (app: App, config: AppConfig): void => { // recursively resolve theme info diff --git a/packages/core/src/page/createPage.ts b/packages/core/src/page/createPage.ts index 600d60d63d..2e9946dec2 100644 --- a/packages/core/src/page/createPage.ts +++ b/packages/core/src/page/createPage.ts @@ -13,6 +13,9 @@ import { resolvePagePermalink } from './resolvePagePermalink.js' import { resolvePageRouteMeta } from './resolvePageRouteMeta.js' import { resolvePageSlug } from './resolvePageSlug.js' +/** + * Create vuepress page object + */ export const createPage = async ( app: App, options: PageOptions, diff --git a/packages/core/src/page/inferPagePath.ts b/packages/core/src/page/inferPagePath.ts index 8d082ea93c..2a406c7b66 100644 --- a/packages/core/src/page/inferPagePath.ts +++ b/packages/core/src/page/inferPagePath.ts @@ -7,6 +7,8 @@ import type { App } from '../types/index.js' /** * Infer page path according to file path + * + * @internal */ export const inferPagePath = ({ app, diff --git a/packages/core/src/page/parsePageContent.ts b/packages/core/src/page/parsePageContent.ts index 8af9d92b96..03e036a87e 100644 --- a/packages/core/src/page/parsePageContent.ts +++ b/packages/core/src/page/parsePageContent.ts @@ -9,6 +9,8 @@ import type { App, PageFrontmatter, PageOptions } from '../types/index.js' /** * Render page content and extract related info + * + * @internal */ export const parsePageContent = ({ app, diff --git a/packages/core/src/page/renderPageSfcBlocksToVue.ts b/packages/core/src/page/renderPageSfcBlocksToVue.ts index 7428643823..79f15e92ca 100644 --- a/packages/core/src/page/renderPageSfcBlocksToVue.ts +++ b/packages/core/src/page/renderPageSfcBlocksToVue.ts @@ -2,6 +2,8 @@ import type { MarkdownSfcBlocks } from '@vuepress/markdown' /** * Render page sfc blocks to vue component + * + * @internal */ export const renderPageSfcBlocksToVue = ( sfcBlocks: MarkdownSfcBlocks, diff --git a/packages/core/src/page/resolvePageChunkInfo.ts b/packages/core/src/page/resolvePageChunkInfo.ts index 037018c6cc..0d5c6b994f 100644 --- a/packages/core/src/page/resolvePageChunkInfo.ts +++ b/packages/core/src/page/resolvePageChunkInfo.ts @@ -3,6 +3,8 @@ import type { App } from '../types/index.js' /** * Resolve page data file path + * + * @internal */ export const resolvePageChunkInfo = ({ app, diff --git a/packages/core/src/page/resolvePageComponentInfo.ts b/packages/core/src/page/resolvePageComponentInfo.ts index b1ee312cc4..1e5d4d6156 100644 --- a/packages/core/src/page/resolvePageComponentInfo.ts +++ b/packages/core/src/page/resolvePageComponentInfo.ts @@ -3,6 +3,8 @@ import type { App } from '../types/index.js' /** * Resolve page component and related info + * + * @internal */ export const resolvePageComponentInfo = ({ app, diff --git a/packages/core/src/page/resolvePageContent.ts b/packages/core/src/page/resolvePageContent.ts index c4bda0732b..51a1aa3ab2 100644 --- a/packages/core/src/page/resolvePageContent.ts +++ b/packages/core/src/page/resolvePageContent.ts @@ -9,6 +9,8 @@ const FALLBACK_CONTENT = '' /** * Resolve page content according to `content` or `filePath` + * + * @internal */ export const resolvePageContent = async ({ filePath, diff --git a/packages/core/src/page/resolvePageDate.ts b/packages/core/src/page/resolvePageDate.ts index d371062f68..f9b2d5312c 100644 --- a/packages/core/src/page/resolvePageDate.ts +++ b/packages/core/src/page/resolvePageDate.ts @@ -10,6 +10,8 @@ const DEFAULT_DATE = '0000-00-00' * Resolve page date according to frontmatter or file path * * It will be resolved as 'yyyy-MM-dd' format + * + * @internal */ export const resolvePageDate = ({ frontmatter, diff --git a/packages/core/src/page/resolvePageFilePath.ts b/packages/core/src/page/resolvePageFilePath.ts index ee5a1b01ed..32d34e23a5 100644 --- a/packages/core/src/page/resolvePageFilePath.ts +++ b/packages/core/src/page/resolvePageFilePath.ts @@ -3,6 +3,8 @@ import type { App, PageOptions } from '../types/index.js' /** * Resolve absolute and relative path of page file + * + * @internal */ export const resolvePageFilePath = ({ app, diff --git a/packages/core/src/page/resolvePageHtmlInfo.ts b/packages/core/src/page/resolvePageHtmlInfo.ts index 677921ed04..0417e4edbe 100644 --- a/packages/core/src/page/resolvePageHtmlInfo.ts +++ b/packages/core/src/page/resolvePageHtmlInfo.ts @@ -3,6 +3,8 @@ import type { App } from '../types/index.js' /** * Resolve page rendered html file path + * + * @internal */ export const resolvePageHtmlInfo = ({ app, diff --git a/packages/core/src/page/resolvePageLang.ts b/packages/core/src/page/resolvePageLang.ts index 3f2f243256..d0774e871c 100644 --- a/packages/core/src/page/resolvePageLang.ts +++ b/packages/core/src/page/resolvePageLang.ts @@ -3,6 +3,8 @@ import type { App, PageFrontmatter } from '../types/index.js' /** * Resolve language of page + * + * @internal */ export const resolvePageLang = ({ app, diff --git a/packages/core/src/page/resolvePagePath.ts b/packages/core/src/page/resolvePagePath.ts index c40e366455..0edac439de 100644 --- a/packages/core/src/page/resolvePagePath.ts +++ b/packages/core/src/page/resolvePagePath.ts @@ -3,6 +3,8 @@ import type { PageOptions } from '../types/index.js' /** * Resolve the final route path of a page + * + * @internal */ export const resolvePagePath = ({ permalink, diff --git a/packages/core/src/page/resolvePagePermalink.ts b/packages/core/src/page/resolvePagePermalink.ts index 15a421dec1..4b62c069f7 100644 --- a/packages/core/src/page/resolvePagePermalink.ts +++ b/packages/core/src/page/resolvePagePermalink.ts @@ -4,6 +4,8 @@ import type { App, PageFrontmatter } from '../types/index.js' /** * Resolve page permalink from frontmatter / options / pattern + * + * @internal */ export const resolvePagePermalink = ({ app, diff --git a/packages/core/src/page/resolvePageRouteMeta.ts b/packages/core/src/page/resolvePageRouteMeta.ts index 76e5030a91..fa66e39d36 100644 --- a/packages/core/src/page/resolvePageRouteMeta.ts +++ b/packages/core/src/page/resolvePageRouteMeta.ts @@ -2,6 +2,8 @@ import type { PageFrontmatter } from '../types/index.js' /** * Resolve page route meta + * + * @internal */ export const resolvePageRouteMeta = ({ frontmatter, diff --git a/packages/core/src/page/resolvePageSlug.ts b/packages/core/src/page/resolvePageSlug.ts index a7329ffbaa..f9c3704571 100644 --- a/packages/core/src/page/resolvePageSlug.ts +++ b/packages/core/src/page/resolvePageSlug.ts @@ -4,6 +4,8 @@ const DATE_RE = /(\d{4}-\d{1,2}(-\d{1,2})?)-(.*)/ /** * Resolve page slug from filename + * + * @internal */ export const resolvePageSlug = ({ filePathRelative, diff --git a/packages/core/src/pluginApi/createHookQueue.ts b/packages/core/src/pluginApi/createHookQueue.ts index 09331eee3e..9fb70cc9ad 100644 --- a/packages/core/src/pluginApi/createHookQueue.ts +++ b/packages/core/src/pluginApi/createHookQueue.ts @@ -10,6 +10,8 @@ const log = debug('vuepress:core/plugin-api') /** * Create hook queue for plugin system + * + * @internal */ export const createHookQueue = <T extends HooksName>(name: T): HookQueue<T> => { const items: HookItem<T>[] = [] diff --git a/packages/core/src/pluginApi/createPluginApi.ts b/packages/core/src/pluginApi/createPluginApi.ts index 4f90cda435..18aa270ed7 100644 --- a/packages/core/src/pluginApi/createPluginApi.ts +++ b/packages/core/src/pluginApi/createPluginApi.ts @@ -2,6 +2,11 @@ import type { PluginApi } from '../types/index.js' import { createPluginApiHooks } from './createPluginApiHooks.js' import { createPluginApiRegisterHooks } from './createPluginApiRegisterHooks.js' +/** + * Create vuepress plugin api + * + * @internal + */ export const createPluginApi = (): PluginApi => { const plugins: PluginApi['plugins'] = [] const hooks = createPluginApiHooks() diff --git a/packages/core/src/pluginApi/createPluginApiHooks.ts b/packages/core/src/pluginApi/createPluginApiHooks.ts index 0cefc33067..d8369985bb 100644 --- a/packages/core/src/pluginApi/createPluginApiHooks.ts +++ b/packages/core/src/pluginApi/createPluginApiHooks.ts @@ -1,6 +1,11 @@ import type { PluginApi } from '../types/index.js' import { createHookQueue } from './createHookQueue.js' +/** + * Create hooks for plugin api + * + * @internal + */ export const createPluginApiHooks = (): PluginApi['hooks'] => ({ // life cycle hooks onInitialized: createHookQueue('onInitialized'), diff --git a/packages/core/src/pluginApi/createPluginApiRegisterHooks.ts b/packages/core/src/pluginApi/createPluginApiRegisterHooks.ts index 79ee344016..22c94c4a01 100644 --- a/packages/core/src/pluginApi/createPluginApiRegisterHooks.ts +++ b/packages/core/src/pluginApi/createPluginApiRegisterHooks.ts @@ -2,6 +2,11 @@ import type { PluginApi } from '../types/index.js' import { normalizeAliasDefineHook } from './normalizeAliasDefineHook.js' import { normalizeClientConfigFileHook } from './normalizeClientConfigFileHook.js' +/** + * Create registerHooks method for plugin api + * + * @internal + */ export const createPluginApiRegisterHooks = ( plugins: PluginApi['plugins'], diff --git a/packages/core/src/pluginApi/normalizeAliasDefineHook.ts b/packages/core/src/pluginApi/normalizeAliasDefineHook.ts index 6c3da1a21b..1b12ac361b 100644 --- a/packages/core/src/pluginApi/normalizeAliasDefineHook.ts +++ b/packages/core/src/pluginApi/normalizeAliasDefineHook.ts @@ -3,6 +3,8 @@ import type { AliasDefineHook } from '../types/index.js' /** * Normalize alias and define hook + * + * @internal */ export const normalizeAliasDefineHook = (hook: AliasDefineHook['exposed']): AliasDefineHook['normalized'] => diff --git a/packages/core/src/pluginApi/normalizeClientConfigFileHook.ts b/packages/core/src/pluginApi/normalizeClientConfigFileHook.ts index bba349702c..a6475852e4 100644 --- a/packages/core/src/pluginApi/normalizeClientConfigFileHook.ts +++ b/packages/core/src/pluginApi/normalizeClientConfigFileHook.ts @@ -4,6 +4,8 @@ import type { ClientConfigFileHook } from '../types/index.js' /** * Normalize hook for client config file + * + * @internal */ export const normalizeClientConfigFileHook = (hook: ClientConfigFileHook['exposed']): ClientConfigFileHook['normalized'] => diff --git a/packages/core/src/types/app/options.ts b/packages/core/src/types/app/options.ts index 4cbcabbc37..74a11478f6 100644 --- a/packages/core/src/types/app/options.ts +++ b/packages/core/src/types/app/options.ts @@ -9,18 +9,98 @@ import type { Theme } from '../theme.js' * Vuepress app common config that shared between dev and build */ export interface AppConfigCommon extends Partial<SiteData> { + /** + * Source directory of the markdown files. + * + * Vuepress will load markdown files from this directory. + * + * @required + */ source: string + + /** + * Destination directory of the output files. + * + * Vuepress will output the static site files to this directory. + * + * @default `${source}/.vuepress/dist` + */ dest?: string + + /** + * Temp files directory. + * + * Vuepress will write temp files to this directory. + * + * @default `${source}/.vuepress/.temp` + */ temp?: string + + /** + * Cache files directory. + * + * Vuepress will write cache files to this directory. + * + * @default `${source}/.vuepress/.cache` + */ cache?: string + + /** + * Public files directory. + * + * Vuepress will copy the files from public directory to the output directory. + * + * @default `${source}/.vuepress/public` + */ public?: string + /** + * Whether to enable debug mode + * + * @default false + */ debug?: boolean + + /** + * Markdown options + * + * @default {} + */ markdown?: MarkdownOptions + + /** + * Patterns to match the markdown files as pages + * + * @default ['**\/*.md', '!.vuepress', '!node_modules'] + */ pagePatterns?: string[] + + /** + * Pattern to generate permalink for pages + * + * @default null + */ permalinkPattern?: string | null + + /** + * Vuepress bundler + * + * @required + */ bundler: Bundler + + /** + * Vuepress theme + * + * @required + */ theme: Theme + + /** + * Vuepress plugins + * + * @default [] + */ plugins?: PluginConfig } @@ -87,17 +167,21 @@ export interface AppConfigBuild { /** * Specify the HTML template renderer to be used for build * - * @default templateRenderer from '@vuepress/utils' + * @default `import { templateRenderer } from '@vuepress/utils'` */ templateBuildRenderer?: TemplateRenderer } /** - * Vuepress app config + * Vuepress app user config. + * + * It would be provided by user, typically via a config file. */ export type AppConfig = AppConfigBuild & AppConfigCommon & AppConfigDev /** - * Vuepress app options + * Vuepress app options that resolved from user config. + * + * It fills all optional fields with a default value. */ export type AppOptions = Required<AppConfig> diff --git a/packages/core/src/types/app/utils.ts b/packages/core/src/types/app/utils.ts index 0e70eb5508..fdfb3fd4a7 100644 --- a/packages/core/src/types/app/utils.ts +++ b/packages/core/src/types/app/utils.ts @@ -7,11 +7,34 @@ export type AppDirFunction = (...args: string[]) => string * Directory utils */ export interface AppDir { + /** + * Resolve file path in cache directory + */ cache: AppDirFunction + + /** + * Resolve file path in temp directory + */ temp: AppDirFunction + + /** + * Resolve file path in source directory + */ source: AppDirFunction + + /** + * Resolve file path in dest directory + */ dest: AppDirFunction + + /** + * Resolve file path in public directory + */ public: AppDirFunction + + /** + * Resolve file path in client directory + */ client: AppDirFunction } diff --git a/packages/core/src/types/bundler.ts b/packages/core/src/types/bundler.ts index cd3b8dc2a7..0f8e64fa78 100644 --- a/packages/core/src/types/bundler.ts +++ b/packages/core/src/types/bundler.ts @@ -8,8 +8,19 @@ import type { App } from './app/index.js' * - build: bundle assets for deployment */ export interface Bundler { + /** + * Name of the bundler + */ name: string + + /** + * Method to run vuepress app in dev mode, starting dev server + */ dev: (app: App) => Promise<() => Promise<void>> + + /** + * Method to run vuepress app in build mode, generating static pages and assets + */ build: (app: App) => Promise<void> } diff --git a/packages/core/src/types/plugin.ts b/packages/core/src/types/plugin.ts index ae52a6b334..99e65f7517 100644 --- a/packages/core/src/types/plugin.ts +++ b/packages/core/src/types/plugin.ts @@ -27,10 +27,14 @@ export type PluginFunction<T extends PluginObject = PluginObject> = ( * Vuepress plugin object */ export interface PluginObject extends Partial<HooksExposed> { - // plugin name + /** + * Name of the plugin + */ name: string - // allow use a plugin multiple times or not + /** + * Allow the plugin to be used multiple times or not + */ multiple?: boolean }