基于.NetCore开发博客项目 StarBlog - (19) Markdown渲染方案探索
前言
笔者认为,一个博客网站,最核心的是阅读体验。
在开发StarBlog的过程中,最耗时的恰恰也是文章的展示部分功能。
最开始还没研究出来如何很好的使用后端渲染,所以只能先用Editor.md组件做前端渲染,过渡一下。前端渲染我是不满意的,因为性能较差,页面加载出来还会闪一下,有割裂感,影响体验。
现在我已经做出了比较完善的后端渲染方案,前端渲染就可以直接退休了。本文介绍StarBlog博客开发过程中的各种Markdown渲染方案(主要是介绍后端渲染)。
两种方案
前端渲染
使用 Editor.md 组件进行渲染,效果差强人意,主要是看中了其可以生成 ToC(文章目录) 的功能,但生成的 ToC 效果也比较差,后面是我fork了一份源码进行魔改才好一点。
魔改过程在这篇文章:魔改editormd组件,优化ToC渲染效果
优化 ToC 的这个功能我给官方提了PR,但没有响应,看了一下GitHub里有几十个PR,上次提交也是快4年前的事了,看来这个项目真的凉了……
除了这个 editor.md ,还有其他几个前端的方案:
看起来都不错,有没有 ToC 我没研究,博客园上有大佬写了一篇比较的文章,有兴趣的同学可以在参考资料中看看~
后端渲染
目前 C# 可用的 Markdown 库似乎只有 Markdig ,一开始我还在吐槽文档缺失导致很难用,甚至一度想自己造轮子重新做一个,不过最近有所改善,在研究了官方新增的几个文档之后,我对这个库的了解又加深了一些,功能确实很多,设计得也不错,扩展性很好~
所以暂时就用这个啦~
目前我的做法是用 Markdig 将 Markdown 生成 HTML,然后前端展示这个 HTML ,再结合 Bootstrap 或者 github-markdown-css 等样式库来美化正文显示效果,用 highlight.js 之类的JS库实现代码高亮。
至于文章的 ToC ,Markdig 没有现成的,我自己造轮子实现~
详见这篇文章:C#实现生成Markdown文档目录树
其实算是一种混合式的方案吧~
接下来介绍的内容围绕后端渲染展开。
处理 ToC
上一篇文章对于生成目录树已经说得比较清楚了,本文不再重复那么多,只说一下有区别的地方~
先解析Markdown文档,拿到所有标题节点
var headings = new List<Heading>();
foreach (var heading in document.Descendants<HeadingBlock>()) {
  var item = new Heading {Level = heading.Level, Text = heading.Inline?.FirstChild?.ToString()};
  headings.Add(item);
}
遍历进行处理。
原本直接把标题作为锚点的 href 属性,实际使用的时候是不行的,根据测试,Markdig生成锚点ID的规则如下
中文按照
- section
 - section-1
 - section-2
 - ...
 
section后面的数字是在所有中文标题里出现的顺序,不是在全部标题里面的顺序。
英文就替换空格 + 转小写 (未考虑其他情况,事实上应该把特殊符号也一并替换掉)
所以处理 href 的时候分两种情况,用正则表达式 [\u4e00-\u9fbb] 检测是否包含中文字符。
var chineseTitleCount = 0;
for (var i = 0; i < headings.Count; i++) {
  var item = headings[i];
  var text = item.Text ?? "";
  if (Regex.IsMatch(text, "[\u4e00-\u9fbb]")) {
    item.Slug = chineseTitleCount == 0 ? "section" : $"section-{chineseTitleCount}";
    chineseTitleCount++;
  }
  else {
    item.Slug = text.Replace(" ", "-").ToLower();
  }
  // ...
}
搞定
样式
bootstrap 默认样式
默认样式还可以,不过会觉得少了点啥,或许可以研究一下各种在线Markdown编辑器的样式~
github-markdown-css
顾名思义是GitHub的markdown样式
地址: https://www.npmjs.com/package/github-markdown-css
安装后有三个文件
- github-markdown.css: (默认) 通过 
@media (prefers-color-scheme)实现自动切换亮色/暗色主题 - github-markdown-light.css: 亮色主题
 - github-markdown-dark.css: 暗色主题
 
官网还有一句话,但我不知道怎么自己生成,难道要我去github扒css下来?
You may know that now GitHub supports more than 2 themes including
dark_dimmed,dark_high_contrastandcolorblindvariants. If you want to try these themes, you can generate them on your own!
安装
yarn add github-markdown-css
引入
<link rel="stylesheet" href="~/lib/github-markdown-css/github-markdown-light.css">
使用
<div class="markdown-body">
  @Html.Raw(Model.ContentHtml)
