[Tutorial] - Como criar um boilerplate para projetos com Next.js

Vou mostrar o passo a passo de como realizar todas as configurações que utilizo nos meus projetos com Next.js

Baseado no boilerplate-apps-router do curso React Avançado

Passo a Passo

Meu ambiente de desenvolvimento é o Windows com WSL e VSCode como editor de código. Escrevi esse tutorial sobre como configurar o WSL.

Inicializar o projeto:

Execute os comandos abaixo:

mkdir next-boilerplate
cd next-boilerplate
npm init
git init

Dessa forma, foi criado o arquivo package.json, onde deixei no seguinte molde:

{
  "name": "next-boilerplate",
  "version": "1.0.0",
  "description": "Next.js Boilerplate",
  "repository": "https://github.com/diasjoaovitor/next-boilerplate.git",
  "author": "João Vitor <jvitordiass@outlook.com.br>",
  "license": "MIT"
}

Especificar a versão do Node:

Crie o arquivo .nvmrc e adicione a versão a ser utilizada:

lts/iron

É obrigatório deixar uma linha em branco ao final do arquivo para que essa configuração funcione corretamente.

Configurar o commit linter

Instale o git-commit-msg-linter:

npm i -D git-commit-msg-linter

Essa biblioteca verifica se a mensagem de um commit contém um prefixo semântico como feat, fix, refactor e demais convenções.

Crie o arquivo .gitignore e adicione a pasta node_modules

Instalar o Next

Instale o next, react e react-dom

npm i next react react-dom

Crie o diretório src/app e adicione os arquivos layout.jsx e page.jsx

export const metadata = {
  title: 'Next Boilerplate',
  description: 'Boilerplate para projetos Next.js'
}

const RootLayout = ({ children }) => {
  return (
    <html lang="pt-br">
      <body>{children}</body>
    </html>
  )
}

export default RootLayout
const Home = () => {
  return <h1>Home</h1>
}

export default Home

Adicione os scripts no arquivo package.json:

{
  "scripts": {
    "dev": "next dev",
    "build": "next build",
    "start": "next start"
  }
}

Ignore a pasta .next no arquivo .gitignore

Configurar as regras do editor e do código

Instale o módulo e a extensão do prettier:

npm i -D prettier

Crie o arquivo .prettierrc.json:

{
  "trailingComma": "none",
  "semi": false,
  "singleQuote": true
}
  • "trailingComma": "none": Isso indica que o Prettier não deve adicionar vírgulas ao final de listas, objetos ou parâmetros de função quando eles são formatados em várias linhas.
  • "semi": false: Isso significa que o Prettier não deve adicionar ponto e vírgula ao final de cada instrução.
  • "singleQuote": true: Isso especifica que o Prettier deve usar aspas simples em vez de aspas duplas, sempre que possível.

Adicione os scripts:

{
  "scripts": {
    "prettier:check": "prettier --check .",
    "prettier:fix": "prettier --write ."
  }
}

Defina o prettier como o formatador padrão do VSCode em Default Formatter e habilite a opção Format On Save

Crie a pasta .vscode e inclua o arquivo settings.json:

{
  "editor.formatOnSave": true,
  "editor.defaultFormatter": "esbenp.prettier-vscode",
  "files.autoSave": "off",
  "git.autofetch": true
}
  • "editor.formatOnSave": true - Esta opção faz com que o VSCode formate automaticamente o código quando você salva um arquivo. Isso ajuda a manter o código limpo e consistente com as regras de formatação definidas.
  • "editor.defaultFormatter": "esbenp.prettier-vscode" - Define o Prettier como o formatador de código padrão. O Prettier é uma ferramenta popular que suporta muitas linguagens e estilos de codificação.
  • "files.autoSave": "off" - Desativa o salvamento automático de arquivos. Com essa configuração, os arquivos não serão salvos automaticamente após um período ou quando o foco é alterado; você precisará salvar manualmente suas alterações.
  • "git.autofetch": true - Quando habilitado, o VSCode buscará automaticamente as alterações mais recentes do seu repositório Git periodicamente. Isso é útil para manter seu repositório local atualizado com as alterações remotas.

