2024-08-30 14:13:35 +08:00
|
|
|
|
from typing import *
|
2024-08-28 15:16:50 +08:00
|
|
|
|
import re
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Header:
|
2024-08-30 13:32:39 +08:00
|
|
|
|
def __init__(self, text: str):
|
|
|
|
|
self.text = text
|
|
|
|
|
|
|
|
|
|
def __call__(self, *args, **kwargs):
|
2024-08-28 15:16:50 +08:00
|
|
|
|
"""
|
|
|
|
|
渲染标题
|
|
|
|
|
:return: 处理后的文本
|
|
|
|
|
"""
|
2024-08-30 13:32:39 +08:00
|
|
|
|
h6 = re.sub(r'###### (.*?)\n', r'<h6>\1</h6>\n', self.text) # H6
|
2024-08-28 15:16:50 +08:00
|
|
|
|
h5 = re.sub(r'##### (.*?)\n', r'<h5>\1</h5>\n', h6) # H5
|
|
|
|
|
h4 = re.sub(r'#### (.*?)\n', r'<h4>\1</h4>\n', h5) # H4
|
|
|
|
|
h3 = re.sub(r'### (.*?)\n', r'<h3>\1</h3>\n', h4) # H3
|
|
|
|
|
h2 = re.sub(r'## (.*?)\n', r'<h2>\1</h2>\n', h3) # H2
|
|
|
|
|
h1 = re.sub(r'# (.*?)\n', r'<h1>\1</h1>\n', h2) # H1
|
|
|
|
|
return h1
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Style:
|
|
|
|
|
"""
|
|
|
|
|
渲染字体样式
|
|
|
|
|
"""
|
2024-08-29 12:36:57 +08:00
|
|
|
|
|
2024-08-28 15:16:50 +08:00
|
|
|
|
def __init__(self, text: str):
|
|
|
|
|
"""
|
|
|
|
|
初始化
|
|
|
|
|
:param text: cd文本
|
|
|
|
|
"""
|
|
|
|
|
self.text = text
|
|
|
|
|
|
|
|
|
|
def italic(self):
|
|
|
|
|
"""
|
|
|
|
|
斜体
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
2024-08-29 12:36:57 +08:00
|
|
|
|
self.text = re.sub(r'\*([^*\n]+)\*', r'<i>\1</i>', self.text)
|
2024-08-28 15:16:50 +08:00
|
|
|
|
|
|
|
|
|
def bold(self):
|
|
|
|
|
"""
|
|
|
|
|
粗体
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
2024-08-29 12:36:57 +08:00
|
|
|
|
self.text = re.sub(r'\*\*([^*\n]+)\*\*', r'<b>\1</b>', self.text)
|
2024-08-28 15:16:50 +08:00
|
|
|
|
|
|
|
|
|
def underline(self):
|
|
|
|
|
"""
|
|
|
|
|
下划线
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
2024-08-29 12:36:57 +08:00
|
|
|
|
self.text = re.sub(r'~([^~\n]+)~', r'<u>\1</u>', self.text)
|
2024-08-28 15:16:50 +08:00
|
|
|
|
|
|
|
|
|
def strikethrough(self):
|
|
|
|
|
"""
|
|
|
|
|
删除线
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
2024-08-29 12:36:57 +08:00
|
|
|
|
self.text = re.sub(r'~~([^~\n]+)~~', r'<s>\1</s>', self.text)
|
2024-08-28 15:16:50 +08:00
|
|
|
|
|
|
|
|
|
def highlight(self):
|
|
|
|
|
"""
|
|
|
|
|
高亮
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
2024-08-29 12:36:57 +08:00
|
|
|
|
self.text = re.sub(r'==([^=\n]+)==', r'<mark>\1</mark>', self.text)
|
2024-08-28 15:16:50 +08:00
|
|
|
|
|
2024-08-31 14:38:03 +08:00
|
|
|
|
def up(self):
|
|
|
|
|
"""
|
|
|
|
|
在文本的正上方添加一行小文本,主要用于标拼音
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
|
|
|
|
self.text = re.sub(r'\[(.*?)]\^\((.*?)\)', r'<ruby>\1<rt>\2</rt></ruby>', self.text)
|
|
|
|
|
|
|
|
|
|
def hide(self):
|
|
|
|
|
"""
|
|
|
|
|
在指定的文本里面隐藏一段文本,只有鼠标放在上面才会显示隐藏文本
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
|
|
|
|
self.text = re.sub(r'\[(.*?)]-\((.*?)\)', r'<span title="\2">\1</span>', self.text)
|
|
|
|
|
|
2024-08-31 19:28:04 +08:00
|
|
|
|
def split_line(self):
|
|
|
|
|
self.text = re.sub(r'([*_-]){3}\n', r'<hr>', self.text)
|
|
|
|
|
|
2024-08-28 15:16:50 +08:00
|
|
|
|
def __call__(self, *args, **kwargs):
|
|
|
|
|
"""
|
|
|
|
|
一键运行
|
|
|
|
|
:param args:
|
|
|
|
|
:param kwargs:
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
|
|
|
|
self.bold()
|
|
|
|
|
self.italic()
|
|
|
|
|
self.strikethrough()
|
|
|
|
|
self.underline()
|
|
|
|
|
self.highlight()
|
2024-08-31 14:38:03 +08:00
|
|
|
|
self.up()
|
|
|
|
|
self.hide()
|
2024-08-31 19:28:04 +08:00
|
|
|
|
self.split_line()
|
2024-08-28 15:16:50 +08:00
|
|
|
|
return self.text
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Function:
|
|
|
|
|
"""
|
|
|
|
|
添加特殊功能
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
def __init__(self, text: str):
|
|
|
|
|
"""
|
|
|
|
|
初始化
|
|
|
|
|
:param text: cd文本
|
|
|
|
|
"""
|
|
|
|
|
self.text = text
|
|
|
|
|
|
2024-08-31 14:38:03 +08:00
|
|
|
|
def image(self):
|
|
|
|
|
"""
|
|
|
|
|
实现链接
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
|
|
|
|
self.text = re.sub(r'!\[([^\[\]\n]+)]\(([^()\n]+)\)', r'<img src="\2" alt="\1">', self.text)
|
|
|
|
|
|
2024-08-28 15:16:50 +08:00
|
|
|
|
def link(self):
|
2024-08-31 14:38:03 +08:00
|
|
|
|
"""
|
|
|
|
|
实现链接
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
2024-08-29 12:36:57 +08:00
|
|
|
|
self.text = re.sub(r'\[([^\[\]\n]+)]\(([^()\n]+)\)', r'<a href="\2">\1</a>', self.text)
|
2024-08-28 15:16:50 +08:00
|
|
|
|
|
|
|
|
|
def __call__(self, *args, **kwargs):
|
|
|
|
|
"""
|
|
|
|
|
一键运行
|
|
|
|
|
:param args:
|
|
|
|
|
:param kwargs:
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
2024-08-31 14:38:03 +08:00
|
|
|
|
self.image()
|
2024-08-28 15:16:50 +08:00
|
|
|
|
self.link()
|
|
|
|
|
return self.text
|
|
|
|
|
|
|
|
|
|
|
2024-08-29 12:36:57 +08:00
|
|
|
|
class Value:
|
|
|
|
|
def __init__(self, text: str):
|
|
|
|
|
self.text = text
|
|
|
|
|
self.values = {
|
2024-08-30 13:32:39 +08:00
|
|
|
|
key: value for key, value in re.findall(r'\{([^{}]+)} ?= ?(.+?)(?=\n|$)', text)
|
2024-08-29 12:36:57 +08:00
|
|
|
|
} # 从text中提取所有变量并转换成字典
|
|
|
|
|
|
2024-08-30 14:13:35 +08:00
|
|
|
|
def __call__(self, *args, **kwargs) -> Tuple[str, Dict[str, str]]:
|
2024-08-29 12:36:57 +08:00
|
|
|
|
"""
|
|
|
|
|
将所有变量赋值并移除变量定义
|
|
|
|
|
:param args:
|
|
|
|
|
:param kwargs:
|
|
|
|
|
:return: 赋值后的正文
|
|
|
|
|
"""
|
|
|
|
|
text = self.text
|
|
|
|
|
for k, v in self.values.items():
|
2024-08-30 14:13:35 +08:00
|
|
|
|
text = re.sub(r'\{' + k + '} ?= ?(.+?)(?=\n|$)', '', text) # 移除变量的定义
|
2024-08-30 13:32:39 +08:00
|
|
|
|
text = re.sub(r'\{' + k + '}', fr'{v}', text) # 给变量赋值
|
2024-08-30 14:13:35 +08:00
|
|
|
|
return text, self.values
|
2024-08-29 12:36:57 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class CodeBlock:
|
|
|
|
|
def __init__(self, text: str):
|
2024-08-29 16:36:37 +08:00
|
|
|
|
"""
|
|
|
|
|
找出代码块并移除代码标识
|
|
|
|
|
:param text: 输入的文本
|
|
|
|
|
"""
|
|
|
|
|
self.codes = [i for i in re.findall(r'`([^`]*)`', text) if i != ''] # 找出代码快
|
|
|
|
|
self.text = re.sub(r'``', '', text) # 移除代码标识`
|
2024-08-29 12:36:57 +08:00
|
|
|
|
|
|
|
|
|
def __call__(self, *args, **kwargs):
|
2024-08-29 16:36:37 +08:00
|
|
|
|
"""
|
|
|
|
|
临时移除代码块
|
|
|
|
|
:param args:
|
|
|
|
|
:param kwargs:
|
|
|
|
|
:return: 不含代码的文本
|
|
|
|
|
"""
|
|
|
|
|
for index, item in enumerate(self.codes): # 替换代码块为-@@-(ID)-@@-
|
2024-08-31 14:20:37 +08:00
|
|
|
|
self.text = re.sub(fr'`{re.escape(item)}`', f'\0\1{index}\1\0', self.text) # 同时转译特殊字符
|
2024-08-29 16:36:37 +08:00
|
|
|
|
return self.text
|
2024-08-29 12:36:57 +08:00
|
|
|
|
|
2024-08-30 14:13:35 +08:00
|
|
|
|
def rendering(self, values: Dict[str, str]):
|
|
|
|
|
"""
|
|
|
|
|
渲染代码
|
|
|
|
|
:param values: 变量字典
|
|
|
|
|
:return:
|
|
|
|
|
"""
|
|
|
|
|
for index, code in enumerate(self.codes):
|
|
|
|
|
if re.match(r'\{[^$]*}', code): # 是变量
|
|
|
|
|
# 给变量赋值
|
|
|
|
|
key = re.findall(r'\{([^}]+)}', code)[0] # 查找变量名
|
|
|
|
|
code = re.sub(r'\{' + key + '}', fr'{values[key]}', code)
|
|
|
|
|
self.codes[index] = code # 给变量赋值
|
|
|
|
|
if re.search(r'\n', code): # 是多行代码
|
|
|
|
|
head = re.findall(r'(.*?)\n', code)[0]
|
2024-08-30 14:53:40 +08:00
|
|
|
|
if head in ('', 'yaml'):
|
2024-08-30 19:15:32 +08:00
|
|
|
|
self.codes[index] = f'<pre><code class="language-yaml">{code}</code></pre>'
|
2024-08-31 13:42:17 +08:00
|
|
|
|
elif head in ('shell', 'python'):
|
2024-08-31 14:16:50 +08:00
|
|
|
|
self.codes[index] = f'<pre><code class="language-{head}">{re.sub(f"({head})", "", code)}</code></pre>'
|
2024-08-31 19:28:04 +08:00
|
|
|
|
elif head in ('mermaid',):
|
|
|
|
|
self.codes[index] = f'<div class="{head}">{re.sub(f"({head})", "", code)}</div>'
|
2024-08-31 13:42:17 +08:00
|
|
|
|
elif re.match(r'\$[^$]*\$', code): # 是LaTex代码(单行)
|
|
|
|
|
self.codes[index] = re.sub(fr'\$([^$]*)\$', r'<p>\(\1\)</p>', code)
|
|
|
|
|
else: # 是突出块
|
|
|
|
|
self.codes[index] = f'<span class="block">{code}</span>'
|
2024-08-30 14:13:35 +08:00
|
|
|
|
|
|
|
|
|
def restore(self, new_text: str):
|
|
|
|
|
"""
|
|
|
|
|
将渲染好的代码重新放回处理好的正文
|
|
|
|
|
:param new_text: 处理好的正文
|
|
|
|
|
:return: 加上代码的文章
|
|
|
|
|
"""
|
|
|
|
|
for index, item in enumerate(self.codes):
|
2024-08-31 14:20:37 +08:00
|
|
|
|
new_text = re.sub(f'\0\1{index}\1\0', f'{item}', new_text, flags=re.DOTALL)
|
2024-08-30 14:13:35 +08:00
|
|
|
|
return new_text
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Escape:
|
|
|
|
|
"""
|
|
|
|
|
转义\后字符
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
def __init__(self, text: str):
|
|
|
|
|
"""
|
|
|
|
|
找出转义符并转义
|
|
|
|
|
:param text: 输入的文本
|
|
|
|
|
"""
|
|
|
|
|
self.text = text
|
|
|
|
|
self.escapes = {
|
2024-08-31 14:20:37 +08:00
|
|
|
|
i: f'\0\1\2{i}\2\1\0' for i in re.findall(r'\\(.)', text)
|
2024-08-30 14:13:35 +08:00
|
|
|
|
} # 找出要转义的字符
|
|
|
|
|
print(self.escapes)
|
|
|
|
|
|
|
|
|
|
def __call__(self, *args, **kwargs):
|
|
|
|
|
"""
|
|
|
|
|
临时移除代码块
|
|
|
|
|
:param args:
|
|
|
|
|
:param kwargs:
|
|
|
|
|
:return: 不含代码的文本
|
|
|
|
|
"""
|
|
|
|
|
# TODO
|
|
|
|
|
for index, item in enumerate(self.escapes): # 替换代码块为-@@-(ID)-@@-
|
2024-08-31 14:20:37 +08:00
|
|
|
|
self.text = re.sub(fr'{index}', f'\0\1\2{index}\2\1\0', self.text) # 同时转译特殊字符
|
2024-08-30 14:13:35 +08:00
|
|
|
|
return self.text
|
|
|
|
|
|
2024-08-29 16:36:37 +08:00
|
|
|
|
def restore(self, new_text: str):
|
|
|
|
|
"""
|
|
|
|
|
将渲染好的代码重新放回处理好的正文
|
|
|
|
|
:param new_text: 处理好的正文
|
|
|
|
|
:return: 加上代码的文章
|
|
|
|
|
"""
|
2024-08-30 19:15:32 +08:00
|
|
|
|
for index, item in enumerate(self.escapes):
|
2024-08-29 16:36:37 +08:00
|
|
|
|
new_text = re.sub(fr'-@@-{index}-@@-', f'{item}', new_text, flags=re.DOTALL)
|
|
|
|
|
return new_text
|
2024-08-29 12:39:59 +08:00
|
|
|
|
|
2024-08-29 12:36:57 +08:00
|
|
|
|
|
2024-08-31 14:16:50 +08:00
|
|
|
|
class Cite:
|
|
|
|
|
def __init__(self, text):
|
|
|
|
|
self.text = text
|
|
|
|
|
|
|
|
|
|
def __call__(self, *args, **kwargs) -> str:
|
|
|
|
|
self.text = re.sub('> (.*?) --\[(.*?)]\n', r'<blockquote>\1<footer><cite>\2</cite></footer></blockquote>', self.text) # 渲染有来源的引用
|
|
|
|
|
self.text = re.sub('> (.*?)\n', r'<blockquote>\1</blockquote>\n', self.text) # 渲染没有来源的引用
|
|
|
|
|
return self.text
|
|
|
|
|
|
|
|
|
|
|
2024-08-28 15:16:50 +08:00
|
|
|
|
class Basic:
|
2024-08-30 13:32:39 +08:00
|
|
|
|
def __init__(self, text: str):
|
|
|
|
|
self.text: str = text
|
|
|
|
|
|
2024-08-31 14:16:50 +08:00
|
|
|
|
@staticmethod
|
|
|
|
|
def strong_annotation(text: str) -> str:
|
|
|
|
|
"""
|
|
|
|
|
移除强注释
|
|
|
|
|
:param text: 原始文本
|
|
|
|
|
:return: 移除强注释后的文本
|
|
|
|
|
"""
|
|
|
|
|
return re.sub('\|=[\s\S]=|', '', text)
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
|
def week_annotation(text: str) -> str:
|
|
|
|
|
"""
|
|
|
|
|
移除弱注释
|
|
|
|
|
:param text: 原始文本
|
|
|
|
|
:return: 移除弱注释后的文本
|
|
|
|
|
"""
|
|
|
|
|
return re.sub('// .*?\n', '', text)
|
|
|
|
|
|
2024-08-30 13:32:39 +08:00
|
|
|
|
def paragraph(self):
|
|
|
|
|
"""
|
|
|
|
|
为普通的行套上段落标签
|
|
|
|
|
"""
|
2024-08-31 14:16:50 +08:00
|
|
|
|
# TODO 有点问题
|
2024-08-30 13:32:39 +08:00
|
|
|
|
self.text = re.sub(r'<p>(<.+?>.*?<.+?>)</p>\n',
|
|
|
|
|
r'\1\n', # 移除已被标签包裹的行的额外的<p>标签
|
|
|
|
|
'\n'.join(
|
|
|
|
|
[
|
2024-08-31 14:20:37 +08:00
|
|
|
|
f'<p>{line}</p>' if not re.search('\0.+?\0', line) else line # 识别-@@-n-@@-并保留
|
2024-08-31 13:42:17 +08:00
|
|
|
|
for line in self.text.splitlines() # 把所有非空的行都套上<p>标签
|
|
|
|
|
if not re.search(r'^\s*\n?$', line) # 识别空行或空白行
|
2024-08-30 13:32:39 +08:00
|
|
|
|
]
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def __call__(self, *args, **kwargs):
|
|
|
|
|
self.paragraph()
|
|
|
|
|
return self.text
|
2024-08-28 15:16:50 +08:00
|
|
|
|
|
|
|
|
|
|
2024-08-29 12:36:57 +08:00
|
|
|
|
def add_indent_to_string(input_string: str, indent_spaces: int = 4):
|
|
|
|
|
"""
|
|
|
|
|
给字符串中的每一行前面加上缩进。
|
|
|
|
|
:param input_string: 原始字符串,可以包含多行。
|
|
|
|
|
:param indent_spaces: 每行前面要添加的空格数,默认为4。
|
|
|
|
|
|
|
|
|
|
:return: 带缩进的新字符串。
|
|
|
|
|
"""
|
|
|
|
|
# 使用字符串的splitlines()方法分割原始字符串为行列表
|
|
|
|
|
lines = input_string.splitlines()
|
|
|
|
|
|
|
|
|
|
# 遍历行列表,给每行前面加上相应的缩进,并重新组合成字符串
|
|
|
|
|
indented_string = "\n".join(f"{' ' * indent_spaces}{line}" for line in lines)
|
|
|
|
|
|
|
|
|
|
return indented_string
|
|
|
|
|
|
|
|
|
|
|
2024-08-30 14:13:35 +08:00
|
|
|
|
def body(text: str) -> Tuple[str, Dict[str, str]]:
|
2024-08-29 12:36:57 +08:00
|
|
|
|
"""
|
|
|
|
|
渲染正文部分
|
|
|
|
|
:param text: 输入正文
|
|
|
|
|
:return: 输出渲染后的正文
|
|
|
|
|
"""
|
2024-08-30 14:13:35 +08:00
|
|
|
|
Escape(text)
|
2024-08-31 14:16:50 +08:00
|
|
|
|
text = Basic.week_annotation(text) # 移除弱注释
|
2024-08-30 14:13:35 +08:00
|
|
|
|
text, values = Value(text)() # 提取变量并赋值到文本中
|
2024-08-30 13:32:39 +08:00
|
|
|
|
text = Header(text)() # 渲染标题
|
2024-08-29 12:36:57 +08:00
|
|
|
|
text = Style(text)() # 渲染字体样式
|
|
|
|
|
text = Function(text)() # 渲染特殊功能
|
2024-08-31 14:16:50 +08:00
|
|
|
|
text = Cite(text)() # 渲染引用
|
2024-08-30 13:32:39 +08:00
|
|
|
|
text = Basic(text)() # 渲染基础格式
|
2024-08-28 15:16:50 +08:00
|
|
|
|
|
|
|
|
|
# text = Basic.paragraph(text) # 渲染段落
|
2024-08-30 14:13:35 +08:00
|
|
|
|
return text, values
|
2024-08-28 15:16:50 +08:00
|
|
|
|
|
|
|
|
|
|
2024-08-29 12:36:57 +08:00
|
|
|
|
def main(origen: str):
|
2024-08-31 14:16:50 +08:00
|
|
|
|
# 预处理、
|
|
|
|
|
origen = Basic.strong_annotation(origen) # 移除强注释
|
2024-08-29 16:36:37 +08:00
|
|
|
|
code_block = CodeBlock(origen) # 获取代码内容
|
|
|
|
|
text = code_block() # 暂时移除代码
|
2024-08-31 14:16:50 +08:00
|
|
|
|
# 处理正文
|
|
|
|
|
text, values = body(text)
|
|
|
|
|
# 后处理
|
2024-08-30 14:13:35 +08:00
|
|
|
|
code_block.rendering(values) # 渲染代码
|
|
|
|
|
return code_block.restore(text) # 放回代码
|
2024-08-29 12:36:57 +08:00
|
|
|
|
|
|
|
|
|
|
2024-08-28 15:16:50 +08:00
|
|
|
|
if __name__ == '__main__':
|
|
|
|
|
with open('test.md', encoding='utf-8') as test:
|
2024-08-29 12:39:59 +08:00
|
|
|
|
cd = main(test.read())
|
2024-08-28 15:16:50 +08:00
|
|
|
|
with open('test.html', 'w', encoding='utf-8') as html:
|
2024-08-29 12:36:57 +08:00
|
|
|
|
html.write(f"""<!DOCTYPE html>
|
|
|
|
|
<html lang="zh-CN">
|
|
|
|
|
<head>
|
|
|
|
|
<meta charset="UTF-8">
|
|
|
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
|
|
|
<title>UTF-8编码示例</title>
|
2024-08-29 16:36:37 +08:00
|
|
|
|
<script type="text/javascript" async
|
|
|
|
|
src="https://cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.7/MathJax.js?config=TeX-MML-AM_CHTML">
|
|
|
|
|
</script>
|
2024-08-31 13:42:17 +08:00
|
|
|
|
<link href="https://cdn.jsdelivr.net/npm/prismjs/themes/prism.css" rel="stylesheet" />
|
|
|
|
|
<script src="https://cdn.jsdelivr.net/npm/prismjs/prism.js"></script>
|
2024-08-30 19:15:32 +08:00
|
|
|
|
<script src="https://cdn.jsdelivr.net/npm/prismjs/components/prism-yaml.min.js"></script>
|
2024-08-31 13:42:17 +08:00
|
|
|
|
<style>
|
|
|
|
|
.block {{
|
|
|
|
|
background-color: grey; /* 灰色背景 */
|
|
|
|
|
color: white; /* 白色文字 */
|
|
|
|
|
}}
|
|
|
|
|
</style>
|
2024-08-31 19:28:04 +08:00
|
|
|
|
<script src="https://cdn.jsdelivr.net/npm/mermaid/dist/mermaid.min.js"></script>
|
2024-08-29 12:36:57 +08:00
|
|
|
|
<!-- 可以在这里添加其他元数据和CSS链接 -->
|
|
|
|
|
</head>
|
2024-08-31 19:28:04 +08:00
|
|
|
|
<body>
|
|
|
|
|
<script>
|
|
|
|
|
mermaid.initialize({{startOnLoad:true}});
|
|
|
|
|
</script>
|
2024-08-29 12:36:57 +08:00
|
|
|
|
{add_indent_to_string(cd, 4)}
|
|
|
|
|
</body>
|
|
|
|
|
</html>
|
2024-08-29 12:39:59 +08:00
|
|
|
|
""")
|