1. 前言
大家好,我是小鑫同学。一位从事过Android开发、混合开发,现在长期从事前端开发的编程爱好者,我觉得在编程之路上最重要的是知识的分享,所谓三人行必有我师。
Vitepress的文档渲染目的就是将程序员日常所写的Markdown文件编译为Html文件,并添加了更多的插件来丰富MD文件的功能,就比如说Vuejs组件在MD文件中渲染等等,为了我们可以在使用Vitepress的时候可以更随心所欲的定制一些功能,我们要先搞一搞明白Vitepress是如何将MD文档渲染成HTML的~
看完可以明白这3点?
- MD文档转HTML文档流程;
- 如何支持代码块高亮;
- 如何实现自定义容器;
演示地址
2. 实现MD文档转HTML文档
2.1 请按如下项目结构准备我们的实验环境~
代码语言:javascript复制├─markdown-it-demo
│ ├─src
│ │ ├─index.ts
│ │ ├─temp.md
│ ├─index.html
└─ └─package.json
2.2 利用markdown-it模块实现文档转换:
markdown-it 是目前比较通用的MD语法解析模块,快速且易于扩展,遵循COmmonMark规范,且有大量的社区插件~
- 执行安装模块命令:
pnpm i markdown-it @types/markdown-it -D
; - 导入
markdown-it
模块并实例化md对象; ```typescript import markdownIt from "markdown-it";
// 实例化md-it对象 const md = new markdownIt();
代码语言:javascript复制3. 通过`fs-extra`模块读取放置在`src`下的`temp.md`文件,读取后的**Buffer**数组通过`toString()`转为字符串;
```typescript
const rawMd = fs.readFileSync(path.resolve(__dirname, "temp.md")).toString();
利用md对象的render
函数来讲rawMd
进行转换;
const output = md.render(rawMd);
转换完成后将output
内容输出到index.html
文件中;
fs.writeFileSync(path.resolve(__dirname, "../index.html"), `
${output}
`);
在转换完成后可以利用child_process.exec(root-path)
自动在浏览器打开index.html文档;
3. 实现MD支持代码块高亮
代码块高亮所使用的模块时highlight.js,该模块同时内置了很多常见的代码块样式文件可供选择~
3.1 第一步改造markdownIt对象的构造函数:
highlight
属性配置的函数传入code片段和代码方言两部分,通过在hljs库中查找对应的方言来利用hljs库实现代码的快速高亮,当无法查找到对应的方言时将返回仅仅转义后的html片段~
const md = new markdownIt({
highlight: (str: string, lang: string) => {
const defaultCode: string = `<pre class="hljs"><code>${md.utils.escapeHtml(str)}</code></pre>`;
if (lang && hljs.getLanguage(lang)) {
try {
return `<pre class="hljs"><code>${hljs.highlight(str, { language: lang, ignoreIllegals: true }).value}</code></pre>`
} catch (__) {
return defaultCode;
}
}
return defaultCode;
}
});
3.2 第二部整合output内容和高亮样式文本:
第一步的操作仅仅完成了由code片段到html结构的转换,但是完成高亮还需要样式配合渲染,我们这里可以通过在输出output内容到index.html时将hljs中喜欢的样式文档路径传入到html文件来加载~
代码语言:javascript复制const output = md.render(rawMd);
const styles = `
<link rel="stylesheet" href="./node_modules/highlight.js/styles/a11y-dark.css">
`;
// 输出html文本
fs.writeFileSync(path.resolve(__dirname, "../index.html"), `
${styles}
${output}
`);
更多的样式文档可以在./node_modules/highlight.js/styles
选择~
4. 实现MD支持自定义容器
自定义容器是MD文档默认并不支持的一种语法,在Vuejs的文档有很多的应用,实现自定义容易需要用到
markdown-it-container
模块~
markdownIt
通过插件的形式利用markdown-it-container
来实现自定义容器,通过配置validate
来做渲染前的语法校验,通过render
函数来组中容器部分的HTML
结构~
::: warning
*here be dragons*
:::
↓↓↓↓↓↓↓↓↓↓转换为↓↓↓↓↓↓↓↓↓↓
<div class="warning">
<em>here be dragons</em>
</div>
复制代码
代码语言:javascript复制md.use(require("markdown-it-container"), "warning", {
validate: (params: string) => {
return params.trim().match(/^warning (.*)$/m);
},
render: (tokens: Array<Token>, idx: number) => {
const m = tokens[idx].info.trim().match(/^warning (.*)$/m);
if (tokens[idx].nesting === 1) {
return `<div class="warning">${md.utils.escapeHtml(m ? m[1] : '')}`
} else {
return '</div>n';
}
}
})
复制代码
提示:通过tokens[idx]
取到的数据如下图所示~
- 上面的处理依旧是MD到HTML结构的转换,在自定义容器的时候我们预留的css名称,我们还是需要在输出
index.html
文件的时候自定义样式文档~
const output = md.render(rawMd);
const styles = `
<link rel="stylesheet" href="./node_modules/highlight.js/styles/a11y-dark.css">
<style>
.warning{
margin: 28px 0;
padding: 10px 14px 4px 22px;
border-radius: 8px;
overflow-x: auto;
transition: color .5s,background-color .5s;
position: relative;
font-size: 14px;
line-height: 1.6;
font-weight: 500;
color: #0000008c;
background-color: #f9f9f9;
border: 1px solid #ffc517;
}
.hljs {
padding: 5px 8px;
border-radius: 5px;
}
</style>
`;
// 输出html文本
fs.writeFileSync(path.resolve(__dirname, "../index.html"), `
${styles}
${output}
`);
复制代码
5. 总结
通过使用markdown-it
、highlight.js
、markdown-it-container
模块实现了Markdown到HTML的文档转换,代码块高亮和自定义容器,VItepress搭建的组件库文档中的组件渲染和源码展示功能就需要用到自定义容器的解析和组装自定义的Vue组件实现高级功能~
本文项目已推送至GitHub,欢迎克隆演示:
git clone https://github.com/OSpoon/awesome-examples.git