快速搭建你的 github pages 个人博客 —— 基于 Create-React-App 的单页面应用实践

相信各位github资深玩家们都有自己基于 github pages 搭建的个人站点。官方推荐的静态站点生成器是 Jekyll,关于 Jekyll 的使用感兴趣的各位请自行 google,这里就不赘述了。本文主要介绍下基于 Create-React-App 搭建个人博客的相关实践,可能更适合做前端开发的伙伴。

github pages

github pagesgithub 推出的静态站点服务,主要的用途在于使用你在 github 仓库中的代码构建你自己的静态站点,为用户提供 github.io 二级域名,您也可以通过添加DNS的 CNAME 记录来绑定自己的域名。

github pages 最简单粗暴的方法就是直接往 github 上方静态页面了,创建一个名为 [您的github账号名].github.io 的github仓库,将您的index.html页面代码扔进master分支,就可以直接通过 https://[您的github账号名].github.io 访问到您的站点了。

对于一个简单的个人博客站点来说,存在以下基本功能特性:

  • 文章的新增、编辑、一键发布
  • 文章的分类、归档
  • 风格良好的博客样式
  • 评论、SEO等等功能

下面介绍基于React如何实现一个简单的静态博客。

1. 创建一个 React 项目

使用 Create-React-App(以下简称CRA) 的generator创建一个React前端项目骨架。对此项目进行一定改造以方便我们日常的开发和使用习惯:

  • 使用react-app-rewired来调整CRA中webpack的配置
    • 对CRA的webpack配置感兴趣的童鞋可以看看这篇文章
  • 使用core-js对浏览器版本进行向下兼容
  • 通过编写不同的React容器组件(container)来实现不同的页面,通过统一的json结构来配置应用的页面路由
  • 使用蚂蚁金服的antd设计语言(React组件)快速实现业务UI
  • 使用axios实现前后端的数据请求

个人改造后的项目代码在这里,您可以直接fork或者down下来使用。

2. 使用 markdown 搞定你的文章

2.1 用于新建文章的交互式命令行(基于 inquirer)

一般的静态博客系统(如gatsby),会给用户提供一个用于创建新文章的交互式命令行,效果大致如下:

readline

类似功能可以使用nodejs中readline模块的原生方法来实现。这里推荐一个第三方工具:inquirer,本质上是对readline模块进行了增强,提供了很多实用的方法用于交互式命令行开发,实现的用户界面(命令行)也比较友好。

对于上面GIF示例的功能,其代码实现如下:

// newPost.js

const inquirer = require('inquirer');
const moment = require('moment');

const questions = [
  {
    type: 'input',
    name: 'post_name',
    message: '请输入您的文章别名(用于创建文章目录,仅限英文,单词间用短横杠‘-’连接):',
    validate: value => {
      if (/(\.|\*|\?|\\|\/)/gi.test(value)) {
        return '文章别名不得包含特殊符号(.*?\\/),请重新输入↑↑';
      }

      if (/(([A-z]+-)+)?[A-z]+/gi.test(value)) {
        return true;
      }

      return '文章别名不合法,请重新输入↑↑';
    },
    filter: value => value.replace(/\s+/gi, '-'),
  },
  {
    type: 'input',
    name: 'create_at',
    message: '请输入文章的发布时间(或者按回车键使用默认值):',
    default: () => {
      return moment().format('YYYY-MM-DDThh:mm:ss');
    },
    validate: value => {
      if (/\d{4}-\d\d-\d\dT\d\d:\d\d:\d\d/gi.test(value)) {
        return true;
      }

      return '时间格式不合法,请重新输入↑↑';
    },
  },
];

inquirer
  .prompt(questions)
  .then(answers => {
    // 获取用户输入
    const { post_name, create_at } = answers;
  
    /* 此处做一些命令行反馈和过程性的工作 */
    /* (如:提示用户输入是否合法、创建文章对应的目录和文件等等) */
  })
  .catch(err => {
    /* 异常处理 */
  });

如是,将此node脚本添加到项目package.jsonscripts中(如:new-post: "node newPost.js"),即可通过npm run命令执行。

2.2 md 转 html(基于 react-markdown)

为使用markdown文档来编辑、存储博客的文章内容,需要将md文档转换为react的JSX对象以渲染到网页中。在此推荐使用react-markdown,功能很6,作者维护得也比较勤。

使用方式如下:

import ReactMarkdown from 'react-markdown';

<ReactMarkdown source={'# 这是文章标题\n\n'} />
// <h1>这是文章标题</h1>

