blog image

2018-06-23

使用Nuxt.js及Nuxtent建立Blog的心得

Nuxt.js 是內建 SSR 的 vue.js 框架,我最感興趣的部份是可輸出靜態的 Html,如此就可以輕鬆的將程式部署至 GitHub Page 上,並且可以使用 module 來擴充功能。

當我看到了 Nuxtent 這個 module,它是可以取得 Markdown 內容的 module,所以想將 Blog 用 Nuxt.js 重新翻寫。

Nuxtent Template

第一步快速建立的話,可使用Nuxtent Template建立一個基本的網站架構,需要事先安裝vue-cli

vue init nuxt-community/nuxtent-template my-site

網站架構

page

建立網站的路由,例如在目錄中建立 about.vue,網址會成為 localhost:3000/about。

static

靜態資源,例如:圖片,路徑會基於根目錄,例如/images/bg.png

content

Nuxtent module 特定的目錄,放置 Markdown 檔案的地方,也就是放入文章的地方

layout

預設 layout 為 layout.vue,也可以自定 layout,之後使用 layout 屬性指定自定的 layout

export default {
  async asyncData({ app }) {
    return {
      posts: await app.$content('/').getAll(),
      name: 'Thomas Blog',
    };
  },
  layout: 'customLayout',
};

設定檔

nuxtent.config.js

const Prism = require('prismjs');

module.exports = {
  content: {
    permalink: '/:slug',
    page: '/_content',
    generate: [
      // for static build
      'get',
      'getAll',
    ],
    isPost: true,
  },
  api: {
    baseURL: 'http://localhost:3200',
  },
  parsers: {
    md: {
      extend(config) {
        config.highlight = (code, lang) => {
          return `<pre class="language-${lang}"><code class="language-${lang}">${Prism.highlight(
            code,
            Prism.languages[lang] || Prism.languages.markup
          )}</code></pre>`;
        };
      },
    },
  },
};

content

  • permalink:設定文章路徑的顯示方式。

  • page:指定顯示文章內容的檔案名稱,預設為_cotent,即指向_cotent.vue。按照文件可以設定為多個。

  • isPost::查看官方文件的說明不是很了解這個屬性的用法,但當看了原始碼後發現

    get date() {
      if (isDev || !cached.date) {
        const { filePath, fileName, section } = meta
        if (options.isPost) {
          const fileDate = fileName.match(/!?(\d{4}-\d{2}-\d{2})/) // YYYY-MM-DD
          if (!fileDate) {
            throw Error(`Post in "${section}" does not have a date!`)
          }
          cached.date = fileDate[0]
        } else {
          const stats = statSync(filePath)
          cached.date = dateFns.format(stats.ctime, 'YYYY-MM-DD')
        }
      }
      return cached.date
    }

當 true 時,需要將 markdown 檔案名稱設成 2017-06-20-HelloWorld.md,即會取得檔名上的日期做為文章日期。

當 false,會使用檔案修改日期做為文章日期。

parser

我是參考官方文件,使用 prismjs 將 Mardown 的程式碼加上著色器功能。

取得內容

可以參考_content.vue 的程式碼,使用get取得目前路徑的內容。

post: await app.$content('/').get(route.path);

如果是要取得所有文章的話,可以使用getAll這個方法。

posts: await app.$content('/').getAll();

評論功能

想使用基於 GitHub issues 的評論功能,有發現gitment這個套件剛好符合我的需要,安裝及設定都很簡單。

<div class="row">
  <div class="col-lg-8 col-lg-offset-2 col-md-10 col-md-offset-1">
    <div id="comments"></div>
  </div>
</div>
import PostHeader from '../components/PostHeader';
import Gitment from 'gitment';

export default {
  async asyncData({ app, route, payload }) {
    return {
      post: await app.$content('/').get(route.path),
    };
  },
  mounted() {
    const gitment = new Gitment({
      id: this.post.title, // optional
      owner: 'thomascsd',
      repo: 'thomascsd.github.io',
      oauth: {
        client_id: 'client_id',
        client_secret: 'client_secret',
      },
    });

    gitment.render(document.getElementById('comments'));
  },
  components: {
    PostHeader,
  },
};

我是參考這篇文章

要注意的點是,第一次會顯示Error:Comments Not Initialized

需要登入自己的 Github 帳號後,啟動應用程式。

成功之後,就會變成下圖

部署

將靜態檔案部署至 Github pages,我遇到了一些問題,按照範例設定會將 NODE_ENV='production'及 baseUrl 會成為正式部署的路徑,例如我的 Blog 網址 thomascsd.github.io。

當使用 npm generate 時,卻會出現下列錯誤

最後發現,使用下列步驟就 OK 了

  • 不需要加上 NODE_ENV
  • baseUrl 設定成預設的 localhost:3200,並執行 npm generate
  • 再使用 VSCode 搜索 localhost:3200,即會發現 dist/api.js 內容有 baseURL:localhost:3200,然後將網址替換成正式環境,如我的 Blog 網址 thomascsd.github.io
  • 最後就可以部署至 GitHub page

結論

很方便將一個 Blog 建立起來,但是缺乏一些功能,例如標籤、Archive 功能,這些都要自己實作出來。原始碼可以參考這裡