Crie o arquivo .editorconfig:

root = true

[*]
indent_style = space
indent_size = 2
end_of_line = lf
charset = utf-8
trim_trailing_whitespace = true
insert_final_newline = true
  • root = true: Esta configuração sinaliza que este é o arquivo de configuração principal e que o EditorConfig não deve procurar por outros arquivos de configuração nas pastas acima.

  • [*]: Este é um padrão de correspondência que se aplica a todos os arquivos no projeto.

  • indent_style = space: Define que o estilo de indentação deve ser feito com espaços em vez de tabulações.

  • indent_size = 2: Especifica que o tamanho da indentação deve ser de dois espaços.

  • end_of_line = lf: Indica que o final de linha deve ser formatado usando LF (Line Feed), que é o padrão para sistemas Unix e macOS.

  • charset = utf-8: Define que o conjunto de caracteres do arquivo deve ser UTF-8.

  • trim_trailing_whitespace = true: Quando verdadeiro, remove automaticamente qualquer espaço em branco no final das linhas ao salvar o arquivo.

  • insert_final_newline = true: Garante que haja uma nova linha no final do arquivo ao salvar.

O linting é o processo de aplicar regras a uma base de código e destacar padrões ou códigos problemáticos que não aderem a determinadas diretrizes de estilo. ESLint permite que os desenvolvedores descubram problemas com seu código sem a necessidade de executá-lo.

Instale o eslint, eslint-config-next e o eslint-config-prettier:

npm i -D eslint eslint-config-next eslint-config-prettier

Crie o arquivo .eslintrc.json:

{
 "extends": [
    "eslint:recommended",
    "next/core-web-vitals",
    "prettier"
  ]]
}

Adicione o script:

{
  "scripts": {
    "lint": "next lint",
  }
}

Instale o lint-staged e o husky


npm i -D lint-staged husky

Crie o arquivo .lintstagedrc.js:

const path = require('path')

const buildCommand = (filenames) => {
  const files = filenames.map((f) => path.relative(process.cwd(), f))
  return [
    `npx prettier --write ${files.join(' --file ')}`,
    `npx next lint --fix --file ${files.join(' --file ')}`,
  ]
}

module.exports = {
  '*.{js,jsx,ts,tsx}': [buildCommand]
}

Inicie as configurações do husky:

npx husky init

Dessa forma, foi criado o script prepare no arquivo package.json e a pasta .husky, onde vamos alterar o conteúdo do arquivo pre-commit para:

npx --no-install lint-staged

Configurar TypeScript

Instale o typescript e demais dependências:

npm i -D typescript @types/node @types/react @types/react-dom @typescript-eslint/eslint-plugin @typescript-eslint/parser

Crie o arquivo tsconfig.json:

{
  "compilerOptions": {
    "target": "ESNext", // Especifica a versão do ECMAScript de destino para a qual o código TypeScript será compilado.
    "lib": ["dom", "dom.iterable", "esnext"], // Define as bibliotecas de declaração de tipo que serão incluídas na compilação.
    "allowJs": true, // Permite a compilação de arquivos JavaScript junto com arquivos TypeScript.
    "skipLibCheck": true, // Pula a verificação de tipos em todas as declarações de bibliotecas de tipos (*.d.ts).
    "strict": true, // Habilita um conjunto de verificações de tipo mais rigorosas.
    "noEmit": true, // Não emite arquivos de saída (como .js) após a compilação.
    "esModuleInterop": true, // Habilita a interoperabilidade de módulos ES6 com módulos CommonJS/AMD/UMD.
    "module": "esnext", // Define o formato do módulo de saída.
    "moduleResolution": "bundler", // Especifica a estratégia de resolução de módulos.
    "resolveJsonModule": true, // Permite a importação de arquivos .json.
    "isolatedModules": true, // Garante que cada arquivo possa ser compilado de forma isolada.
    "jsx": "preserve", // Preserva as anotações JSX no arquivo de saída.
    "incremental": true, // Habilita a compilação incremental para acelerar as compilações subsequentes.
    "plugins": [
      // Lista de plugins do compilador TypeScript.
      {
        "name": "next" // Especifica o plugin 'next' para ser usado durante a compilação.
      }
    ],
    "paths": {
      // Define um conjunto de entradas de caminho que serão resolvidas durante a compilação.
      "@/*": ["./src/*"] // Permite o uso de '@' como um alias para o diretório './src/'.
    }
  },
  "include": [
    // Especifica os arquivos que devem ser incluídos na compilação.
    "next-env.d.ts", // Arquivo de declaração de tipo específico do Next.js.
    "**/*.ts", // Todos os arquivos TypeScript no projeto.
    "**/*.tsx", // Todos os arquivos TypeScript com JSX no projeto.
    ".next/types/**/*.ts" // Arquivos de tipo específicos do Next.js.
  ],
  "exclude": ["node_modules"] // Exclui a pasta 'node_modules' da compilação.
}