2.3 代码块的语法高亮

react-markdown提供了一个renderers属性,用户可以传入一系列renderer组件来自定义文章中一些内容的渲染方式(有兴趣的童鞋可以看下包作者对默认renderer的实现)。

如:自定义md中图片的渲染方式(用法如下)。

// 传入renderer的方式
<ReactMarkdown
  source={'[md文本内容]'}
  renderers={{
    image: ImageRenderer,
  }}
/>
// ImageRenderer的实现

import React, { Component } from 'react';
import PropTypes from 'prop-types';

class ImageRenderer extends Component {
  static propTypes = {
    src: PropTypes.string.isRequired,
  };

  render() {
    return (
      <img
        className="post-content-image"
        src={this.props.src}
        alt={this.props.src}
      />
    );
  }
}

export default ImageRenderer;

与此类似,我们可以通过传入一个自定义的renderer来实现文章中代码块的语法高亮。名为CodeBlock的renderer实现如下:

import React, { Component } from 'react';
import PropTypes from 'prop-types';
import { highlight, languages } from 'prismjs';
import ReactHtmlParser from 'react-html-parser';
import 'prismjs/themes/prism.css';

export class HtmlComponent extends Component {
  static propTypes = {
    html: PropTypes.string.isRequired,
  };

  render() {
    return ReactHtmlParser(this.props.html);
  }
}

export class CodeBlock extends Component {
  static propTypes = {
    literal: PropTypes.string.isRequired,
    language: PropTypes.string.isRequired,
  };

  render() {
    const html = highlight(this.props.literal, languages[this.props.language]);
    const cls = `language-${this.props.language}`;

    return (
      <pre className={cls}>
        <code className={cls}>
          <HtmlComponent html={html} />
        </code>
      </pre>
    );
  }
}

export default CodeBlock;

此处用到了prismjsreact-html-parser两个npm包,前者用于将代码文本转化为html文本,后者用于将html文本转化为React的JSX对象以传入React组件(这样做比直接使用dangerouslySetInnerHTML属性更安全些)。

3. 文章分类

一个友好的站点肯定少不了导航菜单(或文章的分类菜单),本人的实现方式是直接使用文章的“标签”来进行分类统计,并生成站点的顶部导航,效果如下:

nav-top

为此,需要撰写一定的脚本实现文章的分类统计和打包,个人的实现方式是将统计结果和文章内容各自打包为json文件,通过前端组件请求数据并加载。

导航栏组件的具体实现如下:

import React, { Component } from 'react';
import PropTypes from 'prop-types';
import { Link } from 'react-router-dom';
import { Dropdown, Menu, Icon } from 'antd';
import { randomId } from 'utils';
import './style.css';

export class Header extends Component {
  static propTypes = {
    data: PropTypes.array,
    activeTag: PropTypes.string,
  };

  static defaultProps = {
    data: [{ tag: '前端', count: 5 }],
    activeTag: '',
  };

  constructor(props) {
    super(props);
    this.navTotal = 6;
  }

  renderMore() {
    if (this.props.data.length <= this.navTotal) {
      return false;
    }

    const subNavItems = this.props.data.slice(this.navTotal).map(t =>
      <Menu.Item key={`sub_nav_${randomId()}`}>
        <Link
          to={t.linkTo || `/tag/${t.tag}`}
          className={`ant-dropdown-link ${this.props.activeTag === t.tag
            ? 'active'
            : ''}`}
          key={`nav_top_${randomId()}`}>
          {t.tag}({t.count})
        </Link>
      </Menu.Item>
    );

    const SubNav = (
      <Menu>
        {subNavItems}
      </Menu>
    );

    const DropDownBtn = (
      <Dropdown overlay={SubNav} key={`nav_top_${randomId()}`}>
        <div className="header-nav-item">
          更多分类 <Icon type="down" />
        </div>
      </Dropdown>
    );

    return DropDownBtn;
  }

  renderTop5() {
    const items = this.props.data.slice(0, this.navTotal - 1).map(t =>
      <Link
        className={`header-nav-item ${this.props.activeTag === t.tag
          ? 'active'
          : ''}`}
        to={t.linkTo || `/tag/${t.tag}`}
        key={`nav_top_${randomId()}`}>
        {!t.linkTo ? `${t.tag}(${t.count})` : t.tag}
      </Link>
    );

    return (
      <div className="header-nav">
        {items}
        {this.renderMore()}
      </div>
    );
  }

  render = () => this.renderTop5();
}

export default Header;

