用 Org 和 GitPage 搭建简易博客
1. Intro
2. Setup
我的笔记目录结构如下:
tree -L 2
. ├── 20230209T052037--常用工具集合__tool.org ├── 20230209T055812--zoxide__tool.org ├── 20230209T155228--emacs-server__tool.org ├── 20230210T211056--macos私人定制__tool.org ├── 20230211T045151--emacs-插件-easy-kill__tool.org ├── 20230211T133256--emacs-插件集合__tool.org ├── 20230211T143210--emacs__tool.org ├── 20230211T143523--vscode__tool.org ├── 20230211T143600--vimnvim__tool.org ├── 20230212T193131--用org和github-page搭建博客__tool.org ├── LICENSE ├── README.md ├── index.org └── scaffold ├── Makefile ├── attachs ├── css ├── public_html └── publish.el 5 directories, 15 files
在笔记根目录下还有个 github-action 的配置文件:
tree .github
: .github : └── workflows : └── org-denotes-publish.yml : : 2 directories, 1 file
我把所有用于构建的脚本等与笔记正文无关的文件都放在 scaffold 目录下:
Makefile | 构建脚本 |
attachs | 存放外部链接文件 |
css | 存放网站的样式文件 |
public_html | 生成的静态网站 |
publish.el | 网站静态页面生成器 |
org-denotes-publish | github-action脚本 |
首先看 org-denotes-publish 脚本:
name: org-denotes-publish on: push: branches: ["main"] workflow_dispatch: permissions: contents: read pages: write id-token: write # Allow one concurrent deployment concurrency: group: "pages" cancel-in-progress: true jobs: build-and-deploy: runs-on: ubuntu-latest steps: - uses: purcell/setup-emacs@master with: version: 28.2 - uses: actions/checkout@v3 - run: | cd scaffold make clean make - uses: actions/configure-pages@v3 - uses: actions/upload-pages-artifact@v1 with: path: './scaffold/public_html' - uses: actions/deploy-pages@v1
上述脚本逻辑很清晰:配置环境-›构建静态网站-›发布。如何构建静态网站由我们自己控制,其他交由 github 。
再看 Makefile 脚本:
# Makefile for myblog .PHONY: all publish publish_no_init all: publish publish: publish.el @echo "Publishing..." emacs --batch -q --script publish.el clean: @echo "Cleaning up.." @rm -rvf *.elc @rm -rvf ./public_html/ @rm -rvf ~/.org-timestamps/*
这脚本很简单,执行 make 运行命令: emacs --batch -q --script publish.el
,也可以直接把这条命令写入 github-action 脚本中。
接下来就是最重要的 publish.el 文件了:
;;; publish.el --- notes setting. -*- lexical-binding: t no-byte-compile: t -*- (require 'package) (package-initialize) (unless package-archive-contents (package-refresh-contents)) (dolist (pkg '(denote htmlize)) (unless (package-installed-p pkg) (package-install pkg))) (require 'denote) (require 'htmlize) (require 'ox-publish) (setq denote-directory "../") (setq org-export-with-section-numbers t org-export-htmlize-output-type 'css org-export-with-smart-quotes t org-export-with-sub-superscripts nil) (setq org-html-doctype "html5" org-html-html5-fancy t org-html-checkbox-type 'html org-html-htmlize-output-type 'css org-html-container-element "section" org-html-head-include-default-style nil) (defvar yx/html-head "<link rel='stylesheet' href='./css/org.css' type='text/css'/>") (defvar yx/html-postamble "<div id='postamble' class='status'> <hr/> <p class='author'>Created with %c by %a <br\>Updated: %C<br/></p> </div>") (setq org-publish-project-alist `(("yx-notes" :components ("yx-notes-page" "yx-notes-static")) ("yx-notes-page" :base-directory "../" :base-extension "org" :publishing-directory "./public_html/" :recursive nil :publishing-function org-html-publish-to-html :headline-levels 4 :auto-preamble t :auto-sitemap t :sitemap-filename "index.org" :sitemap-sort-files anti-chronologically ;sort the posts from newest to oldest. :html-link-home "/yx-notes" :html-link-up "/yx-notes" :html-head-include-scripts nil :html-head-include-default-style nil :html-head ,yx/html-head :html-postamble ,yx/html-postamble ) ("yx-notes-static" :base-directory "./" :base-extension "css\\|js\\|png\\|jpg\\|gif" :publishing-directory "./public_html/" :recursive t :publishing-function org-publish-attachment )) ) (org-publish "yx-notes" t nil) ;;; publish.el ends here
这个脚本:
- 安装必要插件:
- 我用 org-roam 做笔记,它使用链接的形式是的形式,为了正确导出成 html 识别的链接,必须导入org-roam包,并正确设置 denote-directory 变量。
- 安装 htmlize 插件。
设置导出的一些参数:
org-export-with-section-numbers
t
导出的每个 headline 上都有一个指示层级的数字 org-export-with-sub-superscripts
nil
下划线不要导出成 latex 似的下标 org-html-head-include-default-style
nil
不要使用默认样式 这三个对我比较重要,其他可选。
- 设置要发布的 org 项目。
这一步是通过设置参数
org-publish-project-alist
来完成的,它是一个列表,列表里的每一个子列表是一个项目。比如上面脚本中设置了三个项目:- yx-notes-static
- 从
:base-directory
中,把所有由:base-extension
指定后缀名的文件,移动到由:publishing-directory
指定的发布目录。 - yx-notes-page
- (同上)把基目录下的 org 文件转换成 html 文件,并放入发布目录。
:auto-sitemap
字段指示是否自动生成一个汇总主页。 - yx-notes
- 把上面两个项目汇总成一个项目。
这样,基本上就有一个笔记展示主页的雏形了。接下来就是定制样式(如何添加样式,请看脚本中 yx/html-head
变量的使用 ),添加内容。
3. Code Highlight
org-mode 导出 html 中的代码高亮由变量 org-html-htmlize-output-type
控制:
’inline-css | (default)以 inline 的方式,用当前 emacs 的主题的色彩高亮代码 |
’css | 用类选择器标识代码,用户用自己定义的方案高亮代码 |
nil | 啥都没有 |
为了保证整个页面风格一致,所以选取 ’css 的方式。这部分 css 借鉴于kaushalmodi ,其用 leuven-theme 主题修改的高亮方案。我喜欢它的简洁。
4. Workflow
- 本地编辑笔记
- 本地构建:
- 在 scaffold 目录下执行
make
- 启动 web-server,执行
python -m http.server --directory=public_html
- 打开 localhost:8000 查看修改
- 在 scaffold 目录下执行
- 上传服务器
- 提交,上传(上传的 github 仓库,会自动进行构建)
整个流程简单,清晰。
5. Supplement
参考:
在配置中有思考有收获,或许这就是折腾的意义吧。