本文最后更新于 2025-03-23T13:54:49+08:00
简介
最近想在博客中写一些日语学习笔记,需要用到 ruby 标签来显示假名注音。虽然可以直接在 markdown 中使用 ruby 标签,但写起来实在太麻烦了。网上也有一些 hexo 插件可以实现这个功能,但要么语法太复杂,要么和我用的渲染器不兼容。于是使用 hexo 的 filter 功能写了一个简单的插件。
代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
| 'use strict';
hexo.extend.filter.register('before_post_render', function(data) { if (!data.content) return data;
const contentParts = []; const codeBlockRegex = /```[\s\S]*?```|`[\s\S]*?`/g; let lastIndex = 0; let match;
while ((match = codeBlockRegex.exec(data.content)) !== null) { if (match.index > lastIndex) { contentParts.push({ type: 'text', content: data.content.substring(lastIndex, match.index) }); }
contentParts.push({ type: 'code', content: match[0] });
lastIndex = match.index + match[0].length; }
if (lastIndex < data.content.length) { contentParts.push({ type: 'text', content: data.content.substring(lastIndex) }); }
const pattern = /{([^{}|]+)\|([^{}]+)}/g;
const processedParts = contentParts.map(part => { if (part.type === 'code') { return part.content; } else { return part.content .replace(pattern, (match, base, rt) => { return `<ruby>${base}<rt>${rt}</rt></ruby>`; }); } });
data.content = processedParts.join(''); return data; });;
|
在 hexo 项目的 scripts
目录下新建一个 ruby.js
文件,将上面的代码复制进去即可。
使用方法
{耄|mào}{耋|dié}
耄 耋
{一方通行|アクセラレータ}
一方通行
{这是一段文本|这是一段注音}
这是一段文本
在 tag 插件和表格中也可以使用:
とある魔術の{禁書目録|インデックス}
とある魔術の禁書目録
在 VSCode 中预览
我一般用 VSCode 写文章,希望在 VSCode 中的预览中也能看到注音效果。具体方法如下:
- 安装
Markdown Preview Enhanced
插件
Ctrl + Shift + P
打开命令面板,找到 Markdown Preview Enhanced: Extend Parser
命令
- 根据需要选择在工作区还是全局扩展解析器
- 选择后会自动打开
parser.js
文件,将内容替换为下面的代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47
| ({ onWillParseMarkdown: async function(markdown) { const parts = []; const codeBlockRegex = /```[\s\S]*?```|`[\s\S]*?`/g; let lastIndex = 0; let match;
while ((match = codeBlockRegex.exec(markdown)) !== null) { if (match.index > lastIndex) { parts.push({ type: 'text', content: markdown.substring(lastIndex, match.index) }); } parts.push({ type: 'code', content: match[0] }); lastIndex = match.index + match[0].length; } if (lastIndex < markdown.length) { parts.push({ type: 'text', content: markdown.substring(lastIndex) }); }
const processedParts = parts.map(part => { if (part.type === 'code') { return part.content; } else { return part.content.replace(/\{([^|\{\}]+)\|([^|\{\}]+)\}/g, (match, base, ruby) => { return `<ruby><rb>${base}</rb><rt>${ruby}</rt></ruby>`; }); } });
return processedParts.join(''); },
onDidParseMarkdown: async function(html) { return html; } })
|