注意
如果您选择使用示例或克隆项目快速启动,则可以跳过本指南,直接进入学习基础知识指南。
那么,您想从零开始构建一个 TanStack Start 项目吗?
本指南将帮助您构建一个非常基本的 TanStack Start Web 应用程序。我们将一起使用 TanStack Start 来:
让我们创建一个新的项目目录并进行初始化。
mkdir myApp
cd myApp
npm init -y
mkdir myApp
cd myApp
npm init -y
注意
> 在所有这些示例中,我们都使用 npm,但您可以选择使用您的包管理器。
我们强烈建议您将 TypeScript 与 TanStack Start 配合使用。创建一个至少包含以下设置的 tsconfig.json 文件:
{
"compilerOptions": {
"jsx": "react-jsx",
"moduleResolution": "Bundler",
"module": "ESNext",
"target": "ES2022",
"skipLibCheck": true,
"strictNullChecks": true
}
}
{
"compilerOptions": {
"jsx": "react-jsx",
"moduleResolution": "Bundler",
"module": "ESNext",
"target": "ES2022",
"skipLibCheck": true,
"strictNullChecks": true
}
}
注意
> 启用 verbatimModuleSyntax 可能会导致服务器包泄露到客户端包中。建议禁用此选项。
TanStack Start 由 Vite 和 TanStack Router 提供支持,并需要它们作为依赖项。
要安装它们,请运行
npm i @tanstack/react-start @tanstack/react-router vite
npm i @tanstack/react-start @tanstack/react-router vite
您还需要 React
npm i react react-dom
npm i react react-dom
和一些 TypeScript
npm i -D typescript @types/react @types/react-dom vite-tsconfig-paths @vitejs/plugin-react
npm i -D typescript @types/react @types/react-dom vite-tsconfig-paths @vitejs/plugin-react
然后我们将更新 package.json 以使用 Vite 的 CLI 并设置 "type": "module"
{
// ...
"type": "module",
"scripts": {
"dev": "vite dev",
"build": "vite build"
}
}
{
// ...
"type": "module",
"scripts": {
"dev": "vite dev",
"build": "vite build"
}
}
然后在 vite.config.ts 中配置 TanStack Start 的 Vite 插件
注意
TanStack Start 将停止自动配置 React/Solid Vite 插件。您将获得完全控制权——选择 @vitejs/plugin-react、@vitejs/plugin-react-oxc 等。立即设置 customViteReactPlugin: true 以选择此功能!
// vite.config.ts
import { defineConfig } from 'vite'
import tsConfigPaths from 'vite-tsconfig-paths'
import { tanstackStart } from '@tanstack/react-start/plugin/vite'
import viteReact from '@vitejs/plugin-react'
export default defineConfig({
server: {
port: 3000,
},
plugins: [
tsConfigPaths(),
tanstackStart({ customViteReactPlugin: true }),
viteReact(),
],
})
// vite.config.ts
import { defineConfig } from 'vite'
import tsConfigPaths from 'vite-tsconfig-paths'
import { tanstackStart } from '@tanstack/react-start/plugin/vite'
import viteReact from '@vitejs/plugin-react'
export default defineConfig({
server: {
port: 3000,
},
plugins: [
tsConfigPaths(),
tanstackStart({ customViteReactPlugin: true }),
viteReact(),
],
})
TanStack Start 的使用需要 2 个必需文件
配置完成后,我们的文件树将如下所示:
.
├── src/
│ ├── routes/
│ │ └── `__root.tsx`
│ ├── `router.tsx`
│ ├── `routeTree.gen.ts`
├── `.gitignore`
├── `vite.config.ts`
├── `package.json`
└── `tsconfig.json`
.
├── src/
│ ├── routes/
│ │ └── `__root.tsx`
│ ├── `router.tsx`
│ ├── `routeTree.gen.ts`
├── `.gitignore`
├── `vite.config.ts`
├── `package.json`
└── `tsconfig.json`
此文件将决定 Start 中使用的 TanStack Router 的行为。您可以在此处配置从默认的预加载功能到缓存陈旧性的所有内容。
注意
您还没有 routeTree.gen.ts 文件。此文件将在您首次运行 TanStack Start 时生成。
// src/router.tsx
import { createRouter as createTanStackRouter } from '@tanstack/react-router'
import { routeTree } from './routeTree.gen'
export function createRouter() {
const router = createTanStackRouter({
routeTree,
scrollRestoration: true,
})
return router
}
declare module '@tanstack/react-router' {
interface Register {
router: ReturnType<typeof createRouter>
}
}
// src/router.tsx
import { createRouter as createTanStackRouter } from '@tanstack/react-router'
import { routeTree } from './routeTree.gen'
export function createRouter() {
const router = createTanStackRouter({
routeTree,
scrollRestoration: true,
})
return router
}
declare module '@tanstack/react-router' {
interface Register {
router: ReturnType<typeof createRouter>
}
}
最后,我们需要创建应用程序的根目录。这是所有其他路由的入口点。此文件中的代码将包装应用程序中的所有其他路由。
// src/routes/__root.tsx
/// <reference types="vite/client" />
import type { ReactNode } from 'react'
import {
Outlet,
createRootRoute,
HeadContent,
Scripts,
} from '@tanstack/react-router'
export const Route = createRootRoute({
head: () => ({
meta: [
{
charSet: 'utf-8',
},
{
name: 'viewport',
content: 'width=device-width, initial-scale=1',
},
{
title: 'TanStack Start Starter',
},
],
}),
component: RootComponent,
})
function RootComponent() {
return (
<RootDocument>
<Outlet />
</RootDocument>
)
}
function RootDocument({ children }: Readonly<{ children: ReactNode }>) {
return (
<html>
<head>
<HeadContent />
</head>
<body>
{children}
<Scripts />
</body>
</html>
)
}
// src/routes/__root.tsx
/// <reference types="vite/client" />
import type { ReactNode } from 'react'
import {
Outlet,
createRootRoute,
HeadContent,
Scripts,
} from '@tanstack/react-router'
export const Route = createRootRoute({
head: () => ({
meta: [
{
charSet: 'utf-8',
},
{
name: 'viewport',
content: 'width=device-width, initial-scale=1',
},
{
title: 'TanStack Start Starter',
},
],
}),
component: RootComponent,
})
function RootComponent() {
return (
<RootDocument>
<Outlet />
</RootDocument>
)
}
function RootDocument({ children }: Readonly<{ children: ReactNode }>) {
return (
<html>
<head>
<HeadContent />
</head>
<body>
{children}
<Scripts />
</body>
</html>
)
}
现在我们已经完成了基本模板设置,我们可以编写第一个路由了。这可以通过在 src/routes 目录中创建一个新文件来完成。
// src/routes/index.tsx
import * as fs from 'node:fs'
import { createFileRoute, useRouter } from '@tanstack/react-router'
import { createServerFn } from '@tanstack/react-start'
const filePath = 'count.txt'
async function readCount() {
return parseInt(
await fs.promises.readFile(filePath, 'utf-8').catch(() => '0'),
)
}
const getCount = createServerFn({
method: 'GET',
}).handler(() => {
return readCount()
})
const updateCount = createServerFn({ method: 'POST' })
.validator((d: number) => d)
.handler(async ({ data }) => {
const count = await readCount()
await fs.promises.writeFile(filePath, `${count + data}`)
})
export const Route = createFileRoute('/')({
component: Home,
loader: async () => await getCount(),
})
function Home() {
const router = useRouter()
const state = Route.useLoaderData()
return (
<button
type="button"
onClick={() => {
updateCount({ data: 1 }).then(() => {
router.invalidate()
})
}}
>
Add 1 to {state}?
</button>
)
}
// src/routes/index.tsx
import * as fs from 'node:fs'
import { createFileRoute, useRouter } from '@tanstack/react-router'
import { createServerFn } from '@tanstack/react-start'
const filePath = 'count.txt'
async function readCount() {
return parseInt(
await fs.promises.readFile(filePath, 'utf-8').catch(() => '0'),
)
}
const getCount = createServerFn({
method: 'GET',
}).handler(() => {
return readCount()
})
const updateCount = createServerFn({ method: 'POST' })
.validator((d: number) => d)
.handler(async ({ data }) => {
const count = await readCount()
await fs.promises.writeFile(filePath, `${count + data}`)
})
export const Route = createFileRoute('/')({
component: Home,
loader: async () => await getCount(),
})
function Home() {
const router = useRouter()
const state = Route.useLoaderData()
return (
<button
type="button"
onClick={() => {
updateCount({ data: 1 }).then(() => {
router.invalidate()
})
}}
>
Add 1 to {state}?
</button>
)
}
就是这样!🤯 您现在已经设置了一个 TanStack Start 项目并编写了您的第一个路由。🎉
您现在可以运行 npm run dev 来启动您的服务器,然后导航到 https://:3000 查看您的路由实际运行情况。
您想部署您的应用程序吗?请查看托管指南。
您的每周 JavaScript 资讯。每周一免费发送给超过 10 万开发者。