2024-09-23 12:21:43 +08:00

177 lines
5.4 KiB
TypeScript

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 { 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.isDev
const plugins: PluginConfig = [
markdownTitlePlugin(),
fontsPlugin(),
contentUpdatePlugin(),
activeHeaderLinksPlugin({
headerLinkSelector: 'a.outline-link',
headerAnchorSelector: '.header-anchor',
delay: 200,
offset: 5,
}),
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) {
plugins.push(mdEnhancePlugin(
Object.assign(
{
attrs: true,
align: true,
mark: true,
tasklist: true,
sup: true,
sub: true,
footnote: true,
} as MarkdownEnhancePluginOptions,
pluginOptions.markdownEnhance || {},
{ hint: false, alert: false, imgSize: false, imgLazyload: false, imgMark: false, figure: false, obsidianImgSize: false, katex: false, mathjax: false, tabs: false, codetabs: false } as MarkdownEnhancePluginOptions,
),
))
}
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
}