2021-11-29 13:24:32 +08:00
|
|
|
// ref https://github.com/vuejs/vitepress/blob/main/src/node/markdown/plugins/highlight.ts
|
|
|
|
import chalk from 'chalk'
|
|
|
|
import escapeHtml from 'escape-html'
|
|
|
|
import prism from 'prismjs'
|
|
|
|
|
|
|
|
// prism is listed as actual dep so it's ok to require
|
2021-12-30 19:00:11 +08:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
2021-11-29 13:24:32 +08:00
|
|
|
const loadLanguages = require('prismjs/components/index')
|
|
|
|
|
|
|
|
// required to make embedded highlighting work...
|
|
|
|
loadLanguages(['markup', 'css', 'javascript'])
|
|
|
|
|
|
|
|
function wrap(code: string, lang: string): string {
|
|
|
|
if (lang === 'text') {
|
|
|
|
code = escapeHtml(code)
|
|
|
|
}
|
|
|
|
return `<pre v-pre><code>${code}</code></pre>`
|
|
|
|
}
|
|
|
|
|
|
|
|
export const highlight = (str: string, lang: string) => {
|
|
|
|
if (!lang) {
|
|
|
|
return wrap(str, 'text')
|
|
|
|
}
|
|
|
|
lang = lang.toLowerCase()
|
|
|
|
const rawLang = lang
|
|
|
|
if (lang === 'vue' || lang === 'html') {
|
|
|
|
lang = 'markup'
|
|
|
|
}
|
|
|
|
if (lang === 'md') {
|
|
|
|
lang = 'markdown'
|
|
|
|
}
|
|
|
|
if (lang === 'ts') {
|
|
|
|
lang = 'typescript'
|
|
|
|
}
|
|
|
|
if (lang === 'py') {
|
|
|
|
lang = 'python'
|
|
|
|
}
|
|
|
|
if (!prism.languages[lang]) {
|
|
|
|
try {
|
|
|
|
loadLanguages([lang])
|
|
|
|
} catch (e) {
|
|
|
|
// eslint-disable-next-line no-console
|
|
|
|
console.warn(
|
|
|
|
chalk.yellow(
|
|
|
|
`[vitepress] Syntax highlight for language "${lang}" is not supported.`
|
|
|
|
)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (prism.languages[lang]) {
|
|
|
|
const code = prism.highlight(str, prism.languages[lang], lang)
|
|
|
|
return wrap(code, rawLang)
|
|
|
|
}
|
|
|
|
return wrap(str, 'text')
|
|
|
|
}
|