Recently, I migrated my personal site from Hugo to Jekyll and I was satisfied to make this change. My website seems to be silent for roughly 3 years after I entered Pinduoduo. It is the right time to bring my person page back to the earth as well as my writing habit.

Jekyll is a powerful static websites and blogs generator. It takes text written in your favorite markup language and uses layouts to create a static website. You can tweak the site’s look and feel, URLs, the data displayed on the page, and more. Since I am not really familiar with all the features Jekyll has provided to write a nice post, I will summarize some common usage in this blog.

1. How to use Jekyll ?

Prerequisites

  • Ruby version 2.5.0 or higher
  • RubyGems
  • GCC and Make

Use Jekyll locally

  1. Install all prerequisites.

  2. Install the jekyll and bundler

    gem install jekyll bundler
    
  3. Install mermaid-cli

    npm install -g @mermaid-js/mermaid-cli
    

    If we cannot install chromium, we can ignore it.

    PUPPETEER_SKIP_DOWNLOAD=1 npm install -g @mermaid-js/mermaid-cli
    
  4. Serve in local environment.
    bundle exec jekyll serve --host 0.0.0.0
    
  5. Browse to http://127.0.0.1:4000

Now, we could visit and preview our blog locally before pushing contents to github pages.

2. How blogging work with Jekyll?

You might be wondering how you can have a blog without a database. In true Jekyll style, blogging is powered by text files only.

Post header

All blog posts live in a folder called posts. Jekyll is responsible for transforming text file into rich text web pages without the help of a database. If we want to create a post from scratch, we need one file in a special name format: YYYY-mm-dd-name.md. The date and name part will form up a permanent link to this post.

Following header content:

---
layout: post
title: the title we include in the main page
date: 2023-02-08 15:58:00
description: description showing in the preview mode
tags: tags we want to attach.
categories: categories we want to use.
---

Now, we could proceed our content of this post.

