Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/lorenseanstewart/nextjs-mdx-blog-kit

Starter code for creating a static blog system using Next.js and MDX. A service worker for offline reading is set up, and the blog has Perfect Lighthouse scores.
https://github.com/lorenseanstewart/nextjs-mdx-blog-kit

Last synced: about 2 months ago
JSON representation

Starter code for creating a static blog system using Next.js and MDX. A service worker for offline reading is set up, and the blog has Perfect Lighthouse scores.

Awesome Lists containing this project

README

        

[![code style: prettier](https://img.shields.io/badge/code_style-prettier-ff69b4.svg?style=flat-square)](https://github.com/prettier/prettier)

# Next.js MDX Blog Kit

This kit is designed to give you a fully functional, easy to use (if you are a developer) blog with perfect Lighthouse scores.

![Lighthouse audit with 100% scores.](./static/images/lighthouse.png "Lighthouse Scores")

The idea is to provide a complete blogging system that requires only two things from the developer:

1. **write** some posts; share you knowledge!
2. **style** the blog to make it your own

And, of course, this repo could merely be a starting point for your hacking. Hack away and have fun.

To clone this repo: `git clone --depth=1 https://github.com/lorenseanstewart/nextjs-mdx-blog-kit `

#### Feature list:

- Static website deployment
- Write your posts in markdown, and use React components in your markdown thanks to [MDX](https://github.com/mdx-js/mdx). Or use plain React, if you prefer.
- A CLI for creating new pages or posts. It processes all the necessary meta data for creating the blog list and adding SEO data to all pages.
- Perfect Lighthouse scores.
- Excellent SEO
- Service Worker that caches all pages and posts for offline reading
- Tag search
- A [reading progress indicator](https://github.com/jeremenichelli/scrollProgress) on posts (optional/configurable)
- [Code syntax highlighting](https://github.com/conorhastings/react-syntax-highlighter)
- [Smooth scroll links](https://github.com/mauricevancooten/react-anchor-link-smooth-scroll)
- [Optimized images](https://github.com/cyrilwanner/next-optimized-images)
- Tests are set up using [Jest](https://github.com/facebook/jest)
- Easy deploys with Zeit's [Now](https://zeit.co/now)

## CLI

This blog processes pages based on the meta data found in the page and post component files. To make sure you include all necessary meta data, and that page/post components conform to the requirements of the BlogEngine and its utility functions, **it is important to use the CLI to scaffold all new pages and posts**.

There are two ways to use the CLI:

1. Install the CLI so you can use the `blog` keyword.
- This allows you to use commands like `blog -t post -m`
2. Use the file as a script.
- This allows you to use commands like `./cli.js -t post -m`

#### 1. Install: CLI with keyword

At the root of the project, in your terminal run the command `npm link` or `sudo npm link`. This will allow you to use the `blog` keyword when using the CLI.

If you have trouble installing the CLI, skip the installation and use the file as a script (see #2 below).

#### 2. No Install: CLI file as script

At the root of the project, in your terminal run the command `chmod +x ./cli.js` or `sudo chmod +x ./cli.js`. This allows you to use the CLI like a script: `./cli.js `.

#### CLI Flags

To list the flags and their descriptions, run one of the following commands in the root of the project:

1. `blog help`
2. `./cli.js help`

Either of those commands will display:

```
Flags:
-t ["template" | String | either "page" or "post"]
-m ["mdx file" | n/a | if present file will be .mdx, else .js]
-f ["file name" | String | e.g. "how-to-build-a-nextjs-app"]
```

#### Creating pages and posts

The basic difference between a page and a post is that a **page** lives inside the `pages/` directory, whereas a **post** lives in the `pages/blog/` directory. Only **post** files are included in the blog post list at `/blog`. Use the `-t` flag to indicate whether you want to create a new page or a new post.

```
blog -t page -f newPage // this creates a page named "newPage.js"
blog -t post // this creates a JavaScript post with a randomized file name
```

As you can see above, the `-f` flag allows you to **name your new file**. Your page/post component name will also be the url to that page/post, so name your components strategically. For example, a page named `my-page.js` will have the url `www.mybaseurl.com/my-page`. A post named `my-post.mdx` will have the url `www.mybaseurl.com/blog/my-post`.

If you omit the `-f` flag, the CLI will generate a file name for you but you should rename it immediately.

You also have the option of using a **MDX** file or a **JavaScript** file for your new page/post.

```
blog -t page -m // this creates a MDX file with a randomized name
blog -t page -f contact // this creates a JavaScript file named contact.js
```

#### Meta Data

Each page/post needs to export (not default export) a `meta` object. The **required fields** for content are:

```
title: "Title of Page/Post goes here",
tags: ["tag-1", "tag-2"],
layout: "page",
publishDate: "2011-01-01",
modifiedDate: false,
seoDescription: "In this post I with . And blah blah."
```

- **title**: The title displayed on the page. It is also used for SEO.
- **tags**: This array enables the tag search page. The tags are also used in the SEO header property: ``.
- **layout**: This is used in the `utils/render-app-layout.js` function. There are currently four layouts (and a default layout): `"blog-post-list"`, `"post"`, `"page"` and `"search"`.
- **publishDate**: This is displyed on blog posts. It is also used for SEO. Must be in "YYYY-MM-DD" format.
- **modifiedDate**: Used for SEO. Include this if the post has been modified. If it has not been modified, leave it out or give it the value `false`.
- **seoDescription**: Used for SEO.

Blog posts have optional meta properties. These **optional fields** are:

- **exclude**: Set this property to `true` if you do not want it to appear on the blog post list on the `/blog` page.
- **hideProgressBar**: Set this property to `true` if you do not want the reading progress bar for a particular blog post.

If you do not include the optional properties for a blog post, they are assumed to be false.

## BlogMeta Component

This component should be added to each blog post after the `meta` export. Make sure there is an empty line between the `meta` export and ``, otherwise the MDX parser will throw an error. To illustrate:

```
import BlogMeta from "../../components/BlogMeta";
export const meta = {
title: "Third Post With Image",
tags: ["mdx", "javascript"],
layout: "post",
publishDate: "2017-12-10",
hideProgressBar: true,
seoDescription: "This post demonstrates a photo embed and Twitter card."
};

```

_All this is added for you when you use the CLI, but it's good to know these requirements exist in case you create posts without the CLI._

## Styles

This project uses [styled-jsx](https://github.com/zeit/styled-jsx). The docs are found [here](https://github.com/zeit/styled-jsx#getting-started).

## Config

This blog kit uses `/config/config.yml` for global configuration data.

```
config:
author: Your Name here
siteName: Your Site Name here
siteDescription: The personal blog of Your Name
defaultPageTitle: Add a default page title here
blogTitle: Blog
baseUrl: base url of your site here (e.g. www.mysite.com)

# For Twitter cards, this logo must be a hosted url, not a relative path.
# Must be square. Minimum size is 144x144; maximum size 4096x4096.
websiteLogo: "https://cdn.auth0.com/blog/logos/nextjs-logo.png"

# if your string begins with a special character, you usually have to put it in quotes
twitterHandle: "@your twitter handle"
twitterCardType: summary

# This is list is used to create the links in the navigation panel
navigation:
- { text: Home, link: / }
- { text: Blog, link: /blog }
- { text: About, link: /about }

css:
primaryColor: "#C70039"
accentColor: "#FF5733"
lightGray: "#eeeeee"
backgroundColor: "#ffffff"
black: "#333"
```

All the properties here are required, add more properties as you need them.

## Google Analytics

In `_document.js` there is a pair of commented out script tags. You can put a Google Analytics script there.

## Adding Links to Navigation

The navigation panel slides out from the right side of the website. To add a link to this list, add to the `config.navigation` list in `/config/config.yml` as shown above.

## Code Syntax Highlighting

The syntax highlighting component is found at `/code-snippets/CodeBlocks.js`. The component is currently set up for JavaScript syntax highlighting.

In `/code-snippets/post-one/`, you will find two instances of the CodeBlock component: `EscapedBackticksCode` and `DemoCode`. First, use a template string to wrap your code snippet. Second, use that string in the CodeBlock component.

The `EscapedBackticksCode` component illustrates how to include back ticks within your code snippet. (Back ticks are the ` character used in JavaScript template strings.)

In a page or post component, you can import your code snippet component and use it in a React file or an MDX file.

For more information, refer to the `react-syntax-highlighter` [docs](https://github.com/conorhastings/react-syntax-highlighter).

## Smooth Scroll

Smooth scrolling is achieved through the use of two components: `` and ``. The links that you click in order to scroll somewhere are the ``s. The destination of the scroll in the ``.

The `` takes two props:

- **target**: This is the id you want to scroll to.
- **linkText**: This is the text displayed by the link.

`` can be any html element that is the destination of a scroll. `` takes three props:

- **element**: This is the type of html element the anchor needs to be. In the component file there is a list of valid html tags. To make additional tags valid, add them to the list.
- **id**: The anchor html tag needs an id that connects the `` to the ``.
- **text**: This is the text the html tag will display.

If everything is working correctly, when you click a `` it will scroll the user down to its dedicated ``. For reference, check out an example: `pages/blog/demo-reading-progress-bar.mdx`.

For more information, refer to the `react-anchor-link-smooth-scroll` [docs](https://github.com/mauricevancooten/react-anchor-link-smooth-scroll).

## Images in MDX

To use an image in an MDX file, import the image like this:

```
import imageUrl from "../../public/static/images/mountains.jpg";
```

Then use the image like this:

```
Image alt
```

## Blog Engine

The blog engine extracts the meta objects from each page file, constructs several new properties, and creates an array of all page data. This array is used to construct and sort the post list on the `/blog` page as well as the tag search page at `/search/?q=tagName`.

For individual pages, this data is used to determine the layout for each page. The data also contains SEO information that is injected into the markup of each post.

The blog engine uses `utils/page-list.js` to get data from files in the `/pages` folder, and `utils/post-list.js` to extract data from the `/pages/blog` folder. These files use the [Babel Preval plugin](https://github.com/kentcdodds/babel-plugin-preval) to get the filenames from the `/pages` (and `/pages/blog`) directory. This list of filenames is used to import meta data from component files.

Checkout the comments in `utils/page-list.js`, `utils/post-list.js`, and `utils/blog-engine.js` to get a better idea how the BlogEngine works.

## Icon Pack

This blog kit uses SVG icons from `react-icons`. Information about usage can be found in the project docs [here](https://github.com/react-icons/react-icons).

## Branded Icons (favicon, etc.)

In `static/icons` you will see many png files with name like `icon_144`. The `144` in the name means it is a 144px x 144px square (all of the images are square). Keep the file names, but replace all these icons in all the noted sizes with your own brand images/icons.

These icons are displayed if a user desides to save your blog to their homescreen on mobile devices and some desktops.

## Custom Fonts

To load custom fonts, follow these two steps:

- Add the font files to `/static/fonts`
- Add font face CSS rules to `/styles/index.js`

Keep in mind custom fonts affect your site's performance, so use be careful with your CSS `font-display` property. This project uses `font-display: auto;` to help make font loading more performant.

If you do not want to use custom fonts, delete the files from `/static/fonts`, and remove both (1) the `@font-face` rules and (2) the `font-family: "Abril Fatface"` reference from `/styles/index.js`.

## Precaching Files for Offline Use

See the comments in `my-worker.js`.

All pages cache for desktop, so offline the entire blog should be available.

For mobile users, only pages that have been visited are cached.

**Disclaimer**: the notes above about what caches vs. what does not are gleaned from some informal investigating on my part. I'm sure there are a variety of limitations based on browser and device.

## Deploying with Now

First sign up for a Now account [here](https://zeit.co/now), and install the Now CLI. Installing the Now desktop app will also install the Now CLI.

Once you have an account and the CLI installed, open your terminal at the root of your project and issue the command `now`.

For more information see the Now [docs](https://zeit.co/docs/v2/deployments/official-builders/static-build-now-static-build/).

## Warnings and Gotchas:

- Custom routes for static sites do seem possible at the moment. As the Now v2 ecosystem develops, this will probably become manageable.
- Do not put regular React components in the `pages` directory, as they are expected to be full web pages. Next.js creates a route for each of these pages/components. Put your non-page components in the `components` directory.
- Blog posts must have an empty line between the meta export and the component. The MDX parser errors if there is not a space there.
- If you add a new page/post and it does not appear as a page or post, run `npm run clean` or delete `.next/` and run `npm run dev`.