Skip to content

stanleyfok/nextjs-template

Repository files navigation

Nextjs Template

Nextjs already provide a very easy way to use react to create a universal app. However, there are still much to setup if you want to put your app to a production environment. I have spent some time myself to come up with a project template which is able to suit my business needs.

This project template is actually a simple universal webapp with 2 screens. I hope it will be a useful example for those who are interested to build your next webapp with Nextjs.

Highlights

  • a project skeleton on Next.js SSR supported React framework
  • with eslint setup with rules from airbnb-base and react/recommended
  • integrated with express, with i18n and compression middlewares
  • demonstrate using redux in Next.js
  • use axios for HTTP client
  • error handling for API
  • use next-route for dynamic routes
  • use of scss and bootstrap
  • use of nprogress for progress bar
  • i18n support by integrating i18next
  • webpack-bundle-analyzer support
  • a sample dockerfile and docker-compose.yml, which the docker-compose.yml simulates a environment having an application server and a CDN server

Getting Started

To run this example, simply do:

npm install
npm run dev

Then you will see a simple web app with 2 pages like this:

IDE setup

VSCode is highly preferred. Please ensure you have installed these extensions:

  • Prettier
  • stylelint
  • eslint

Useful Commands

#to run in dev mode
npm run dev

#to run in qa or production env. Expect to run this in docker
npm run start

#to run the webpack analyzer
npm run analyze

#to clear cache and .next folder
npm run cc

Design Philosophy - Dev Environment

πŸ’‘ Rule: Hot Reload should be supported during development

Hot Reload is important for frontend development as the changes can be reflected in a very quick manner. Hot reload feature is already supported by the latest version of Nextjs. Any changes on the source files would trigger rebuilding the application automatically.

πŸ’‘ Rule: Code Linting is required

Linters are added into this project template to enhance code quality.

JS

For javascript, eslint is used, extending rules from airbnb-base and react/recommended

#To check js syntax
npm run eslint

#To try fixing js syntax
npm run eslint-fix

SCSS

For scss, sass-lint is used.

#To check scss syntax
npm run sasslint

Design Philosophy - Application Design

πŸ’‘ Rule: Need a clear layout structure

Every page in the same project actually shares the same structure. I make good use of some Nextjs features and some custom components to organize the pages:

pages/_document.js

This is provided by the Nextjs framework. It defines root skeleton of any page. For example, it contains markup like <html>, <head>, <body>

components/Layout.js

This is a custom component, which is more about look and feel structure of any page. For a typical webpage, we usually need a define the header, the footer and where to show the main content.

components/Meta.js

This components is representing the meta data to be placed in the section. In this example, I have only added page title and page description. You can extend it to hold more meta fields, for example, the facebook open graph markup

Individual Page

This is where we define the actual page content. A page needs to include the Meta component and Layout component. The Meta component defines those required meta fields, such as page title and description, while the Layout component describes the look and feel of the page.

Here is how it looks in the render method of a page:

render() {
  return (
    <Layout>
      <Meta
        title="Page Title"
        description="Page Description"
      />
      <p>Hello World</p>
    </Layout>,
  );
}

πŸ’‘ Rule: Error Page has to be customized

Nextjs allows us to define our own error page. The custom error page is located at /pages/_error.js. It looks very similar to the normal pages as you can also define it's own Meta and Layout.

πŸ’‘ Rule: Routes pattern has to be SEO friendly

Custom server and routing are needed because we need to support dynamic routing like '/shows/3557-Beware-the-Batman' for better SEO. The routing config file is just a simple object, having each row in the following format:

'[http method] [pattern]': '[actual nextjs page path]'

This is how it actually looks in our example:

module.exports = {
  "GET /": "/index",
  "GET /shows/(:id)(-*)?": "/show"
};

The server.js takes this routing file and tells the Express engine how to response when seeing the url patterns.

Related Files

Design Philosophy - JS

πŸ’‘ Rule: Ensure imported modules are universal

Not all javascript libraries can be run on both server and client side. When you pick a module to use, make sure it is universal.

πŸ’‘ Rule: Config file cannot expose parameters from other environments

In a universal application, every files will be exposed to the frontend. You can actually define your config file like below, which contains all parameters under different environment:

export default {
  dev: { apiHost: "http://localhost:3001/" },
  staging: { apiHost: "https://www.myinternalsite.com/" },
  production: { apiHost: "https://www.mysite.com/" }
};

However, the problem is that this would leak internal information to the public, which maybe potentially a security issue.

