import type { App, PluginConfig } from 'vuepress/core' import type { PlumeThemePluginOptions } from '../../shared/index.js' import { activeHeaderLinksPlugin } from '@vuepress/plugin-active-header-links' import { cachePlugin } from '@vuepress/plugin-cache' import { commentPlugin } from '@vuepress/plugin-comment' import { docsearchPlugin } from '@vuepress/plugin-docsearch' import { gitPlugin } from '@vuepress/plugin-git' import { markdownHintPlugin } from '@vuepress/plugin-markdown-hint' import { markdownImagePlugin } from '@vuepress/plugin-markdown-image' import { markdownMathPlugin } from '@vuepress/plugin-markdown-math' import { nprogressPlugin } from '@vuepress/plugin-nprogress' import { photoSwipePlugin } from '@vuepress/plugin-photo-swipe' import { readingTimePlugin } from '@vuepress/plugin-reading-time' import { seoPlugin } from '@vuepress/plugin-seo' import { sitemapPlugin } from '@vuepress/plugin-sitemap' import { watermarkPlugin } from '@vuepress/plugin-watermark' import { contentUpdatePlugin } from '@vuepress-plume/plugin-content-update' import { fontsPlugin } from '@vuepress-plume/plugin-fonts' import { searchPlugin } from '@vuepress-plume/plugin-search' import { shikiPlugin } from '@vuepress-plume/plugin-shikiji' import { type MarkdownEnhancePluginOptions, mdEnhancePlugin } from 'vuepress-plugin-md-enhance' import { markdownPowerPlugin } from 'vuepress-plugin-md-power' import { resolveDocsearchOptions, resolveSearchOptions } from '../config/index.js' import { deleteAttrs } from '../utils/index.js' import { customContainerPlugins } from './containerPlugins.js' import { markdownTitlePlugin } from './markdown-title.js' export interface SetupPluginOptions { app: App pluginOptions: PlumeThemePluginOptions hostname?: string cache?: false | 'memory' | 'filesystem' } export function getPlugins({ app, pluginOptions, hostname, cache, }: SetupPluginOptions): PluginConfig { const isProd = app.env.isBuild const plugins: PluginConfig = [ markdownTitlePlugin(), fontsPlugin(), contentUpdatePlugin(), activeHeaderLinksPlugin({ headerLinkSelector: 'a.outline-link', headerAnchorSelector: '.header-anchor', }), markdownHintPlugin({ hint: true, alert: true, injectStyles: false }), ...customContainerPlugins, ] if (pluginOptions.readingTime !== false) { plugins.push(readingTimePlugin({ locales: { '/zh/': { word: '$word字', less1Minute: '小于1分钟', time: '约$time分钟', }, }, ...pluginOptions.readingTime, })) } if (pluginOptions.nprogress !== false) { plugins.push(nprogressPlugin()) } if (pluginOptions.git ?? isProd) { plugins.push(gitPlugin({ createdTime: true, updatedTime: true, contributors: true, })) } if (pluginOptions.photoSwipe !== false) { plugins.push(photoSwipePlugin({ selector: '.plume-content > img, .plume-content :not(a) > img', delay: 300, })) } if (pluginOptions.docsearch) { if (pluginOptions.docsearch.appId && pluginOptions.docsearch.apiKey) plugins.push(docsearchPlugin(resolveDocsearchOptions(app, pluginOptions.docsearch))) else console.error('docsearch plugin: appId and apiKey are both required') } else if (pluginOptions.search !== false) { plugins.push(searchPlugin(resolveSearchOptions(app, pluginOptions.search))) } const shikiOption = pluginOptions.shiki let shikiTheme: any = { light: 'vitesse-light', dark: 'vitesse-dark' } if (shikiOption !== false) { shikiTheme = shikiOption?.theme ?? shikiTheme plugins.push(shikiPlugin({ theme: shikiTheme, ...(shikiOption ?? {}), })) } if (pluginOptions.markdownEnhance !== false) { const options: MarkdownEnhancePluginOptions = { ...pluginOptions.markdownEnhance, } plugins.push(mdEnhancePlugin(deleteAttrs(options, 'hint', 'alert', 'imgSize', 'imgLazyload', 'imgMark', 'figure', 'obsidianImgSize', 'katex', 'mathjax', 'tabs', 'codetabs', 'align', 'mark', 'sub', 'sup', 'attrs', 'tasklist', 'footnote'))) } if (pluginOptions.markdownPower !== false) { plugins.push(markdownPowerPlugin({ fileTree: true, plot: true, icons: true, ...pluginOptions.markdownPower || {}, repl: pluginOptions.markdownPower?.repl ? { theme: shikiTheme, ...pluginOptions.markdownPower?.repl } : pluginOptions.markdownPower?.repl, })) } if (pluginOptions.markdownMath !== false) { plugins.push(markdownMathPlugin(pluginOptions.markdownMath ?? { type: 'katex' })) } if (pluginOptions.markdownImage) { plugins.push(markdownImagePlugin(pluginOptions.markdownImage)) } if (pluginOptions.watermark) { plugins.push(watermarkPlugin({ delay: 300, enabled: true, ...typeof pluginOptions.watermark === 'object' ? pluginOptions.watermark : {}, })) } if (pluginOptions.comment) { plugins.push(commentPlugin(pluginOptions.comment)) } if (pluginOptions.sitemap !== false && hostname && isProd) { plugins.push(sitemapPlugin({ hostname })) } if (pluginOptions.seo !== false && hostname && isProd) { plugins.push(seoPlugin({ hostname })) } if (cache !== false) { plugins.push(cachePlugin({ type: cache || 'filesystem' })) } return plugins }