NextJS Tutorial for Beginners with Examples | Starter
Using NextJS, you can skip the configuration headaches associated with traditional React apps…
You don’t need to worry about configuring Babel or Webpack.
You don’t need to worry about installing react or react-dom .
You don’t need to worry about implementing a server like Express.
Instead, you simply run:
npx create-next-app your-app
and all of that gets handled for you.
Note: npx commands only work with npm versions > 5.2
How NextJS Works
Creating a NextJS app with the CLI generates the following folder structure…
├── components
└── nav.js
├── pages
└── index.js
├── public
├── next.config.js
├── node_modules
├── package.json
Sharable components go in the components folder. Different pages like (Home, About, Contact) go in the pages folder.
Static assets go in the public folder. Custom configuration for webpack etc. goes in the next.config.js file.
Inside the package.json file, you’ll see:
"scripts": {
"dev": "next dev",
"build": "next build",
"start": "next start"
}
By simply running:
npm run dev
NextJS will run your app in development mode using Webpack under the hood.
Similarly, if you run:
npm run build
NextJS will package your app into an optimized .next folder so it is production ready.
Finally, running:
npm run start
will run the optimized output on the server. No deployment needed!
Creating pages with NextJS
The heart of NextJS magic is the folder structure. Any file you define under the pages folder will map to a route based on the name of the file.
If you create a about.js file like this…
const About = () => (
<div><h1>About us</h1></div>
)
export default About
and place it under the pages directory pages like this…
├── pages
│ └── index.js
│ └── about.js
then when you run npm run dev and navigate to http://localhost:3000/about you’ll see your “About” page.
Creating API endpoints with NextJS
Using this same folder “magic”, you can create API endpoints with no configuration…
Let’s say your app is an online book store. In addition to having different pages (about us, contact, etc) you’d have API endpoints for fetching data.
If you create a books.js file like this…
export default (req, res) => {
res.setHeader('Content-Type', 'application/json')
res.statusCode = 200
res.end(JSON.stringify({ title: 'The Hobbit', author: 'Tolkien' }))
}
and place it under pages/api/ subdirectory like this…
├── pages
│ └── index.js
│ └── about.js
│ └── api/
│ └── books.js
then http://localhost:3000/api/books will return the following JSON response:
{
"title": "The Hobbit",
"author": "Tolkien"
}
It’s important to remember that NextJS magic works based on the folder names. You MUST follow this structure for the routes to configure automatically. Using a subdirectory name other than api/ will not work!
Serving static files with NextJS
Images and other static assets are automatically configured with NextJS…
By placing an image some-image.png under a public folder like this…
├── public
└── some-image.png
you can reference it like this:
<img src="/some-image.png" alt="some image" />
Customizing and extending NextJS
While NextJS abstracts away a lot of the configuration for you, you can still extend NextJS to work with any implementation.
Specifically the next.config.js file lets you specify configuration settings for webpack and other things that are “hidden from view”.
For example, say you want disable the automatic routing generated from the pages directory. In the next.config.js file, you’d specify:
// next.config.js
module.exports = {
useFileSystemPublicRoutes: false,
}
The Benefits of Using NextJS
If you don’t use NextJS, you have to configure a lot of things manually. For example, without the magic of the pages directory, you’d have to manually configure a backend server to serve those pages…
On a similar note, you’d also have to configure that server to serve static assets like images etc. as well as configure the development environment for running a Webpack locally, etc.
Apart from saving you A LOT of configuration headaches, NextJS optimizes your code for performance.
For instance, NextJS automatically does server side rendering and indexing for every “page” you define under the pages directory.
NextJS also performs automatic code splitting so you aren’t loading unnecessary modules / code with each page.
Using NextJS, you can also leverage client side rendering with the framework’s built in component.
NextJS also offers built in CSS support and integrates nicely with Sass/Less.
Conclusion
NextJS lets you focus more on code and less on configuration. Things like Webpack, Babel, and Express, are automatically configured for you.
NextJS gives you both convenience and flexibility. While things come pre-configured, you can easily extend NextJS to work with your own Express implementation or customize Webpack etc.
NextJS Tutorial for Beginners with Examples | Layout
Using layouts, you can easily create reusable templates for different pages…
In the components/ folder, create a new file MyLayout.js
import Nav from './nav'
const MyLayout = props => (
<div>
<Nav />
<div>
{props.children}
</div>
</div>
)
export default MyLayout
then in the pages/ directory you can reference MyLayout like this…
import React from 'react'
import Head from 'next/head'
import Nav from '../components/nav'
import MyLayout from '../components/MyLayout'
const Home = () => (
<MyLayout>
<Head>
<title>Home</title>
</Head>
hello
</MyLayout>
)
export default Home
Notice how we include a
in our MyLayout component. By wrapping our Home component in tags, we don’t have to include with every page. Instead, we use a common MyLayout across different pages.
Notice the reference to {props.children} from the MyLayout component. This represents all child content that will be displayed in between the tags.
You can also create a layout like this…
import Nav from './nav';
const withLayout = Page => {
return () => (
<div>
<Nav />
<Page />
</div>
);
};
export default withLayout;
and then in your pages/ file…
import withLayout from '../components/MyLayout';
const PageContent = () => <p>Content for my layout!</p>;
export default withLayout(PageContent);
The main difference with this approach is that we are treating the page content as input to a function.
Yet another approach to creating layouts…
import Nav from './nav';
const MyLayout = props => (
<div>
<Nav />
{props.content}
</div>
);
export default MyLayout;
and then in your pages/ file…
import MyLayout from '../components/MyLayout.js';
const pageContent = <p>Hello Next.js</p>;
export default function Index() {
return <MyLayout content={pageContent} />;
}
This approach is similar to the previous example in that we are passing the pageContent into our MyLayout component. The main difference is that we are passing the contents as props rather than a function param.
All of these examples are valid approaches to implementing sharable components in NextJS.
NextJS Tutorial for Beginners with Examples | Express
At the root of your NextJS project, create a server.js file. This will be the entry point for your app.
Following is an example Express configuration defined in the server.js file…
const express = require('express')
const next = require('next')
const bodyParser = require('body-parser')
const posts = require('./api/post')
//next.js configuration
const dev = process.env.NODE_DEV !== 'production'
const nextApp = next({ dev })
const handle = nextApp.getRequestHandler()
nextApp.prepare().then(() => {
const app = express()
app.use(bodyParser.json())
app.use(bodyParser.urlencoded({ extended: true }))
app.use('/posts', posts)
//catch-all for nextJS /pages
app.get('*', (req, res) => {
return handle(req, res)
})
app.listen(process.env.PORT, err => {
if (err) throw err
console.log('listening on port ' + process.env.port)
})
})
Notice how we first define a NextJS instance…
const nextApp = next({ dev })
We pass in a single option for dev which specifies to run NextJS in development mode.
We then call prepare() on our instance. This method “prepares” the NextJS server with configuration options for dev environment, directory, configuration etc.
In this example, all we’ve passed is the dev flag which tells the NextJS server to run in development mode if process.env.NODE_DEV !== ‘production’ .
Notice how we create our app after nextApp.prepare() resolves. From here, we can set up our Express server for different API routes defined outside of Next.
Notice how we declare a “catch all” route…
app.get('*', (req, res) => {
return handle(req, res)
})
Express forwards request to Next via the request handler…
const handle = nextApp.getRequestHandler()
By calling handle(req, res) , we forward any request not defined through Express to NextJS.
From here, the routes defined in pages/ are applied and Next “takes over” the request if not explicitly defined somewhere else by Express.
You can even “forward” requests to the NextJS routing system directly inside an Express route…
app.get('/post', (req, res) => {
return nextApp.render(req, res, '/post')
})
Using render() will “forward” the request to a pages/post.js file.
Updating package.json
You must update the run scripts to point to the new server.js file…
"scripts": {
"dev": "node server.js",
"test": "jest",
"build": "next build",
"start": "NODE_ENV=production node server.js -p $PORT"
}
Notice how both the dev and start script have been updated to point to our server.js file.
This allows us to deploy the application as an Express app using NextJS. If we don’t update these scripts then NextJS will run using the default next start script.
NextJS Tutorial for Beginners with Examples | MongoDB
In this post, you’ll see how to include MongoDB with a NextJS project…
const express = require('express')
const next = require('next')
const bodyParser = require('body-parser')
const posts = require('./api/post')
const mongoose = require('mongoose')
//next.js configuration
const dev = process.env.NODE_DEV !== 'production'
const nextApp = next({ dev })
const handle = nextApp.getRequestHandler()
mongoose.connect(process.env.DB_STRING, {
useNewUrlParser: true,
useUnifiedTopology: true
})
nextApp.prepare().then(() => {
const app = express()
app.use(bodyParser.json())
app.use(bodyParser.urlencoded({ extended: true }))
app.use('/posts', posts)
//catch-all for nextJS /pages
app.get('*', (req, res) => {
return handle(req, res)
})
app.listen(process.env.PORT, err => {
if (err) throw err
console.log('listening on port ' + process.env.port)
})
})
That’s it! In this example, we’ve included Mongoose, a popular ODM tool for MongoDB. By connecting to our MongoDB instance via…
mongoose.connect(process.env.DB_STRING, {
useNewUrlParser: true,
useUnifiedTopology: true
})
we can share our MongoDB connection across different parts of the application…
api/post.js
const Post = require('../db/model/Post.js')
const express = require('express')
const router = express.Router()
router.get('/', async (req, res) => {
try {
let q = Post.find()
q.sort({createdAt:-1})
let posts = await q.exec()
res.json(posts)
} catch(e) {
console.log(e)
res.json({err:e})
}
})
module.exports = router
This sample demonstrates how the same DB connection can be referenced in different files.
#next #react #javascript #web-development