</div>
效果
确实有GitHub内味了,但还没代码高亮

代码高亮
目前使用 highlight.js 包,官网: https://www.npmjs.com/package/highlight.js
有很多其他的工具,不展开了,用这个足够了~
下载
要在网页上直接用没办法通过安装NPM包的方式,只能通过网址下载: https://highlightjs.org/download/
如果不想下载的话可以用CDN,但就只能支持部分语言高亮。
里面有好多种语言,竟然没全选按钮,一个个选太麻烦了,我写了个全选脚本,复制到浏览器控制台执行就能全选,然后下载。
document.querySelectorAll('input').forEach( item => {
    if(item.getAttribute('type')==='checkbox') item.checked=true
})
引入
下载后把zip解压放到 wwwroot/lib 下
<link rel="stylesheet" href="~/lib/highlight/styles/default.min.css">
<script src="~/lib/highlight/highlight.min.js"></script>
<script>hljs.highlightAll();</script>
或者不下载,直接使用 CDN
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.6.0/styles/default.min.css">
<script src="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.6.0/highlight.min.js"></script>
<script>hljs.highlightAll();</script>
效果
有了代码高亮,一下就不一样了

还有很多其他主题,styles 目录下很多,引入css的时候自行选择即可
我来换个深色的主题看看

系列文章
- 基于.NetCore开发博客项目 StarBlog - (1) 为什么需要自己写一个博客?
 - 基于.NetCore开发博客项目 StarBlog - (2) 环境准备和创建项目
 - 基于.NetCore开发博客项目 StarBlog - (3) 模型设计
 - 基于.NetCore开发博客项目 StarBlog - (4) markdown博客批量导入
 - 基于.NetCore开发博客项目 StarBlog - (5) 开始搭建Web项目
 - 基于.NetCore开发博客项目 StarBlog - (6) 页面开发之博客文章列表
 - 基于.NetCore开发博客项目 StarBlog - (7) 页面开发之文章详情页面
 - 基于.NetCore开发博客项目 StarBlog - (8) 分类层级结构展示
 - 基于.NetCore开发博客项目 StarBlog - (9) 图片批量导入
 - 基于.NetCore开发博客项目 StarBlog - (10) 图片瀑布流
 - 基于.NetCore开发博客项目 StarBlog - (11) 实现访问统计
 - 基于.NetCore开发博客项目 StarBlog - (12) Razor页面动态编译
 - 基于.NetCore开发博客项目 StarBlog - (13) 加入友情链接功能
 - 基于.NetCore开发博客项目 StarBlog - (14) 实现主题切换功能
 - 基于.NetCore开发博客项目 StarBlog - (15) 生成随机尺寸图片
 - 基于.NetCore开发博客项目 StarBlog - (16) 一些新功能 (监控/统计/配置/初始化)
 - 基于.NetCore开发博客项目 StarBlog - (17) 自动下载文章里的外部图片
 - 基于.NetCore开发博客项目 StarBlog - (18) 实现本地Typora文章打包上传
 - 基于.NetCore开发博客项目 StarBlog - (19) Markdown渲染方案探索
 
参考资料
- JavaScript解析和渲染Markdown - https://www.cnblogs.com/makalochen/p/14464519.html#5117019
 
基于.NetCore开发博客项目 StarBlog - (19) Markdown渲染方案探索的更多相关文章
- 基于.NetCore开发博客项目 StarBlog - (4) markdown博客批量导入
		
系列文章 基于.NetCore开发博客项目 StarBlog - (1) 为什么需要自己写一个博客? 基于.NetCore开发博客项目 StarBlog - (2) 环境准备和创建项目 基于.NetC ...
 - 基于.NetCore开发博客项目 StarBlog - (5) 开始搭建Web项目
		
系列文章 基于.NetCore开发博客项目 StarBlog - (1) 为什么需要自己写一个博客? 基于.NetCore开发博客项目 StarBlog - (2) 环境准备和创建项目 基于.NetC ...
 - 基于.NetCore开发博客项目 StarBlog - (6) 页面开发之博客文章列表
		
系列文章 基于.NetCore开发博客项目 StarBlog - (1) 为什么需要自己写一个博客? 基于.NetCore开发博客项目 StarBlog - (2) 环境准备和创建项目 基于.NetC ...
 - 基于.NetCore开发博客项目 StarBlog - (7) 页面开发之文章详情页面
		
