Build­ing an in­ter­ac­tive per­son­al blog with Re­act and Gats­by JS

I've al­ways want­ed a per­son­al blog. A small piece of the in­ter­net that I could call home. I also re­al­ly want­ed a blog that I cre­at­ed from scratch. While I could have very eas­i­ly turned to tools like Square­space or Medi­um, the idea of own­ing my own web­site that I built from scratch was some­thing that was very ap­peal­ing to me. I want­ed to learn how to build web­sites, and the tools need­ed to build them.

Un­like ex­ist­ing blogs, I also want­ed to build a blog that could be in­ter­ac­tive. I be­lieve that blogs could have an in­cred­i­ble amount of po­ten­tial as an in­ter­ac­tive form of read­ing. As a sim­ple ex­am­ple, you could press a but­ton in or­der to change its in­ter­nal state (some­thing like this but­ton here ). A more com­pli­cat­ed ex­am­ple might in­volve a read­er in­ter­act­ing with vi­su­al­iza­tions, code ex­am­ples, graphs, or oth­er ob­jects. Many ex­ist­ing blog­ging tools just did­n't have this op­tion (or if they did, I could­n't find it), so I de­cid­ed to build my own.

It seems fit­ting then that my first blog ar­ti­cle should be about its own cre­ation. In this ar­ti­cle, I dis­cuss how I built this in­ter­ac­tive per­son­al blog us­ing the Re­act and Gats­by JS frame­works. I'll go from the de­sign process, to its im­ple­men­ta­tion, to how I host­ed it on GitHub Pages. I've also open sourced this blog un­der the MIT Li­cense, and it is avail­able at GitHub for any­one to use or learn from.

Re­quire­ments

Off the bat, I knew I want­ed two things from my blog.

  1. The abil­i­ty to write in­ter­ac­tive con­tent. Vi­su­al­iza­tions, graphs, and cus­tom cal­cu­la­tors for ex­am­ple.
  2. For peo­ple to reach my pages via search en­gines.

I also want­ed to use Re­act for my in­ter­ac­tive con­tent.

Gats­by helps solve all of the above re­quire­ments. It gen­er­ates sta­t­ic pages that are then "hy­drat­ed" with Re­act code on the client side. Hav­ing sta­t­ic, pre-ren­dered pages is im­por­tant for search en­gine web-crawlers, since they can't nor­mal­ly read the JS code in the oth­er­wise emp­ty sin­gle page app that Re­act would gen­er­ate.

I, of­course, did not start with Gats­by. I de­cid­ed to make my life dif­fi­cult by think­ing I could cre­ate my own so­lu­tion to my prob­lems. I want­ed to use my own cus­tom web­pack, ba­bel, and re­act con­fig­u­ra­tions, de­spite know­ing of Gats­by's ex­is­tence. I strug­gled with this for weeks be­fore swal­low­ing my pride. Do not be like me. Do not make your life dif­fi­cult. Just use Gats­by.

De­sign

De­sign is an in­cred­i­bly im­por­tant and of­ten over­looked step of cre­at­ing a web­site, par­tic­u­lar­ly for be­gin­ner web devs or peo­ple (like me) who like cod­ing more than looks. If you are cre­at­ing a blog from scratch, you need to start with de­sign. I use Fig­ma to de­sign my web­sites. It's a fair­ly ubiq­ui­tous in­dus­try tool that has the dis­tinct ad­van­tage of be­ing free (as com­pared to some­thing like Adobe XD).

To be­gin, I col­lect­ed ref­er­ences of blogs that would ap­peal to my tar­get au­di­ence. Since my tar­get au­di­ence would be peo­ple like me, I col­lect­ed ref­er­ences from the blogs I like. Once I had my ref­er­ence ma­te­r­i­al, I de­cid­ed what el­e­ments I liked and dis­liked. I would in­clude the el­e­ments I like, and cut what I dis­like.

I used Red Blob Games and Medi­um as my two main sources of ref­er­ences. The for­mer as an ex­am­ple of a per­son­al blog that I re­al­ly like (that also had in­ter­ac­tive con­tent). The lat­ter I used for some­thing with more broad ap­peal.

My ini­tial de­sign for this blog can be seen here. There are a few dif­fer­ences here and there be­tween the ini­tial de­sign and the fi­nal prod­uct you're now read­ing. I opt­ed for a small­er foot­er bar with­out a nav menu and slight­ly less width for the page con­tents, but the over­all wire­frame re­mained in­tact.

Dur­ing this process, I end­ed up ask­ing my­self way more ques­tions about text dis­play than I thought I would ini­tial­ly. What size of text is ide­al? How should it re­ac­tive­ly scale ac­cord­ing to the size of the web brows­er? What's the ide­al num­ber of char­ac­ters per line (ap­par­ent­ly, around 65 char­ac­ters). I used my browser's dev tools on my ref­er­ence blogs in or­der to help an­swer these ques­tion. I would­n't have made any progress had I not had those ref­er­ences. Long sto­ry short, use ref­er­ences!

