Gatsby + AsciiDoc
AsciiDoc 比 Markdown 更规范、灵活、严谨,缺点是插件没有 Markdown 丰富。
创建
生成项目
$ gatsby new
What would you like to call your site?
? ‣ notfound 1
What would you like to name the folder where your site will be created?
? test/ notfound 2
? Will you be using JavaScript or TypeScript?
(Single choice) Arrow keys to move, enter to confirm
JavaScript
▸ TypeScript 3
? Will you be using a CMS?
(Single choice) Arrow keys to move, enter to confirm
▸ No (or I'll add it later) 4
–
Contentful
DatoCMS
Netlify CMS
Sanity
Shopify
WordPress
? Would you like to install a styling system?
(Single choice) Arrow keys to move, enter to confirm
▸ No (or I'll add it later) 5
–
Emotion
PostCSS
Sass
styled-components
Theme UI
vanilla-extract
? Would you like to install additional features with other plugins?
(Multiple choice) Use arrow keys to move, spacebar to select, and confirm with an enter on "Done"
◯ Add the Google Analytics tracking script
◉ Add responsive images 6
◉ Add page meta tags with React Helmet 7
◉ Add an automatic sitemap 8
◉ Generate a manifest file 9
◯ Add Markdown and MDX support
◯ Add Markdown support (without MDX)
─────
▸ Done
1 | 网站名称 notfound |
2 | 目录名称 notfound |
3 | 使用 TypeScript |
4 | 不用 CMS |
5 | 不用 sytling system |
6 | 插件:responsive images |
7 | 插件:react helmet |
8 | 插件:sitemap |
9 | 插件:manifest file |
asciidoc 插件
npm install gatsby-transformer-asciidoc 1
npm i font-awesome 2
1 | asciidoc 插件 |
2 | 添加 font-awesome, 提供给 asciidoc 使用 |
修改配置:
gatsby-config.ts
graphqlTypegen: true, 1
plugins: [
{
resolve: "gatsby-transformer-asciidoc",
options: {
attributes: {
icons: "font", 2
showtitle: false, 3
},
},
},
{ 4
resolve: "gatsby-source-filesystem",
options: {
name: "posts",
path: "./src/pages/posts",
},
__key: "posts",
},
]
1 | GraphQL 类型生成 |
2 | 使用 font awesome |
3 | 不显示一级标题 |
4 | 将 asciidoc 文件(后缀为 .adoc ) 放在 ./src/pages/posts 目录下,作为 graphql 资源 |
添加 adoc 以及页面
添加 adoc 文件:
./src/pages/posts/hello.adoc
= Hello, AsciiDoc!
Firstname Lastname <author@asciidoctor.org> 1
1.0, 1970-01-01: Asciidoctor article template 2
:page-slug: hello 3
:page-category: asciidoc 4
:page-draft: true 5
This is an interactive editor.
Use it to try https://asciidoc.org[AsciiDoc].
== Section Title
* A list item
* Another list item
[,ruby]
----
puts 'Hello, World!'
----
1 | 姓、名和邮箱,用空格分隔 |
2 | 版本号、日期和备注,分别用 , 和 : 分隔 |
3 | 自定义属性: slug 用作 'url' |
4 | 自定义属性: category 用作分类 |
5 | 自定义属性: draft 用作草稿 |
npm start 1
1 | 开发模式启动服务 |
访问 http://localhost:8000/___graphql ,输入 graphql 查询,可以看到结果
query Post {
asciidoc {
pageAttributes {
slug 1
}
}
}
1 | 自定义页面属性 |
创建文件 src/pages/posts/{asciidoc.pageAttributes__slug}.tsx
gatsby 使用 asciidoc.pageAttributes.slug
的值作为路由的一部分,如 src/pages/posts/hello.adoc
路由为 /posts/hello
。
src/pages/posts/{asciidoc.pageAttributes__slug}.tsx
@import "font-awesome/css/font-awesome.css";
@import "./asciidoctor.css"; 1
const Template = (props: PageProps<Queries.PostQuery>) => { 2
const post = props.data.asciidoc; 3
return (
<article className="post">
<header>
<h1>{post?.document?.title}</h1>
<div className="post-meta">
<div className="post-meta-item">版本 {post?.revision?.number}</div>
<div className="post-meta-item">
发表于
<time dateTime={post?.revision?.date || "1970-01-01"}>
{post?.revision?.date || "1970-01-01"}
</time>
</div>
</div>
</header>
<div dangerouslySetInnerHTML={{ __html: post?.html || "" }} />
</article>
);
};
4
export const query = graphql`
query Post($id: String!) {
asciidoc(id: { eq: $id }) {
id
html
document {
title
}
pageAttributes {
slug
category
}
revision {
date
number
}
}
}
`;
export default Template;
1 | 页面样式来自 asciidoctor ,也可通过 asciidoctor-skins 获取不同主题 |
2 | 类型 Queries.PostQuery 在运行 npm start 后,会根据 graphql 自动生成 |
3 | graphql 查询结果 |
4 | grapqhl 页面查询语句。启动 gatsby 后,会根据该语句自动生成类型文件 src/gatsby-types.d.ts 。查询参数 id 来自 pageContext ,可通过 props.pageContext.id 访问。 |
gatsby 将会遍历 src/pages/posts/*.adoc
为每个文件生成页面(如 public/posts/hello/index.html
)和数据(如 public/page-data/posts/hello/page-data.json
)。
语法高亮
语法高了可通过后端或者前端处理。
后端处理
通过自定义 Converter在处理代码相关 node 时进行语法高亮。
前端处理
可以使用 highlight.js 或者 prismjs 等。这里我们使用 prismjs
npm install prismjs
adoc 支持页面中插入 标注,解析为 HTML 会生成如 <i class="conum" data-value="1"></i><b>1</b>
这样的 HTML 标签,但 prismjs 在高亮前会 剔除掉 HTML 导致标注无法正常显示,所以需要对标注进行特殊处理。
const placeholder = "PLACEHOLDER_COLUMN{n}";
const placeholderRegex = RegExp("PLACEHOLDER_COLUMN(\\d+)", "g");
const conum = '<i class="conum" data-value="{n}"></i><b>{n}</b>';
const conumRegex = RegExp(
'<i class="conum" data-value="(\\d+)"></i><b>\\(\\d+\\)</b>',
"g"
);
const Template = (props: PageProps<Queries.PostQuery>) => {
const post = props.data.asciidoc;
const [html, setHtml] = React.useState("");
React.useEffect(() => {
if (post?.html) {
const doc = new DOMParser().parseFromString(post.html, "text/html");
doc.querySelectorAll("pre.highlight > code").forEach(function (el) {
if (!el.getAttribute("data-lang")) return;
1
el.innerHTML = el.innerHTML.replace(conumRegex, (_, i) =>
placeholder.replace("{n}", i)
);
2
Prism.highlightElement(el, false);
3
el.innerHTML = el.innerHTML.replace(placeholderRegex, (_, i) =>
conum.replaceAll("{n}", i)
);
});
setHtml(doc.body.innerHTML);
} else {
setHtml("");
}
}, [post?.html]);
return <>...</>
}
1 | 前置处理:调标注相关的 HTML 用特殊字符替换 |
2 | 高亮 |
3 | 后置处理:特殊字符还原成标注 |
如果特殊字符被高亮了,就会无法替换回来,所以特殊字符需要足够特殊。