Adicione as configurações de lint recomendadas no arquivo .eslintrc.json:

{
  "extends": [
    "eslint:recommended",
    "next/core-web-vitals",
    "next/typescript",
    "prettier"
  ]
}

Altere as extensões dos arquivos layout e page para .tsx e defina os tipos:

import type { Metadata } from 'next'
import { Inter } from 'next/font/google'

const inter = Inter({ subsets: ['latin'] })

export const metadata: Metadata = {
  title: 'Next Boilerplate',
  description: 'Boilerplate para projetos Next.js'
}

const RootLayout = ({
  children
}: Readonly<{
  children: React.ReactNode
}>) => {
  return (
    <html lang="pt-br">
      <body className={inter.className}>{children}</body>
    </html>
  )
}

export default RootLayout

Configurar testes automatizados

Instale as bibliotecas de testes:

npm i -D jest @types/jest jest-environment-jsdom @testing-library/react @testing-library/jest-dom @testing-library/user-event eslint-plugin-jest

Crie os arquivos jest.config.js e jest.setup.ts:

const nextJest = require('next/jest')

const createJestConfig = nextJest({
  dir: '.'
})

const jestConfig = createJestConfig({
  testEnvironment: 'jsdom',
  setupFilesAfterEnv: ['<rootDir>/jest.setup.ts']
})

module.exports = jestConfig
import '@testing-library/jest-dom'

Adicione os scripts:

{
  "scripts": {
    "test": "jest",
    "test:watch": "jest --watchAll"
  }
}

Adicione o plugin do eslint no arquivo .eslintrc.json:

{
  "extends": [
    "eslint:recommended",
    "next/core-web-vitals",
    "next/typescript",
    "plugin:jest/recommended",
    "prettier"
  ]
}

Adicione o comando para rodar os testes no arquivo lintstagedrc.json:

const path = require('path')

const buildCommand = (filenames) => {
  const files = filenames.map((f) => path.relative(process.cwd(), f))
  return [
    `npx prettier --write ${files.join(' --file ')}`,
    `npx next lint --fix --file ${files.join(' --file ')}`,
    `npx jest --runInBand --findRelatedTests ${files.join(' ')} --passWithNoTests`
  ]
}

module.exports = {
  '*.{js,jsx,ts,tsx}': [buildCommand]
}

Configurar Integração Contínua

Adicione o script "test:ci": "jest --runInBand" no package.json e crie o arquivo ci.yml na pasta .github/workflows:

name: ci
on: [push, pull_request] # O workflow é acionado quando um push ou pull request é aberto, sincronizado ou reaberto.

jobs:
  build: # Este é o trabalho de construção que será executado.
    runs-on: ubuntu-latest # O trabalho será executado na última versão do Ubuntu disponível.
    steps: # Seguem os passos que serão executados em sequência.
        uses: actions/checkout@v4 # Este passo faz o checkout do seu repositório usando a ação checkout v4.
        uses: actions/setup-node@v4 # Este passo configura o ambiente Node.js usando a ação setup-node v4.
        with:
          node-version: lts/iron # Define a versão do Node.js para a versão LTS mais recente chamada "Iron".
          cache: 'npm' # Habilita o cache para o gerenciador de pacotes NPM.

      - name: Install dependencies
        run: npm ci # Este passo instala as dependências listadas no arquivo package-lock.json.

      - name: Prettier
        run: npm run prettier:check  # Este passo executa o linting no código para verificar erros de formatação.
        
      - name: Linting
        run: npm run lint # Este passo executa o linting no código para verificar erros

      - name: Testing
        run: npm run test:ci # Este passo executa os testes definidos para o projeto.

      - name: Build
        run: npm run build # Este passo constrói o projeto.