Im­ple­men­ta­tion

I'll cov­er the tech­ni­cal im­plemeta­tion of the blog here. I'll only be cov­er­ing the "in­ter­ac­tive blog" bit, and won't be cov­er­ing like the styling, css, SEO op­ti­miza­tion, etc.

I did­n't start a Gats­by project the nor­mal way ( i.e. us­ing npm init gats­by). In­stead, I start­ed a Gats­by project from scratch, us­ing:

npm i gats­by re­act

In or­der for gats­by to ac­tu­al­ly build web­sites from my code, I need­ed a par­tic­u­lar fold­er struc­ture. I end­ed up with this:

ja­son­rob­web­ster.github.io/
  ├── src/
  |   ├── com­po­nents/
  |   ├── pages/
  |   |   ├── blog/
  |   |   |   ├── in­dex.jsx
  |   |   |   ├── blog-ar­ti­cle-1.jsx
  |   |   |   ├── blog-ar­ti­cle-2.jsx
  |   |   |   ├── ...
  |   |   ├── con­tact.jsx
  |   |   ├── in­dex.jsx
  |   |   ├── tools.jsx
  |   ├── shared
  |   |   ├── styles/
  |   |   ├── de­vices.js
  |   |   ├── theme.js
  ├── .es­lin­trc
  ├── .git
  ├── .git­ig­nore
  ├── .pret­tierig­nore
  ├── .pret­ter­rc.js
  ├── gats­by-brows­er.js
  ├── gats­by-con­fig.js
  ├── gats­by-ssr.js
  ├── LI­CENSE
  ├── pack­age-lock.json
  ├── pack­age.json
  └── root-wrap­per.js

For those new to Gats­by, the src/pages/ di­rec­to­ry is most im­por­tant. Re­act com­po­nents writ­ten there will be au­to­mat­i­cal­ly com­piled into their own html pages that are then hy­drat­ed on the client side. In oth­er words, I'd be able to write Re­act code in a file named src/pages/page-name.jsx and have it ren­dered at the /page-name/ end­point.

The key idea to cre­ate my in­ter­ac­tive blog posts was to write blogs in Re­act un­der the src/pages/blog/ di­rec­to­ry. I'd then use the /blog/ end­point (gen­er­at­ed by the src/pages/blog/in­dex.jsx file) to route users to the posts. This is­n't the most ide­al so­lu­tion: I'll have to write my blogs in code (in­stead of some­thing like Mark­down or an in brows­er ed­i­tor) and will have to push my changes to GitHub every­time I write a new ar­ti­cle. But, im­por­tant­ly, the so­lu­tion works, and you would­n't be read­ing this blog with­out this so­lu­tion.

An ex­am­ple of an in­ter­ac­tive blog post would be some­thing that uses Re­ac­t's us­eS­tate hook:

// src/pages/ex­am­ple-blog.jsx

im­port Re­act, { us­eS­tate } from "re­act"

im­port { Ar­ti­cleWrap­per } from "../../com­po­nents"

con­st ex­am­ple­Blog = () => {
  con­st [val­ue, set­Val­ue] = us­eS­tate(0)

  con­st in­crea­se­Val­ue = () => {
    set­Val­ue(val­ue + 1)
  }

  re­turn (
    <Ar­ti­cleWrap­per las­tUp­dat­ed={new Date(2021, 8, 7)}>
      <but­ton onClick={in­crea­se­Val­ue}>{`Val­ue: ${val­ue}`}</but­ton>
    </Ar­ti­cleWrap­per>
  )
}

ex­port de­fault ex­am­ple­Blog

That's it for how I wrote the in­ter­ac­tive blog posts. To ac­tu­al­ly con­nect the blog posts to my blog root end­point, I set up a "data­s­tore" to hold the lo­ca­tion and meta­da­ta for each of my blogs. The slug stored the end­point, which as dis­cussed above was ba­si­cal­ly the name of the file that con­tained my page. I also in­clud­ed a pub­lished tag that would hide it from the root page, in case I ever want­ed to do that. The "blog data­s­tore" end­ed up just be­ing a vari­able that looked like:

// src/pages/blog/in­dex.jsx

// blog card data­s­tore
con­st blogs = [
  {
    ti­tle: "Ex­am­ple blog ti­tle",
    de­scrip­tion: "Ex­am­ple blog de­scrip­tion",
    tag: "Tech",
    slug: "ex­am­ple-blog", // for src/pages/blog/ex­am­ple-blog.jsx
    im­age: "im­ages/blog.png",
    pub­lished: false,
    las­tUp­dat­ed: new Date(2021, 8, 3),
  },
]

I'd then be able to use this to gen­er­ate each link to my blog, us­ing a cus­tom built <Blog­Card /> com­po­nent (in­spired di­rect­ly by my Red Blob Games ref­er­ence, and my love of cards in gen­er­al):

// src/pages/blog/in­dex.jsx

im­port Re­act from "re­act"