系列文章 基于.NetCore开发博客项目 StarBlog - (1) 为什么需要自己写一个博客? 基于.NetCore开发博客项目 StarBlog - (2) 环境准备和创建项目 基于.NetC ...
 - 基于.NetCore开发博客项目 StarBlog - (8) 分类层级结构展示
		
系列文章 基于.NetCore开发博客项目 StarBlog - (1) 为什么需要自己写一个博客? 基于.NetCore开发博客项目 StarBlog - (2) 环境准备和创建项目 基于.NetC ...
 - 基于.NetCore开发博客项目 StarBlog - (9) 图片批量导入
		
系列文章 基于.NetCore开发博客项目 StarBlog - (1) 为什么需要自己写一个博客? 基于.NetCore开发博客项目 StarBlog - (2) 环境准备和创建项目 基于.NetC ...
 - 基于.NetCore开发博客项目 StarBlog - (10) 图片瀑布流
		
系列文章 基于.NetCore开发博客项目 StarBlog - (1) 为什么需要自己写一个博客? 基于.NetCore开发博客项目 StarBlog - (2) 环境准备和创建项目 基于.NetC ...
 - 基于.NetCore开发博客项目 StarBlog - (11) 实现访问统计
		
系列文章 基于.NetCore开发博客项目 StarBlog - (1) 为什么需要自己写一个博客? 基于.NetCore开发博客项目 StarBlog - (2) 环境准备和创建项目 基于.NetC ...
 - 基于.NetCore开发博客项目 StarBlog - (12) Razor页面动态编译
		
系列文章 基于.NetCore开发博客项目 StarBlog - (1) 为什么需要自己写一个博客? 基于.NetCore开发博客项目 StarBlog - (2) 环境准备和创建项目 基于.NetC ...
 
随机推荐
- 基于EasyExcel实现的分页数据下载封装
			
功能概述 主要实现的功能: 1.分页查询,避免一次性查询全部数据加载到内存引起频繁FULL GC甚至OOM 2.当数据量超过单个工作簿最大行数(1048575)时,自动将数据写入新的工作簿 3.支持百 ...
 - feign远程调用出错
			
如果你传递的参数,比较复杂时,默认会采用POST的请求方式. 传递单个参数时,推荐使用@PathVariable,如果传递的单个参数比较多,这里也可以采用@RequestParam,Feign接口中不 ...
 - ASP.NET CORE在docker中的健康检查(healthcheck)
			
在使用docker-compose的过程中,很多程序都提供了健康检查(healthcheck)的方法,通过健康检查,应用程序能够在确保其依赖的程序都已经启动的前提下启动,减少各种错误的发生,同时,合理 ...
 - 【JDBC】学习路径9-dbcp数据源的使用
			
第一章:下载 要下载三个东西:commons pool.commons log.dbcp dbcp中有些东西是依赖于commons pool 和 commons log 的. 缺一不可,否则无法正确运 ...
 - Session认证机制与JWT认证机制
			
一.什么是身份认证? 身份认证(Authentication)又称"身份验证"."鉴权",是指通过一定的手段,完成对用户身份的确认.日常生活中的身份认证随处可见 ...
 - OpenDataV低代码平台新增组件流程
			
OpenDataV计划采用子库的方式添加子组件,即每一个组件都当做一个子库,子库有自己的依赖,而项目本身的依赖只针对框架,因此每一个组件我们都当做一个子库来开发.下面我带着大家一步步详细的开发一个数字 ...
 - 详谈 MySQL 8.0 原子 DDL 原理
			
柯煜昌 青云科技研发顾问级工程师 目前从事 RadonDB 容器化研发,华中科技大学研究生毕业,有多年的数据库内核开发经验. 文章字数 3800+,阅读时间 15 分钟 背景 MySQL 5.7 的字 ...
 - 004-GoingDeeperConvolutions2014(googLeNet)
			
Going Deeper with Convolutions #paper 1. paper-info 1.1 Metadata Author:: [[Christian Szegedy]], [[W ...
 - 现有rabbitmq集群添加新节点,移除旧节点(可以作为rabbitmq集群迁移使用)
			
原有集群安装步骤:https://www.cnblogs.com/sanduzxcvbnm/p/15797788.html 1.拉取镜像 集群中新节点需要执行 docker pull rabbitmq ...
 - MySQL集群搭建(4)-MMM+LVS+Keepalived
			
1 LVS 介绍 1.1 简介 LVS 是 Linux Virtual Server 的简写,意即 Linux 虚拟服务器,是一个虚拟的服务器集群系统.本项目在 1998 年 5 月由章文嵩博士成立, ...