Przeglądaj źródła

📝 docs(story): 新增商品详情页规格选择集成故事文档

- 创建故事文档 006.006,详细描述商品详情页规格选择功能的集成需求
- 定义验收标准,涵盖组件集成、购买流程支持、价格库存正确性及多租户数据获取
- 列出详细的任务清单,包括组件集成验证、购买功能测试、数据正确性验证和多租户测试
- 提供技术栈、源码树、现有组件分析、数据模型、API信息等开发上下文
- 明确测试要求,包括集成测试、功能测试、数据正确性测试和多租户场景测试
- 记录项目结构注意事项,确保多租户支持完整和API向后兼容性
yourname 1 miesiąc temu
rodzic
commit
34e8d9b356

+ 168 - 0
docs/stories/006.006.goods-detail-spec-integration.story.md

@@ -0,0 +1,168 @@
+# Story 006.006: 商品详情页规格选择集成
+
+## Status
+Draft
+
+## Story
+**As a** 商品购买用户
+**I want** 在商品详情页选择商品规格并确保系统使用正确的子商品价格和库存
+**so that** 我可以准确购买特定规格的商品
+
+## Acceptance Criteria
+1. 在商品详情页集成规格选择组件
+2. "立即购买"和"加入购物车"支持规格选择
+3. 规格选择后使用子商品的价格和库存信息
+4. 多租户环境下的商品规格数据获取
+
+## Tasks / Subtasks
+- [ ] 验证GoodsSpecSelector组件在商品详情页的正确集成 (AC: 1)
+  - [ ] 确认组件已取消注释并正确导入
+  - [ ] 验证组件props传递正确(parentGoodsId、currentSpec等)
+  - [ ] 测试组件显示/隐藏状态管理
+- [ ] 验证"立即购买"和"加入购物车"的规格选择支持 (AC: 2)
+  - [ ] 检查handleAddToCart函数正确处理规格选择逻辑
+  - [ ] 检查handleBuyNow函数正确处理规格选择逻辑
+  - [ ] 验证选择规格后使用正确的商品ID、名称、价格和库存
+  - [ ] 测试无规格商品时的向后兼容性
+- [ ] 验证规格选择后的价格和库存信息正确性 (AC: 3)
+  - [ ] 确认子商品价格正确显示和计算
+  - [ ] 验证库存限制基于子商品库存
+  - [ ] 测试价格计算正确性(总价 = 单价 × 数量)
+  - [ ] 验证数量选择器基于子商品库存限制
+- [ ] 验证多租户环境下的数据获取 (AC: 4)
+  - [ ] 确认API调用包含正确的tenantId参数
+  - [ ] 验证父子商品在同一租户下的数据一致性
+  - [ ] 测试多租户数据隔离机制保持完整
+- [ ] 添加集成测试和E2E测试 (AC: 1-4)
+  - [ ] 创建商品详情页集成测试,验证规格选择功能
+  - [ ] 添加E2E测试验证完整用户流程(选择规格 → 加入购物车/立即购买)
+  - [ ] 测试多租户场景下的规格选择
+  - [ ] 验证所有测试通过
+
+## Dev Notes
+
+### 技术栈信息 [Source: architecture/tech-stack.md]
+- **前端框架**: React 19.1.0 + TypeScript
+- **小程序框架**: Taro(微信小程序)
+- **构建工具**: Vite 7.0.0
+- **状态管理**: @tanstack/react-query (服务端状态)
+- **UI组件库**: shadcn/ui (基于Radix UI)
+- **样式**: Tailwind CSS 4.1.11
+- **HTTP客户端**: 基于Hono Client的封装 + axios适配器
+
+### 源码树信息 [Source: architecture/source-tree.md]
+- **小程序项目位置**: `mini/` - 小程序项目 (Taro + React)
+- **组件位置**: `mini/src/components/goods-spec-selector/index.tsx` - 规格选择器组件(故事5已实现)
+- **页面位置**: `mini/src/pages/goods-detail/index.tsx` - 商品详情页面(已集成组件)
+- **API客户端位置**: `mini/src/api.ts` - API客户端配置
+- **测试位置**: `mini/tests/` - 小程序测试文件
+- **多租户商品模块**: `packages/goods-module-mt/` - 多租户商品管理模块
+
+### 现有组件分析(故事5已完成)
+- **`GoodsSpecSelector`组件状态** [Source: mini/src/components/goods-spec-selector/index.tsx]:
+  - 已实现真实API调用,替换模拟数据
+  - Props: `parentGoodsId`(父商品ID)、`visible`、`onClose`、`onConfirm`、`currentSpec`、`currentQuantity`
+  - API调用: `GET /api/v1/goods/{id}/children` 获取子商品列表
+  - 支持加载状态、错误处理、空状态显示
+  - 规格选择实际选择子商品ID,而不仅仅是规格名称
+- **`GoodsDetailPage`集成状态** [Source: mini/src/pages/goods-detail/index.tsx]:
+  - 组件已取消注释并集成(第11行导入,第496-503行使用)
+  - 规格选择状态管理已实现(`selectedSpec`、`showSpecModal`)
+  - "加入购物车"逻辑已支持规格选择(第249-283行)
+  - "立即购买"逻辑已支持规格选择(第285-327行)
+  - 规格选择按钮和当前规格显示已实现(第402-418行)
+
+### 数据模型信息 [Source: docs/stories/006.005.parent-child-goods-multi-spec-selector.story.md:69-78]
+- **商品实体字段**:
+  - `id`: number - 商品ID
+  - `spuId`: number - 主商品ID,0表示父商品或单规格商品,>0表示子商品
+  - `spuName`: string | null - 主商品名称
+  - `tenantId`: number - 租户ID,用于多租户数据隔离
+  - `state`: number - 状态(1可用,2不可用)
+  - `name`: string - 商品名称(子商品名称作为规格名称)
+  - `price`: number - 商品价格
+  - `stock`: number - 商品库存
+- **父子商品关系**: 通过spuId字段建立父子关系,子商品的spuId指向父商品的id
+
+### API信息 [Source: docs/prd/epic-006-parent-child-goods-multi-spec-support.md:79]
+- **获取子商品列表API**: `GET /api/v1/goods/{id}/children` - 已实现,返回指定父商品的子商品列表
+- **API响应格式**: 返回子商品数组,每个子商品包含id、name、price、stock等字段
+- **API路由位置**: `packages/goods-module-mt/src/routes/public-goods-children.mt.ts`
+- **多租户路由聚合**: `public-goods-aggregated.mt.ts` 聚合基础CRUD路由和子商品列表路由(故事5已创建)
+
+### 多租户支持要求 [Source: docs/stories/006.005.parent-child-goods-multi-spec-selector.story.md:96-99]
+- 保持多租户支持完整,所有查询必须包含tenantId过滤
+- 父子商品必须在同一租户下
+- API调用需要传递正确的租户上下文
+- 数据权限机制保持完整
+
+### RPC客户端架构 [Source: architecture/coding-standards.md:28-33]
+- 使用单例模式的客户端管理器确保全局唯一的客户端实例
+- 组件中应使用`clientManager.get().api.$method`调用API
+- 类型安全:使用Hono的InferRequestType和InferResponseType确保类型一致性
+- 测试Mock:在测试中正确mock客户端管理器的get()方法调用链
+
+### 文件位置
+- **商品详情页面**: `mini/src/pages/goods-detail/index.tsx`
+- **规格选择器组件**: `mini/src/components/goods-spec-selector/index.tsx`
+- **API客户端**: `mini/src/api.ts`
+- **测试文件**:
+  - `mini/tests/pages/goods-detail.test.tsx`(需要创建/更新)
+  - `mini/tests/components/goods-spec-selector.test.tsx`(已存在,故事5创建)
+- **多租户商品API路由**: `packages/goods-module-mt/src/routes/public-goods-aggregated.mt.ts`
+
+### 编码标准 [Source: architecture/coding-standards.md]
+- **测试框架**: Vitest + Testing Library
+- **测试位置**: `tests`文件夹与源码并列(例如:`mini/tests/pages/`)
+- **覆盖率目标**: 核心业务逻辑 > 80%
+- **测试类型**: 单元测试、集成测试、E2E测试
+
+### Testing
+- **测试框架**: Vitest + Testing Library + Playwright (E2E)
+- **测试文件位置**:
+  - 页面集成测试: `mini/tests/pages/goods-detail.test.tsx`(需要创建)
+  - 组件单元测试: `mini/tests/components/goods-spec-selector.test.tsx`(已存在)
+  - E2E测试: `web/tests/e2e/goods-detail.e2e.test.ts`(如果适用)
+- **测试标准**:
+  - 组件集成测试:验证商品详情页正确集成规格选择器
+  - 功能测试:验证"加入购物车"和"立即购买"的规格选择支持
+  - 数据正确性测试:验证选择规格后使用正确的子商品价格和库存
+  - 多租户测试:验证API调用包含正确的tenantId参数
+  - 向后兼容性测试:验证无规格商品保持现有行为
+- **测试模式**:
+  - 使用`vi.mock()` mock API客户端和依赖
+  - 使用`render`函数渲染页面组件
+  - 使用`fireEvent`模拟用户交互(选择规格、点击按钮等)
+  - 验证状态变化、回调调用和API调用参数
+- **具体测试要求**:
+  - 测试规格选择器正确显示子商品作为规格选项
+  - 测试选择规格后,"加入购物车"使用子商品信息
+  - 测试选择规格后,"立即购买"使用子商品信息
+  - 测试多租户参数正确传递到API调用
+  - 测试无规格商品时使用父商品信息(向后兼容性)
+  - 测试库存限制基于子商品库存
+  - 测试价格计算正确性(总价 = 单价 × 数量)
+
+### 项目结构注意事项
+- 保持多租户支持完整,所有API调用必须包含tenantId参数
+- 验证父子商品数据一致性(同一租户)
+- 保持API向后兼容性,不影响现有功能
+- 组件修改保持现有接口兼容性(尽可能)
+- 确保路由聚合正确,子商品API可正常访问
+
+## Change Log
+| Date | Version | Description | Author |
+|------|---------|-------------|--------|
+| 2025-12-12 | 1.0 | 初始故事创建 | Bob (Scrum Master) |
+
+## Dev Agent Record
+
+### Agent Model Used
+
+### Debug Log References
+
+### Completion Notes List
+
+### File List
+
+## QA Results