im­port fp from "lo­dash/fp"

im­port { Blog­Card, PageWrap­per } from "../../com­po­nents"

con­st Blog = () => {
  con­st pub­lished­Blogs = fp.fil­ter((blog) =>
    is­Pro­duc­tion ? blog.pub­lished : true
  )(blogs)
  con­st last­Blog = fp.max­By((blog) => blog.las­tUp­dat­ed)(blogs)

  re­turn (
    <Re­act.Frag­men­t>
      <PageWrap­per las­tUp­dat­ed={last­Blog.las­tUp­dat­ed}>
        <Card­Con­tent>
          {fp.flow(
            fp.sort­By((blog) => -(blog.las­tUp­dat­ed || new Date())),
            fp.map((blog) => (
              <Blog­Card
                key={blog.slug}
                url={blog.slug}
                ti­tle={blog.ti­tle}
                im­ageLink={blog.im­age}
                de­scrip­tion={blog.de­scrip­tion}
                tag={blog.tag}
              />
            ))
          )(pub­lished­Blogs)}
        </Card­Con­tent>
      </PageWrap­per>
    </Re­act.Frag­men­t>
  )
}

ex­port de­fault Blog
// src/com­po­nents/Blog­Card.jsx

im­port Re­act from "re­act"

im­port { Link } from "gats­by"

im­port { Hy­phen­at­ed­Text } from "./text"

con­st Blog­Card = ({ url, ti­tle, im­ageLink, de­scrip­tion, tag }) => {
  re­turn (
    <Re­act.Frag­men­t>
      <Card to={url}>
        <Card.Con­tent>
          <Card.Im­age src={im­ageLink} />
          <Card.Tex­t>
            <Card.Ti­tle>
              <Hy­phen­at­ed­Tex­t>{ti­tle}</Hy­phen­at­ed­Tex­t>
            </Card.Ti­tle>
            <Card.De­scrip­tion>
              <Hy­phen­at­ed­Tex­t>{de­scrip­tion}</Hy­phen­at­ed­Tex­t>
            </Card.De­scrip­tion>
            <Card.Tag>{tag}</Card.Tag>
          </Card.Tex­t>
        </Card.Con­tent>
      </Card>
    </Re­act.Frag­men­t>
  )
}

ex­port de­fault Blog­Card

When all com­bined (with a lit­tle css styling), I'd have a nice set of cards at the /blog/ end­point, each link­ing to my pub­lished in­ter­ac­tive blog posts:

And... that's about it for the "in­ter­ac­tive blog" part of the im­ple­men­ta­tion. The rest of the im­ple­men­ta­tion was ba­si­cal­ly get­ting the styles and UI/UX to match my ini­tial de­sign, which I won't cov­er here.

De­ploy­ing to GitHub Pages

GitHub pages is a free way to host a per­son­al blog. In or­der to set this up, you need to have a repo named <user­name>.github.io. In my case, that's ja­son­rob­web­ster.github.io. Then, in the re­po's Set­tings > Pages, you need to spec­i­fy the branch that the your site will be built from. Set it to the gh-pages branch. If you don't have this branch yet, cre­ate it.

We'll de­ploy our site us­ing the gh-pages pack­age. This will send a lo­cal di­rec­to­ry to our re­po's gh-pages branch. Since we've set up our repo with the gh-pages branch, this should work off the bat, but we'll re­quire a small con­fig­u­ra­tion line to make sure we up­load the pub­lic/ di­rec­to­ry.

First, we'll in­stall the gh-pages pack­age

npm i gh-pages

Be­fore we de­ploy our blog, we need to build our pages from our source code. In our pack­age.json, set up the fol­low­ing scripts. Note that we are up­load­ing the pub­lic/ di­rec­to­ry to the gh-pages branch us­ing the gh-pages -d pub­­lic script.

// pack­age.json
"scripts": {
  "dev": "gats­by de­vel­op",
  "build": "gats­by build",
  "pre­de­ploy": "npm run build",
  "de­ploy": "gh-pages -d pub­lic",
  "serve": "gats­by serve",
  "clean": "gats­by clean"
}

Then run

npm run build

Our blog will be built in the pub­lic/ di­rec­to­ry. To de­ploy our built web­site, we run

npm run de­ploy

Go­ing to <user­name>.github.io, you should see your built web­site, ready for any­one to use. And be­cause we're us­ing Gats­by, we have pre­ren­dered html pages for search en­gines to crawl (we're still not op­ti­mised for SEO though, to do that we can fol­low this guide).

Con­clu­sion

We've man­aged to de­sign, im­ple­ment, and de­ploy an in­ter­ac­tive per­son­al blog us­ing the Re­act and Gats­by frame­work. We can use it to write in­ter­ac­tive blog posts in Re­act code, and gen­er­ate an SEO op­ti­mized page that we host on GitHub. For any more de­tails, feel free to browse this blog's source code on GitHub.

Copyright © Jason Webster 2021
Last updated: 7 September 2021