Gar

Gar 是一个 Bash 脚本程序,用于管理 Markdown 文档项目,可将 Markdown 文档集合其转化为 HTML 文档集合。Gar 的运行,依赖 pandoc,git,tree 以及一个能在 Shell(命令行)里打开指定网页文件的网页浏览器。Gar 默认将 Firefox 作为网页浏览器,但是可在文档项目根目录的 gar.conf 文件中指定其它符合要求的网页浏览器。

文档项目初始化

命令:gar init 文档项目名

例如:

$ gar init demo
[master (root-commit) 6f7dd1c] init
 1 file changed, 2 insertions(+)
 create mode 100644 gar.conf

以下命令可观察 gar init 创造了什么:

$ cd demo
$ ls -a
  ..  gar.conf  .git  output  source  图片
$ gar tree
demo
├── gar.conf
├── output
├── source
└── 图片
$ git log
commit 6f7dd1c23c0cc8b18eb84b6a5236605eebd2bfbf (HEAD -> master)
Author: xxx 
Date:   Tue Mar 9 07:30:53 2021 +0800

    init

文档项目初始化后,文档的撰写和编辑工作主要在 source 目录进行。Gar 将 Markdown 文档转化为 HTML 文档后,放在 output 子目录内。

文集创建与删除

进入 source 目录:

$ cd source

创建文集 foo:

$ gar new-class foo

可使用 gar tree 查看文档项目的目录变化,观察 gar new-class 创造了什么:

$ gar tree
demo
├── gar.conf
├── output
│   └── foo
├── source
│   └── foo
└── 图片
    └── foo

可一次创建多个文集:

$ gar new-class a b c

结果为:

$ gar tree
demo
├── gar.conf
├── output
│   ├── a
│   ├── b
│   ├── c
│   └── foo
├── source
│   ├── a
│   ├── b
│   ├── c
│   └── foo
└── 图片
    ├── a
    ├── b
    ├── c
    └── foo

删除文集:

$ gar remove-class a b c
$ gar tree
demo
├── gar.conf
├── output
│   └── foo
├── source
│   └── foo
└── 图片
    └── foo

可在文集里创建子文集:

$ cd foo
$ gar new-class a
$ gar tree
demo
├── gar.conf
├── output
│   └── foo
│       └── a
├── source
│   └── foo
│       └── a
└── 图片
    └── foo
        └── a

可创建文集层次序列:

$ gar new-class b/c/d/e/f
$ gar tree
demo
├── gar.conf
├── output
│   └── foo
│       ├── a
│       └── b
│           └── c
│               └── d
│                   └── e
│                       └── f
├── source
│   └── foo
│       ├── a
│       └── b
│           └── c
│               └── d
│                   └── e
│                       └── f
└── 图片
    └── foo
        ├── a
        └── b
            └── c
                └── d
                    └── e
                        └── f

将试验复盘:

$ gar remove-class a b
$ gar tree
demo
├── gar.conf
├── output
│   └── foo
├── source
│   └── foo
└── 图片
    └── foo

提示,目前工作目录仍为 source/foo。

创建和删除文档

在文集目录内,使用 gar new-post 创建内容为空的文档。例如,在 source/foo 内创建 test.md 文档:

$ gar new-post test.md
[master 6a894eb] Added test.md
 1 file changed, 0 insertions(+), 0 deletions(-)
 create mode 100644 source/foo/test.md

查看发生了什么:

$ gar tree
demo
├── gar.conf
├── output
│   └── foo
├── source
│   └── foo
│       └── test.md
└── 图片
    └── foo
        └── test
$ git log
commit 6a894eb44cf66e95b4ff938e05c019b7218039e0 (HEAD -> master)
Author: xxx 
Date:   Tue Mar 9 08:16:49 2021 +0800

    Added test.md

commit 6f7dd1c23c0cc8b18eb84b6a5236605eebd2bfbf
Author: xxx 
Date:   Tue Mar 9 07:30:53 2021 +0800

    init

每次创建文档时,Gar 会调用 git 记录文档创建历史。

可一次创建多份空文档:

$ gar new-post a.md b.md c.md
[master 25e7d65] Added a.md b.md c.md
 3 files changed, 0 insertions(+), 0 deletions(-)
 create mode 100644 source/foo/a.md
 create mode 100644 source/foo/b.md
 create mode 100644 source/foo/c.md
$ gar tree
demo
├── gar.conf
├── output
│   └── foo
├── source
│   └── foo
│       ├── a.md
│       ├── b.md
│       ├── c.md
│       └── test.md
└── 图片
    └── foo
        ├── a
        ├── b
        ├── c
        └── test

使用 gar remove-post 可删除当前工作目录下的文档。以下命令可将上述创建的文档一举删除:

$ gar remove-post test.md a.md b.md c.md
[master 189da8b] Remove test.md a.md b.md c.md
 4 files changed, 0 insertions(+), 0 deletions(-)
 delete mode 100644 source/foo/a.md
 delete mode 100644 source/foo/b.md
 delete mode 100644 source/foo/c.md
 delete mode 100644 source/foo/test.md

每次删除文档,git 会记录文档的删除历史。

经过上述操作后,这个试验性的文档项目又复盘为:

$ gar tree
demo
├── gar.conf
├── output
│   └── foo
├── source
│   └── foo
└── 图片
    └── foo

网页的生成和预览

记住,当前的工作目录依然是 source/foo。下面的命令重新创建 test.md:

$ gar new-post test.md

然后用文本编辑器打开 test.md,增加以下内容:



这只是一份无用的示例文档。

使用 gar convert 命令可将文档 test.md 转换为网页文件 test.html,并将其置于 文档项目根目录/output/foo 目录内:

$ gar convert test.md

查看文档目录发生的变化:

$ gar tree
demo
├── gar.conf
├── output
│   └── foo
│       └── test.html
├── source
│   └── foo
│       └── test.md
└── 图片
    └── foo
        └── test

倘若工作目录(当前文集)内有多份文档,也可以一次性将其转换为一组网页文件,例如:

$ gar convert test.md a.md b.md c.md

使用 gar preview 命令,可将文档转化为网页文件,并由 Gar 默认的网页浏览器打开:

$ gar preview test.md

Gar_第1张图片

gar preview 不支持多份文档一次性转换和预览。

附录

Gar 的全部代码:

#!/bin/bash

SCRIPT_PATH=$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)

GAR_CONF=gar.conf
SOURCE=source
IMAGES=图片
OUTPUT=output

function error_msg {
    echo $1
    exit -1
}

function goto_root {
    if [ ! -e $GAR_CONF ]
    then
        if [ $(pwd) = / ]
        then
            echo "gar.conf Not found!"
        else
            cd ..
            goto_root  
        fi
    fi
}

function class_name {
    TARGET_PATH=$1
    goto_root
    SOURCE_PATH=$(pwd)/$SOURCE/
    echo ${TARGET_PATH//$SOURCE_PATH/}
}

function get_title {
    NAME=${1%.md}
    # 将文件名里用于间隔中文和英文的短线替换为空格
    L_SPACE_PAT='s/\([^a-zA-Z]\)-/\1 /g'
    R_SPACE_PAT='s/-\([^a-zA-Z]\)/ \1/g'
    FANCY_NAME="$(echo $NAME | sed -e "$L_SPACE_PAT; $R_SPACE_PAT")"
    # 文章标题
    PREFIX="^[[:space:]]*\.\.[[:space:]]*title:[[:space:]]*"
    TITLE=$(grep -o "${PREFIX}.*$" $1 | sed -e "s/${PREFIX}\(.*\)$/\1/g")
    if [ -z "$TITLE" ]
    then
        TITLE=$FANCY_NAME
    fi
    echo $TITLE
}

function markdown_to_html {
    HTML=$OUTPUT/$1/${2%.*}.html
    pandoc $SOURCE/$1/$2 -s --mathjax \
           -c $SCRIPT_PATH/gar.css \
           --highlight-style pygments \
           --metadata title="$TITLE" -o $HTML
}

case $1 in
    init)
        case $2 in
            "")
                error_msg "You should tell me the name of the project!"
                ;;
            *)
                mkdir $2
                cd $2
                echo '#!/bin/bash' > $GAR_CONF
                echo "BROWSER_FOR_GAR=firefox" >> $GAR_CONF
                mkdir $SOURCE $OUTPUT $IMAGES
                git init -q
                git add .
                git commit -a -m "init"
                ;;
        esac
        ;;
    new-class)
        case $2 in
            "")
                error_msg "You should tell me the name of the posts class!"
                ;;
            *)
                for i in ${@:2}
                do
                    MARK=$(pwd)
                    mkdir -p $i
                    cd $i
                    CLASS_NAME=$(class_name $MARK/$i)
                    
                    goto_root
                    mkdir -p $IMAGES/$CLASS_NAME
                    mkdir -p $OUTPUT/$CLASS_NAME
                    cd $MARK
                done
                ;;
        esac
        ;;
    remove-class)
        case $2 in
            "")
                error_msg "You should tell me the name of the posts class!"
                ;;
            *)
                for i in ${@:2}
                do
                    MARK=$(pwd)
                    rm -rf $i
                    CLASS_NAME=$(class_name $MARK/$i)
                    
                    goto_root
                    rm -rf $IMAGES/$CLASS_NAME
                    rm -rf $OUTPUT/$CLASS_NAME
                    cd $MARK
                done
                ;;
        esac
        ;;
    new-post)
        case $2 in
            "")
                error_msg "You should tell me the name of the post!"
                ;;
            *)
                MARK=$(pwd)
                CLASS_NAME=$(class_name $MARK)
                for i in ${@:2}
                do
                    touch $i
                    goto_root
                    mkdir -p $IMAGES/$CLASS_NAME/${i%.*}
                    cd $MARK
                done
                goto_root
                git add .
                DOCUMENTS="${@:2}"
                git commit -a -m "Added $DOCUMENTS"
                ;;
        esac
        ;;
    remove-post)
        case $2 in
            "")
                error_msg "You should tell me the name of the post!"
                ;;
            *)
                MARK=$(pwd)
                for i in ${@:2}
                do
                    if [ ! -e $i ]
                    then
                        error_msg "The file you want to remove is not found!"
                    fi
                    rm $i
                    CLASS_NAME=$(class_name $MARK)
                    goto_root
                    rm -rf $IMAGES/$CLASS_NAME/${i%.*}
                    HTML_OF_POST=$OUTPUT/$CLASS_NAME/${i%.*}.html
                    if [ -e HTML_OF_POST ]
                    then
                        rm $HTML_OF_POST
                    fi
                    cd $MARK
                done
                goto_root
                git add .
                DOCUMENTS="${@:2}"
                git commit -a -m "Remove $DOCUMENTS"
                ;;
        esac
        ;;
    rename)
        case $2 in
            "")
                error_msg "You should tell me the name of the post you want to rename!"
                ;;
            *)
                if [ ! -e $2 ]
                then
                    error_msg "The file you want to rename is not found!"
                fi
                case $3 in
                    "")
                        error_msg "You should tell me new name of the post!"
                        ;;
                    *)
                        mv $2 $3
                        CLASS_NAME=$(class_name $(pwd))
                        goto_root
                        mv $IMAGES/$CLASS_NAME/${2%.*} $IMAGES/$CLASS_NAME/${3%.*}
                        mv $OUTPUT/$CLASS_NAME/${2%.*} $OUTPUT/$CLASS_NAME/${3%.*}
                        git add .
                        git commit -a -m "Rename $2 $3"
                        ;;
                esac
                ;;
        esac
        ;;
    convert)
        case $2 in
            "")
                error_msg "You should tell me the name of the post!"
                ;;
            *)
                MARK=$(pwd)
                for i in ${@:2}
                do
                    TITLE=$(get_title $i)
                    CLASS_NAME=$(class_name $MARK)
                    goto_root
                    markdown_to_html $CLASS_NAME $i
                    cd $MARK
                done
                ;;
            esac
        ;;
    preview)
        case $2 in
            "")
                error_msg "You should tell me the name of the post!"
                ;;
            *)  
                TITLE=$(get_title $2)
                CLASS_NAME=$(class_name $(pwd))
                goto_root
                source gar.conf
                $BROWSER_FOR_GAR $OUTPUT/$CLASS_NAME/${2%.*}.html
                ;;
            esac
        ;;
    tree)
        goto_root
        GAR_ROOT=$(basename $(pwd))
        cd ..
        tree $GAR_ROOT
        cd $GAR_ROOT
        ;;
    *)
        error_msg "I do not know what you want to do!"
        ;;