We can write markdown scripts directly, but we still need to use the power of Liquid templating language sometimes. For example, if we want to open an link in another tab, we need to append {:target="_blank"} at the end of a hyperlink [description](http://link.com). We can also click the default cv link without opening a new tab.

3. HTML Support

We can take full advantage of html with markdown. Here are some example to use html elements:

  • <ul> and <li>

    source:

    <ul>
      <li>brunch</li>
      <li>fixie</li>
      <li>raybans</li>
      <li>messenger bag</li>
    </ul>
    

    display:

    • brunch
    • fixie
    • raybans
    • messenger bag
  • <a href="">

    source:

    <a href="https://github.com">Github</a>
    

    display:

    Github

  • <hr> Separator

    source:// or mermaid-cli

    <hr />
    

    display:


  • <blockquote> for quote

    source:

    <blockquote>
      We do not grow absolutely, chronologically. We grow sometimes in one
      dimension, and not in another, unevenly. We grow partially. We are relative.
      We are mature in one realm, childish in another. —Anais Nin
    </blockquote>
    

    display:

We do not grow absolutely, chronologically. We grow sometimes in one dimension, and not in another, unevenly. We grow partially. We are relative. We are mature in one realm, childish in another. —Anais Nin

HTML Formatting

  • <b> - Bold text
  • <strong> - Important text
  • <i> - Italic text
  • <em> - Emphasized text
  • <mark> - Marked text
  • <small> - Smaller text
  • <del> - Deleted text
  • <ins> - Here are Inserted text between normal text.
  • <sub> - Here isSubscript text
  • <sup> - Here isSuperscript text
  • <u> - Underline text

Text Size

We can also set text size with attribute style.

  1. normal mode.

    <p style="font-size:70px">Hello World</p>
    

    Hello World

  2. Responsive size.

    <h1 style="font-size:10vw">Hello World</h1>
    

    Hello World

    It’s responsive font size. The text size can be set with a vw unit, which means the “viewpoint width”. If we resize the browser, the font size scales to the same viewpoint width.

    Viewport is the browser window size. 1vw = 1% of viewport width. If the viewport is 50cm wide, 1vw is 0.5cm.

4. Math

With the support of MathJax 3 engine, out post can render beautiful math inline and display mode. We just need to surround math expressions with $$ , like $$ E = mc^2 $$. If we leave it inside a paragraph, it will produce an inline expression, just like \(E = mc^2\).

  • Display mode: place $$ as a separate paragraph
\[\sum_{k=1}^\infty |\langle x, e_k \rangle|^2 \leq \|x\|^2\]
  • MathJax mode: use \begin{equation}...\end{equation} ; add \label{...} inside the equation environment, we can now refer to the equation using \eqref.
\begin{equation}
\label{eq:cauchy-schwarz}
\left( \sum_{k=1}^n a_k b_k \right)^2 \leq \left( \sum_{k=1}^n a_k^2 \right) \left( \sum_{k=1}^n b_k^2 \right)
\end{equation}

\begin{equation} \label{eq:cauchy-schwarz} \left( \sum_{k=1}^n a_k b_k \right)^2 \leq \left( \sum_{k=1}^n a_k^2 \right) \left( \sum_{k=1}^n b_k^2 \right) \end{equation}

We need to refer to the equation above:

\begin{align}
x& = y_1-y_2+y_3-y_5+y_8-\dots
&& \text{by \eqref{eq:cauchy-schwarz}}\\
& = y'\circ y^* && \text{(by \eqref{eq:cauchy-schwarz})}\\
& = y(0) y' && \text {by Axiom 1.}
\end{align}

\begin{align} x& = y_1-y_2+y_3-y_5+y_8-\dots && \text{by \eqref{eq:cauchy-schwarz}}
& = y’\circ y^* && \text{(by \eqref{eq:cauchy-schwarz})}
& = y(0) y’ && \text {by Axiom 1.} \end{align}

Note that MathJax 3 is a major re-write of MathJax that brought a significant improvement to the loading and rendering speed, which is now on par with KaTeX.

5. Code highlight

We can use a liquid tag to highlight our code:


{% highlight c++ linenos %}
code code code
{% endhighlight %}


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
int main(int argc, char const \*argv[]) {
string myString;

    cout << "input a string: ";
    getline(cin, myString);
    int length = myString.length();
    
    char charArray = new char * [length];
    
    charArray = myString;
    for(int i = 0; i < length; ++i){
        cout << charArray[i] << " ";
    }
    
    return 0;

}

Rust code:

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
#![crate_name = "doc"]

/// A human being is represented here
pub struct Person {
/// A person must have a name, no matter how much Juliet may hate it
name: String,
}

impl Person {
/// Returns a person with the name given them
///
/// # Arguments
///
/// \* `name` - A string slice that holds the name of the person
///
/// # Examples
///
/// `` /// // You can have rust code between fences inside the comments /// // If you pass --test to `rustdoc`, it will even test it for you! /// use doc::Person; /// let person = Person::new("name"); ///``
pub fn new(name: &str) -> Person {
Person {
name: name.to_string(),
}
}

    /// Gives a friendly hello!
    ///
    /// Says "Hello, [name](Person::name)" to the `Person` it is called on.
    pub fn hello(& self) {
        println!("Hello, {}!", self.name);
    }

}

fn main() {
let john = Person::new("John");

    john.hello();

}

6. Images layout

We can use different layout to form a group of images, here are some exmaples.

:arrow_right_hook: Images in the same row with a caption.

<div class="row mt-3">
       <div class="col-sm mt-3 mt-md-0">
           {% include figure.html path="assets/img/9.jpg" class="img-fluid rounded z-depth-1" %}
       </div>
       <div class="col-sm mt-3 mt-md-0">
           {% include figure.html path="assets/img/7.jpg" class="img-fluid rounded z-depth-1" %}
       </div>
   </div>
   <div class="caption">
       A simple, elegant caption looks good between image rows, after each row, or doesn't have to be there at all.
</div>
A simple, elegant caption looks good between image rows, after each row, or doesn't have to be there at all.

:arrow_right_hook: Images can be zoomable. Simply add data-zoomable to <img> tags that you want to make zoomable.

<div class="row mt-3">
       <div class="col-sm mt-3 mt-md-0">
           {% include figure.html path="assets/img/8.jpg" class="img-fluid rounded z-depth-1" zoomable=true %}
       </div>
       <div class="col-sm mt-3 mt-md-0">
           {% include figure.html path="assets/img/10.jpg" class="img-fluid rounded z-depth-1" zoomable=true %}
       </div>
</div>

:arrow_right_hook: Images are arranged into different mini-galleries.

<div class="row mt-3">
       <div class="col-sm mt-3 mt-md-0">
           {% include figure.html path="assets/img/11.jpg" class="img-fluid rounded z-depth-1" zoomable=true %}
       </div>
       <div class="col-sm mt-3 mt-md-0">
           {% include figure.html path="assets/img/12.jpg" class="img-fluid rounded z-depth-1" zoomable=true %}
       </div>
       <div class="col-sm mt-3 mt-md-0">
           {% include figure.html path="assets/img/7.jpg" class="img-fluid rounded z-depth-1" zoomable=true %}
       </div>
</div>

7. Diagrams

We can generate great diagrams using jekyll-diagrams plugin. We can use different languages such as mermaid, plantuml, vega-lite, etc.

We prepared mermaid-cli before, so here are examples showing how to use mermaid.

Mermaid

Sequence Diagram

{% mermaid %}
sequenceDiagram
    participant John
    participant Alice
    Alice->>John: Hello John, how are you?
    John-->>Alice: Great!
{% endmermaid %}
AliceJohnAliceJohn Hello John, how are you?Great!

flowchart

flowchart TD
    A[Start] --> B{Is it?}
    B -- Yes --> C[OK]
    C --> D[Rethink]
    D --> B
    B -- No ----> E[End]

Yes

No

Start

Is it?

OK

Rethink

End

Class Diagrams

---
title: Animal example
---
classDiagram
    note "From Duck till Zebra"
    Animal <|-- Duck
    note for Duck "can fly\ncan swim\ncan dive\ncan help in debugging"
    Animal <|-- Fish
    Animal <|-- Zebra
    Animal : +int age
    Animal : +String gender
    Animal: +isMammal()
    Animal: +mate()
    class Duck{
        +String beakColor
        +swim()
        +quack()
    }
    class Fish{
        -int sizeInFeet
        -canEat()
    }
    class Zebra{
        +bool is_wild
        +run()
    }
Rendering Failed: mmdc --puppeteerConfigFile /home/runner/work/kobehub.github.io/kobehub.github.io/vendor/bundle/ruby/3.0.0/gems/jekyll-diagrams-0.10.0/vendor/mermaid_puppeteer_config.json --input /tmp/input20241213-2643-3352ei --output /tmp/output20241213-2643-ealrpq.svg: Error: Parse error on line 1: ---title: Animal ex ^ Expecting 'CLASS_DIAGRAM', 'acc_title', 'acc_descr', 'acc_descr_multiline_value', 'NAMESPACE', 'CLASS', 'ANNOTATION_START', 'MEMBER', 'SEPARATOR', 'NOTE_FOR', 'NOTE', 'CLASSDEF', 'ALPHA', 'direction_tb', 'direction_bt', 'direction_rl', 'direction_lr', 'CALLBACK', 'LINK', 'CLICK', 'STYLE', 'CSSCLASS', 'NUM', 'MINUS', 'UNICODE_TEXT', 'BQUOTE_STR', got 'LINE' Parser3.parseError (/usr/local/lib/node_modules/@mermaid-js/mermaid-cli/node_modules/mermaid/dist/mermaid.js:90350:28) at #evaluate (file:///usr/local/lib/node_modules/@mermaid-js/mermaid-cli/node_modules/puppeteer-core/lib/esm/puppeteer/cdp/ExecutionContext.js:388:19) at async ExecutionContext.evaluate (file:///usr/local/lib/node_modules/@mermaid-js/mermaid-cli/node_modules/puppeteer-core/lib/esm/puppeteer/cdp/ExecutionContext.js:275:16) at async IsolatedWorld.evaluate (file:///usr/local/lib/node_modules/@mermaid-js/mermaid-cli/node_modules/puppeteer-core/lib/esm/puppeteer/cdp/IsolatedWorld.js:97:16) at async CdpJSHandle.evaluate (file:///usr/local/lib/node_modules/@mermaid-js/mermaid-cli/node_modules/puppeteer-core/lib/esm/puppeteer/api/JSHandle.js:146:20) at async CdpElementHandle.evaluate (file:///usr/local/lib/node_modules/@mermaid-js/mermaid-cli/node_modules/puppeteer-core/lib/esm/puppeteer/api/ElementHandle.js:340:20) at async CdpElementHandle.$eval (file:///usr/local/lib/node_modules/@mermaid-js/mermaid-cli/node_modules/puppeteer-core/lib/esm/puppeteer/api/ElementHandle.js:494:24) at async CdpFrame.$eval (file:///usr/local/lib/node_modules/@mermaid-js/mermaid-cli/node_modules/puppeteer-core/lib/esm/puppeteer/api/Frame.js:450:20) at async CdpPage.$eval (file:///usr/local/lib/node_modules/@mermaid-js/mermaid-cli/node_modules/puppeteer-core/lib/esm/puppeteer/api/Page.js:450:20) at async renderMermaid (file:///usr/local/lib/node_modules/@mermaid-js/mermaid-cli/src/index.js:252:22) at fromText (/usr/local/lib/node_modules/@mermaid-js/mermaid-cli/node_modules/mermaid/dist/mermaid.js:146633:21)