Prechádzať zdrojové kódy

docs(story): 创建故事017.002 - 登录与首页实现

- 实现人才用户登录功能(身份证号/残疾证号密码登录)
- 实现首页/个人主页页面(个人概览、打卡状态、功能入口、通知列表)
- 完善认证状态管理(AuthContext、token存储和验证)
- 集成人才用户认证API和个人信息查询API
- 参照yongren系列UI包的实现模式
- 遵循UI包开发规范(Jest测试框架、相对路径导入、RPC类型推断)

验收标准:
- 登录页面功能完整,支持身份证号/残疾证号密码登录
- 认证状态管理正常,token存储和验证可靠
- 首页展示个人概览数据、打卡状态、功能入口、通知列表
- 页面设计符合原型标准,已集成Navbar导航栏组件
- 与基础框架(故事017.001)无缝集成

状态: Approved
清晰度评分: 9/10

🤖 Generated with [Claude Code](https://claude.com/claude-code)
via [Happy](https://happy.engineering)

Co-Authored-By: Claude <noreply@anthropic.com>
Co-Authored-By: Happy <yesreply@happy.engineering>
yourname 3 týždňov pred
rodič
commit
7ed32239b7
1 zmenil súbory, kde vykonal 797 pridanie a 0 odobranie
  1. 797 0
      docs/stories/017.002.story.md

+ 797 - 0
docs/stories/017.002.story.md

@@ -0,0 +1,797 @@
+# 故事017.002: 登录与首页实现
+
+## 元信息
+- **史诗**: 017 - 人才小程序功能实现
+- **优先级**: P1 - 核心功能
+- **状态**: Approved
+- **创建日期**: 2025-12-25
+- **负责人**: 开发团队
+
+## 故事描述
+
+**作为** 人才小程序开发者,
+**我想要** 实现人才用户登录功能和首页/个人主页页面,
+**以便** 人才用户能够通过身份证号/残疾证号密码登录系统,并查看个人概览、打卡状态、快捷功能入口和最新通知。
+
+### 背景
+
+**现有系统状态:**
+- 故事017.001已完成rencai mini ui包基础框架搭建
+- rencai系列7个UI包基础目录结构已创建
+- API客户端文件已创建(`src/api/index.ts`)
+- mini-talent项目路由结构已配置完成
+- 基础布局组件(TabBarLayout、Navbar等)已就绪
+- 人才用户认证框架已建立
+
+**原型设计参考:**
+- `docs/小程序原型/rencai.html` 提供了登录页和首页的完整原型设计
+- 登录页 (原型行115-158): 身份证号/残疾证号输入框、密码输入框、登录按钮
+- 首页/个人主页 (原型行160-301): 个人概览、打卡模块、功能入口网格、最新通知列表
+
+**技术集成模式:**
+- 参照`yongren-auth-ui`和`yongren-dashboard-ui`的实现模式
+- 登录页面使用`@d8d/rencai-auth-ui`包,集成人才用户认证API
+- 首页使用`@d8d/rencai-dashboard-ui`包,集成个人信息查询API
+- API调用逻辑封装在页面组件内部
+- 认证状态通过AuthContext管理,token存储和验证
+
+**依赖API (史诗015):**
+- ✅ 人才用户认证API (POST /api/v1/rencai/auth/login) - 已就绪
+- ✅ 个人信息查询API (GET /api/v1/rencai/personal/info) - 已就绪
+- ⚠️ 打卡记录API - P2延期功能,当前使用前端模拟数据
+
+## 验收标准
+
+### 登录功能
+- [ ] 登录页面功能完整,支持人才用户身份证号/残疾证号密码登录
+- [ ] 登录表单验证正常,包括输入验证和错误提示
+- [ ] 登录成功后跳转到首页,并保存token到本地存储
+- [ ] 登录失败时显示错误提示信息
+
+### 认证状态管理
+- [ ] 登录状态管理正常,token存储和验证可靠
+- [ ] AuthContext提供登录状态、用户信息和登出方法
+- [ ] token过期时自动跳转到登录页
+
+### 首页/个人主页
+- [ ] 首页展示个人概览数据(姓名、残疾类型、出勤统计、本月薪资)
+- [ ] 打卡状态显示正常(已打卡/未打卡,上班/下班打卡时间)- 使用前端模拟数据
+- [ ] 快捷功能入口可用(个人信息、考勤记录、薪资查询、企业信息)
+- [ ] 最新通知列表显示正常 - 使用前端模拟数据
+
+### 页面设计与布局
+- [ ] 页面设计符合原型标准,移动端体验良好
+- [ ] 主页面已集成Navbar导航栏组件(无返回按钮)
+- [ ] TabBarLayout底部导航正常工作(首页、考勤、我的、更多)
+
+### 集成与兼容性
+- [ ] 与基础框架(故事017.001)无缝集成,建立统一的页面层级结构规范
+- [ ] 现有mini-talent项目功能不受影响
+
+## 任务列表
+
+### 任务1: 实现登录页面组件 (AC: 登录功能)
+- [ ] 1.1 在`@d8d/rencai-auth-ui`中实现`LoginPage`页面组件 (`src/pages/LoginPage/LoginPage.tsx`)
+- [ ] 1.2 创建登录表单,包含以下字段:
+  - 身份证号/残疾证号输入框(必填)
+  - 密码输入框(必填,密码类型)
+  - 登录按钮
+  - 忘记密码链接(占位功能)
+- [ ] 1.3 集成人才用户认证API (`talentAuthClient.auth.login.$post`)
+- [ ] 1.4 实现表单验证逻辑(使用Zod schema)
+- [ ] 1.5 实现登录成功处理:
+  - 保存token到本地存储
+  - 保存用户信息到AuthContext
+  - 跳转到首页 (`/pages/index/index`)
+- [ ] 1.6 实现登录失败处理:
+  - 显示错误提示信息
+  - 清空密码输入框
+- [ ] 1.7 更新`@d8d/rencai-auth-ui/package.json`的exports字段,添加LoginPage导出路径:
+  ```json
+  "./pages/LoginPage/LoginPage": {
+    "types": "./dist/src/pages/LoginPage/LoginPage.d.ts",
+    "import": "./dist/src/pages/LoginPage/LoginPage.js",
+    "require": "./dist/src/pages/LoginPage/LoginPage.js"
+  }
+  ```
+
+### 任务2: 完善认证状态管理 (AC: 认证状态管理)
+- [ ] 2.1 完善`@d8d/rencai-auth-ui/src/utils/AuthContext.tsx`:
+  - 添加登录状态(`isLoggedIn`)
+  - 添加用户信息(`userInfo`)
+  - 添加登录方法(`login`)
+  - 添加登出方法(`logout`)
+  - 添加token验证逻辑
+- [ ] 2.2 实现token存储机制(localStorage)
+- [ ] 2.3 实现token过期检测和自动跳转
+- [ ] 2.4 创建`useAuth` hook便于组件使用认证状态
+- [ ] 2.5 更新`@d8d/rencai-auth-ui/src/utils/index.ts`,导出AuthContext和useAuth
+
+### 任务3: 实现首页/个人主页组件 (AC: 首页/个人主页)
+- [ ] 3.1 在`@d8d/rencai-dashboard-ui`中实现`Dashboard`页面组件 (`src/pages/Dashboard/Dashboard.tsx`)
+- [ ] 3.2 创建个人概览卡片组件:
+  - 显示姓名
+  - 显示残疾类型
+  - 显示出勤统计(本月出勤天数)
+  - 显示本月薪资(前端模拟数据)
+- [ ] 3.3 创建打卡状态模块组件:
+  - 显示打卡状态(已打卡/未打卡)
+  - 显示上班打卡时间(如已打卡)
+  - 显示下班打卡时间(如已打卡)
+  - 远程打卡按钮(占位功能,使用前端模拟数据)
+- [ ] 3.4 创建快捷功能入口网格组件:
+  - 个人信息入口(跳转到`/pages/personal-info/index`)
+  - 考勤记录入口(跳转到`/pages/attendance/index`)
+  - 薪资查询入口(跳转到`/pages/employment/index`)
+  - 企业信息入口(占位功能)
+- [ ] 3.5 创建最新通知列表组件:
+  - 显示通知列表(最多5条)
+  - 显示通知标题和时间
+  - 使用前端模拟数据
+- [ ] 3.6 集成个人信息查询API (`talentDashboardClient.personal.info.$get`)
+- [ ] 3.7 使用TabBarLayout包装Dashboard组件(无返回按钮的Navbar)
+- [ ] 3.8 更新`@d8d/rencai-dashboard-ui/package.json`的exports字段,添加Dashboard导出路径:
+  ```json
+  "./pages/Dashboard/Dashboard": {
+    "types": "./dist/src/pages/Dashboard/Dashboard.d.ts",
+    "import": "./dist/src/pages/Dashboard/Dashboard.js",
+    "require": "./dist/src/pages/Dashboard/Dashboard.js"
+  }
+  ```
+
+### 任务4: 更新mini-talent页面集成 (AC: 集成与兼容性)
+- [ ] 4.1 更新`mini-talent/src/pages/login/index.tsx`:
+  - 从`@d8d/rencai-auth-ui/pages/LoginPage/LoginPage`导入LoginPage组件
+  - 用AuthContextProvider包装页面
+  - 导出LoginPage组件
+- [ ] 4.2 更新`mini-talent/src/pages/index/index.tsx`:
+  - 从`@d8d/rencai-dashboard-ui/pages/Dashboard/Dashboard`导入Dashboard组件
+  - 用AuthContextProvider包装页面
+  - 添加认证检查(未登录跳转到登录页)
+  - 导出Dashboard组件
+- [ ] 4.3 验证页面路由配置(已在故事017.001中配置完成)
+- [ ] 4.4 验证底部TabBar导航正常工作
+
+### 任务5: 实现页面样式和移动端适配 (AC: 页面设计与布局)
+- [ ] 5.1 参照原型设计实现登录页面样式(原型行115-158)
+  - 蓝色渐变背景
+  - 圆角白色卡片表单
+  - 表单字段间距和样式
+  - 登录按钮样式
+- [ ] 5.2 参照原型设计实现首页样式(原型行160-301)
+  - 个人概览卡片样式(蓝色渐变背景)
+  - 打卡模块样式(白色卡片,状态图标)
+  - 功能入口网格样式(2x2网格布局)
+  - 通知列表样式(白色卡片,简洁列表)
+- [ ] 5.3 确保页面设计符合移动端规范:
+  - 宽度参考: 375px
+  - 圆角规范: 12px (卡片)、40px (移动框架)
+  - 颜色主题: 蓝色渐变 (#3b82f6 → #1e40af)
+  - 字体规范: 标题18-24px, 正文14px, 小字12px
+- [ ] 5.4 使用Navbar组件确保统一的页面层级结构:
+  - 登录页:不使用Navbar(全屏页面)
+  - 首页:使用TabBarLayout(无返回按钮)
+
+### 任务6: 编写测试 (AC: 集成与兼容性)
+- [ ] 6.1 为LoginPage编写组件测试 (`tests/pages/LoginPage/LoginPage.test.tsx`)
+  - 测试表单渲染
+  - 测试表单验证
+  - 测试登录成功流程(Mock API响应)
+  - 测试登录失败处理(Mock API错误响应)
+- [ ] 6.2 为Dashboard编写组件测试 (`tests/pages/Dashboard/Dashboard.test.tsx`)
+  - 测试组件渲染
+  - 测试个人概览数据展示(Mock API响应)
+  - 测试打卡状态显示
+  - 测试功能入口导航
+- [ ] 6.3 为AuthContext编写hook测试 (`tests/utils/AuthContext.test.tsx`)
+  - 测试登录状态管理
+  - 测试token存储
+  - 测试登出功能
+- [ ] 6.4 编写集成测试验证现有功能不受影响
+- [ ] 6.5 运行`pnpm typecheck`确保类型检查通过
+
+## 开发者笔记
+
+### 前置故事见解
+
+**故事017.001完成状态:**
+- ✅ rencai系列7个UI包基础结构已创建
+- ✅ API客户端文件已创建(`src/api/index.ts`)
+- ✅ mini-talent项目路由结构已配置完成
+- ✅ 基础布局组件(TabBarLayout、Navbar等)已就绪
+- ✅ 人才用户认证框架已建立(AuthContext基础框架)
+- ✅ package.json的exports字段已配置
+- ✅ Jest测试框架已配置(`jest.config.cjs`)
+
+**故事017.001关键实现细节:**
+1. API客户端模式:使用简单的RPC客户端导出模式,参照`yongren-statistics-ui`
+2. UI包内部导入规范:必须使用相对路径,不要使用别名
+3. 类型推断规范:必须使用RPC推断类型,避免直接导入schema类型
+4. 页面组件导出:通过package.json的exports字段配置完整路径
+5. 测试框架选择:**mini项目使用Jest**(不是Vitest)
+
+**实现经验总结:**
+- API客户端导入路径修正:从相应的后端模块包导入,而不是`@d8d/server`
+- 页面文件简化设计:采用"薄包装层",仅导入并导出组件
+- 复用现有共享组件:StatusBar、PageContainer、Navbar使用`@d8d/mini-shared-ui-components`中的实现
+
+### 技术栈要求
+
+**来源**: [architecture/tech-stack.md](../architecture/tech-stack.md)
+
+**运行时和框架:**
+- **Node.js**: 20.18.3
+- **Hono**: 4.8.5 (RPC客户端)
+- **React**: 19.1.0 (UI组件)
+- **Taro**: 4.1.4 (小程序框架)
+- **Tailwind CSS**: 4.1.11 (样式)
+
+**测试框架:**
+- **Jest**: 30.2.0 (mini项目使用Jest,不是Vitest!)
+- **ts-jest**: 29.4.5 (TypeScript预处理器)
+- **@testing-library/react**: 16.3.0 (React组件测试)
+- **@d8d/mini-testing-utils**: workspace包 (Taro小程序测试工具)
+
+**重要**: **mini项目使用Jest测试框架**,与web应用使用的Vitest不同。
+
+### UI包开发规范
+
+**来源**: [architecture/ui-package-standards.md](../architecture/ui-package-standards.md)
+
+**关键规范要求:**
+
+#### 1. UI包内部导入规范
+**重要**: UI包内部导入必须使用相对路径,不要使用别名。
+
+**正确示例**:
+```typescript
+// ✅ 正确: 使用相对路径导入同一包内的模块
+import { talentAuthClient } from '../../api'
+import { SomeComponent } from '../components/SomeComponent'
+```
+
+**错误示例**:
+```typescript
+// ❌ 错误: 不要使用别名导入UI包内部的模块
+import { talentAuthClient } from '@/api'
+import { SomeComponent } from '@/pages/Dashboard/components'
+```
+
+#### 2. package.json exports配置规范
+```json
+{
+  "exports": {
+    ".": {
+      "types": "./dist/src/index.d.ts",
+      "import": "./dist/src/index.js",
+      "require": "./dist/src/index.js"
+    },
+    "./api": {
+      "types": "./src/api/index.ts",
+      "import": "./src/api/index.ts",
+      "require": "./src/api/index.ts"
+    },
+    "./pages/<PageName>/<PageName>": {
+      "types": "./dist/src/pages/<PageName>/<PageName>.d.ts",
+      "import": "./dist/src/pages/<PageName>/<PageName>.js",
+      "require": "./dist/src/pages/<PageName>/<PageName>.js"
+    }
+  }
+}
+```
+
+#### 3. RPC客户端实现规范
+使用简单的RPC客户端导出模式 (参照`yongren-statistics-ui/src/api/enterpriseStatisticsClient.ts`):
+
+```typescript
+// src/api/talentAuthClient.ts
+import type { rencaiAuthRoutes } from '@d8d/core-module/auth-module'
+import { rpcClient } from '@d8d/mini-shared-ui-components/utils/rpc/rpc-client'
+
+// 人才小程序API路径前缀: /api/v1/rencai
+export const talentAuthClient = rpcClient<typeof rencaiAuthRoutes>('/api/v1/rencai')
+```
+
+**src/api/index.ts导出**:
+```typescript
+export { talentAuthClient } from './talentAuthClient'
+```
+
+#### 4. 类型定义规范
+**必须使用RPC推断类型**,避免直接导入schema类型:
+
+```typescript
+// ✅ 正确:使用RPC推断类型 + 相对路径导入
+import type { InferResponseType } from 'hono'
+import { talentAuthClient } from '../api/talentAuthClient'
+
+export type LoginResponse = InferResponseType<typeof talentAuthClient.auth.login.$post, 200>
+
+// ❌ 错误:直接导入schema类型 (可能导致Date/string不匹配)
+import type { LoginSchema } from '@d8d/core-module/auth-module/schemas'
+```
+
+#### 5. Jest配置规范
+每个UI包必须创建`jest.config.cjs`配置文件,参照`yongren-dashboard-ui/jest.config.cjs`:
+
+```javascript
+module.exports = {
+  preset: 'ts-jest',
+  testEnvironment: 'jsdom',
+  setupFilesAfterEnv: ['@d8d/mini-testing-utils/setup'],
+  moduleNameMapper: {
+    '^@/(.*)$': '<rootDir>/src/$1',
+    '^~/(.*)$': '<rootDir>/tests/$1',
+    '^@tarojs/taro$': '@d8d/mini-testing-utils/testing/taro-api-mock.ts',
+    '\\.(css|less|scss|sass)$': '@d8d/mini-testing-utils/testing/style-mock.js',
+    '\\.(jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$':
+      '@d8d/mini-testing-utils/testing/file-mock.js'
+  },
+  testMatch: [
+    '<rootDir>/tests/**/*.spec.{ts,tsx}',
+    '<rootDir>/tests/**/*.test.{ts,tsx}'
+  ],
+  collectCoverageFrom: [
+    'src/**/*.{ts,tsx}',
+    '!src/**/*.d.ts',
+    '!src/**/index.{ts,tsx}',
+    '!src/**/*.stories.{ts,tsx}'
+  ],
+  coverageDirectory: 'coverage',
+  coverageReporters: ['text', 'lcov', 'html'],
+  testPathIgnorePatterns: [
+    '/node_modules/',
+    '/dist/',
+    '/coverage/'
+  ],
+  transform: {
+    '^.+\\.(ts|tsx)$': 'ts-jest',
+    '^.+\\.(js|jsx)$': 'babel-jest'
+  },
+  transformIgnorePatterns: [
+    '/node_modules/(?!(swiper|@tarojs)/)'
+  ],
+  moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx', 'json']
+}
+```
+
+**关键配置说明**:
+- `setupFilesAfterEnv`: 使用`@d8d/mini-testing-utils/setup`进行测试环境初始化
+- `moduleNameMapper`:
+  - `^@/(.*)$` 和 `^~/(.*)$`: 仅用于测试文件的路径映射,不在源代码中使用
+  - `^@tarojs/taro$`: 映射Taro API到mock
+  - 样式和文件映射: 将样式文件和静态文件映射到mock
+- **重要**: 源代码中不使用`@/`或`~/`别名,只使用相对路径
+- `testMatch`: 支持`.spec.{ts,tsx}`和`.test.{ts,tsx}`两种测试文件格式
+
+### 史诗015 API集成要点
+
+**来源**: [docs/prd/epic-015-talent-mini-program-api-support.md](../prd/epic-015-talent-mini-program-api-support.md)
+
+**API路径约定:**
+所有人才小程序API使用 `api/v1/rencai` 前缀:
+- 人才用户登录: `POST /api/v1/rencai/auth/login`
+- 人才个人信息: `GET /api/v1/rencai/personal/info`
+
+**已完成的API (可直接集成):**
+- ✅ 人才用户认证API (故事015-02)
+  - 路由定义: `@d8d/core-module/auth-module/src/routes/mini-login.route.ts`
+  - RPC路由: `rencaiAuthRoutes.auth.login`
+  - 请求格式: `{ idCard: string, password: string }`
+  - 响应格式: `{ token: string, user: UserInfo }`
+- ✅ 个人信息查询API (故事015-03)
+  - 路由定义: `@d8d/allin-disability-module/src/routes/talent-personal-info.route.ts`
+  - RPC路由: `talentPersonalInfoRoutes.personal.info`
+  - 响应格式: `{ name: string, disabilityType: string, ... }`
+
+**只读设计原则:**
+遵循与用人方小程序相同的设计原则,人才小程序API以查询功能为主。
+
+### 原型设计参考
+
+**来源**: [docs/小程序原型/rencai.html](../小程序原型/rencai.html)
+
+**登录页** (原型行115-158):
+```html
+<!-- 关键样式 -->
+<div style="background: linear-gradient(135deg, #3b82f6 0%, #1e40af 100%);">
+  <div style="background: white; border-radius: 12px;">
+    <!-- 表单字段 -->
+    <input type="text" placeholder="身份证号/残疾证号" />
+    <input type="password" placeholder="密码" />
+    <button>登录</button>
+  </div>
+</div>
+```
+
+**首页/个人主页** (原型行160-301):
+```html
+<!-- 个人概览卡片 -->
+<div style="background: linear-gradient(135deg, #3b82f6 0%, #1e40af 100%); border-radius: 12px;">
+  <div>姓名: 张三</div>
+  <div>残疾类型: 肢体残疾</div>
+  <div>出勤: 20天</div>
+  <div>薪资: 3000元</div>
+</div>
+
+<!-- 打卡模块 -->
+<div style="background: white; border-radius: 12px;">
+  <div>打卡状态: 已打卡</div>
+  <div>上班: 08:30</div>
+  <div>下班: 18:00</div>
+  <button>远程打卡</button>
+</div>
+
+<!-- 功能入口网格 -->
+<div style="display: grid; grid-template-columns: 1fr 1fr;">
+  <div>个人信息</div>
+  <div>考勤记录</div>
+  <div>薪资查询</div>
+  <div>企业信息</div>
+</div>
+
+<!-- 通知列表 -->
+<div style="background: white; border-radius: 12px;">
+  <div>通知标题1 - 时间</div>
+  <div>通知标题2 - 时间</div>
+  ...
+</div>
+```
+
+**移动端设计规范:**
+- 宽度参考: 375px
+- 状态栏高度: 44px
+- 底部导航高度: 60px
+- 圆角规范: 12px (卡片)、40px (移动框架)
+- 颜色主题: 蓝色渐变 (#3b82f6 → #1e40af)
+- 字体规范: 标题18-24px, 正文14px, 小字12px
+
+### 项目结构指南
+
+**来源**: [architecture/source-tree.md](../architecture/source-tree.md)
+
+**mini-talent项目结构:**
+```
+mini-talent/                   # 人才小程序项目
+├── src/
+│   ├── app.tsx                # 小程序入口
+│   ├── app.config.ts          # 小程序配置 (已在故事017.001中更新)
+│   ├── app.css                # 全局样式
+│   ├── pages/                 # 页面目录
+│   │   ├── login/             # 登录页 (从UI包导入)
+│   │   │   └── index.tsx
+│   │   └── index/             # 首页/个人主页 (从UI包导入)
+│   │       └── index.tsx
+├── package.json
+├── jest.config.js             # Jest配置
+└── tsconfig.json
+```
+
+**mini-ui-packages目录结构:**
+```
+mini-ui-packages/
+├── rencai-auth-ui/            # 人才认证UI包
+│   ├── src/
+│   │   ├── api/
+│   │   │   ├── talentAuthClient.ts
+│   │   │   └── index.ts
+│   │   ├── pages/
+│   │   │   └── LoginPage/
+│   │   │       ├── LoginPage.tsx
+│   │   │       └── index.ts (可选)
+│   │   ├── utils/
+│   │   │   ├── AuthContext.tsx
+│   │   │   └── index.ts
+│   │   └── index.ts
+│   ├── package.json           # 包含exports配置
+│   ├── jest.config.cjs
+│   └── tsconfig.json
+├── rencai-dashboard-ui/       # 人才仪表板UI包
+│   ├── src/
+│   │   ├── api/
+│   │   │   ├── talentDashboardClient.ts
+│   │   │   └── index.ts
+│   │   ├── pages/
+│   │   │   └── Dashboard/
+│   │   │       ├── Dashboard.tsx
+│   │   │       └── index.ts (可选)
+│   │   ├── components/       # UI组件
+│   │   │   ├── PersonalOverviewCard.tsx
+│   │   │   ├── ClockInStatus.tsx
+│   │   │   ├── QuickActionsGrid.tsx
+│   │   │   └── NotificationList.tsx
+│   │   └── index.ts
+│   ├── package.json           # 包含exports配置
+│   ├── jest.config.cjs
+│   └── tsconfig.json
+└── mini-shared-ui-components/ # 通用小程序UI组件
+    ├── src/
+    │   └── components/
+    │       ├── status-bar.tsx
+    │       ├── page-container.tsx
+    │       ├── navbar.tsx
+    │       └── tab-bar.tsx
+    └── ...
+```
+
+**mini-talent页面导入方式:**
+```typescript
+// mini-talent/src/pages/login/index.tsx
+import LoginPage from '@d8d/rencai-auth-ui/pages/LoginPage/LoginPage'
+import { AuthContextProvider } from '@d8d/rencai-auth-ui/utils'
+
+export default function Login() {
+  return (
+    <AuthContextProvider>
+      <LoginPage />
+    </AuthContextProvider>
+  )
+}
+
+// mini-talent/src/pages/index/index.tsx
+import Dashboard from '@d8d/rencai-dashboard-ui/pages/Dashboard/Dashboard'
+import { AuthContextProvider, useAuth } from '@d8d/rencai-auth-ui/utils'
+
+function IndexPage() {
+  const { isLoggedIn } = useAuth()
+
+  // 未登录跳转到登录页
+  if (!isLoggedIn) {
+    Taro.navigateTo({ url: '/pages/login/index' })
+    return null
+  }
+
+  return <Dashboard />
+}
+
+export default function Index() {
+  return (
+    <AuthContextProvider>
+      <IndexPage />
+    </AuthContextProvider>
+  )
+}
+```
+
+### 测试策略
+
+**来源**: [architecture/testing-strategy.md](../architecture/testing-strategy.md)
+
+**测试框架选择:**
+- **mini项目使用Jest** (不是Vitest!)
+- 测试命令: `pnpm test` (在UI包目录下运行)
+- 测试配置: `jest.config.cjs`
+
+**测试文件位置:**
+```
+mini-ui-packages/<package-name>/
+└── tests/
+    ├── unit/                  # 单元测试
+    │   ├── utils/
+    │   │   └── AuthContext.test.tsx
+    │   └── components/
+    │       └── *.test.tsx
+    └── pages/                 # 页面组件测试
+        ├── LoginPage/
+        │   └── LoginPage.test.tsx
+        └── Dashboard/
+            └── Dashboard.test.tsx
+```
+
+**测试要求:**
+1. 为每个页面组件编写Jest测试
+2. 测试API客户端的集成(使用Mock响应)
+3. 测试认证状态管理
+4. 验证mini-talent项目现有功能不受影响
+5. 运行`pnpm typecheck`确保类型检查通过
+
+**Mock响应工具函数:**
+```typescript
+const createMockResponse = (status: number, data?: any) => ({
+  status,
+  ok: status >= 200 && status < 300,
+  body: null,
+  bodyUsed: false,
+  statusText: status === 200 ? 'OK' : 'Error',
+  headers: new Headers(),
+  url: '',
+  redirected: false,
+  type: 'basic' as ResponseType,
+  json: async () => data || {},
+  text: async () => '',
+  blob: async () => new Blob(),
+  arrayBuffer: async () => new ArrayBuffer(0),
+  formData: async () => new FormData(),
+  clone: function() { return this; }
+})
+```
+
+### 编码标准
+
+**来源**: [architecture/coding-standards.md](../architecture/coding-standards.md)
+
+**关键编码规范:**
+
+#### 1. 必须遵循UI包开发规范
+开发UI包时,**必须**参考并遵循[UI包开发规范](../architecture/ui-package-standards.md),该规范基于史诗008(AllIn UI模块移植)的经验总结。
+
+#### 2. 关键检查点 (基于史诗008经验)
+- **API路径映射验证**: 开发前必须验证故事中的API路径映射与实际后端路由定义的一致性
+- **类型推断最佳实践**: 必须使用RPC推断类型,而不是直接导入schema类型
+- **测试选择器优化**: 必须为关键交互元素添加`data-testid`属性
+- **表单组件模式**: 必须使用条件渲染两个独立的Form组件
+- **API调用一致性**: 必须根据实际路由名称修正API调用
+
+#### 3. 常见错误避免
+- ❌ 不要直接导入schema类型 (可能导致Date/string类型不匹配)
+- ❌ 不要使用`getByText()`查找可能重复的文本元素
+- ❌ 不要在单个Form组件上动态切换props
+- ❌ 不要使用故事中描述但实际不存在的路由名称
+- ❌ **不要在UI包内部导入中使用别名** (`@/`, `~/`等),必须使用相对路径
+
+**路径使用示例**:
+```typescript
+// ✅ 正确: UI包内部使用相对路径
+import { apiClient } from '../../api'
+import { MyComponent } from '../components'
+
+// ✅ 正确: 跨包导入使用workspace包名
+import { SharedComponent } from '@d8d/mini-shared-ui-components'
+
+// ❌ 错误: UI包内部使用别名
+import { apiClient } from '@/api'
+import { MyComponent } from '~/pages/Dashboard/components'
+```
+
+#### 4. 参考实现
+- **用人方认证UI包**: `mini-ui-packages/yongren-auth-ui`
+  - API客户端: `src/api/enterpriseAuthClient.ts`
+  - 认证上下文: 参考其AuthContext实现模式
+  - 主入口文件: `src/index.ts`
+
+- **用人方仪表板UI包**: `mini-ui-packages/yongren-dashboard-ui`
+  - 组件: `src/pages/Dashboard/Dashboard.tsx`
+  - package.json exports配置
+  - 目录结构参考
+  - Jest配置: `jest.config.cjs`
+
+### 技术约束
+
+1. **向后兼容**: 不影响现有mini-talent项目功能
+2. **类型安全**: 使用TypeScript严格模式,所有API调用必须有类型定义
+3. **模块独立性**: 每个UI包独立管理自己的API客户端和类型定义
+4. **测试覆盖**: 所有新增代码必须有测试覆盖
+5. **代码规范**: 遵循项目编码标准和UI包开发规范
+
+### 风险和缓解措施
+
+**主要风险:**
+1. **API集成风险**: 史诗015的API可能存在接口变更
+2. **认证状态管理风险**: token过期和自动跳转逻辑可能复杂
+3. **UI组件复用风险**: rencai系列UI包可能与现有yongren系列UI包存在差异
+
+**缓解措施:**
+1. **API验证**: 开发前验证API路径映射与实际后端路由定义的一致性
+2. **分阶段实现**: 先实现登录功能,再实现首页功能
+3. **参考现有模式**: 参照yongren系列UI包的实现模式和架构
+4. **类型安全**: 使用RPC推断类型,避免类型不匹配问题
+5. **测试驱动**: 编写完整的测试,确保功能正确性
+
+## 测试
+
+### 测试框架和模式
+
+**来源**: [architecture/testing-strategy.md](../architecture/testing-strategy.md)
+
+**测试框架:**
+- **Jest**: 30.2.0 (mini项目使用Jest,不是Vitest!)
+- **ts-jest**: 29.4.5 (TypeScript预处理器)
+- **@testing-library/react**: 16.3.0 (React组件测试)
+- **@d8d/mini-testing-utils**: workspace包 (Taro小程序测试工具)
+
+**测试文件位置:**
+```
+mini-ui-packages/<package-name>/
+└── tests/
+    ├── unit/                      # 单元测试
+    │   ├── utils/
+    │   │   └── AuthContext.test.tsx
+    │   └── components/
+    │       ├── PersonalOverviewCard.test.tsx
+    │       ├── ClockInStatus.test.tsx
+    │       ├── QuickActionsGrid.test.tsx
+    │       └── NotificationList.test.tsx
+    └── pages/                     # 页面组件测试
+        ├── LoginPage/
+        │   └── LoginPage.test.tsx
+        └── Dashboard/
+            └── Dashboard.test.tsx
+```
+
+### 测试要求
+
+1. **组件测试**:
+   - 测试组件渲染正确
+   - 测试用户交互(点击、输入等)
+   - 测试API集成(使用Mock响应)
+   - 测试错误处理
+
+2. **Hook测试**:
+   - 测试AuthContext的状态管理
+   - 测试token存储和验证
+   - 测试登录和登出功能
+
+3. **集成测试**:
+   - 测试登录成功后跳转到首页
+   - 测试未登录时跳转到登录页
+   - 测试token过期处理
+
+4. **回归测试**:
+   - 验证mini-talent项目现有功能不受影响
+   - 运行`pnpm typecheck`确保类型检查通过
+
+### Mock响应工具函数
+
+```typescript
+const createMockResponse = (status: number, data?: any) => ({
+  status,
+  ok: status >= 200 && status < 300,
+  body: null,
+  bodyUsed: false,
+  statusText: status === 200 ? 'OK' : status === 201 ? 'Created' : status === 204 ? 'No Content' : 'Error',
+  headers: new Headers(),
+  url: '',
+  redirected: false,
+  type: 'basic' as ResponseType,
+  json: async () => data || {},
+  text: async () => '',
+  blob: async () => new Blob(),
+  arrayBuffer: async () => new ArrayBuffer(0),
+  formData: async () => new FormData(),
+  clone: function() { return this; }
+})
+```
+
+### 测试执行
+
+```bash
+# 运行所有测试
+cd mini-ui-packages/rencai-auth-ui && pnpm test
+cd mini-ui-packages/rencai-dashboard-ui && pnpm test
+
+# 运行特定测试
+pnpm test --testNamePattern="LoginPage"
+
+# 生成覆盖率报告
+pnpm test:coverage
+```
+
+## 变更日志
+
+| 日期 | 版本 | 描述 | 作者 |
+|------|------|------|------|
+| 2025-12-25 | 1.0 | 创建故事文档 | Bob (Scrum Master) |
+
+## 开发者记录
+
+*此部分由开发代理在实施过程中填写*
+
+### 使用的代理模型
+
+待填写
+
+### 调试日志引用
+
+待填写
+
+### 完成说明列表
+
+待填写
+
+### 文件列表
+
+待填写
+
+## QA结果
+
+*此部分由QA代理在审查完成后填写*