---
phase: 03-dialog-feedback
plan: 01
type: execute
wave: 1
depends_on: []
files_modified:
- app/layout.tsx
autonomous: true
requirements:
- CRED-FE-05
must_haves:
truths:
- "调用 toast.success(...) / toast.error(...) 后屏幕能看到 Sonner 通知"
- "Toaster 在所有路由下均可用(挂在 RootLayout 顶层)"
- "不破坏现有 RootLayout 渲染(children 仍正常显示)"
artifacts:
- path: "app/layout.tsx"
provides: "RootLayout 注入 portal"
contains: "Toaster"
key_links:
- from: "app/layout.tsx"
to: "@/components/ui/sonner"
via: "import { Toaster }"
pattern: "from \"@/components/ui/sonner\""
---
在 `app/layout.tsx` 的 `` 内挂载 Sonner ``,让全局 `toast.success(...)` / `toast.error(...)` 命令式调用真正能在屏幕上显示。
**Purpose**:仓库 9 处 `toast(...)` 调用当前全部是 dead code(仅 `components/ui/sonner.tsx` 定义了 Toaster 包装但**从未挂载**),不挂载 Phase 3 的成功 / 失败反馈完全静默;这是 Phase 3 业务功能跑通的硬前置。
**Output**:修改后的 `app/layout.tsx`,新增 1 行 import + 1 个 JSX 元素挂载点。
@$HOME/.claude/get-shit-done/workflows/execute-plan.md
@$HOME/.claude/get-shit-done/templates/summary.md
@.planning/PROJECT.md
@.planning/ROADMAP.md
@.planning/STATE.md
@.planning/phases/03-dialog-feedback/03-CONTEXT.md
@.planning/phases/03-dialog-feedback/03-RESEARCH.md
@CLAUDE.md
@app/layout.tsx
@components/ui/sonner.tsx
From components/ui/sonner.tsx:
```typescript
type ToasterProps = React.ComponentProps
const Toaster: ({ ...props }: ToasterProps) => JSX.Element
export { Toaster }
```
调用形态:``(无 props 即可,theme 内部已读 next-themes context;本仓库未挂 ThemeProvider,会回退到默认 "system",无错误)
Task 1:在 RootLayout 挂载 Sonner Toaster
app/layout.tsx
必读 `app/layout.tsx` 全文(仅 20 行)确认当前结构。当前内容:
```tsx
import type { Metadata } from 'next'
import './globals.css'
export const metadata: Metadata = {
title: 'v0 App',
description: 'Created with v0',
generator: 'v0.dev',
}
export default function RootLayout({
children,
}: Readonly<{
children: React.ReactNode
}>) {
return (
{children}
)
}
```
**精确改动 2 处**:
**改动 1**:在 `import './globals.css'` 之后追加 1 行 import:
```tsx
import { Toaster } from '@/components/ui/sonner'
```
**改动 2**:把 `{children}` 改为 `{children}`(即在 `{children}` 之后、`` 之前插入 ``)。
**最终全文(应该是这样)**:
```tsx
import type { Metadata } from 'next'
import './globals.css'
import { Toaster } from '@/components/ui/sonner'
export const metadata: Metadata = {
title: 'v0 App',
description: 'Created with v0',
generator: 'v0.dev',
}
export default function RootLayout({
children,
}: Readonly<{
children: React.ReactNode
}>) {
return (
{children}
)
}
```
**严格约束**:
- **不**挂 Radix Toast ``(来自 `@/components/ui/toaster`)—— 那是另一套实现,与 Sonner 不通信;本 phase 锁定 Sonner(CONTEXT D-Toast 决策)
- **不**新增 ThemeProvider / next-themes 包装 —— `components/ui/sonner.tsx:9` 有 `useTheme()` fallback 到 `"system"`,无 ThemeProvider 也能跑(本仓库目前确实没挂 ThemeProvider)
- **不**改 `metadata` / `html lang` / `globals.css` import 顺序
- **不**给 RootLayout 加 `"use client"` —— `` 自身就是 client component(`components/ui/sonner.tsx:1` 顶部已 `"use client"`),React Server Component 可以直接渲染 client child,无需 RootLayout 自己 client 化
- 这是本 phase 唯一改动 `app/layout.tsx` 的 task,**不**在此挂任何其他 provider
# Windows PowerShell(项目不含 .eslintrc*,lint 跳过沿用 Phase 1+2 判定)
cd C:\Users\admin\Desktop\Lila-Server\qy-lty-admin
# A 段:tsc 整体 + 反向断言(必须 0 条指向 app/layout.tsx)
npx tsc --noEmit 2>&1 | Select-String -Pattern 'app/layout\.tsx|app\\layout\.tsx'
# 期望:无任何输出(0 条新错误指向本文件)
# B 段:grep 验证 import + Toaster 元素都已落地(PowerShell Select-String)
Select-String -Path 'app/layout.tsx' -Pattern 'from "@/components/ui/sonner"'
# 期望:1 行命中 import 行
Select-String -Path 'app/layout.tsx' -Pattern ''
# 期望:1 行命中 标签
# C 段:lockfile 未动(不引入新依赖)—— Sonner 已在 deps(^1.7.1)
git diff --stat HEAD -- package.json yarn.lock package-lock.json pnpm-lock.yaml
# 期望:0 行(无 diff)
- `app/layout.tsx` 包含 `import { Toaster } from "@/components/ui/sonner"` 一行
- `` 内 `{children}` 之后渲染 ``
- `npx tsc --noEmit` 输出过滤 `app/layout.tsx` 后 **0 条新错误**(67 条存量错误与本 task 无关)
- 4 个 manifest+lockfile 在 git diff 中 0 行 diff(不引入新依赖)
**Phase 3 Plan 1 整体验证**(Plan 内已涵盖,此处汇总):
1. **类型检查**:`npx tsc --noEmit` exit 非 0(67 条存量错误,与本 phase 无关),但 `Select-String` 过滤 `app/layout.tsx` 命中 0 行
2. **挂载位置正确**:grep `` 命中且位于 `` 内、`{children}` 之后(不是 `` 内、不在 children 之前)
3. **不动 lockfile**:`git diff --stat HEAD -- package.json *.lock` 输出 0 行
4. **lint 跳过**:项目无 `.eslintrc*` / `eslint-config-next`,沿用 Phase 1+2 判定(不阻塞)
**关键失败模式**(如果出现,回头修):
- 如果挂在 `` 之外或 `` 内 → 渲染失败 → 修
- 如果误用 `import { Toaster } from "@/components/ui/toaster"`(Radix Toast)→ 与 Sonner toast() 不通信 → 修
- 如果给 layout.tsx 加了 `"use client"` → 改回 RSC(无必要)
- [ ] `app/layout.tsx` import 块包含 `from "@/components/ui/sonner"`
- [ ] `app/layout.tsx` `` 内含 `` 元素
- [ ] `npx tsc --noEmit` 过滤后 0 条新错误指向 `app/layout.tsx`
- [ ] 4 个 lockfile 在 git diff 中 0 行 diff
- [ ] Plan 03-02 的 toast 调用上线后能在屏幕显示(本 plan 单独无法跑通端到端,由 03-02 联动验证)