新增:管理后台前端页面,以及openspec内容。

This commit is contained in:
2025-12-24 20:15:28 +08:00
parent 6d7ed38105
commit 845f1c6618
64 changed files with 9017 additions and 6 deletions

View File

@@ -0,0 +1,377 @@
# 后台管理Admin API + 前端)规范
## Purpose
本能力描述"后台管理侧"的完整实现:包括后端接口和前端管理网站。
- **后端接口**:用于查询用户与会话记录,供管理端页面/外部系统使用,以 `AdminController` 提供的 API 为主。
- **前端管理网站**:基于 Vue 3.x + Element Plus + Vite 构建的管理员操作界面,用于系统配置、数据管理和监控。
## Requirements
### Requirement: 前端技术架构
前端管理网站 SHALL 使用以下技术栈开发:
- **前端框架**Vue 3.x使用 Composition API 和 `<script setup>` 语法)
- **UI组件库**Element Plus
- **路由管理**vue-router
- **状态管理**Pinia用于深色/浅色模式状态)
- **HTTP客户端**axios
- **构建工具**Vite
- **开发环境**Node.js
- **后端服务**:与微信小程序共用 ASP.NET Core MVC 后端AdminController
#### Scenario: 前端项目初始化
- **WHEN** 创建前端项目
- **THEN** 使用 Vite 初始化 Vue 3.x 项目
- **AND** 安装 Element Plus、vue-router、Pinia、axios 等依赖
- **AND** 配置项目目录结构src/router、src/components、src/store、src/styles
### Requirement: 用户认证(前端验证)
前端管理网站 SHALL 提供管理员登录功能,使用前端验证方式。
#### Scenario: 管理员登录
- **WHEN** 管理员访问登录页面
- **THEN** 显示登录表单(用户名、密码字段)
- **AND** 输入正确的用户名和密码(均为 Admin后能够成功登录
- **AND** 登录成功后跳转到首页
- **AND** 登录状态保存在 Pinia store 中
#### Scenario: 登录验证失败
- **WHEN** 管理员输入错误的用户名或密码
- **THEN** 显示错误提示信息
- **AND** 不允许进入系统
#### Scenario: 未登录访问保护页面
- **WHEN** 未登录用户访问受保护页面
- **THEN** 自动重定向到登录页面
### Requirement: 响应式设计
前端管理网站 SHALL 支持响应式设计,优先适配手机宽度。
#### Scenario: 手机端访问
- **WHEN** 管理员使用手机(宽度 < 768px访问后台管理网站
- **THEN** 网站布局能够自动调整,完美适配手机屏幕
- **AND** 菜单采用折叠或抽屉式布局
- **AND** 表格支持横向滚动
#### Scenario: 平板端访问
- **WHEN** 管理员使用平板768px <= 宽度 < 1024px访问后台管理网站
- **THEN** 网站布局能够自动调整,适配平板屏幕
- **AND** 菜单采用侧边栏布局
#### Scenario: 桌面端访问
- **WHEN** 管理员使用桌面(宽度 >= 1024px访问后台管理网站
- **THEN** 网站布局能够自动调整,适配桌面屏幕
- **AND** 菜单采用侧边栏布局
### Requirement: 深色/浅色模式切换
前端管理网站 SHALL 支持深色模式和浅色模式的自由切换。
#### Scenario: 切换到深色模式
- **WHEN** 管理员点击深色模式切换按钮
- **THEN** 网站切换到深色主题
- **AND** 所有组件和页面使用深色配色
- **AND** 主题状态保存在 Pinia store 中
#### Scenario: 切换到浅色模式
- **WHEN** 管理员点击浅色模式切换按钮
- **THEN** 网站切换到浅色主题
- **AND** 所有组件和页面使用浅色配色
- **AND** 主题状态保存在 Pinia store 中
#### Scenario: 主题状态持久化
- **WHEN** 管理员刷新页面
- **THEN** 网站保持上次选择的主题模式
### Requirement: 路由管理
前端管理网站 SHALL 使用 vue-router 管理多页面路由。
#### Scenario: 路由配置
- **WHEN** 配置路由
- **THEN** 定义登录页、首页、会话记录管理页、用户管理页等路由
- **AND** 设置路由守卫,未登录用户重定向到登录页
#### Scenario: 菜单导航
- **WHEN** 管理员点击菜单项
- **THEN** 路由跳转到对应页面
- **AND** 菜单高亮显示当前页面
### Requirement: 菜单布局
前端管理网站 SHALL 使用合理的菜单布局来控制多页面。
#### Scenario: 侧边栏菜单
- **WHEN** 管理员在桌面或平板端访问
- **THEN** 显示侧边栏菜单
- **AND** 菜单包含:首页、会话记录管理、用户管理等功能入口
#### Scenario: 抽屉式菜单
- **WHEN** 管理员在手机端访问
- **THEN** 显示抽屉式菜单(点击菜单按钮展开)
- **AND** 菜单包含:首页、会话记录管理、用户管理等功能入口
### Requirement: 模块化设计
前端管理网站 SHALL 采用模块化设计思路。
#### Scenario: 组件模块化
- **WHEN** 开发页面功能
- **THEN** 将可复用的UI元素封装为独立组件
- **AND** 每个组件职责单一,可复用
#### Scenario: 页面模块化
- **WHEN** 开发页面
- **THEN** 每个页面独立管理自己的状态和逻辑
- **AND** 通过 Pinia store 共享全局状态
### Requirement: 查询会话记录(管理端)
系统 SHALL 提供接口按条件查询会话记录,并返回会话与用户的联合视图。
接口:`POST /api/Admin/QueryConversations`
过滤规则(均为可选):
- `UserKey`:用户唯一标识键
- `MessageType`消息类型1-公有2-私有)
- `StartTime``EndTime`(按 `RecordTimeUTCStamp` 过滤datetime参数转换为UTC时间戳
- `Department`:用户部门(来自用户表)
默认规则:
- 仅返回 `xcx_conversation.IsDeleted = 0` 的记录
-`RecordTimeUTCStamp DESC` 排序
- 通过 `LEFT JOIN xcx_users` 补全用户字段
返回字段(会话):
- Id, Guid, UserKey, ConversationContent, SendMethod
- UserLocation, Latitude, Longitude
- RecordTime, RecordTimeUTCStamp, IsDeleted, CreateTime
- MessageType, SpeakingTime
返回字段用户通过LEFT JOIN关联
- UserName, WeChatName, PhoneNumber, AvatarUrl, Department
#### Scenario: 管理端按时间范围查询
- **WHEN** 管理端提交包含 `StartTime``EndTime` 的查询请求
- **THEN** 系统将datetime参数转换为UTC时间戳
- **AND** 返回 `RecordTimeUTCStamp` 落在区间内的会话记录
- **AND** 结果按 `RecordTimeUTCStamp` 倒序
#### Scenario: 管理端按部门筛选
- **WHEN** 管理端提交包含 `Department` 的查询请求
- **THEN** 系统仅返回用户部门匹配的会话记录
#### Scenario: 管理端按UserKey筛选
- **WHEN** 管理端提交包含 `UserKey` 的查询请求
- **THEN** 系统仅返回指定用户的会话记录
#### Scenario: 管理端按MessageType筛选
- **WHEN** 管理端提交包含 `MessageType` 的查询请求
- **THEN** 系统仅返回指定消息类型的会话记录
#### Scenario: 管理端查询全部会话记录
- **WHEN** 管理端提交不包含任何筛选条件的查询请求
- **THEN** 系统返回所有 `IsDeleted = 0` 的会话记录
- **AND** 结果按 `RecordTimeUTCStamp` 倒序
#### Scenario: 管理端组合多条件查询
- **WHEN** 管理端提交包含多个筛选条件的查询请求
- **THEN** 系统返回同时满足所有条件的会话记录
### Requirement: 查询用户列表(管理端)
系统 SHALL 提供接口返回可用用户列表,用于管理侧选择/筛选。
接口:`GET /api/Admin/QueryUsers`
过滤规则(内置):
- `PhoneNumber` 不为空且不为空字符串
- `UserName` 不为空且不为空字符串
- `UserKey` 不为空且不为空字符串
- `IsDisabled = 0`(未禁用)
-`FirstLoginTime DESC` 排序
返回字段:
- Id, UserName, UserKey, WeChatName, PhoneNumber
- FirstLoginTime, IsDisabled, CreateTime, UpdateTime
- AvatarUrl, Department
#### Scenario: 获取用户列表
- **WHEN** 管理端请求用户列表
- **THEN** 返回已完善基础信息且未禁用的用户
- **AND** 结果按首次登录时间倒序排列
## Known Limitations
- 当前接口未提供分页与导出能力(若管理端数据量很大,需在后续能力中补齐)。
- 当前未强制鉴权JWT 配置存在但未启用认证中间件/授权标注)。
- 前端验证仅使用固定账号密码Admin/Admin安全性较低后续应考虑接入后端JWT认证。
## Implementation Details
### 后端技术实现
- **控制器文件**: `WxCheckMvc/Controllers/AdminController.cs`
- **数据库连接**: 使用 `MySqlConnection` 进行数据库操作
- **路由模式**: `[Route("api/[controller]/[action]")]`
- **API特性**: `[ApiController]` 提供自动模型验证和绑定
### 前端技术实现
- **项目目录**: `admin-web/`
- **构建工具**: Vite
- **路由配置**: `src/router/index.js`
- **状态管理**: `src/store/index.js`Pinia
- **样式目录**: `src/styles/`
- **组件目录**: `src/components/`
- **页面目录**: `src/views/`
#### 项目结构
```
admin-web/
├── src/
│ ├── assets/ # 静态资源
│ ├── components/ # 可复用组件
│ │ ├── Layout/ # 布局组件(侧边栏、顶部栏)
│ │ ├── ThemeSwitcher.vue # 主题切换组件
│ │ └── ...
│ ├── router/ # 路由配置
│ │ └── index.js
│ ├── store/ # Pinia store
│ │ ├── index.js
│ │ ├── auth.js # 认证状态
│ │ └── theme.js # 主题状态
│ ├── styles/ # 全局样式
│ │ ├── variables.scss # CSS变量深色/浅色主题)
│ │ ├── responsive.scss # 响应式样式
│ │ └── main.scss
│ ├── utils/ # 工具函数
│ │ └── request.js # axios封装
│ ├── views/ # 页面组件
│ │ ├── Login.vue # 登录页
│ │ ├── Home.vue # 首页
│ │ ├── ConversationList.vue # 会话记录管理页
│ │ └── UserList.vue # 用户管理页
│ ├── App.vue
│ └── main.js
├── index.html
├── package.json
└── vite.config.js
```
#### Pinia Store 结构
##### auth.js认证状态
```javascript
import { defineStore } from 'pinia'
export const useAuthStore = defineStore('auth', {
state: () => ({
isLoggedIn: false,
username: ''
}),
actions: {
login(username, password) {
if (username === 'Admin' && password === 'Admin') {
this.isLoggedIn = true
this.username = username
return true
}
return false
},
logout() {
this.isLoggedIn = false
this.username = ''
}
}
})
```
##### theme.js主题状态
```javascript
import { defineStore } from 'pinia'
export const useThemeStore = defineStore('theme', {
state: () => ({
isDark: localStorage.getItem('theme') === 'dark'
}),
actions: {
toggleTheme() {
this.isDark = !this.isDark
localStorage.setItem('theme', this.isDark ? 'dark' : 'light')
document.documentElement.setAttribute('data-theme', this.isDark ? 'dark' : 'light')
},
initTheme() {
const theme = localStorage.getItem('theme') || 'light'
this.isDark = theme === 'dark'
document.documentElement.setAttribute('data-theme', theme)
}
}
})
```
#### 响应式断点
- **手机端**: < 768px
- **平板端**: 768px - 1023px
- **桌面端**: >= 1024px
### 数据模型
#### ConversationQueryRequest
```csharp
public class ConversationQueryRequest
{
public DateTime? StartTime { get; set; }
public DateTime? EndTime { get; set; }
public string UserKey { get; set; }
public int? MessageType { get; set; }
public string Department { get; set; }
}
```
#### ConversationQueryResponse
```csharp
public class ConversationQueryResponse
{
public long Id { get; set; }
public string Guid { get; set; }
public string UserKey { get; set; }
public string ConversationContent { get; set; }
public string SendMethod { get; set; }
public string UserLocation { get; set; }
public string Latitude { get; set; }
public string Longitude { get; set; }
public DateTime RecordTime { get; set; }
public long RecordTimeUTCStamp { get; set; }
public bool IsDeleted { get; set; }
public DateTime CreateTime { get; set; }
public int MessageType { get; set; }
public int? SpeakingTime { get; set; }
public string UserName { get; set; }
public string WeChatName { get; set; }
public string PhoneNumber { get; set; }
public string AvatarUrl { get; set; }
public string Department { get; set; }
}
```
#### UserQueryResponse
```csharp
public class UserQueryResponse
{
public long Id { get; set; }
public string UserName { get; set; }
public string UserKey { get; set; }
public string WeChatName { get; set; }
public string PhoneNumber { get; set; }
public DateTime FirstLoginTime { get; set; }
public bool IsDisabled { get; set; }
public DateTime CreateTime { get; set; }
public DateTime UpdateTime { get; set; }
public string AvatarUrl { get; set; }
public string Department { get; set; }
}
```
### 异常处理
- 所有方法使用 try-catch-finally 模式
- 异常时返回 HTTP 500 状态码
- 响应格式:`{ success: false, message: "错误描述", error: "异常详情" }`
- finally 块确保数据库连接正确关闭
### 数据库操作
- 使用参数化查询防止 SQL 注入
- 动态构建 SQL 查询条件,仅添加非空参数
- 使用 `LEFT JOIN` 关联 `xcx_users` 表获取用户信息
- 时间转换:`DateTimeOffset.UtcNow.ToUnixTimeMilliseconds()`