大家可以根据实际需要实现自己的文章打包方式(这里就不奉上我的脚本了😝)。

4. 更多功能

对于个人博客来说,到这里为止还有很多功能没有实现,这里偷个懒,奉上一些相关的链接吧:

4.1 关于文章评论

4.2 关于文章结构树

我最近应该会实现一个React用途的markdown树组件,大家不妨期待下☺️

5. 发布你的个人静态站点

5.1 部署到 github pages(基于 gh-pages)

CRA针对github pages用途专门推荐了一个包:gh-pages,使用方法如下:

(1)修改项目的package.json文件,添加homepage属性:

"homepage": "https://parksben.github.io",

(2)项目安装gh-pages依赖后修改,在package.json中添加如下配置:

  "scripts": {
+   "predeploy": "npm run build",
+   "deploy": "gh-pages -d build",
    "start": "react-scripts start",
    "build": "react-scripts build",

(3)将本地代码上传到github博客仓库的某个分支(只要不是master分支就行),然后执行:

yarn deploy

gh-pages会将CRA项目build到仓库的master分支,然后,你就可以访问你的站点了(有关 CRA 项目部署到 github pages 的详细描述可以看这里)。

5.2 如何兼容 React 的客户端路由(一种比较 hack 的方法)

单页面应用一般需要设置服务端路由,将应用的所有页面路径都重定向到index.html,而github pages并没有这样的默认设置。

因而,当你使用React的客户端路由(React的createBrowserHistory方法创建前端路由)时,除根路径以外的页面,github都会返回自己的404页面。

为此,CRA项目提供了一种比较hack的方法来支持React的客户端路由(通过操作window.history来强行匹配url)。也算是一种奇技淫巧吧☺️。

(1)在CRA项目的public目录下添加一个404.html,其内容如下:

<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <title>矮大紧的日常 | parksben's blog</title>
    <script type="text/javascript">
      var segmentCount = 0;
      var l = window.location;
      l.replace(
        l.protocol + '//' + l.hostname + (l.port ? ':' + l.port : '') +
        l.pathname.split('/').slice(0, 1 + segmentCount).join('/') + '/?p=/' +
        l.pathname.slice(1).split('/').slice(segmentCount).join('/').replace(/&/g, '~and~') +
        (l.search ? '&q=' + l.search.slice(1).replace(/&/g, '~and~') : '') +
        l.hash
      );
    </script>
  </head>
  <body>
  </body>
</html>

(2)在index.html的head中添加如下代码:

<script type="text/javascript">
  (function(l) {
    if (l.search) {
      var q = {};
      l.search.slice(1).split('&').forEach(function(v) {
        var a = v.split('=');
        q[a[0]] = a.slice(1).join('=').replace(/~and~/g, '&');
      });
      if (q.p !== undefined) {
        window.history.replaceState(null, null,
          l.pathname.slice(0, -1) + (q.p || '') +
          (q.q ? ('?' + q.q) : '') +
          l.hash
        );
      }
    }
  }(window.location))
</script>

大功告成,你的github站点支持React的客户端路由了。

除此之外,也可以改为使用createHashHistory方法来创建客户端路由,这样前端路由就与服务端路由没多大关系了,不过url里面一串hash毕竟不够优雅。

有兴趣了解奇技淫巧的童鞋,可以点这里

5.3 部署到自己的服务

与CRA项目的生产环境部署方式一样:

  • 线上执行 yarn build 命令,站点的所有静态资源将打包到 build 目录下
  • 将你的站点的入口配置到 build 目录下

6. 项目参考(源码奉上)

parksben.github.io

这是我的github博客(基于上述过程实现的静态站点),感兴趣的伙伴可以点击这里查看项目源码,觉得有用也可以fork或star一下下。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 170,569评论 25 707
  • Swift版本点击这里欢迎加入QQ群交流: 594119878最新更新日期:18-09-17 About A cu...
    ylgwhyh阅读 24,861评论 7 249
  • 同样是做事,有人顺风顺水,有人效率低下。 例如同在销售部门,有人每月收入颇丰,有人仅够温饱,因为对时间的态度不...
    斐丽希娅阅读 131评论 0 0
  • 北京时间2017年3月22日武汉江夏公司李倩 导语:从昨天一天补听鲲鹏大哥的分享,以及今日听了舒月姐的分享,总有一...
    叁栎阅读 166评论 0 0
  • “举杯邀明月,对影成三人。”明月空照,影徒随身,你还是孤独的。也就是这孤独成就了你精彩的一生。 “仰...
    幸存者XCZ阅读 145评论 0 0