To solve this problem, the babel plugin transform define is used. It helps to rewrite a variable to its actual value during babel compilation. The variables which can be replaced during the compile time are defined at env.config.js

Related Files

Design Philosophy - CSS and other media files

πŸ’‘ Rule: Use of SCSS

SCSS is a powerful CSS syntax extension which makes our lives easier. I have given up the use of styled-jsx as I found it easier to organize my styles with scss. For example, I can change the whole site's look and feel by replacing the css file. I couldn't figure out a nice way to do this in styled-jsx. It would be great if someone can share me your experience :)

πŸ’‘ Rule: Use new path for a new version of asset files

When using a CDN, a URL can be cached for a long period of time. Even you replace with a new file content, the CDN will still serve the previous copy of the files stored in the CDN. The safest way is to have different file paths for different releases.

I have introduced 'version hash' to solve it, which is simply the md5 hash of the version number in package.json. A folder named by the version hash will be generated under the path /static. For example: /static/47cd76e43f74bbc2e1baaf194d07e1fa/images/favicon.png

If you have any new release, remember to bump up the version number. Your static files will then be placed in a new url, and then be cached by the CDN.

Also, do not check in any code under the static folder. Instead, please put your resources file under the assets folder. The webpack rules defined in next.config.js will move your files to the static version hash folder.

Design Philosophy - i18N

πŸ’‘ Rule: Every string should be placed in translation files

No matter the site is built for one or more locales, it is still a good practice to extract text into translation files. It will help us to better organize the string and make changes very easily. The JSX files should be clean and without any text content.

πŸ’‘ Rule: Make sure no translation files are loaded if the page is rendered with SSR

If the page is rendered by server side, the translation should be ready at server side already. Therefore, we should not expect a translation file is loaded in the client side. You can verify this by inspecting the network calls in your browser. This can ensure the page load speed is faster by loading less files in the browser.

πŸ’‘ Rule: Translation should be organized with hierarchy

By using the library i18next, translation files are stored in JSON format. We should group the translation string, rather than constructing a long translation key

Good:

{
  "header": {
    "title": "Batman TV Programs"
  },
  "footer": {
    "text": "This is footer"
  }
}

Bad:

{
  "header_title": "Batman TV Programs",
  "footer_text": "This is footer"
}

Design Philosophy - User Experience

πŸ’‘ Rule: Need to show progress bar when navigating between pages

If data is fetched from client side in getInitialProps(), the UI is frozen util API is fetched completely. Therefore, we have to ensure progress bar is shown when users click on any links. In this example, the plugin NProgress is used.

Design Philosophy - Quality Control

πŸ’‘ Rule: Ensure every page can be loaded from server side and client side

The great thing Nextjs provides to us is React with SSR, which means any page can be access from server side and client side. This also means that you have to ensure your page should be functional when entering from server side or client side.

πŸ’‘ Rule: Test Driven Development is encouraged

Need to write unit test for every modules or UI components

πŸ’‘ Rule: Need a file structure favoring writing test cases

This article says well how to manage the test files: http://www.tysoncadenhead.com/blog/where-should-i-put-javascript-unit-tests/

I take the approach having the __tests__ folder in each folders, so unit testing files are closer to the modules

πŸ’‘ Rule: Using Jest and Enzyme for unit testing tools

The library enzyme and enzyme-to-json are mainly used to write unit tests

πŸ’‘ Rule: Need to export the undecorated component for every component file to facilitate unit testing

Each react component has to export the undecorated component. For example:

export default withI18N(Footer);
export const undecorated = Footer;

The exported undecorated will be only used for unit testing. This can help us to test a component by passing the required props. Shallow rendering is used whenever possible.

πŸ’‘ Rule: API client library has to be mocked

API client library has to be mocked so that unit testing is not related to the real api. It will also speed up the time running your test cases.

Reference: https://hackernoon.com/api-testing-with-jest-d1ab74005c0a

Design Philosophy - Optimization

πŸ’‘ Rule: Analyzer

It is always a good practice to run the webpack-bundle-analyzer to understand how to optimize your app. To learn some optimization skills on Nextjs, please refer to:

#To start the analyzer
npm run analyze

Design Philosophy - Deployment

πŸ’‘ Rule: Using Docker for deployment strategy

A sample dockerfile is created for reference

πŸ’‘ Rule: Using Docker Compose to simulate a close to production env

Docker compose can help you to simulate an environment closer to your production environment. In this example, I have added a CDN server to simulate serving static files through CDN.

Related Files

About

A comprehensive Nextjs project template

Topics

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published