Configurar gerador de componentes

Instale o Plop

npm i -D plop

Crie a pasta generators na raiz do projeto e adicione o arquivo plopfile.js:

const fs = require('fs')
const path = require('path')

module.exports = (plop) => {
  plop.setGenerator('component', {
    description: 'Create a component',
    prompts: [
      {
        type: 'input',
        name: 'name',
        message: 'What is your component name?'
      }
    ],
    actions: [
      {
        type: 'add',
        path: '../src/app/components/{{pascalCase name}}/index.tsx',
        templateFile: 'templates/index.tsx.hbs'
      },
      {
        type: 'add',
        path: '../src/app/components/{{pascalCase name}}/{{kebabCase name}}.test.tsx',
        templateFile: 'templates/test.tsx.hbs'
      },
      {
        type: 'append',
        path: '../src/app/components/index.ts',
        template: "export * from './{{pascalCase name}}'",
        separator: ''
      },
      () => {
        const indexPath = path.resolve(
          __dirname,
          '../src/app/components/index.ts'
        )
        const content = fs.readFileSync(indexPath, 'utf-8')
        const lines = content.split('\n')
        const updatedContent =
          lines
            .filter((line) => line.trim() !== '')
            .sort()
            .join('\n') + '\n'
        fs.writeFileSync(indexPath, updatedContent)
      }
    ]
  })
}

Crie o arquivo .prettierignore e ignore todos templates:

*.hbs

Dentro de generators, crie o templates index.tsx.hbs e test.tsx.hbs:

export const {{pascalCase name}} = () => {
  return (
    <div>{{pascalCase name}}</div>
  )
}
import { render, screen } from '@testing-library/react'
import { {{pascalCase name}} } from '.'

describe('<{{pascalCase name}} />', () => {
  it('should render component', () => {
    render(<{{pascalCase name}} />)
    expect(screen.getByText('{{pascalCase name}}')).toBeInTheDocument()
  })
})

Adicione o script:

{
  "scripts": {
    "generate": "plop --plopfile generators/plopfile.js"
  }
}

Crie um componente:

npm run generate MyComponent

Assim, foi gerado um componente e um teste seguindo o padrão dos templates

Dentro da pasta components, crie o arquivo index.ts e exporte o componente:

export * from './MyComponent'

Essa configuração ajuda a importar todos os componentes exportados nesse arquivo através do alias @/app/components

Para finalizar, altere o componente MyComponent e seu arquivo de teste para:

import { PropsWithChildren } from 'react'

export const MyComponent = ({ children }: PropsWithChildren) => {
  return <div>{children}</div>
}
import { render, screen } from '@testing-library/react'
import { MyComponent } from '.'

describe('<MyComponent />', () => {
  it('should render component', () => {
    render(<MyComponent>MyComponent</MyComponent>)
    expect(screen.getByText('MyComponent')).toBeInTheDocument()
  })
})

Dessa forma, importe o componente no arquivo page.ts na raiz da pasta app e crie o teste home.test.tsx:

import { MyComponent } from '@/app/components'

const Home = () => {
  return <MyComponent>Home</MyComponent>
}

export default Home
import { render, screen } from '@testing-library/react'
import Home from './page'

describe('<MyComponent />', () => {
  it('should render component', () => {
    render(<Home />)
    expect(screen.getByText('Home')).toBeInTheDocument()
  })
})

Execute npm run dev para iniciar o app e npm test para executar os testes.

Considerações finais

Este é um boilerplate genérico para projetos NextJS.

O repositório está disponível no meu perfil do GitHub e está aberto a contribuições.