esac

Gar 在使用 pandoc 将 Markdown 文档转化为网页时,需要一个 CSS 文件 gar.css,其内容如下:

html {
    font-size: 16px;
    line-height: 1.8rem;
}

body {
    margin: 0 auto;
    max-width: 50rem;
    padding: 50px;
    hyphens: auto;
    word-wrap: break-word;
    font-kerning: normal;
}

header {
    text-align: center;
    margin-bottom: 4rem;
}

h1, h2, h3, h4, h5 {
    margin-top: 2rem;
    margin-bottom: 2rem;
    color: #d35400;
}

h1.title { font-size: 2.5rem; }
h1 { font-size: 1.8rem; }
h2 { font-size: 1.65rem; }
h3 { font-size: 1.5em; }
h4 { font-size: 1.35rem; }
h5 { font-size: 1.2rem; }

p {
    margin: 1.3rem 0;
    text-align: justify;
}

figure {
    text-align: center;
}
figure img {
    width: 80%;
}
figure figcaption {
    font-size: 0.9rem;
}

pre {
    padding: 1rem;
    font-size: 0.9rem;
    line-height: 1.6em;
    overflow:auto;
    background: #f8f8f8;
    border: 1px solid #ccc;
    border-radius: 0.25rem;
}

p code {
    color: #d35400;
}

/* 文章里小节标题的序号与标题名称之间的间距 */
span.section-sep { margin-left: 0.5rem; margin-right: 0.5rem; }


blockquote {
    margin: 0px !important;
    border-left: 4px solid #009A61;
}

blockquote p {
    font-size: 1rem;
    line-height: 1.8rem;
    margin: 0px !important;
    text-align: justify;
    padding:0.5em;
}

上述 gar.css 并无特别之处,完全可根据自己对 css 的熟悉程度并结合需要自行定制,但是要记得将它放在 gar 脚本同一目录下。

你可能感兴趣的:(Gar)