Next.js 已經成為 React 應用程序最重要的框架之一。它可以幫助開發人員在沒有模板的情況下構建更好的服務器端渲染 React 應用程序。
Next.js 之所以能成為目前最好的 React 框架之一,與其很多特性離不開,比如打包構建、路由預取、TypeScript、seo 等。
對于那些想要擁有一個簡單但功能強大的博客的人來說,使用 Next.js 創建博客是當今的最佳選擇。
SEO(搜索引擎優化)是改進應用程序在搜索引擎排名的過程。對于任何想要在搜索引擎上獲得更好排名并帶來更多流量的博客來說,這都是非常重要的。
我們將在本文中使用 Next.js 來構建博客。我們將介紹 SSG(靜態站點生成)的工作原理,并完成 SEO 友好的博客。
入門
使用官方推薦的Create Next App創建項目
npx create-next-app@latest --typescript
# or
yarn create next-app --typescript
# or
pnpm create next-app --typescript
復制代碼
為什么要使用Create Next App
- 交互式體驗:不帶任何參數運行npx create-next-app@latest,將會開啟交互模式,引導創建項目
- 零依賴:Create Next App沒有依賴,毫秒級創建項目
- 離線支持:Create Next App偵測網絡狀態,無網狀態將使用本地依賴緩存
- 支持模板:通過加入--example參數,可以拉取官方倉庫任何模板
- 集成測試:集成測試功能
創建完成后項目目錄構造如下:
.
├── README.md
├── next-env.d.ts
├── next.config.js
├── node_modules
├── package.json
├── pages
├── pnpm-lock.yaml
├── public
├── styles
└── tsconfig.json
復制代碼
安裝依賴
pnpm install globby gray-matter dayjs @chakra-ui/react prismjs @emotion/react @emotion/styled framer-motion next-mdx-remote remark-gfm
復制代碼
創建文章
根目錄新增_posts目錄,在_posts目錄下創建兩個mdx文件(_posts/js/helloWorld.mdx,_posts/demo.mdx),為什么是mdx文件呢?mdx支持渲染組件,支持引入導出組件,詳細文檔參考MDX
創建公共函數目錄
根目錄新增utils目錄,在utils目錄下創建getAllPosts.js并寫入如下函數
import fs from 'fs'
import {globby} from 'globby'
import matter from 'gray-matter'
const dayjs = require('dayjs')
const relativeTime = require('dayjs/plugin/relativeTime')
dayjs.extend(relativeTime)
//獲取所有文章
const GetAllPosts = async () => {
const posts = awAIt globby(['_posts'])
return posts
.reduce((prev, next) => {
const fileContents = fs.readFileSync(next, 'utf8')
const {data, content} = matter(fileContents)
const postData = {
...data,
group: dayjs(data.date).format('MMM/YYYY'),
date: dayjs(data.date).format('MMM DD, YYYY'),
fromNow: dayjs(data.date).fromNow(),
modified: dayjs(data.modified).format('MMM DD, YYYY'),
content,
slug: next.replace(/^_posts//, '').replace(/.mdx$/, '')
}
!data.draft && prev.push(postData)
return prev
}, [])
.sort((a, b) => dayjs(b.date) - dayjs(a.date))
}
// 根據slug導出文章
const GetPostBySlug = (slug) => {
// eslint-disable-next-line no-undef
return new Promise((resolve, reject) => {
GetAllPosts()
.then((posts) => {
const post = posts.find((post) =>
post.slug.includes(`${slug.join('/')}`)
)
resolve(post)
})
.catch(() => {
reject({})
})
})
}
export {GetAllPosts, GetPostBySlug}
復制代碼
創建組件
根目錄新增components目錄
- 創建PostPage.tsx組件,內容如下:
import React, {useEffect} from 'react'
import Prism from 'prismjs'
import {Box} from '@chakra-ui/react'
// 以下按需引入
require('prismjs/components/prism-go')
require('prismjs/components/prism-Python/ target=_blank class=infotextkey>Python')
require('prismjs/components/prism-JAVAscript')
require('prismjs/components/prism-css')
require('prismjs/components/prism-bash')
require('prismjs/components/prism-swift')
require('prismjs/components/prism-tsx')
require('prismjs/components/prism-jsx')
require('prismjs/components/prism-typescript')
require('prismjs/components/prism-sql')
require('prismjs/themes/prism-okaidia.min.css')
const PostPage = ({children}) => {
useEffect(() => {
const highlight = async () => {
await Prism.highlightAll()
}
highlight().then(() => {})
}, [children])
return (
<Box position="relative" w="2/3" fontSize="text.sm">
{children}
</Box>
)
}
export default PostPage
復制代碼
- 創建pages/index.tsx
import NextLink from 'next/link'
import {Fragment} from 'react'
import {
List,
LinkOverlay,
ListItem,
Container,
Heading,
Image
} from '@chakra-ui/react'
const IndexPage = ({groupByMonthPosts}) => {
return (
<Container>
{Object.keys(groupByMonthPosts).map((group) => {
return (
<Fragment key={group}>
<Heading as="h3" mt={12} mb={4}>
{group}
</Heading>
<List spacing={3}>
{groupByMonthPosts[group].map((post) => {
return (
<ListItem
position="relative"
display="flex"
gap={2}
alignItems="center"
key={post.title}
>
<NextLink
legacyBehavior
href={`/${post.slug}`}
passHref
>
<LinkOverlay>{post.title}</LinkOverlay>
</NextLink>
{post.tags.map((tag) => {
return (
<Image
key={tag}
boxSize={4}
objectFit="cover"
alt={tag}
src={`https://pics-Rust.vercel.app/uPic/icons/${tag}.svg`}
/>
)
})}
</ListItem>
)
})}
</List>
</Fragment>
)
})}
</Container>
)
}
export default IndexPage
export async function getStaticProps() {
const {GetAllPosts} = await import('utils/getAllPosts')
const posts = await GetAllPosts()
const groupByMonthPosts = posts.reduce((prev, next) => {
if (Array.isArray(prev[next.group])) {
prev[next.group].push(next)
} else {
prev[next.group] = []
prev[next.group].push(next)
}
return prev
}, {})
return {
props: {
groupByMonthPosts
}
}
}
復制代碼
- 創建pages/[...slug].tsx
import {MDXRemote} from 'next-mdx-remote'
import {serialize} from 'next-mdx-remote/serialize'
import dynamic from 'next/dynamic'
import ErrorPage from 'next/error'
import NextLink from 'next/link'
import {useRouter} from 'next/router'
import React from 'react'
import remarkGfm from 'remark-gfm'
import components from 'utils/components'
import {
Container,
Box,
Heading,
Text,
Link,
Image,
Center
} from '@chakra-ui/react'
const PostPage = dynamic(() => import('components/PostPage'))
const Post = ({title, description, date, originalUrl, mdxSource, cover}) => {
const router = useRouter()
if (!router.isFallback && !mdxSource) {
return <ErrorPage statusCode={404} />
}
return (
<Container
mt={20}
maxW={{
sm: 'container.sm',
md: 'container.md',
lg: 'container.2xl',
xl: 'container.xl'
}}
className="post"
>
<NextSeo
title={title}
description={description}
openGraph={{title, description}}
/>
<Box as="hgroup">
<Text textAlign="center" color="gray.500" fontSize="xs" as="p">
Published {date}
</Text>
<Heading textAlign="center" as="h1" mt={4} mb={2}>
{title}
</Heading>
{originalUrl && (
<Center color="gray.500" fontSize="sm" mb={8}>
本文翻譯自:
<NextLink legacyBehavior href={originalUrl} passHref>
<Link>{originalUrl}</Link>
</NextLink>
</Center>
)}
</Box>
<Image
boxSize="100%"
src={
cover ??
'https://cdn.jsdelivr.NET/gh/manonicu/pics@master/uPic/NhSU3O.jpg'
}
alt={title}
/>
<PostPage>
<MDXRemote {...mdxSource} components={components} />
</PostPage>
</Container>
)
}
export const getStaticPaths = async () => {
const {GetAllPosts} = await import('utils/getAllPosts')
const allPosts = await GetAllPosts()
const paths = allPosts.map((post) => ({
params: {
slug: post.slug.split('/')
}
}))
return {
paths,
fallback: false
}
}
export const getStaticProps = async ({params}) => {
const {GetPostBySlug} = await import('utils/getAllPosts')
const {content, ...data} = await GetPostBySlug(params.slug)
const mdxSource = await serialize(content, {
mdxOptions: {
remarkPlugins: [[remarkGfm]],
rehypePlugins: []
},
scope: data
})
return {
props: {
...data,
mdxSource
}
}
}
export default Post
復制代碼
至此,基本框架搭建完成,接下來調整樣式及組件的引入,以及 mdx 渲染修正。
- 調整樣式
可選
引入tailwind.css,執行pnpm install -D tailwindcss postcss autoprefixer && npx tailwindcss init -p
修改tailwind.config.js,如下:
/** @type {import('tailwindcss').Config} */
module.exports = {
content: [
'./pages/**/*.{js,ts,jsx,tsx}',
'./components/**/*.{js,ts,jsx,tsx}'
],
theme: {
extend: {}
},
plugins: []
}
復制代碼
修改全局樣式styles/globals.scss
@tailwind base;
@tailwind components;
@tailwind utilities;
復制代碼
必須
修改pages/_app.tsx,引入chakra-ui的配置
// pages/_app.js
import {ChakraProvider} from '@chakra-ui/react'
function MyApp({Component, pageProps}) {
return (
<ChakraProvider>
<Component {...pageProps} />
</ChakraProvider>
)
}
export default MyApp
復制代碼
到這里,不出意外,你的界面應該是長這樣
點擊鏈接,應該會報錯,未引入utils/components,這個是配置 mdx 內元素渲染的組件,參考MDX Components,mdx 提供默認的渲染組件,所以,這個是非必須的,不需要刪除即可
個人比較喜歡 chakra-ui,所以將組件都轉成了 chakra-ui 提供的組件,配置如下:
import CanIUse from 'components/CanIUse'
import {Heading, Link, Box} from '@chakra-ui/react'
import {FiExternalLink} from 'react-icons/fi'
const components = {
CanIUse,
h2: (props) => (
<Heading as="h2" mb={4}>
{props.children}
</Heading>
),
h3: (props) => (
<Heading as="h3" mb={4}>
{props.children}
</Heading>
),
h4: (props) => (
<Heading as="h4" mb={4}>
{props.children}
</Heading>
),
h5: (props) => (
<Heading as="h5" mb={4}>
{props.children}
</Heading>
),
p: (props) => (
<Box as="div" mb={4}>
{props.children}
</Box>
),
div: (props) => <Box mb={4}>{props.children}</Box>,
a: (props) => {
return (
<Link
display="inline-flex"
alignItems="center"
href={props.href}
gap={2}
isExternal
>
{props.children}
<FiExternalLink />
</Link>
)
}
}
export default components
復制代碼
好了,到這里基本完成了基于Next.js的博客搭建。
部署到Vercel
Next.js部署到Vercel無需更改和配置,無縫銜接。
【Source Code】
也可以參考我的個人網站Manon.icu | Home