34 Commits

Author SHA1 Message Date
yuzhiran 1e8e22c9ed docs: update project status v4.8, AGENTS.md build notes, deployment docs
- PROJECT-STATUS.md: v4.8 changelog (SEO, share hooks, version injection)
- AGENTS.md: update version to v1.0.17, add build notes for SEO/version/share
- DEPLOYMENT.md: update deploy instructions for robots.txt/sitemap.xml/static, version bump
2026-06-21 20:09:45 +08:00
yuzhiran d8fb8e3bba feat: enable WeChat share/forward on all pages via onShareAppMessage + onShareTimeline 2026-06-21 18:51:35 +08:00
yuzhiran 214571688c feat: SEO optimization - canonical URL, robots.txt, sitemap, structured data, keyword-rich titles, manifest update 2026-06-21 09:48:41 +08:00
yuzhiran 8532776fa1 refactor: remove duplicate contact button in about page 2026-06-21 09:38:24 +08:00
yuzhiran b8667395ac feat: unify contact button styles, dynamic version from git tag, enrich AI positions & homepage UX 2026-06-21 09:16:31 +08:00
yuzhiran 1d1c4ab590 fix: bump patch version automatically on mp upload (v1.0.16 → 1.0.17) 2026-06-21 09:03:34 +08:00
yuzhiran 19b087a589 docs: update project status to v4.7 - gravity pay-per-use, full production deployment 2026-06-21 08:58:33 +08:00
yuzhiran 310176a11b chore: production cleanup - remove debug logs, add DB cleanup script
- App.vue: remove console.log on launch/show/hide, drop unused onShow/onHide imports
- interview.vue: remove verbose console.log('[ASR] upload response')
- login.vue: remove debug console.log('[wxLogin]') logs (keep error logs)
- scripts/cleanup-test-data.ts: new script to identify and remove test data
  while preserving admin accounts; supports --dry-run preview mode
- All 43 backend tests pass
2026-06-20 23:08:44 +08:00
yuzhiran ef4d22a633 feat(admin): enrich admin panel fields; add user index constraint and customer service
- admin controller: add updatedAt to interview/resume selects; add orderCount,
  todayOrders, totalRevenue to overview
- admin.vue: enrich all tabs with more fields
  - overview: order cards (count, revenue)
  - users: wxOpenid, email, createdAt, interviewCount, vipExpireAt, role badge
  - interviews: user email, updatedAt, summary preview
  - orders: title, type, channel, paidAt, wxTransactionId, refund info
  - resumes: user email, updatedAt
  - share: sharer phone, shareCode, isActive, visitorId(IP), creditedAt
  - admins: email, createdAt
- user.schema: add unique indexes on phone/wxOpenid/email; pre-save hook
  requiring at least one contact method
- user/about: add WeChat contact button (open-type=contact) for customer service
2026-06-20 22:38:33 +08:00
yuzhiran 8ee27fdd32 feat: refactor member to pay-per-use gravity purchase; mv webview to clipboard+browser
- member.vue: rewrite from subscription plans (free/growth/sprint) to
  H5-only pay-per-use gravity purchase with quantity selector + QR code
- user.vue: gravity card replacing quota card, add share/contribute/H5-buy
  entry points, plus gravity acquisition modal (share/contribute/buy)
- share.vue: layout fix (flex column), smarter copyLink with cached URL,
  WeChat timeline hint instead of open-type
- share.controller.ts: add GET /:shareCode redirect route (IP record + 302)
- interview.vue: guest mode fix, H5 buy modal, clipboard copy instead of
  webview for mini-program
- App.vue: handleH5UrlParams for ?token=&buy=gravity auto-login
- composables/useGravityPurchase.ts: reusable gravity purchase composable
- remove webview.vue (no longer used), replace with clipboard+browser flow
- AGENTS.md: sync all above changes, fix duplicate numbering
2026-06-20 20:49:15 +08:00
yuzhiran a1e1f0b3c3 docs: update project status to v4.6 - gravity unification, admin panel improvements 2026-06-19 22:52:57 +08:00
yuzhiran 2fbab1072f feat: unified gravity system - VIP members consume gravity instead of unlimited; add monthly gravity top-up cron 2026-06-19 22:43:52 +08:00
yuzhiran c2ba810a02 fix: use onShow lifecycle to refresh login state in member center
1. member.vue: added onShow to refresh login state when returning from login page

2. manifest.json: bump to v1.0.15

Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent)

Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
2026-06-18 21:23:20 +08:00
yuzhiran 3f1239c35e fix: add network error detection and timeout for mini program JSAPI payment
1. member.vue: check for res.statusCode === 0 (network failure) before accessing data, add 30s timeout

2. manifest.json: bump to v1.0.14

Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent)

Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
2026-06-18 20:54:42 +08:00
yuzhiran 1be5b34906 chore: bump version to 1.0.13
Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent)

Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
2026-06-18 20:39:56 +08:00
yuzhiran c58bb27575 fix: admin page reactive import + member payment toast show real error
1. admin.vue: 添加缺失的reactive导入,修复管理后台只显示标题不显示功能的问题

2. member.vue: 支付失败toast改为显示后端真实错误信息,而非硬编码'创建订单失败'

Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent)

Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
2026-06-18 20:39:49 +08:00
yuzhiran e0de29fdd0 fix: AI LLM backup model not producing content; add retry for thinking models
- Replace backup model stepfun-ai/step-3.5-flash with meta/llama-3.1-8b-instruct
  (stepfun is a thinking model that uses all tokens on reasoning and
  never outputs content, causing all 3 fallthroughs to fail)
- Add retry with doubled max_tokens when primary model returns empty
  content (deepseek-v4-flash thinking can exhaust token budget)
- Increase backup timeout to 120s and max_tokens to min 2048
- Move callApi error handling to return null instead of throw for
  cleaner fallthrough logic with timeout logging
2026-06-18 19:42:40 +08:00
yuzhiran 6a3cc8544e fix: handle WeChat Pay public key mode in callback
- verifyAndDecrypt now processes decryption even when signature
  verification fails (decryption key is separate from signature key)
- Notify handler uses returnRaw flag to always decrypt resource
- Loud log when pub_key.pem verification fails, directs admin
  to download correct public key from merchant platform
2026-06-18 19:36:19 +08:00
yuzhiran c161ffbc3c feat: payment refund support + admin payment management
- Add refund()/queryRefund()/downloadPlatformCerts() to WechatPayService
- Add refundId field to PaymentOrder schema
- Fix WeChat Pay callback to auto-download platform certs on verification failure
- Fix syncOrder to handle sprint plan properly
- Add admin refund, refund-query, order-detail endpoints
- Add refund UI (button, modal, query) to admin.vue orders tab
- Fix member.vue MP payment: pass outTradeNo instead of prepayId to pollPayResult
2026-06-18 19:33:10 +08:00
yuzhiran 7e1bf669ab fix: show toast when clicking login without agreeing to privacy terms
Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent)

Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
2026-06-18 18:54:44 +08:00
yuzhiran bed9dce943 chore: bump version to 1.0.12
Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent)

Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
2026-06-18 18:48:27 +08:00
yuzhiran 4ac42f6575 fix: privacy policy compliance - checkbox must be manually checked, add WeChat privacy API
Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent)

Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
2026-06-18 18:46:57 +08:00
yuzhiran f72312ea52 docs: update with production mode, test accounts, admin auto-verify
Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent)

Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
2026-06-18 18:21:48 +08:00
yuzhiran 7cf4636b8c fix: auto-verify admin on mount (onMounted doVerify)
Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent)

Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
2026-06-18 18:21:41 +08:00
yuzhiran 27e4d06da7 fix: show devCode modal in email login for non-production mode
Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent)

Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
2026-06-18 18:21:34 +08:00
yuzhiran e049be280e fix: auto-copy avatars on build, show masked user ID on profile page 2026-06-18 17:58:54 +08:00
yuzhiran b9651a9ff3 chore: hide company-bank and internship entries from homepage (content empty) 2026-06-18 17:44:11 +08:00
yuzhiran 54c21e2953 refactor: rewrite company-bank and internship pages
- bank.vue: Composition API, design tokens, 2-col grid, better UX
- internship.vue: search bar, category tabs, card list layout
2026-06-18 17:27:37 +08:00
yuzhiran 0616fd955c fix: homepage layout optimization (2-col grid, guest CTA, daily question for guests)
- Unified feature grid to consistent 2-column layout
- Improved hero guest card with benefit tags
- Show daily question section for guests with login prompt
- Added AGENTS.md: H5 deployment docs, git remote, build gotchas
2026-06-18 15:53:11 +08:00
wlt df1b37fe79 feat: add positions management admin tab + career advisor homepage entry
- admin.vue: new '岗位' tab with CRUD list/modal/api functions/styles
- index.vue: add AI择业顾问 entry card linking to career page
- Backend CRUD endpoints already exist, no backend changes needed
2026-06-18 15:13:29 +08:00
wlt 103dbd3b34 feat: AI岗位专区 — 5个AI岗位置顶 + 首页分组展示
- schema: HotPosition 新增 category 字段 (ai/traditional)
- positions: 5 AI岗位 (AI算法/大模型应用/Prompt/AI产品/AI运维) + 7传统岗位
- frontend: 首页拆分 "🔥 AI热门岗位" 置顶高亮 + "更多岗位" 折叠
- ai服务: 新增 primaryFallbackModel (sensenova-6.7-flash-lite) 降级链路
2026-06-17 13:57:18 +08:00
wlt a5c4bcb821 feat: AI 择业顾问 MVP — 专业分析 + 岗位匹配 + 多轮对话
- backend: career-advice module with analyze/chat/positions endpoints
- frontend: career.vue page with profile form, AI advice, recommendation cards
- config/api/pages/user.vue: full integration into existing flow
- docs: PROJECT-STATUS v4.5, FEATURE-LIST v4.3, ROADMAP v4.3
- AGENTS.md: updated module count and career link paths
2026-06-17 10:32:23 +08:00
wlt 4cd889c081 feat: interview review module with whisper.cpp ASR + AI analysis + frontend page
New backend module 'interview-review' provides:
- Audio upload (50MB limit, MP3/M4A/WAV/AAC/OGG/MP4/WebM)
- Text transcript submission
- whisper.cpp local ASR integration (tiny + base models)
- AI analysis (4-dimension scoring: logic/expression/professionalism/stability)
- Speech analysis (filler words detection, pace, duration)
- Async processing pipeline with status polling
- Graceful fallback to mock ASR when whisper unavailable

New frontend page 'pages/review/review.vue' with 3 modes:
- List mode: review history with status indicators
- Upload mode: audio file upload or text paste
- Report mode: score radar, dimension bars, analysis details

Docs updated: PROJECT-STATUS.md v4.4, FEATURE-LIST.md v4.2, ROADMAP.md v4.2
2026-06-16 18:32:25 +08:00
yuzhiran 96c367e0f8 feat: latest code update 2026-06-16 13:18:36 +08:00
67 changed files with 5605 additions and 1243 deletions
@@ -0,0 +1,11 @@
{
"sessionID": "ses_130b21833ffe0epQvGBjpGtogZ",
"updatedAt": "2026-06-17T05:54:06.381Z",
"sources": {
"background-task": {
"state": "active",
"reason": "2 background task(s) active",
"updatedAt": "2026-06-17T05:54:06.381Z"
}
}
}
@@ -0,0 +1,10 @@
{
"sessionID": "ses_15492f54bffepl01q9FkaRyN28",
"updatedAt": "2026-06-19T07:14:26.627Z",
"sources": {
"background-task": {
"state": "idle",
"updatedAt": "2026-06-19T07:14:26.627Z"
}
}
}
+312 -42
View File
@@ -1,56 +1,326 @@
## 开发交付流程 # 职引 (ZhiYin) — AGENTS.md
每次实施/开发完功能后,必须按以下步骤执行: > AI 模拟面试教练,专注校招。NestJS + uni-app(Vue3) + MongoDB。
### Step 1: 代码评审 ---
- 检查变更是否符合现有代码模式(schema、service、controller、module 的结构一致性)
- 检查是否有未使用的变量、import、参数
- 检查命名规范是否与项目一致
- 检查是否有重复代码或可复用逻辑
### Step 2: 安全评审 ## 一、项目结构
- **注入防护**:检查所有外部输入(请求 body、query、params)是否有注入风险(HTML、Shell、NoSQL
- **认证/授权**:检查端点是否正确地加了 JWT guard 或 `@Public()`;管理端接口是否有 AdminGuard
- **并发安全**:检查所有修改用户额度/积分的操作是否使用原子操作(`findOneAndUpdate + $inc`),而非 read-modify-write
- **敏感信息**:检查是否误将密钥、凭据、token 暴露到响应或日志中
- **XSS/CSRF**:检查用户内容输出到 HTML/PDF 时是否做了转义
### Step 3: 性能优化 ```
- 检查是否有 N+1 查询(循环内查数据库),应使用批量查询或聚合 zhiyin/
- 检查大表查询是否有索引覆盖 ├── backend/ # NestJS 10.x 后端 (端口 3006, 前缀 /api)
- 检查是否有不必要的 `.lean()` 缺失(读操作用 `exec()` 但不需要 mongoose document 方法时应加 `.lean()` │ └── src/
- 检查是否有内存泄漏风险(如 puppeteer browser 未在 finally 中 close │ ├── main.ts # 入口:DOMMatrix polyfill → NestFactory → CORS → ValidationPipe
- Throttler/限流是否合理 │ ├── app.module.ts # 根模块:导入全部子模块 + JWT/Throttler/Mongoose
│ ├── common/
### Step 4: 完整测试 │ │ ├── guards/ # JwtAuthGuard (全局), admin.guard.ts
```bash │ │ ├── strategies/ # JwtStrategy
# 构建检查(注意内存限制,服务器 OOM 时加 --max-old-space-size │ │ ├── decorators/ # @CurrentUser, @Public()
cd /root/opencode-workspace/zhiyin/backend && NODE_OPTIONS="--max-old-space-size=2048" npx nest build 2>&1 │ │ └── filters/ # AllExceptionsFilter
│ └── modules/ # 20 个模块(详见下文)
# 单元测试 ├── zhiyin-app/ # uni-app 3.x 前端 (H5 + 微信小程序)
npm test -- --forceExit --detectOpenHandles 2>&1 │ └── src/
│ ├── pages/ # 20 个页面 (pages.json 路由)
# 如果有变更的模块,验证关键 endpoint curl 可访问 │ ├── composables/ # 可复用组合式函数(如 useGravityPurchase
│ ├── services/api.ts # API 调用封装 (uni.request)
│ ├── config.ts # 端点定义 + api() 辅助函数
│ └── App.vue # 设计 Token + 全局样式 + H5 URL 参数处理
└── docs/ # 产品/架构/部署/路线图文档
``` ```
### Step 5: 同步修复 ### 后端模块清单
- 上述步骤发现的问题必须修复后再发布
- 修复后重新执行 Step 4 验证构建通过
### Step 6: 部署到生产 | 模块 | 职责 |
|------|------|
| `user` | 手机/邮箱/密码/微信登录, JWT, 配额 |
| `interview` | AI 面试核心(多轮对话 + 评分 + 报告 + 进度) |
| `ai` | AI 调用封装(deepseek-v4-flash 主 + step-3.5-flash 备) |
| `analyze` | 简历诊断 / 优化 / 技能缺口分析 |
| `resume` | 简历 CRUD |
| `member` | 会员套餐 / 权益扣减 |
| `payment` | 微信支付 v3Native + JSAPI + 回调) |
| `progress` | 进步轨迹雷达图 / 打卡日历 / 行业基准 / 岗位匹配 |
| `contribution` | 面经贡献 + 公司题库(数据飞轮核心) |
| `schedule` | 定时任务:VIP 过期降级、每日一题推送、微信 token 刷新、月度引力值补给 |
| `share` | 分享链接生成 / 访问追踪 / 积分奖励 |
| `tts` | 语音合成(TTS |
| `admin` | 管理后台 API |
| `positions` | 热门岗位维护 |
| `interview-review` | 面试复盘(音频上传 -> whisper.cpp ASR -> AI 评析 -> 口语分析) |
|`career-advice` | AI 择业顾问:专业分析 + 岗位匹配 + 推荐对话 |
| `upload` | 文件上传(PDF/图片) |
| `email` | 邮件发送 |
| `daily-question` | 每日一题 API |
| `schemas/` | 共享 Schemapricing 定价、site-config、company-bank 等) |
### 前端页面(3 Tab + 18 子页)
- **Tab1 面试**: pages/index/index → interview → report → career
- **Tab2 面经**: pages/history/history → contribute → company-bank
- **Tab3 我的**: pages/user/user → login/member/progress/resume/review/career/about/agreement/privacy/admin/share
- 其他: internship, result
---
## 二、架构约定(必须遵守)
### 模块模式
每个业务模块遵循 NestJS 标准结构:
```
模块名/
├── 模块名.module.ts # @Module({ imports: [MongooseModule.forFeature(...)], controllers, providers, exports })
├── 模块名.controller.ts # @Controller('prefix'),注入 service
├── 模块名.service.ts # @Injectable(),注入 Model
└── 模块名.schema.ts # @Schema({ timestamps: true })class + SchemaFactory
```
### 认证体系
- **全局守卫**: `JwtAuthGuard` 默认拦截所有路由(在 `app.module.ts``APP_GUARD` 注册)
- **白名单**: 公开接口加 `@Public()` 装饰器(登录、注册、支付回调、分享访问等)
- **管理员**: 管理接口加 `@UseGuards(AdminGuard)`admin controller 内部)
- **当前用户**: `@CurrentUser('userId')` 从 JWT payload 提取用户 ID
- **JWT 过期**: 7 天,在 `app.module.ts` 和每个模块的 `JwtModule.register` 中配置
### 安全硬性要求
1. **JWT_SECRET 必须来自环境变量**,不允许任何硬编码 fallback(已有历史漏洞修复)
2. **所有外部输入**经过 class-validator `ValidationPipe({ whitelist: true, forbidNonWhitelisted: true })`
3. **修改用户额度/积分**使用 `findOneAndUpdate + $inc` 原子操作,禁止 read-modify-write
4. **支付订单查询**校验 userId 归属,防止 IDOR
5. **CORS** 生产环境必须配置白名单(当前在 `main.ts` 中从 `CORS_ORIGINS` 环境变量读取)
6. **用户内容输出到响应**时避免泄漏敏感信息(验证码、密钥等)
7. **MongoDB 查询**中对外部输入的字符串做特殊字符转义(尤其在 admin 模块)
### AI 调用
- 主模型: `opencode-go` (deepseek-v4-flash)
- 备用模型: NVIDIA (stepfun-ai/step-3.5-flash)
- 主用不可用时自动切换(在 `ai` 模块处理)
- 环境变量: `AI_PRIMARY_KEY`, `AI_BACKUP_KEY`
### 支付(微信支付 v3
- Native 支付(H5 扫码): `POST /payment/create-product`(按量购买引力值)
- JSAPI 支付(小程序内): `POST /payment/jsapi-product`(按量购买引力值)
- 支付回调: `POST /payment/notify`@Public,验签 + 解密 + 自动到账)
- 支付结果轮询: `GET /payment/check/:outTradeNo`
- 产品定价: `GET /member/plans`(含 products 字段,定义引力值单价和赠送量)
- 需要微信商户证书文件(通过 postbuild 复制到 dist
- **注意**: 当前会员体系已从按月订阅制改为按量购买引力值制(小程序内复制链接到浏览器打开购买,H5 直接扫码支付)
---
## 三、开发命令
> ⚠️ **构建铁律:必须始终使用 `npm run build:*` 命令,禁止直接调用 `npx uni build` 或 `npx nest build`。**
> 前端 `npm run build:mp-weixin` 和 `npm run build:h5` 脚本包含头像文件(`avatar-*.png`)复制步骤,
> 后端 `npm run build` 是 `nest build` 的别名。直接使用 `npx` 会遗漏这些关键步骤,导致线上数字人头像不显示等问题。
### 后端
```bash ```bash
# 构建 # 路径: backend/
cd /root/opencode-workspace/zhiyin/backend && npx nest build npm run start:dev # 开发模式(watch
npm run build # 编译到 dist/
npm test # 单元测试(43 个,jest --forceExit --detectOpenHandles
npm run test:watch # 监听模式
npm run test:cov # 覆盖率报告
npm run test:e2e # 集成测试(需 MongoDB 运行)
npm run test:browser # Playwright API 测试(需后端运行)
```
# 同步到生产目录 ### 前端
```bash
# 路径: zhiyin-app/
npm run dev:mp-weixin # 微信小程序开发(uni -p mp-weixin
npm run build:mp-weixin # 构建小程序(输出 dist/build/mp-weixin/
npm run dev:h5 # H5 开发(端口 8888,带 /api 代理到 localhost:3006
npm run build:h5 # 构建 H5(输出 dist/build/h5/
npm test # 前端单元测试(vitest,7 个)
```
### 构建检查
```bash
# 后端构建(注意 OOM:需 NODE_OPTIONS="--max-old-space-size=2048"
cd backend && npm run build
```
### 构建产物说明
- **H5 构建** (`npm run build:h5`):自动复制 `avatar-*.png``static/`,复制 `robots.txt``sitemap.xml` 到输出根目录
- **版本号注入**`vite.config.js``define.__APP_VERSION__` 自动从 `git describe --tags` 获取版本号,`about.vue` 页面实时显示
- **小程序上传版本** (`scripts/upload-mp.js`):自动从 git tag 获取基础版本,末位自增 1(如 tag v1.0.16 → 上传版本 1.0.17
- **微信分享**:所有主页面均已注册 `onShareAppMessage` + `onShareTimeline`,右上角菜单转发/分享到朋友圈可用
### 部署后端
```bash
cd backend && npm run build
cp -rf dist/* /www/wwwroot/server/zhiyin/backend/dist/ cp -rf dist/* /www/wwwroot/server/zhiyin/backend/dist/
# 复制证书(postbuild 替代)
cp -r certs /www/wwwroot/server/zhiyin/backend/dist/src/certs cp -r certs /www/wwwroot/server/zhiyin/backend/dist/src/certs
# 重启
pm2 restart yhl-backend pm2 restart yhl-backend
# 验证
sleep 3 && curl -s http://localhost:3006/api/user/wx-login -X POST -H "Content-Type: application/json" -d '{"code":"test"}' sleep 3 && curl -s http://localhost:3006/api/user/wx-login -X POST -H "Content-Type: application/json" -d '{"code":"test"}'
``` ```
### 部署前端 H5
```bash
cd zhiyin-app && npm run build:h5
rm -rf /www/wwwroot/zhiyin.yzrcloud.cn/assets
cp -r dist/build/h5/index.html /www/wwwroot/zhiyin.yzrcloud.cn/
cp -r dist/build/h5/assets /www/wwwroot/zhiyin.yzrcloud.cn/
cp -r dist/build/h5/static /www/wwwroot/zhiyin.yzrcloud.cn/
cp -f dist/build/h5/robots.txt dist/build/h5/sitemap.xml /www/wwwroot/zhiyin.yzrcloud.cn/
chown -R www:www /www/wwwroot/zhiyin.yzrcloud.cn/
# 验证无缺失文件
grep -oP '["'"'"']([a-zA-Z0-9_-]+\.[a-z]+(\.js|\.css|\.png|\.svg))["'"'"']' /www/wwwroot/zhiyin.yzrcloud.cn/assets/index-*.js | sort -u
```
### 小程序上传(先 build 后 upload,两步分开更安全)
```bash
cd zhiyin-app && npm run build:mp-weixin && node scripts/upload-mp.js
# 注意:build:mp-weixin 已自动复制 avatar-*.png 到 dist/build/mp-weixin/static/
# 如遇数字人头像不显示,检查是否漏了 cp 步骤,重新用 npm run build:mp-weixin 构建
```
---
## 四、测试注意事项
- **e2e 测试**需要 MongoDB 运行 + `jest-setup.ts` 设置测试 JWT_SECRET
- `payment.controller.spec.ts` 涉及微信支付,需 mock 外部依赖
- `benchmark.service.spec.ts` 涉及行业基准计算
- Playwright 测试需要后端已在运行(测试 `api.browser.spec.ts`
- 测试文件位置:`backend/src/**/*.spec.ts`(单元),`backend/test/*.e2e-spec.ts`(集成),`backend/test/*.browser.spec.ts`(浏览器)
- 前端测试:`zhiyin-app/src/**/*.spec.ts`vitest + jsdom
---
## 五、定时任务(4 个 cron,在 schedule 模块)
| 服务 | 周期 | 职责 |
|------|------|------|
| `VipExpiryService` | 每日 00:00 | 扫描过期 VIP 并降级为 free 计划 |
| `DailyQuestionPushService` | 每日 09:00 | 通过微信订阅消息推送每日一题(需配置模板 ID) |
| `WechatTokenService` | 每 2 小时 | 刷新微信 access_token(缓存到 Redis |
| `GravityTopUpService` | 每日 02:00 | 给所有未过期的成长版/冲刺版用户补给月度引力值 |
---
## 六、项目状态与开发阶段
**当前**: Phase 1.5(商业化 + 全量部署)— v1.0.17
| 阶段 | 状态 | 关键交付 |
|------|------|---------|
| Phase 0: 战略升级 | ✅ 完成 | 定价重构(免费 + 按量购买),三层壁垒设计 |
| Phase 0.5: 壁垒构建 | ✅ 完成 | 数据飞轮(面经贡献+题库),留存入围(进步轨迹+打卡日历+每日一题) |
| Phase 1: MVP 上线 | ✅ 完成 | 面试复盘(whisper.cpp ASR + AI 评析)、AI 择业顾问 |
| Phase 1.5: 商业化 | 🚧 当前 | 按量购买引力值(¥5/份)、管理后台完善、H5/小程序全量部署上线、清理脚本 |
| Phase 2: 增强 + 题库 | 📋 规划 | 50+ 校招岗位、技能缺口分析、公司真题库建设 |
| Phase 3: 秋招冲刺 | 📋 规划 | 高校合作、B 端服务、KOC 推广 |
详细产品规划见 `docs/PRODUCT-PLAN.md`,路线图见 `docs/ROADMAP.md`
---
## 七、环境变量
### 后端(`backend/.env`,不提交 git,在 `.gitignore` 中)
```
MONGODB_URI=mongodb://localhost:27017/zhiyin
JWT_SECRET=your-strong-secret-at-least-32-chars
NODE_ENV=production
PORT=3006
AI_PRIMARY_KEY=xxx
AI_BACKUP_KEY=xxx
WECHAT_APPID=wxf466b3c3bc411ffc
WECHAT_MCHID=xxx
WECHAT_API_KEY=xxx
WECHAT_SERIAL_NO=xxx
WECHAT_PRIVATE_KEY_PATH=/path/to/apiclient_key.pem
WX_DAILY_QUESTION_TMPL=微信订阅消息模板 ID
CORS_ORIGINS=http://localhost:8888,https://zhiyin.yzrcloud.cn
EMAIL_HOST=smtp.qiye.aliyun.com
EMAIL_PORT=465
EMAIL_SECURE=true
EMAIL_USER=contact@yuzhiran.com
EMAIL_PASSWORD=xxx
EMAIL_FROM=宇之然AI磁场 <contact@yuzhiran.com>
WHISPER_CPP_PATH=/home/wlt/whisper.cpp # whisper.cpp 路径
WHISPER_MODEL=base # ASR 模型:tiny / base / small
WHISPER_LANGUAGE=zh # ASR 语言
WHISPER_THREADS=4 # ASR CPU 线程数
```
### 前端(`zhiyin-app/.env.production`,已提交 git
```
VITE_API_BASE_URL=https://zhiyinwx.yzrcloud.cn/api
VITE_APP_NAME=AI磁场
```
---
## 八、测试 / 管理员账号
| 账号 | 密码 | 角色 | 说明 |
|------|------|------|------|
| `13701190814@139.com` | `Zhiyin2024!` | admin | 管理员,可访问管理后台 |
| `test@yzrcloud.cn` | `123456` | user | 测试账号 |
| `test@test.com` | 验证码 `123456` | admin | 旧管理员(dev 模式可用) |
管理后台路径:`/pages/admin/admin`,进入后自动验证管理员身份(`onMounted``doVerify`)。
---
## 九、Git
- 远程仓库: `http://127.0.0.1:2999/txai-dev/zhiyin.git`(本机 Gitea,带 token 认证)
- 默认分支: `master`
- 最新 tag: `v1.0.16`(小程序上传版本 v1.0.17 源自 git tag + 末位自增 1
---
## 十、技术细节与坑
1. **DOMMatrix polyfill**: `main.ts` 顶部有 pdf-parse 所需的浏览器 API polyfillDOMMatrix / DOMPoint),新增 PDF 相关功能时注意兼容性
2. **postbuild**: `backend/package.json` 中的 `postbuild` 脚本自动复制 `certs/``dist/src/certs/`,这是微信支付证书的必要步骤
3. **微信小程序 appid**: `zhiyin-app/manifest.json``mp-weixin.appid = wxf466b3c3bc411ffc`;开发模式 `appid = __UNI__DEV__`
4. **前端 API 调用**: `zhiyin-app/src/services/api.ts` 封装了 `uni.request`,自动处理 token 注入(从 `uni.getStorageSync('token')`)和 401 过期跳转
5. **前端环境判断**: `config.ts` 中使用 `// #ifdef H5` / `// #ifdef MP-WEIXIN` 条件编译区分 H5 和小程序
6. **API 限流**: 100 次/60 秒(在 `app.module.ts` 中配置),注意避免在定时任务和批量操作中被限
7. **验证码**: 生产模式(`NODE_ENV=production`)使用真实 SMTP 发邮件验证码;非生产模式手机验证码固定为 `123456`、邮件验证码在响应中返回 `devCode`
8. **MongoDB**: 8 个核心集合 + 2 个分享集合
9. **引力值体系**: 所有计划统一走引力值消耗(面试 5、优化 3、下载 2)。VIP 不再免额度,成长版每月 250 引力值,冲刺版每月 600 引力值,每日凌晨 2 点定时补给。免费用户注册送 5 引力值。小程序内通过分享得引力值/贡献面经/复制官网链接到浏览器打开购买三种方式获取引力值;H5 直接扫码支付按量购买(¥5/份)。
10. **api.ts 陷阱**: 对象字面量必须在 `export const apiService = {``const apiService = { ... export default apiService` 中包裹,否则 uni-app 构建报错 `Expected ";" but found ":"`。git pull 后经常丢失这行声明,需手动补回
11. **H5 构建 assets 清理**: `assets/` 中的旧 hash 文件不能随意删除——`index-*.js`(主 bundle)动态 import 了所有 page chunk,删除仍在引用的文件会导致浏览器 `NS_ERROR_CORRUPTED_CONTENT`
12. **管理后台自动验证**: `admin.vue``onMounted` 自动调用 `doVerify()`,进入后台即检测 JWT 中 `role` 是否为 `admin`,不再需要手动点击"验证管理员身份"按钮
13. **分享重定向路由**: `share.controller.ts``GET /api/share/:shareCode` 是公开路由(泛匹配,放最后避免拦截其他路由),访问时记录访问者 IP → 302 重定向到 H5 首页
14. **小程序官网购买走剪贴板**: 小程序内"官网购买"不再使用 webview 内嵌 H5,改为 `uni.setClipboardData` 复制带 JWT token 的 URL 到剪贴板,提示用户在手机浏览器中打开购买(`App.vue``handleH5UrlParams` 解析 `?token=``?buy=gravity` 参数自动登录跳转)
---
## 十一、交付检查清单(每次实施/修改后执行)
### Step 1: 代码评审
- [ ] 是否符合现有模块模式(schema→service→controller→module
- [ ] 是否有多余变量、import、参数
- [ ] 命名是否与项目一致
- [ ] 是否有重复代码或可复用逻辑
### Step 2: 安全评审
- [ ] 外部输入是否有注入风险(HTML、Shell、NoSQL
- [ ] 接口是否正确加了 JWT guard 或 `@Public()`
- [ ] 管理端接口是否有 AdminGuard
- [ ] 修改用户额度/积分是否用 `$inc` 原子操作
- [ ] 敏感信息是否泄漏到响应或日志中
- [ ] 用户内容输出是否做了转义
### Step 3: 性能评审
- [ ] 是否存在 N+1 查询
- [ ] 大表查询是否有索引覆盖
- [ ] 读操作是否该加 `.lean()`
- [ ] 是否有内存泄漏风险(puppeteer 等资源未释放)
### Step 4: 测试验证
```bash
cd backend && npm run build
npm test -- --forceExit --detectOpenHandles
```
### Step 5: LSP 诊断
- [ ] Changed files diagnostics clean
- [ ] 无 `as any` / `@ts-ignore` / `@ts-expect-error`
+156
View File
@@ -0,0 +1,156 @@
/**
* 测试数据清理脚本
* 识别并清理测试用户及其关联数据(订单/面试/简历/分享)
* 保留所有管理员账号
*
* 用法:
* 预览模式(推荐先执行): npx ts-node --project tsconfig.json scripts/cleanup-test-data.ts
* 执行清理: npx ts-node --project tsconfig.json scripts/cleanup-test-data.ts --execute
*/
import { NestFactory } from '@nestjs/core'
import { AppModule } from '../src/app.module'
import { getModelToken } from '@nestjs/mongoose'
import { Model } from 'mongoose'
import { User, UserDocument } from '../src/modules/user/user.schema'
import { Interview, InterviewDocument } from '../src/modules/interview/interview.schema'
import { Resume, ResumeDocument } from '../src/modules/resume/resume.schema'
import { PaymentOrder, PaymentOrderDocument } from '../src/modules/payment/payment-order.schema'
import { ShareRecord, ShareRecordDocument, ShareVisit, ShareVisitDocument } from '../src/modules/share/share.schema'
// 已知测试账号(保留这些不清理——防止误伤真实用户)
const KNOWN_TEST_ACCOUNTS = [
'test@yzrcloud.cn',
'test@test.com',
]
// 管理员账号(始终保留)
const ADMIN_EMAILS = [
'13701190814@139.com',
]
async function bootstrap() {
const isExecute = process.argv.includes('--execute')
const app = await NestFactory.createApplicationContext(AppModule)
const userModel = app.get<Model<UserDocument>>(getModelToken(User.name))
const interviewModel = app.get<Model<InterviewDocument>>(getModelToken(Interview.name))
const resumeModel = app.get<Model<ResumeDocument>>(getModelToken(Resume.name))
const orderModel = app.get<Model<PaymentOrderDocument>>(getModelToken(PaymentOrder.name))
const shareModel = app.get<Model<ShareRecordDocument>>(getModelToken(ShareRecord.name))
const shareVisitModel = app.get<Model<ShareVisitDocument>>(getModelToken(ShareVisit.name))
// Step 1: 找到所有管理员(保留)
const admins = await userModel.find({ role: 'admin' }).lean().exec()
const adminIds = admins.map(a => a._id.toString())
const adminEmails = admins.map(a => a.email).filter(Boolean)
// Step 2: 找到测试用户
const allUsers = await userModel.find().sort({ createdAt: -1 }).lean().exec()
const allAdminEmails = [...new Set([...ADMIN_EMAILS, ...adminEmails])]
const knownTestEmails = KNOWN_TEST_ACCOUNTS
// 测试用户识别策略:
// a) 已知测试邮箱
// b) 非管理员
// 实际用户自己补充识别条件
const testUserIds: string[] = []
const skippedUserIds: string[] = []
for (const u of allUsers) {
const uid = u._id.toString()
// 跳过管理员
if (adminIds.includes(uid)) {
skippedUserIds.push(uid)
continue
}
const userEmail = (u.email || '').toLowerCase().trim()
// 已知测试邮箱 → 标记为测试
if (knownTestEmails.includes(userEmail)) {
testUserIds.push(uid)
console.log(` [TEST] ${u.phone || '--'} / ${userEmail} / ${u.nickname || '--'}`)
continue
}
// 找不到匹配条件的用户属于真正用户
skippedUserIds.push(uid)
}
console.log(`\n========================================`)
console.log(`总用户: ${allUsers.length}`)
console.log(`管理员: ${adminIds.length}`)
console.log(`测试用户: ${testUserIds.length}`)
console.log(`真实用户: ${skippedUserIds.length}`)
console.log(`========================================\n`)
if (testUserIds.length === 0) {
console.log('未发现测试用户,无需清理。')
await app.close()
return
}
// Step 3: 统计关联数据
const testInterviews = await interviewModel.find({ userId: { $in: testUserIds } }).lean().exec()
const testResumes = await resumeModel.find({ userId: { $in: testUserIds } }).lean().exec()
const testOrders = await orderModel.find({ userId: { $in: testUserIds } }).lean().exec()
const testShares = await shareModel.find({ userId: { $in: testUserIds } }).lean().exec()
console.log('将删除的关联数据:')
console.log(` - 面试记录: ${testInterviews.length}`)
console.log(` - 简历: ${testResumes.length}`)
console.log(` - 订单: ${testOrders.length}`)
console.log(` - 分享记录: ${testShares.length}`)
console.log(` - 用户账号: ${testUserIds.length}\n`)
if (!isExecute) {
console.log('⚠️ 预览模式,未执行删除。')
console.log(' 确认无误后执行: npx ts-node --project tsconfig.json scripts/cleanup-test-data.ts --execute\n')
await app.close()
return
}
// Step 4: 执行删除(先删关联数据再删用户)
console.log('正在清理...')
// 分享访问记录(通过分享记录找)
const shareIds = testShares.map(s => s._id)
if (shareIds.length > 0) {
const visits = await shareVisitModel.deleteMany({ shareId: { $in: shareIds } }).exec()
console.log(` - 删除分享访问记录: ${visits.deletedCount}`)
}
if (testShares.length > 0) {
const r = await shareModel.deleteMany({ _id: { $in: shareIds } }).exec()
console.log(` - 删除分享记录: ${r.deletedCount}`)
}
if (testInterviews.length > 0) {
const r = await interviewModel.deleteMany({ _id: { $in: testInterviews.map(i => i._id) } }).exec()
console.log(` - 删除面试记录: ${r.deletedCount}`)
}
if (testResumes.length > 0) {
const r = await resumeModel.deleteMany({ _id: { $in: testResumes.map(r => r._id) } }).exec()
console.log(` - 删除简历: ${r.deletedCount}`)
}
if (testOrders.length > 0) {
const r = await orderModel.deleteMany({ _id: { $in: testOrders.map(o => o._id) } }).exec()
console.log(` - 删除订单: ${r.deletedCount}`)
}
if (testUserIds.length > 0) {
const r = await userModel.deleteMany({ _id: { $in: testUserIds } }).exec()
console.log(` - 删除测试用户: ${r.deletedCount}`)
}
console.log('\n✅ 清理完成!')
await app.close()
}
bootstrap().catch((err) => {
console.error('清理失败:', err)
process.exit(1)
})
+54
View File
@@ -0,0 +1,54 @@
/**
* 引力值迁移脚本
* 将现有用户的多维额度合并到 gravity 字段
* 公式: gravity = interviewCredits×5 + resumeOptimizeCredits×3 + resumeDownloadCredits×2 + shareCredits×1 + remaining×5
* 用法: npx ts-node --project tsconfig.json scripts/migrate-gravity.ts
*/
import { NestFactory } from '@nestjs/core'
import { AppModule } from '../src/app.module'
import { getModelToken } from '@nestjs/mongoose'
import { User, UserDocument } from '../src/modules/user/user.schema'
import { Model } from 'mongoose'
async function bootstrap() {
const app = await NestFactory.createApplicationContext(AppModule)
const userModel = app.get<Model<UserDocument>>(getModelToken(User.name))
const total = await userModel.countDocuments().exec()
console.log(`Total users: ${total}`)
let migrated = 0
let skipped = 0
const cursor = userModel.find().cursor()
for await (const user of cursor) {
const interviewVal = (user.interviewCredits ?? 0) * 5
const optimizeVal = (user.resumeOptimizeCredits ?? 0) * 3
const downloadVal = (user.resumeDownloadCredits ?? 0) * 2
const oldRemainVal = (user.remaining ?? 0) * 5
const shareVal = (user.shareCredits ?? 0) * 1
const totalGravity = interviewVal + optimizeVal + downloadVal + oldRemainVal + shareVal
if (totalGravity <= 0 && (user.gravity ?? 0) === 0) {
skipped++
continue
}
await userModel.findByIdAndUpdate(user._id, {
$set: {
gravity: Math.max(user.gravity ?? 0, totalGravity),
interviewCredits: 0,
resumeOptimizeCredits: 0,
resumeDownloadCredits: 0,
remaining: 0,
shareCredits: 0,
},
}).exec()
migrated++
}
console.log(`Migrated: ${migrated}, Skipped (no credits): ${skipped}`)
await app.close()
}
bootstrap().catch(console.error)
+4
View File
@@ -26,6 +26,8 @@ import { ScheduleModule } from './modules/schedule/schedule.module'
import { TtsModule } from './modules/tts/tts.module' import { TtsModule } from './modules/tts/tts.module'
import { PricingModule } from './modules/schemas/pricing.module' import { PricingModule } from './modules/schemas/pricing.module'
import { ShareModule } from './modules/share/share.module' import { ShareModule } from './modules/share/share.module'
import { InterviewReviewModule } from './modules/interview-review/interview-review.module'
import { CareerAdviceModule } from './modules/career-advice/career-advice.module'
const MONGODB_URI = process.env.MONGODB_URI || 'mongodb://localhost:27017/zhiyin' const MONGODB_URI = process.env.MONGODB_URI || 'mongodb://localhost:27017/zhiyin'
@@ -60,6 +62,8 @@ const MONGODB_URI = process.env.MONGODB_URI || 'mongodb://localhost:27017/zhiyin
TtsModule, TtsModule,
PricingModule, PricingModule,
ShareModule, ShareModule,
InterviewReviewModule,
CareerAdviceModule,
], ],
providers: [ providers: [
JwtStrategy, JwtStrategy,
+79 -25
View File
@@ -1,4 +1,4 @@
import { Controller, Get, Post, Body, Query, HttpException, HttpStatus, UseGuards } from '@nestjs/common' import { Controller, Get, Post, Body, Query, Param, HttpException, HttpStatus, UseGuards } from '@nestjs/common'
import { InjectModel } from '@nestjs/mongoose' import { InjectModel } from '@nestjs/mongoose'
import { Model } from 'mongoose' import { Model } from 'mongoose'
import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard' import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard'
@@ -48,7 +48,7 @@ export class AdminController {
const [ const [
userCount, interviewCount, todayUsers, todayInterviews, userCount, interviewCount, todayUsers, todayInterviews,
resumeCount, paidDownloadCount, resumeCount, paidDownloadCount,
planStats, planStats, orderCount, todayOrders, totalRevenue,
] = await Promise.all([ ] = await Promise.all([
this.userModel.countDocuments().exec(), this.userModel.countDocuments().exec(),
this.interviewModel.countDocuments().exec(), this.interviewModel.countDocuments().exec(),
@@ -59,12 +59,19 @@ export class AdminController {
this.userModel.aggregate([ this.userModel.aggregate([
{ $group: { _id: '$plan', count: { $sum: 1 } } }, { $group: { _id: '$plan', count: { $sum: 1 } } },
]).exec(), ]).exec(),
this.orderModel.countDocuments().exec(),
this.orderModel.countDocuments({ status: 'success', createdAt: { $gte: new Date(Date.now() - 24 * 60 * 60 * 1000) } }).exec(),
this.orderModel.aggregate([
{ $match: { status: 'success' } },
{ $group: { _id: null, total: { $sum: '$amount' } } },
]).exec(),
]) ])
const planBreakdown: Record<string, number> = {} const planBreakdown: Record<string, number> = {}
planStats.forEach(p => { planBreakdown[p._id || 'free'] = p.count }) planStats.forEach(p => { planBreakdown[p._id || 'free'] = p.count })
return { return {
userCount, interviewCount, todayUsers, todayInterviews, userCount, interviewCount, todayUsers, todayInterviews,
resumeCount, paidDownloadCount, resumeCount, paidDownloadCount, orderCount, todayOrders,
totalRevenue: totalRevenue[0]?.total || 0,
planBreakdown, planBreakdown,
} }
} }
@@ -95,8 +102,8 @@ export class AdminController {
this.interviewModel.find() this.interviewModel.find()
.sort({ createdAt: -1 }) .sort({ createdAt: -1 })
.skip(skip).limit(+limit) .skip(skip).limit(+limit)
.populate('userId', 'phone nickname') .populate('userId', 'phone nickname email wxOpenid')
.select('position status totalScore questionCount fillerScore fillerDensity summary createdAt') .select('position status totalScore questionCount fillerScore fillerDensity summary createdAt updatedAt')
.lean().exec(), .lean().exec(),
this.interviewModel.countDocuments().exec(), this.interviewModel.countDocuments().exec(),
]) ])
@@ -113,7 +120,7 @@ export class AdminController {
expireAt.setDate(expireAt.getDate() + (pricing.plans?.growth?.durationDays || VIP_DURATION_DAYS)) expireAt.setDate(expireAt.getDate() + (pricing.plans?.growth?.durationDays || VIP_DURATION_DAYS))
user.plan = 'growth' user.plan = 'growth'
user.vipExpireAt = expireAt user.vipExpireAt = expireAt
await this.quotaService.setPlanQuota(targetUserId, 'growth', credits) await this.quotaService.setPlanQuota(targetUserId, pricing.plans?.growth?.gravityPerMonth || 250)
return { success: true, plan: 'growth', expireAt } return { success: true, plan: 'growth', expireAt }
} }
@@ -122,7 +129,7 @@ export class AdminController {
if (!userId || !type || amount === undefined) { if (!userId || !type || amount === undefined) {
throw new HttpException('参数不完整', HttpStatus.BAD_REQUEST) throw new HttpException('参数不完整', HttpStatus.BAD_REQUEST)
} }
const validTypes = ['interviewCredits', 'resumeOptimizeCredits', 'resumeDownloadCredits', 'shareCredits'] const validTypes = ['interviewCredits', 'resumeOptimizeCredits', 'resumeDownloadCredits', 'shareCredits', 'gravity']
if (!validTypes.includes(type)) { if (!validTypes.includes(type)) {
throw new HttpException('无效的额度类型', HttpStatus.BAD_REQUEST) throw new HttpException('无效的额度类型', HttpStatus.BAD_REQUEST)
} }
@@ -217,8 +224,8 @@ export class AdminController {
this.resumeModel.find() this.resumeModel.find()
.sort({ createdAt: -1 }) .sort({ createdAt: -1 })
.skip(skip).limit(+limit) .skip(skip).limit(+limit)
.populate('userId', 'phone nickname') .populate('userId', 'phone nickname email')
.select('title targetPosition version paidDownload createdAt') .select('title targetPosition version paidDownload createdAt updatedAt contentHash')
.lean().exec(), .lean().exec(),
this.resumeModel.countDocuments().exec(), this.resumeModel.countDocuments().exec(),
]) ])
@@ -226,7 +233,7 @@ export class AdminController {
} }
@Get('user/:id') @Get('user/:id')
async getUserDetail(@Query('id') id: string) { async getUserDetail(@Param('id') id: string) {
const user = await this.userModel.findById(id).select('-password -openid').lean().exec() const user = await this.userModel.findById(id).select('-password -openid').lean().exec()
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND) if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
const [interviews, resumes] = await Promise.all([ const [interviews, resumes] = await Promise.all([
@@ -279,29 +286,76 @@ export class AdminController {
const user = await this.userModel.findById(order.userId).exec() const user = await this.userModel.findById(order.userId).exec()
if (user && user.plan === 'free') { if (user && user.plan === 'free') {
const pricing = await this.pricingService.getConfig() const pricing = await this.pricingService.getConfig()
const credits = pricing.plans?.growth?.credits || { interview: 999, resumeOptimize: 20, resumeDownload: 10 } const planId = order.plan === 'sprint' ? 'sprint' : 'growth'
const planCfg = pricing.plans?.[planId]
const credits = planCfg?.credits || { interview: 999, resumeOptimize: 20, resumeDownload: 10 }
const expireAt = new Date() const expireAt = new Date()
expireAt.setDate(expireAt.getDate() + (pricing.plans?.growth?.durationDays || VIP_DURATION_DAYS)) expireAt.setDate(expireAt.getDate() + (planCfg?.durationDays || VIP_DURATION_DAYS))
user.plan = 'growth' user.plan = planId
if (planId === 'sprint') {
user.sprintExpireAt = expireAt
user.sprintRemaining = 10
} else {
user.vipExpireAt = expireAt user.vipExpireAt = expireAt
await this.quotaService.setPlanQuota(order.userId, 'growth', credits) }
await this.quotaService.setPlanQuota(order.userId, planCfg.gravityPerMonth)
} }
} else { } else {
const pricing = await this.pricingService.getConfig() const pricing = await this.pricingService.getConfig()
const creditMap: Record<string, number> = { const gravityMap: Record<string, number> = {
interview: pricing.interview?.creditsPerPurchase || 1, interview: pricing.gravityRates?.interviewPerUse || 5,
optimize: pricing.resumeOptimize?.creditsPerPurchase || 1, optimize: pricing.gravityRates?.optimizePerUse || 3,
download: pricing.resumeDownload?.creditsPerPurchase || 1, download: pricing.gravityRates?.downloadPerUse || 2,
} }
const credits = creditMap[order.type] const g = gravityMap[order.type]
if (credits) { if (g) {
await this.quotaService.grantCredits(order.userId, order.type as any, credits) await this.quotaService.grantGravity(order.userId, g)
} }
} }
} }
return { order, wxResult } return { order, wxResult }
} }
/** 订单详情(含用户信息) */
@Get('order/:outTradeNo')
async getOrderDetail(@Param('outTradeNo') outTradeNo: string) {
const order = await this.orderModel.findOne({ outTradeNo }).lean().exec()
if (!order) throw new HttpException('订单不存在', HttpStatus.NOT_FOUND)
const user = await this.userModel.findById(order.userId).select('phone nickname plan').lean().exec()
return { order, user }
}
/** 发起退款 */
@Post('order/refund')
async refundOrder(@Body('outTradeNo') outTradeNo: string, @Body('amount') amount?: number, @Body('reason') reason?: string) {
const order = await this.orderModel.findOne({ outTradeNo }).exec()
if (!order) throw new HttpException('订单不存在', HttpStatus.NOT_FOUND)
if (order.status !== 'success') throw new HttpException('仅支付成功的订单可退款', HttpStatus.BAD_REQUEST)
if (order.refundAmount && order.refundAmount > 0) throw new HttpException('该订单已退款', HttpStatus.BAD_REQUEST)
const result = await this.wechatPay.refund(outTradeNo, order.amount, amount || order.amount, reason)
const refundId = result?.refund_id || ''
order.status = 'refunded'
order.refundAmount = amount || order.amount
order.refundedAt = new Date()
order.refundReason = reason || ''
order.refundId = refundId
await order.save()
return { success: true, refundId }
}
/** 查询微信侧退款状态 */
@Get('order/refund/:outTradeNo')
async queryRefund(@Param('outTradeNo') outTradeNo: string) {
const order = await this.orderModel.findOne({ outTradeNo }).lean().exec()
if (!order) throw new HttpException('订单不存在', HttpStatus.NOT_FOUND)
if (!order.refundId) return { localStatus: order.status, message: '无微信退款单号' }
const wxResult = await this.wechatPay.queryRefund(order.refundId)
return { localStatus: order.status, wxRefund: wxResult }
}
@Get('config') @Get('config')
async getConfig() { async getConfig() {
const cfg = await this.configModel.findOne({ key: 'site_config' }).exec() const cfg = await this.configModel.findOne({ key: 'site_config' }).exec()
@@ -362,8 +416,8 @@ const DEFAULT_CONFIG = {
optimize: { dailyFreeLimit: 2 }, optimize: { dailyFreeLimit: 2 },
price: { monthly: 1990 }, price: { monthly: 1990 },
plans: { plans: {
free: { name: '免费版', price: 0, features: ['AI 模拟面试 1 次(体验)', '每场最多 5 轮 AI 对话', '基础面试报告', '简历优化(限 3 次)'] }, free: { name: '免费版', price: 0, features: ['AI 模拟面试 1 次(体验)', '基础面试报告', '通用题库随机出题', '简历优化(限 3 次免费'] },
growth: { name: '成长版', price: 1990, features: ['免费版全部权益', 'AI 数字人面试无限次', '每场最多 10 轮 AI 对话', '详细面试报告(四维评分)', '进步轨迹雷达图 + 打卡', '参考回答思路', '公司真题库', '简历优化 20 次/月', '简历下载 10 次/月'] }, growth: { name: '成长版', price: 1990, features: ['免费版全部权益', 'AI 数字人面试无限次', '详细面试报告(四维评分)', '进步轨迹雷达图 + 打卡', '每日一题', '参考回答思路', '公司真题库', '简历优化 20 次/月', '简历下载 10 次/月'] },
}, },
} }
@@ -375,12 +429,12 @@ const DEFAULT_PRICING = {
growth: { growth: {
price: 1990, durationDays: 30, price: 1990, durationDays: 30,
credits: { interview: 999, resumeOptimize: 20, resumeDownload: 10 }, credits: { interview: 999, resumeOptimize: 20, resumeDownload: 10 },
features: ['免费版全部权益', 'AI 数字人面试无限次', '详细面试报告(四维评分)', '进步轨迹雷达图 + 打卡', '每日一题推送', '参考回答思路', '公司真题库', '简历优化 20 次/月', '简历下载 10 次/月'], features: ['免费版全部权益', 'AI 数字人面试无限次', '详细面试报告(四维评分)', '进步轨迹雷达图 + 打卡', '每日一题', '参考回答思路', '公司真题库', '简历优化 20 次/月', '简历下载 10 次/月'],
}, },
sprint: { sprint: {
price: 4990, durationDays: 30, price: 4990, durationDays: 30,
credits: { interview: 999, resumeOptimize: 50, resumeDownload: 30 }, credits: { interview: 999, resumeOptimize: 50, resumeDownload: 30 },
features: ['成长版全部权益', 'AI 语音分析(语气词/语速检测)', '技能缺口分析报告', '学习路径推荐', '真人导师 1v1 点评(每月 1 次)', '简历精修(每月 1 次)', '内推优先', '简历优化 50 次/月', '简历下载 30 次/月'], features: ['成长版全部权益', 'AI 语音分析(语气词/语速检测)', '技能缺口分析报告', '公司真题库精选', '简历优化 50 次/月', '简历下载 30 次/月'],
}, },
}, },
} }
+40 -22
View File
@@ -1,6 +1,6 @@
import { Injectable, Logger } from '@nestjs/common' import { Injectable, Logger } from "@nestjs/common"
import axios from 'axios' import axios from "axios"
import https from 'https' import https from "https"
interface AiCallOptions { interface AiCallOptions {
systemPrompt: string systemPrompt: string
@@ -15,63 +15,81 @@ const httpAgent = new https.Agent({ rejectUnauthorized: true, keepAlive: true })
export class AiService { export class AiService {
private readonly logger = new Logger(AiService.name) private readonly logger = new Logger(AiService.name)
private readonly primaryUrl = process.env.AI_PRIMARY_URL || 'https://token.sensenova.cn/v1' private readonly primaryUrl = process.env.AI_PRIMARY_URL || "https://token.sensenova.cn/v1"
private readonly primaryKey = process.env.AI_PRIMARY_KEY || '' private readonly primaryKey = process.env.AI_PRIMARY_KEY || ""
private readonly primaryModel = process.env.AI_PRIMARY_MODEL || 'deepseek-v4-flash' private readonly primaryModel = process.env.AI_PRIMARY_MODEL || "deepseek-v4-flash"
private readonly primaryFallbackModel = process.env.AI_PRIMARY_FALLBACK_MODEL || "sensenova-6.7-flash-lite"
private readonly backupUrl = process.env.AI_BACKUP_URL || 'https://integrate.api.nvidia.com/v1' private readonly backupUrl = process.env.AI_BACKUP_URL || "https://integrate.api.nvidia.com/v1"
private readonly backupKey = process.env.AI_BACKUP_KEY || '' private readonly backupKey = process.env.AI_BACKUP_KEY || ""
private readonly backupModel = process.env.AI_BACKUP_MODEL || 'stepfun-ai/step-3.5-flash' private readonly backupModel = process.env.AI_BACKUP_MODEL || "meta/llama-3.1-8b-instruct"
async call(options: AiCallOptions): Promise<string> { async call(options: AiCallOptions): Promise<string> {
const { systemPrompt, userMessage, temperature = 0.7, maxTokens = 2048 } = options const { systemPrompt, userMessage, temperature = 0.7, maxTokens = 2048 } = options
// Try primary AI // Try primary AI (deepseek-v4-flash on sensenova)
try { try {
const result = await this.callApi(this.primaryUrl, this.primaryKey, this.primaryModel, systemPrompt, userMessage, temperature, maxTokens) const result = await this.callApi(this.primaryUrl, this.primaryKey, this.primaryModel, systemPrompt, userMessage, temperature, maxTokens, 60000)
if (result) return result if (result) return result
// Primary returned empty content (thinking model exhausted tokens); retry with more tokens
const retry = await this.callApi(this.primaryUrl, this.primaryKey, this.primaryModel, systemPrompt, userMessage, temperature, Math.min(maxTokens * 2, 4096), 60000)
if (retry) return retry
} catch (e) { } catch (e) {
this.logger.warn(`Primary AI failed: ${(e as Error).message}, trying backup...`) this.logger.warn(`Primary AI failed: ${(e as Error).message}, trying primary fallback...`)
} }
// Try backup AI // Try primary fallback model (sensenova-6.7-flash-lite, same provider)
try { try {
const result = await this.callApi(this.backupUrl, this.backupKey, this.backupModel, systemPrompt, userMessage, temperature, maxTokens) const result = await this.callApi(this.primaryUrl, this.primaryKey, this.primaryFallbackModel, systemPrompt, userMessage, temperature, maxTokens, 60000)
if (result) return result
} catch (e) {
this.logger.warn(`Primary fallback AI also failed: ${(e as Error).message}, trying backup...`)
}
// Try backup AI (NVIDIA - meta/llama-3.1-8b-instruct)
try {
const result = await this.callApi(this.backupUrl, this.backupKey, this.backupModel, systemPrompt, userMessage, temperature, Math.max(maxTokens, 2048), 120000)
if (result) return result if (result) return result
} catch (e) { } catch (e) {
this.logger.warn(`Backup AI also failed: ${(e as Error).message}`) this.logger.warn(`Backup AI also failed: ${(e as Error).message}`)
} }
// Final fallback throw new Error("AI 服务暂时不可用,请稍后重试")
throw new Error('AI 服务暂时不可用,请稍后重试')
} }
private async callApi( private async callApi(
baseUrl: string, apiKey: string, model: string, baseUrl: string, apiKey: string, model: string,
systemPrompt: string, userMessage: string, systemPrompt: string, userMessage: string,
temperature: number, maxTokens: number, temperature: number, maxTokens: number, timeout: number,
): Promise<string | null> { ): Promise<string | null> {
try {
const res = await axios.post( const res = await axios.post(
`${baseUrl}/chat/completions`, `${baseUrl}/chat/completions`,
{ {
model, model,
messages: [ messages: [
{ role: 'system', content: systemPrompt }, { role: "system", content: systemPrompt },
{ role: 'user', content: userMessage }, { role: "user", content: userMessage },
], ],
temperature, temperature,
max_tokens: maxTokens, max_tokens: maxTokens,
}, },
{ {
headers: { headers: {
'Authorization': `Bearer ${apiKey}`, "Authorization": `Bearer ${apiKey}`,
'Content-Type': 'application/json', "Content-Type": "application/json",
}, },
timeout: 60000, timeout,
httpsAgent: httpAgent, httpsAgent: httpAgent,
transitional: { clarifyTimeoutError: true }, transitional: { clarifyTimeoutError: true },
}, },
) )
return res.data?.choices?.[0]?.message?.content || null return res.data?.choices?.[0]?.message?.content || null
} catch (e: any) {
if (e.code === 'ECONNABORTED') {
this.logger.warn(`AI call timeout (${timeout}ms): ${model}`)
}
return null
}
} }
} }
@@ -0,0 +1,37 @@
import { Controller, Post, Get, Body } from "@nestjs/common"
import { Public } from "../../common/decorators/public.decorator"
import { CareerAdviceService, CareerProfile, ChatMessage } from "./career-advice.service"
import { CurrentUser } from "../../common/decorators/current-user.decorator"
@Controller("career-advice")
export class CareerAdviceController {
constructor(private service: CareerAdviceService) {}
@Post("analyze")
async analyze(
@Body() profile: CareerProfile,
@CurrentUser("userId") userId: string,
) {
if (!profile.major || !profile.major.trim()) {
return { error: "请填写你的专业" }
}
return this.service.analyze(profile)
}
@Post("chat")
async chat(
@Body() body: { message: string; history: ChatMessage[] },
@CurrentUser("userId") userId: string,
) {
if (!body.message || !body.message.trim()) {
return { error: "请输入消息" }
}
return this.service.chat(body.message, body.history || [])
}
@Get("positions")
@Public()
async positions() {
return this.service.getHotPositions()
}
}
@@ -0,0 +1,18 @@
import { Module } from '@nestjs/common'
import { MongooseModule } from '@nestjs/mongoose'
import { CareerAdviceController } from './career-advice.controller'
import { CareerAdviceService } from './career-advice.service'
import { HotPosition, HotPositionSchema } from '../positions/positions.schema'
import { AiModule } from '../ai/ai.module'
@Module({
imports: [
MongooseModule.forFeature([
{ name: HotPosition.name, schema: HotPositionSchema },
]),
AiModule,
],
controllers: [CareerAdviceController],
providers: [CareerAdviceService],
})
export class CareerAdviceModule {}
@@ -0,0 +1,163 @@
import { Injectable, Logger } from '@nestjs/common'
import { InjectModel } from '@nestjs/mongoose'
import { Model } from 'mongoose'
import { AiService } from '../ai/ai.service'
import { HotPosition } from '../positions/positions.schema'
export interface CareerProfile {
major: string
grade?: string
interests?: string
gpa?: string
goal?: string
}
export interface CareerPath {
name: string
reason: string
matchScore: number
salary?: string
}
export interface ChatMessage {
role: 'user' | 'assistant'
content: string
}
@Injectable()
export class CareerAdviceService {
private readonly logger = new Logger(CareerAdviceService.name)
constructor(
@InjectModel(HotPosition.name) private positionModel: Model<HotPosition>,
private aiService: AiService,
) {}
/**
* Analyze user profile and return career advice with position recommendations.
*/
async analyze(profile: CareerProfile) {
const positions = await this.positionModel.find({ active: true })
.sort({ sort: 1 })
.lean()
.exec()
const positionsContext = positions.map(p =>
`- ${p.name}${p.salary ? ` (薪资: ${p.salary})` : ''}${p.company ? ` - ${p.company}` : ''}`
).join('\n')
const systemPrompt = `你是一位资深的中国大学生职业规划顾问。你的任务是根据学生的专业背景和个人情况,给出个性化的择业建议。
你的建议需要涵盖:
1. 该专业的典型职业方向和发展路径
2. "考研vs就业vs考公"的决策分析(基于学生具体情况)
3. 当前就业市场的真实形势(如AI对各行业的影响)
4. 具体可行动的建议(实习、技能提升等)
你需要从以下可用岗位中,推荐最匹配该学生的3-5个方向:
${positionsContext}
注意:
- 你的回答要诚恳务实,不要画大饼
- 指出每个选择的利弊和风险
- 结合AI时代对各行业的影响给出建议
- 最后以JSON格式输出推荐岗位列表
回复格式:
先输出一段详细的个性化分析建议(中文,300-500字)。
然后在最后单独一行输出JSON数组(不要任何其他内容):
---JSON---
[{"name":"岗位名","reason":"推荐理由简要说明","matchScore":85,"salary":"薪资范围"},...]
---JSON---`
const userMessage = this.buildProfileMessage(profile)
const rawReply = await this.aiService.call({
systemPrompt,
userMessage,
temperature: 0.7,
maxTokens: 2048,
})
return this.parseResponse(rawReply, positions)
}
/**
* Continue a chat conversation about career choices.
*/
async chat(message: string, history: ChatMessage[]) {
const positions = await this.positionModel.find({ active: true })
.sort({ sort: 1 })
.lean()
.exec()
const positionsContext = positions.map(p =>
`- ${p.name}${p.salary ? ` (薪资: ${p.salary})` : ''}`
).join('\n')
const systemPrompt = `你是一位资深的中国大学生职业规划顾问。继续与学生的对话,回答他们的择业相关问题。
可推荐的岗位列表:
${positionsContext}
注意:
- 回答诚恳务实,结合AI时代背景
- 给出具体可操作的建议
- 如果学生提到具体岗位,可以从推荐列表中选择匹配的
- 保持对话亲切自然,用中文回复`
const historyMessages = history.map(h =>
`${h.role === 'user' ? '学生' : '顾问'}: ${h.content}`
).join('\n')
const userMessage = `对话历史:\n${historyMessages}\n\n学生最新提问: ${message}`
const reply = await this.aiService.call({
systemPrompt,
userMessage,
temperature: 0.7,
maxTokens: 1024,
})
return { reply }
}
private buildProfileMessage(profile: CareerProfile): string {
const parts: string[] = [`学生专业: ${profile.major}`]
if (profile.grade) parts.push(`年级: ${profile.grade}`)
if (profile.interests) parts.push(`兴趣方向: ${profile.interests}`)
if (profile.gpa) parts.push(`GPA/成绩: ${profile.gpa}`)
if (profile.goal) parts.push(`目标/困惑: ${profile.goal}`)
return parts.join('\n')
}
private parseResponse(rawReply: string, allPositions: any[]): {
reply: string
careerPaths: CareerPath[]
} {
// Extract JSON array from ---JSON--- markers
const jsonMatch = rawReply.match(/---JSON---\s*(\[[\s\S]*?\])\s*---JSON---/)
let careerPaths: CareerPath[] = []
if (jsonMatch) {
try {
careerPaths = JSON.parse(jsonMatch[1])
} catch {
this.logger.warn('Failed to parse career paths JSON')
}
}
// Clean reply (remove --JSON-- markers)
const reply = rawReply.replace(/---JSON---[\s\S]*?---JSON---/g, '').trim()
return { reply, careerPaths }
}
/** Get hot positions (delegated from positions module) */
async getHotPositions() {
return this.positionModel.find({ active: true })
.sort({ sort: 1 })
.lean()
.exec()
}
}
@@ -2,6 +2,7 @@ import { Controller, Post, Get, Body, Param, UseGuards, Logger } from '@nestjs/c
import { InjectModel } from '@nestjs/mongoose' import { InjectModel } from '@nestjs/mongoose'
import { Model } from 'mongoose' import { Model } from 'mongoose'
import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard' import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard'
import { Public } from '../../common/decorators/public.decorator'
import { CurrentUser } from '../../common/decorators/current-user.decorator' import { CurrentUser } from '../../common/decorators/current-user.decorator'
import { AiService } from '../ai/ai.service' import { AiService } from '../ai/ai.service'
import { Contribution, ContributionDocument } from '../schemas/contribution.schema' import { Contribution, ContributionDocument } from '../schemas/contribution.schema'
@@ -186,4 +187,19 @@ export class ContributionController {
.select('company position rounds experience createdAt') .select('company position rounds experience createdAt')
.exec() .exec()
} }
@Public()
@Get('companies/hot')
async getHotCompanies() {
const banks = await this.companyBankModel.aggregate([
{ $group: { _id: '$company', positionCount: { $sum: 1 }, totalContributions: { $sum: '$contributionCount' } } },
{ $sort: { totalContributions: -1, positionCount: -1 } },
{ $project: { _id: 0, name: '$_id', positionCount: 1 } },
]).exec()
if (banks.length > 0) return banks
const DEFAULT_COMPANIES = ['腾讯', '字节跳动', '阿里巴巴', '美团', '百度', '京东', '网易', '小红书']
return DEFAULT_COMPANIES.map((name, i) => ({ name, positionCount: 0, sort: i }))
}
} }
@@ -0,0 +1,218 @@
import { Injectable, Logger } from '@nestjs/common'
import { execSync } from 'child_process'
import * as path from 'path'
import * as fs from 'fs'
export interface AsrSegment {
startTime: number
endTime: number
speaker: 'interviewer' | 'candidate'
text: string
}
export interface AsrResult {
fullText: string
segments: AsrSegment[]
duration: number
}
export interface AsrConfig {
/** Path to whisper.cpp build/bin/ directory */
whisperCppPath?: string
/** Model name: tiny | base | small | medium */
model?: string
/** Language code (zh, en, auto) */
language?: string
}
@Injectable()
export class AsrService {
private readonly logger = new Logger(AsrService.name)
private readonly whisperCppPath: string
private readonly modelPath: string
private readonly language: string
private readonly cliPath: string
constructor() {
// Configuration via env vars with sensible defaults
this.whisperCppPath = process.env.WHISPER_CPP_PATH || '/home/wlt/whisper.cpp'
this.language = process.env.WHISPER_LANGUAGE || 'zh'
const modelName = process.env.WHISPER_MODEL || 'base'
this.modelPath = path.join(this.whisperCppPath, 'models', `ggml-${modelName}.bin`)
this.cliPath = path.join(this.whisperCppPath, 'build', 'bin', 'whisper-cli')
// Validate whisper.cpp installation on startup
if (!fs.existsSync(this.cliPath)) {
this.logger.warn(`whisper-cli not found at ${this.cliPath}. ASR will fall back to mock.`)
}
if (!fs.existsSync(this.modelPath)) {
this.logger.warn(`Whisper model not found at ${this.modelPath}. ASR will fall back to mock.`)
}
}
async transcribe(audioPath: string, _mimeType: string): Promise<AsrResult> {
this.logger.log(`Transcribing audio: ${audioPath}`)
// Try companion .txt file first (for debugging/testing)
const txtPath = audioPath.replace(/\.(mp3|m4a|wav|aac|ogg|mp4|webm)$/i, '.txt')
try {
if (fs.existsSync(txtPath)) {
const text = fs.readFileSync(txtPath, 'utf-8')
this.logger.log(`Found companion transcript: ${txtPath}`)
return {
fullText: text,
segments: [{
startTime: 0,
endTime: Math.max(text.length / 3.5, 10),
speaker: 'candidate',
text,
}],
duration: Math.max(text.length / 3.5, 10),
}
}
} catch { /* ignore */ }
// Try whisper.cpp
if (fs.existsSync(this.cliPath) && fs.existsSync(this.modelPath)) {
try {
return await this.transcribeWithWhisper(audioPath)
} catch (err: any) {
this.logger.error(`whisper.cpp transcription failed: ${err.message}, falling back to mock`)
}
}
// Fallback to mock
this.logger.warn('Using MOCK ASR — whisper.cpp not available')
return this.mockTranscribe()
}
private async transcribeWithWhisper(audioPath: string): Promise<AsrResult> {
// Ensure audio file exists
if (!fs.existsSync(audioPath)) {
throw new Error(`Audio file not found: ${audioPath}`)
}
// Convert to WAV if needed (whisper.cpp works best with WAV)
const wavPath = await this.ensureWav(audioPath)
// Run whisper-cli with JSON output
const cmd = [
this.cliPath,
'-m', this.modelPath,
'-f', wavPath,
'-l', this.language,
'-oj', // JSON output
'-t', String(Math.max(1, this.getCpuThreads())), // thread count
'--no-prints', // suppress timing info on stderr
].join(' ')
this.logger.log(`Running: ${this.cliPath} -m ${this.modelPath} -f ${wavPath} -l ${this.language}`)
const stdout = execSync(cmd, { timeout: 600000, encoding: 'utf-8' }) // 10 min timeout
// Parse the JSON output
const segments = this.parseWhisperOutput(stdout)
const fullText = segments.map(s => s.text).join(' ')
const duration = segments.length > 0
? segments[segments.length - 1].endTime
: 0
return { fullText, segments, duration }
}
private parseWhisperOutput(stdout: string): AsrSegment[] {
try {
// whisper.cpp -oj outputs one JSON object per line
const lines = stdout.trim().split('\n')
const segments: AsrSegment[] = []
for (const line of lines) {
try {
const parsed = JSON.parse(line)
if (parsed.text && parsed.offsets) {
segments.push({
startTime: parsed.offsets.from / 1000, // ms to seconds
endTime: parsed.offsets.to / 1000,
speaker: 'candidate',
text: parsed.text.trim(),
})
} else if (parsed.text && parsed.start !== undefined) {
// Alternative format
segments.push({
startTime: parsed.start,
endTime: parsed.end || parsed.start + 2,
speaker: 'candidate',
text: parsed.text.trim(),
})
}
} catch { /* skip unparseable lines */ }
}
if (segments.length > 0) return segments
} catch { /* fall through */ }
// Fallback: treat entire output as raw text
this.logger.warn('Could not parse structured JSON output, using raw text')
return [{
startTime: 0,
endTime: 0,
speaker: 'candidate',
text: stdout.trim(),
}]
}
/**
* Convert audio to WAV format if needed.
* Uses ffmpeg if available, otherwise returns original path.
*/
private async ensureWav(audioPath: string): Promise<string> {
const ext = path.extname(audioPath).toLowerCase()
if (ext === '.wav') return audioPath
const wavPath = audioPath.replace(/\.[^.]+$/, '.wav')
try {
execSync(`ffmpeg -y -i "${audioPath}" -ar 16000 -ac 1 -c:a pcm_s16le "${wavPath}"`, {
timeout: 300000,
encoding: 'utf-8',
stdio: 'pipe',
})
this.logger.log(`Converted ${audioPath} to WAV: ${wavPath}`)
return wavPath
} catch (err: any) {
this.logger.warn(`ffmpeg conversion failed: ${err.message}. Trying original format.`)
return audioPath
}
}
private getCpuThreads(): number {
try {
return parseInt(process.env.WHISPER_THREADS || '', 10) ||
require('os').cpus().length || 4
} catch {
return 4
}
}
/** Mock transcription for development/testing when whisper.cpp is not available */
private mockTranscribe(): AsrResult {
const paragraphs = [
'我毕业于计算机科学与技术专业,大学期间主要学习了数据结构、算法、操作系统、计算机网络等核心课程。',
'在项目经验方面,我参与过一个电商平台的开发,主要负责后端接口的设计和实现,使用了 Node.js 和 MongoDB 技术栈。',
'这个项目的难点在于高并发场景下的性能优化,我通过引入 Redis 缓存和数据库索引优化,将接口响应时间从 2 秒降低到了 200 毫秒。',
'关于这个岗位,我了解到贵公司主要使用 React 技术栈,我之前在两个项目中使用过 React,对 Hooks、状态管理、组件化开发都比较熟悉。',
]
const fullText = paragraphs.join('\n')
return {
fullText,
segments: [{
startTime: 0,
endTime: 120,
speaker: 'candidate',
text: fullText,
}],
duration: 120,
}
}
}
@@ -0,0 +1,100 @@
import {
Controller, Post, Get, Delete, Param, Query,
UseInterceptors, UploadedFile, Body,
HttpException, HttpStatus,
} from '@nestjs/common'
import { FileInterceptor } from '@nestjs/platform-express'
import { diskStorage } from 'multer'
import { extname, join } from 'path'
import * as fs from 'fs'
import { randomUUID } from 'crypto'
import { InterviewReviewService } from './interview-review.service'
import { CurrentUser } from '../../common/decorators/current-user.decorator'
const UPLOAD_DIR = join(process.cwd(), 'uploads', 'reviews')
if (!fs.existsSync(UPLOAD_DIR)) {
fs.mkdirSync(UPLOAD_DIR, { recursive: true })
}
@Controller('interview-review')
export class InterviewReviewController {
constructor(private service: InterviewReviewService) {}
/** Upload audio file + metadata */
@Post()
@UseInterceptors(FileInterceptor('file', {
storage: diskStorage({
destination: (_req, _file, cb) => cb(null, UPLOAD_DIR),
filename: (_req, file, cb) => {
const name = randomUUID() + extname(file.originalname || '.mp3')
cb(null, name)
},
}),
limits: { fileSize: 50 * 1024 * 1024 },
fileFilter: (_req, file, cb) => {
const allowed = /\.(mp3|m4a|wav|aac|ogg|mp4|webm)$/i
if (allowed.test(extname(file.originalname))) {
cb(null, true)
} else {
cb(new HttpException('仅支持 mp3/m4a/wav/aac/ogg 格式', HttpStatus.BAD_REQUEST), false)
}
},
}))
async uploadFile(
@UploadedFile() file: any,
@Body('position') position: string,
@Body('company') company: string,
@CurrentUser('userId') userId: string,
) {
if (!file) {
throw new HttpException('请上传录音文件', HttpStatus.BAD_REQUEST)
}
if (!position || !position.trim()) {
throw new HttpException('请填写面试岗位', HttpStatus.BAD_REQUEST)
}
return this.service.create(userId, position.trim(), company?.trim(), file)
}
/** Submit text transcript directly (no audio) */
@Post('text')
async submitText(
@Body('position') position: string,
@Body('company') company: string,
@Body('text') text: string,
@CurrentUser('userId') userId: string,
) {
if (!position || !position.trim()) {
throw new HttpException('请填写面试岗位', HttpStatus.BAD_REQUEST)
}
if (!text || !text.trim()) {
throw new HttpException('请填写面试转录文本', HttpStatus.BAD_REQUEST)
}
return this.service.createFromText(userId, position.trim(), text.trim(), company?.trim())
}
@Get('list')
async list(
@Query('page') page: string,
@Query('limit') limit: string,
@CurrentUser('userId') userId: string,
) {
return this.service.listByUser(userId, parseInt(page) || 1, parseInt(limit) || 20)
}
@Get(':id')
async getDetail(
@Param('id') id: string,
@CurrentUser('userId') userId: string,
) {
return this.service.getDetail(id, userId)
}
@Delete(':id')
async delete(
@Param('id') id: string,
@CurrentUser('userId') userId: string,
) {
return this.service.delete(id, userId)
}
}
@@ -0,0 +1,19 @@
import { Module } from '@nestjs/common'
import { MongooseModule } from '@nestjs/mongoose'
import { InterviewReviewController } from './interview-review.controller'
import { InterviewReviewService } from './interview-review.service'
import { InterviewReview, InterviewReviewSchema } from './interview-review.schema'
import { AsrService } from './asr.service'
import { AiModule } from '../ai/ai.module'
@Module({
imports: [
MongooseModule.forFeature([
{ name: InterviewReview.name, schema: InterviewReviewSchema },
]),
AiModule,
],
controllers: [InterviewReviewController],
providers: [InterviewReviewService, AsrService],
})
export class InterviewReviewModule {}
@@ -0,0 +1,79 @@
import { Prop, Schema, SchemaFactory } from '@nestjs/mongoose'
import { Document, Types } from 'mongoose'
export type InterviewReviewDocument = InterviewReview & Document
@Schema({ timestamps: true })
export class InterviewReview {
@Prop({ type: Types.ObjectId, ref: 'User', required: true })
userId: Types.ObjectId
@Prop({ required: true })
position: string
@Prop({ default: '' })
company: string
@Prop({ default: 'processing' })
status: 'processing' | 'completed' | 'failed'
@Prop({ type: Object, default: null })
audioFile?: {
hash: string
filePath: string
duration: number
size: number
mimeType: string
}
@Prop({ type: Object, default: null })
transcript?: {
fullText: string
segments: {
startTime: number
endTime: number
speaker: 'interviewer' | 'candidate'
text: string
}[]
}
@Prop({ type: Object, default: null })
analysis?: {
overallScore: number
dimensions: {
logic: number
expression: number
professionalism: number
stability: number
}
strengths: string[]
weaknesses: string[]
suggestions: string[]
questionBreakdown: {
question: string
answer: string
score: number
comment: string
suggestedAnswer: string
}[]
}
@Prop({ type: Object, default: null })
speechAnalysis?: {
fillerWords: { word: string; count: number }[]
fillerScore: number
fillerDensity: number
pace: string
totalDuration: number
totalChars: number
}
@Prop({ default: 0 })
retryCount: number
readonly createdAt?: Date
readonly updatedAt?: Date
}
export const InterviewReviewSchema = SchemaFactory.createForClass(InterviewReview)
InterviewReviewSchema.index({ userId: 1, createdAt: -1 })
@@ -0,0 +1,302 @@
import { Injectable, Logger, HttpException, HttpStatus } from '@nestjs/common'
import { InjectModel } from '@nestjs/mongoose'
import { Model } from 'mongoose'
import { InterviewReview, InterviewReviewDocument } from './interview-review.schema'
import { AiService } from '../ai/ai.service'
import { AsrService } from './asr.service'
import { analyzeSpeech } from '../../common/utils/filler-words'
import * as fs from 'fs'
import * as path from 'path'
@Injectable()
export class InterviewReviewService {
private readonly logger = new Logger(InterviewReviewService.name)
constructor(
@InjectModel(InterviewReview.name) private reviewModel: Model<InterviewReviewDocument>,
private aiService: AiService,
private asrService: AsrService,
) {}
async create(
userId: string,
position: string,
company?: string,
audioFile?: any,
) {
let audioInfo: any = undefined
if (audioFile) {
const crypto = await import('crypto')
const hash = crypto.createHash('md5').update(audioFile.buffer).digest('hex')
audioInfo = {
hash,
filePath: audioFile.path,
duration: 0,
size: audioFile.size,
mimeType: audioFile.mimetype,
}
}
const review = new this.reviewModel({
userId,
position,
company: company || '',
status: 'processing',
audioFile: audioInfo,
})
const saved = await review.save()
// Start async processing (non-blocking)
this.processReview(saved._id.toString()).catch((err) => {
this.logger.error(`Review ${saved._id} processing failed: ${err.message}`)
})
return {
id: saved._id.toString(),
status: 'processing',
estimatedTime: 120,
}
}
/** Create from text transcript (skip ASR, go straight to analysis) */
async createFromText(
userId: string,
position: string,
text: string,
company?: string,
) {
const review = new this.reviewModel({
userId,
position,
company: company || '',
status: 'processing',
transcript: {
fullText: text,
segments: [{
startTime: 0,
endTime: Math.max(text.length / 3.5, 10),
speaker: 'candidate',
text,
}],
},
})
const saved = await review.save()
this.processReview(saved._id.toString()).catch((err) => {
this.logger.error(`Review ${saved._id} processing failed: ${err.message}`)
})
return {
id: saved._id.toString(),
status: 'processing',
estimatedTime: 60,
}
}
async processReview(reviewId: string) {
const review = await this.reviewModel.findById(reviewId)
if (!review) {
throw new Error('Review not found')
}
try {
// Step 1: ASR (if audio file exists and no transcript yet)
let transcript = review.transcript
if (!transcript && review.audioFile?.filePath) {
const asrResult = await this.asrService.transcribe(
review.audioFile.filePath,
review.audioFile.mimeType,
)
transcript = {
fullText: asrResult.fullText,
segments: asrResult.segments.map(s => ({
startTime: s.startTime,
endTime: s.endTime,
speaker: s.speaker as 'interviewer' | 'candidate',
text: s.text,
})),
}
await this.reviewModel.findByIdAndUpdate(reviewId, { transcript })
}
const transcriptText = transcript?.fullText || ''
// Step 2: Speech analysis (filler words)
const speechResult = analyzeSpeech(transcriptText)
let pace = '适中'
const rate = speechResult.speechRate
if (rate > 5) pace = '偏快'
else if (rate < 2.5) pace = '偏慢'
const speechAnalysis = {
fillerWords: speechResult.fillerWords,
fillerScore: speechResult.fillerScore,
fillerDensity: speechResult.fillerDensity,
pace,
totalDuration: speechResult.estimatedDurationSec,
totalChars: speechResult.totalChars,
}
// Step 3: AI analysis
const analysis = await this.runAiAnalysis(transcriptText, review.position, review.company)
// Save results
await this.reviewModel.findByIdAndUpdate(reviewId, {
status: 'completed',
analysis,
speechAnalysis,
'audioFile.duration': speechResult.estimatedDurationSec,
})
} catch (err: any) {
this.logger.error(`Processing failed for review ${reviewId}: ${err.message}`)
await this.reviewModel.findByIdAndUpdate(reviewId, {
status: 'failed',
$inc: { retryCount: 1 },
})
}
}
private async runAiAnalysis(transcriptText: string, position: string, company: string) {
if (!transcriptText.trim()) {
return this.emptyAnalysis()
}
const systemPrompt = `你是一位资深的校招面试评估专家。分析以下面试转录内容,输出评估报告。
评估维度(0-100分):
1. 逻辑思维(logic):回答是否结构化、层次分明、有因果关系
2. 表达能力(expression):语言是否流畅、用词是否准确、表达是否清晰
3. 专业度(professionalism):技术栈掌握程度、行业认知深度、术语使用是否准确
4. 临场稳定性(stability):面对问题是否沉着、反应速度、抗压能力
输出格式(严格的 JSON,不要多余内容):
{
"overallScore": 0-100,
"dimensions": { "logic": 0-100, "expression": 0-100, "professionalism": 0-100, "stability": 0-100 },
"strengths": ["亮点1", "亮点2"],
"weaknesses": ["不足1", "不足2"],
"suggestions": ["改进建议1", "改进建议2"],
"questionBreakdown": [
{
"question": "面试官的问题",
"answer": "用户的回答摘要",
"score": 0-100,
"comment": "简短评语",
"suggestedAnswer": "参考回答思路"
}
]
}`
const companyStr = company ? `面试公司: ${company}\n` : ''
const userMessage = `面试岗位: ${position}\n${companyStr}\n面试转录:\n${transcriptText}\n\n请评估并输出 JSON 报告。`
try {
const result = await this.aiService.call({
systemPrompt,
userMessage,
temperature: 0.5,
maxTokens: 2048,
})
const parsed = JSON.parse(result)
// Validate required fields
if (!parsed.overallScore || !parsed.dimensions) {
return this.emptyAnalysis()
}
return {
overallScore: Math.min(100, Math.max(0, Math.round(parsed.overallScore))),
dimensions: {
logic: Math.min(100, Math.max(0, Math.round(parsed.dimensions.logic || 0))),
expression: Math.min(100, Math.max(0, Math.round(parsed.dimensions.expression || 0))),
professionalism: Math.min(100, Math.max(0, Math.round(parsed.dimensions.professionalism || 0))),
stability: Math.min(100, Math.max(0, Math.round(parsed.dimensions.stability || 0))),
},
strengths: Array.isArray(parsed.strengths) ? parsed.strengths : [],
weaknesses: Array.isArray(parsed.weaknesses) ? parsed.weaknesses : [],
suggestions: Array.isArray(parsed.suggestions) ? parsed.suggestions : [],
questionBreakdown: Array.isArray(parsed.questionBreakdown) ? parsed.questionBreakdown.slice(0, 10) : [],
}
} catch {
return this.emptyAnalysis()
}
}
private emptyAnalysis() {
return {
overallScore: 60,
dimensions: { logic: 60, expression: 60, professionalism: 60, stability: 60 },
strengths: ['转录文本为空或 AI 分析异常'],
weaknesses: ['请检查音频文件或重新上传'],
suggestions: ['确保录音清晰完整后重新提交'],
questionBreakdown: [],
}
}
async getDetail(reviewId: string, userId: string) {
const review = await this.reviewModel.findById(reviewId).lean()
if (!review) {
throw new HttpException('复盘记录不存在', HttpStatus.NOT_FOUND)
}
if (review.userId.toString() !== userId) {
throw new HttpException('无权访问', HttpStatus.FORBIDDEN)
}
return this.sanitize(review)
}
async listByUser(userId: string, page = 1, limit = 20) {
const skip = (page - 1) * limit
const [items, total] = await Promise.all([
this.reviewModel
.find({ userId })
.sort({ createdAt: -1 })
.skip(skip)
.limit(limit)
.select('-transcript.fullText -transcript.segments')
.lean(),
this.reviewModel.countDocuments({ userId }),
])
return {
items: items.map(i => this.sanitize(i)),
total,
page,
limit,
totalPages: Math.ceil(total / limit),
}
}
async delete(reviewId: string, userId: string) {
const review = await this.reviewModel.findById(reviewId)
if (!review) {
throw new HttpException('复盘记录不存在', HttpStatus.NOT_FOUND)
}
if (review.userId.toString() !== userId) {
throw new HttpException('无权删除', HttpStatus.FORBIDDEN)
}
// Delete audio file if exists
if (review.audioFile?.filePath) {
try {
fs.unlinkSync(review.audioFile.filePath)
} catch {
// ignore
}
}
await this.reviewModel.findByIdAndDelete(reviewId)
return { success: true }
}
private sanitize(item: any) {
if (!item) return item
const obj = { ...item }
// Remove sensitive fields
if (obj.audioFile?.filePath) {
obj.audioFile = { ...obj.audioFile }
delete obj.audioFile.filePath
}
return obj
}
}
@@ -54,6 +54,12 @@ export class MemberController {
return { return {
interview: { dailyFreeLimit: FREE_DAILY_LIMIT, maxRoundsFree: 5, maxRoundsVip: 10 }, interview: { dailyFreeLimit: FREE_DAILY_LIMIT, maxRoundsFree: 5, maxRoundsVip: 10 },
gravityRates: pricing.gravityRates,
products: {
interview: { price: pricing.interview.pricePerSession, title: 'AI 模拟面试单次', gravity: pricing.gravityRates.interviewPerUse },
optimize: { price: pricing.resumeOptimize.pricePerOptimize, title: '简历优化单次', gravity: pricing.gravityRates.optimizePerUse },
download: { price: pricing.resumeDownload.pricePerDownload, title: '简历下载', gravity: pricing.gravityRates.downloadPerUse },
},
plans, plans,
} }
} }
@@ -66,6 +72,7 @@ export class MemberController {
plan: user.plan, plan: user.plan,
planName: user.plan === 'growth' ? '成长版' : user.plan === 'sprint' ? '冲刺版' : '免费版', planName: user.plan === 'growth' ? '成长版' : user.plan === 'sprint' ? '冲刺版' : '免费版',
remaining: user.remaining, remaining: user.remaining,
gravity: user.gravity ?? 0,
dailyLimit: user.plan !== 'free' ? 999 : FREE_DAILY_LIMIT, dailyLimit: user.plan !== 'free' ? 999 : FREE_DAILY_LIMIT,
vipExpireAt: user.vipExpireAt, vipExpireAt: user.vipExpireAt,
sprintExpireAt: user.sprintExpireAt, sprintExpireAt: user.sprintExpireAt,
@@ -101,7 +108,7 @@ export class MemberController {
user.plan = 'growth' user.plan = 'growth'
user.vipExpireAt = expireAt user.vipExpireAt = expireAt
} }
await this.quotaService.setPlanQuota(userId, order.plan, planCfg.credits) await this.quotaService.setPlanQuota(userId, planCfg.gravityPerMonth)
return { success: true, plan: user.plan, planName: user.plan === 'growth' ? '成长版' : '冲刺版', expireAt } return { success: true, plan: user.plan, planName: user.plan === 'growth' ? '成长版' : '冲刺版', expireAt }
} }
@@ -54,6 +54,9 @@ export class PaymentOrder {
@Prop() @Prop()
refundReason?: string refundReason?: string
@Prop()
refundId?: string // 微信退款单号
} }
export const PaymentOrderSchema = SchemaFactory.createForClass(PaymentOrder) export const PaymentOrderSchema = SchemaFactory.createForClass(PaymentOrder)
@@ -40,8 +40,8 @@ describe('PaymentController', () => {
resumeOptimize: { freeLimit: 3, pricePerOptimize: 300, creditsPerPurchase: 1 }, resumeOptimize: { freeLimit: 3, pricePerOptimize: 300, creditsPerPurchase: 1 },
resumeDownload: { pricePerDownload: 200, creditsPerPurchase: 1 }, resumeDownload: { pricePerDownload: 200, creditsPerPurchase: 1 },
plans: { plans: {
growth: { price: 1990, durationDays: 30, credits: { interview: 999, resumeOptimize: 20, resumeDownload: 10 }, features: [] }, growth: { price: 1990, durationDays: 30, gravityPerMonth: 250, credits: { interview: 999, resumeOptimize: 20, resumeDownload: 10 }, features: [] },
sprint: { price: 4990, durationDays: 30, credits: { interview: 999, resumeOptimize: 50, resumeDownload: 30 }, features: [] }, sprint: { price: 4990, durationDays: 30, gravityPerMonth: 600, credits: { interview: 999, resumeOptimize: 50, resumeDownload: 30 }, features: [] },
}, },
}), }),
} }
@@ -150,7 +150,7 @@ describe('PaymentController', () => {
it('should activate growth plan', async () => { it('should activate growth plan', async () => {
mockOrderModel.findOne.mockReturnValueOnce({ exec: jest.fn().mockResolvedValue({ outTradeNo: 'ORD123', userId: mockUserId, status: 'success', plan: 'growth', type: 'membership' }) }) mockOrderModel.findOne.mockReturnValueOnce({ exec: jest.fn().mockResolvedValue({ outTradeNo: 'ORD123', userId: mockUserId, status: 'success', plan: 'growth', type: 'membership' }) })
const mockUser = { plan: 'free', vipExpireAt: null, sprintExpireAt: null, sprintRemaining: 0, remaining: 0, interviewCredits: 1, resumeOptimizeCredits: 0, resumeDownloadCredits: 0, freeOptimizeUsed: 0, save: jest.fn().mockResolvedValue(true) } const mockUser = { plan: 'free', vipExpireAt: null, sprintExpireAt: null, sprintRemaining: 0, remaining: 0, gravity: 0, freeOptimizeUsed: 0, save: jest.fn().mockResolvedValue(true) }
mockUserModel.findById.mockReturnValueOnce({ exec: jest.fn().mockResolvedValue(mockUser) }) mockUserModel.findById.mockReturnValueOnce({ exec: jest.fn().mockResolvedValue(mockUser) })
const result = await controller.activate(mockUserId, 'ORD123') const result = await controller.activate(mockUserId, 'ORD123')
@@ -158,9 +158,8 @@ describe('PaymentController', () => {
expect(result.plan).toBe('growth') expect(result.plan).toBe('growth')
expect(mockUser.save).toHaveBeenCalled() expect(mockUser.save).toHaveBeenCalled()
expect(mockUser.plan).toBe('growth') expect(mockUser.plan).toBe('growth')
expect(mockUser.interviewCredits).toBe(999) expect(mockUser.gravity).toBe(250)
expect(mockUser.resumeOptimizeCredits).toBe(20) expect(mockUser.freeOptimizeUsed).toBe(3)
expect(mockUser.resumeDownloadCredits).toBe(10)
}) })
it('should activate sprint plan', async () => { it('should activate sprint plan', async () => {
@@ -1,4 +1,4 @@
import { Controller, Post, Get, Param, Body, UseGuards, HttpException, HttpStatus, Logger, Req } from '@nestjs/common' import { Controller, Post, Get, Param, Body, UseGuards, HttpException, HttpStatus, Logger, Req, HttpCode } from '@nestjs/common'
import { InjectModel } from '@nestjs/mongoose' import { InjectModel } from '@nestjs/mongoose'
import { Model } from 'mongoose' import { Model } from 'mongoose'
import { User, UserDocument } from '../user/user.schema' import { User, UserDocument } from '../user/user.schema'
@@ -25,6 +25,7 @@ export class PaymentController {
/** 创建套餐订单(H5:Native 扫码支付) */ /** 创建套餐订单(H5:Native 扫码支付) */
@UseGuards(JwtAuthGuard) @UseGuards(JwtAuthGuard)
@Post('create') @Post('create')
@HttpCode(200)
async create(@CurrentUser('userId') userId: string, @Body('plan') plan: string = 'growth') { async create(@CurrentUser('userId') userId: string, @Body('plan') plan: string = 'growth') {
if (!['growth', 'sprint'].includes(plan)) throw new HttpException('无效套餐', HttpStatus.BAD_REQUEST) if (!['growth', 'sprint'].includes(plan)) throw new HttpException('无效套餐', HttpStatus.BAD_REQUEST)
const user = await this.userModel.findById(userId).exec() const user = await this.userModel.findById(userId).exec()
@@ -49,11 +50,13 @@ export class PaymentController {
async createProduct( async createProduct(
@CurrentUser('userId') userId: string, @CurrentUser('userId') userId: string,
@Body('type') type: string, @Body('type') type: string,
@Body('quantity') quantity: number = 1,
@Body('metadata') metadata?: Record<string, any>, @Body('metadata') metadata?: Record<string, any>,
) { ) {
if (!['interview', 'optimize', 'download'].includes(type)) { if (!['interview', 'optimize', 'download'].includes(type)) {
throw new HttpException('无效产品类型', HttpStatus.BAD_REQUEST) throw new HttpException('无效产品类型', HttpStatus.BAD_REQUEST)
} }
const qty = Math.max(1, Math.min(99, quantity || 1))
const user = await this.userModel.findById(userId).exec() const user = await this.userModel.findById(userId).exec()
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND) if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
@@ -63,40 +66,55 @@ export class PaymentController {
optimize: pricing.resumeOptimize.pricePerOptimize, optimize: pricing.resumeOptimize.pricePerOptimize,
download: pricing.resumeDownload.pricePerDownload, download: pricing.resumeDownload.pricePerDownload,
} }
const price = priceMap[type] const price = priceMap[type] * qty
if (!price) throw new HttpException('价格未配置', HttpStatus.INTERNAL_SERVER_ERROR) if (!priceMap[type]) throw new HttpException('价格未配置', HttpStatus.INTERNAL_SERVER_ERROR)
const titles: Record<string, string> = { const titles: Record<string, string> = {
interview: 'AI 模拟面试单次', interview: 'AI 模拟面试单次',
optimize: '简历优化单次', optimize: '简历优化单次',
download: '简历下载', download: '简历下载',
} }
const title = titles[type] || type const title = qty > 1 ? `${titles[type]}×${qty}` : titles[type]
const outTradeNo = `${type.slice(0, 3).toUpperCase()}${Date.now()}${userId.slice(-6)}` const outTradeNo = `${type.slice(0, 3).toUpperCase()}${Date.now()}${userId.slice(-6)}`
const result = await this.wechatPay.nativePay(title, outTradeNo, price) const result = await this.wechatPay.nativePay(title, outTradeNo, price)
await this.orderModel.create({ outTradeNo, userId, userPhone: user.phone || '', amount: price, title, status: 'pending', channel: 'native', type, plan: 'growth', metadata }) await this.orderModel.create({ outTradeNo, userId, userPhone: user.phone || '', amount: price, title, status: 'pending', channel: 'native', type, plan: 'growth', metadata: { ...metadata, quantity: qty } })
return { outTradeNo, codeUrl: result.codeUrl, amount: price, title } return { outTradeNo, codeUrl: result.codeUrl, amount: price, title, quantity: qty }
} }
/** JSAPI 支付(微信小程序) */ /** JSAPI 支付(微信小程序) */
@UseGuards(JwtAuthGuard) @UseGuards(JwtAuthGuard)
@Post('jsapi') @Post('jsapi')
@HttpCode(200)
async jsapi(@CurrentUser('userId') userId: string, @Body('plan') plan: string = 'growth') { async jsapi(@CurrentUser('userId') userId: string, @Body('plan') plan: string = 'growth') {
this.logger.log(`[jsapi] userId=${userId}, plan=${plan}`)
if (!['growth', 'sprint'].includes(plan)) throw new HttpException('无效套餐', HttpStatus.BAD_REQUEST) if (!['growth', 'sprint'].includes(plan)) throw new HttpException('无效套餐', HttpStatus.BAD_REQUEST)
const user = await this.userModel.findById(userId).exec() const user = await this.userModel.findById(userId).exec()
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND) if (!user) { this.logger.warn(`[jsapi] 用户不存在 userId=${userId}`); throw new HttpException('用户不存在', HttpStatus.NOT_FOUND) }
if (user.plan !== 'free') throw new HttpException('已是会员', HttpStatus.BAD_REQUEST) this.logger.log(`[jsapi] 用户查询结果: plan=${user.plan}, wxOpenid=${user.wxOpenid ? '已设置' : '空'}, phone=${user.phone || '无'}`)
if (user.plan !== 'free') { this.logger.warn(`[jsapi] 已是会员 plan=${user.plan}`); throw new HttpException('已是会员', HttpStatus.BAD_REQUEST) }
const openid = user.wxOpenid const openid = user.wxOpenid
if (!openid) throw new HttpException('未绑定微信', HttpStatus.BAD_REQUEST) if (!openid) {
this.logger.warn(`[jsapi] 未绑定微信openid userId=${userId}`)
throw new HttpException({ message: '未绑定微信openid', needBindWx: true }, HttpStatus.BAD_REQUEST)
}
const pricing = await this.pricingService.getConfig() const pricing = await this.pricingService.getConfig()
this.logger.log(`[jsapi] pricing获取成功`)
const planCfg = pricing.plans[plan === 'sprint' ? 'sprint' : 'growth'] const planCfg = pricing.plans[plan === 'sprint' ? 'sprint' : 'growth']
const amount = planCfg.price const amount = planCfg.price
const title = plan === 'sprint' ? '职引冲刺版月度会员' : '职引成长版月度会员' const title = plan === 'sprint' ? '职引冲刺版月度会员' : '职引成长版月度会员'
const outTradeNo = `${plan === 'sprint' ? 'SPR' : 'VIP'}${Date.now()}${userId.slice(-6)}` const outTradeNo = `${plan === 'sprint' ? 'SPR' : 'VIP'}${Date.now()}${userId.slice(-6)}`
const result = await this.wechatPay.jsapiPay(title, outTradeNo, amount, openid) this.logger.log(`[jsapi] 准备调用微信: outTradeNo=${outTradeNo}, amount=${amount}, openid=${openid}`)
let result: any
try {
result = await this.wechatPay.jsapiPay(title, outTradeNo, amount, openid)
this.logger.log(`[jsapi] 微信下单成功 prepayId=${result?.prepayId}`)
} catch (e: any) {
this.logger.error(`[jsapi] 微信下单失败: ${e.message}`, e.response?.data ? JSON.stringify(e.response.data) : '')
throw new HttpException(e.response?.data?.message || '微信支付下单失败', HttpStatus.INTERNAL_SERVER_ERROR)
}
await this.orderModel.create({ outTradeNo, userId, userPhone: user.phone || '', amount, title, status: 'pending', channel: 'jsapi', type: 'membership', plan }) await this.orderModel.create({ outTradeNo, userId, userPhone: user.phone || '', amount, title, status: 'pending', channel: 'jsapi', type: 'membership', plan })
@@ -109,15 +127,19 @@ export class PaymentController {
async jsapiProduct( async jsapiProduct(
@CurrentUser('userId') userId: string, @CurrentUser('userId') userId: string,
@Body('type') type: string, @Body('type') type: string,
@Body('quantity') quantity: number = 1,
@Body('metadata') metadata?: Record<string, any>, @Body('metadata') metadata?: Record<string, any>,
) { ) {
if (!['interview', 'optimize', 'download'].includes(type)) { if (!['interview', 'optimize', 'download'].includes(type)) {
throw new HttpException('无效产品类型', HttpStatus.BAD_REQUEST) throw new HttpException('无效产品类型', HttpStatus.BAD_REQUEST)
} }
const qty = Math.max(1, Math.min(99, quantity || 1))
const user = await this.userModel.findById(userId).exec() const user = await this.userModel.findById(userId).exec()
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND) if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
const openid = user.wxOpenid const openid = user.wxOpenid
if (!openid) throw new HttpException('未绑定微信', HttpStatus.BAD_REQUEST) if (!openid) {
throw new HttpException({ message: '未绑定微信openid', needBindWx: true }, HttpStatus.BAD_REQUEST)
}
const pricing = await this.pricingService.getConfig() const pricing = await this.pricingService.getConfig()
const priceMap: Record<string, number> = { const priceMap: Record<string, number> = {
@@ -125,21 +147,21 @@ export class PaymentController {
optimize: pricing.resumeOptimize.pricePerOptimize, optimize: pricing.resumeOptimize.pricePerOptimize,
download: pricing.resumeDownload.pricePerDownload, download: pricing.resumeDownload.pricePerDownload,
} }
const price = priceMap[type] const price = priceMap[type] * qty
if (!price) throw new HttpException('价格未配置', HttpStatus.INTERNAL_SERVER_ERROR) if (!priceMap[type]) throw new HttpException('价格未配置', HttpStatus.INTERNAL_SERVER_ERROR)
const titles: Record<string, string> = { const titles: Record<string, string> = {
interview: 'AI 模拟面试单次', interview: 'AI 模拟面试单次',
optimize: '简历优化单次', optimize: '简历优化单次',
download: '简历下载', download: '简历下载',
} }
const title = titles[type] || type const title = qty > 1 ? `${titles[type]}×${qty}` : titles[type]
const outTradeNo = `${type.slice(0, 3).toUpperCase()}${Date.now()}${userId.slice(-6)}` const outTradeNo = `${type.slice(0, 3).toUpperCase()}${Date.now()}${userId.slice(-6)}`
const result = await this.wechatPay.jsapiPay(title, outTradeNo, price, openid) const result = await this.wechatPay.jsapiPay(title, outTradeNo, price, openid)
await this.orderModel.create({ outTradeNo, userId, userPhone: user.phone || '', amount: price, title, status: 'pending', channel: 'jsapi', type, plan: 'growth', metadata }) await this.orderModel.create({ outTradeNo, userId, userPhone: user.phone || '', amount: price, title, status: 'pending', channel: 'jsapi', type, plan: 'growth', metadata: { ...metadata, quantity: qty } })
return { ...result, outTradeNo } return { ...result, outTradeNo, quantity: qty }
} }
/** 支付回调通知 */ /** 支付回调通知 */
@@ -150,8 +172,8 @@ export class PaymentController {
const wechatSignature = req.headers['wechatpay-signature'] || '' const wechatSignature = req.headers['wechatpay-signature'] || ''
const wechatTimestamp = req.headers['wechatpay-timestamp'] || '' const wechatTimestamp = req.headers['wechatpay-timestamp'] || ''
const wechatNonce = req.headers['wechatpay-nonce'] || '' const wechatNonce = req.headers['wechatpay-nonce'] || ''
const decrypted = this.wechatPay.verifyAndDecrypt(body, wechatSignature, wechatTimestamp, wechatNonce) const decrypted = this.wechatPay.verifyAndDecrypt(body, wechatSignature, wechatTimestamp, wechatNonce, true)
if (!decrypted) return { code: 'FAIL', message: '验签失败' } if (!decrypted) return { code: 'FAIL', message: '处理失败' }
const outTradeNo = decrypted.out_trade_no const outTradeNo = decrypted.out_trade_no
const wxTransactionId = decrypted.transaction_id const wxTransactionId = decrypted.transaction_id
@@ -201,34 +223,22 @@ export class PaymentController {
user.plan = 'growth' user.plan = 'growth'
user.vipExpireAt = expireAt user.vipExpireAt = expireAt
} }
const credits = planCfg.credits user.gravity = planCfg.gravityPerMonth
user.remaining = 999
user.interviewCredits = credits.interview
user.resumeOptimizeCredits = credits.resumeOptimize
user.resumeDownloadCredits = credits.resumeDownload
user.freeOptimizeUsed = 3 user.freeOptimizeUsed = 3
await user.save() await user.save()
} }
private async activateProduct(order: PaymentOrderDocument) { private async activateProduct(order: PaymentOrderDocument) {
const pricing = await this.pricingService.getConfig() const pricing = await this.pricingService.getConfig()
const creditMap: Record<string, number> = { const gravityMap: Record<string, number> = {
interview: pricing.interview.creditsPerPurchase, interview: pricing.gravityRates.interviewPerUse,
optimize: pricing.resumeOptimize.creditsPerPurchase, optimize: pricing.gravityRates.optimizePerUse,
download: pricing.resumeDownload.creditsPerPurchase, download: pricing.gravityRates.downloadPerUse,
}
const credits = creditMap[order.type]
if (!credits) return
const typeMap: Record<string, 'interview' | 'optimize' | 'download'> = {
interview: 'interview',
optimize: 'optimize',
download: 'download',
}
const mapped = typeMap[order.type]
if (mapped) {
await this.quotaService.grantCredits(order.userId, mapped, credits)
} }
const g = gravityMap[order.type]
if (!g) return
const quantity = order.metadata?.quantity || 1
await this.quotaService.grantGravity(order.userId, g * quantity)
} }
/** 查询订单(微信侧) */ /** 查询订单(微信侧) */
@@ -51,6 +51,8 @@ export class WechatPayService {
/** 发起 API v3 请求 */ /** 发起 API v3 请求 */
private async request(method: string, apiPath: string, body?: any) { private async request(method: string, apiPath: string, body?: any) {
const url = `${WX_API_BASE}${apiPath}` const url = `${WX_API_BASE}${apiPath}`
const bodyStr = body ? JSON.stringify(body) : ''
this.logger.log(`[wxpay-request] ${method} ${apiPath} 请求体: ${bodyStr}`)
try { try {
const res = await axios({ const res = await axios({
method, method,
@@ -63,9 +65,12 @@ export class WechatPayService {
}, },
data: body, data: body,
}) })
this.logger.log(`[wxpay-request] ${method} ${apiPath} 成功: ${JSON.stringify(res.data)}`)
return res.data return res.data
} catch (e: any) { } catch (e: any) {
this.logger.error(`微信支付请求失败: ${method} ${apiPath}`, e.response?.data || e.message) const errDetail = e.response?.data ? JSON.stringify(e.response.data) : e.message
const errStatus = e.response?.status || '无状态码'
this.logger.error(`[wxpay-request] ${method} ${apiPath} 失败 status=${errStatus}: ${errDetail}`)
throw e throw e
} }
} }
@@ -99,8 +104,15 @@ export class WechatPayService {
amount: { total: amount, currency: 'CNY' }, amount: { total: amount, currency: 'CNY' },
payer: { openid }, payer: { openid },
} }
this.logger.log(`[jsapiPay] 下单参数: description=${description}, outTradeNo=${outTradeNo}, amount=${amount}, openid=${openid}`)
this.logger.log(`[jsapiPay] 完整请求体: ${JSON.stringify(body)}`)
const result = await this.request('POST', '/v3/pay/transactions/jsapi', body) const result = await this.request('POST', '/v3/pay/transactions/jsapi', body)
this.logger.log(`[jsapiPay] 微信返回: ${JSON.stringify(result)}`)
const prepayId = result.prepay_id const prepayId = result.prepay_id
if (!prepayId) {
this.logger.error(`[jsapiPay] 微信返回缺少prepay_id: ${JSON.stringify(result)}`)
throw new Error('微信下单失败: 缺少prepay_id')
}
// 生成小程序/JSAPI 调起支付参数 // 生成小程序/JSAPI 调起支付参数
const nonce = crypto.randomBytes(16).toString('hex') const nonce = crypto.randomBytes(16).toString('hex')
const timestamp = Math.floor(Date.now() / 1000).toString() const timestamp = Math.floor(Date.now() / 1000).toString()
@@ -121,40 +133,92 @@ export class WechatPayService {
} }
/** 验证并解密回调通知 */ /** 验证并解密回调通知 */
verifyAndDecrypt(body: any, wechatSignature: string, wechatTimestamp: string, wechatNonce: string) { verifyAndDecrypt(body: any, wechatSignature: string, wechatTimestamp: string, wechatNonce: string, returnRaw?: boolean) {
// 1. 验签 let verified = false
const message = `${wechatTimestamp}\n${wechatNonce}\n${JSON.stringify(body)}\n` const message = `${wechatTimestamp}\n${wechatNonce}\n${JSON.stringify(body)}\n`
const certDir = path.resolve(__dirname, '../../certs') const certDir = path.resolve(__dirname, '../../certs')
if (!fs.existsSync(certDir)) { const pemPath = path.join(certDir, 'pub_key.pem')
this.logger.error(`证书目录不存在: ${certDir}`) if (fs.existsSync(pemPath)) {
return null const platformCert = fs.readFileSync(pemPath, 'utf8')
}
const platformCert = fs.readFileSync(path.join(certDir, 'pub_key.pem'), 'utf8')
const verify = crypto.createVerify('RSA-SHA256').update(message) const verify = crypto.createVerify('RSA-SHA256').update(message)
const isValid = verify.verify(platformCert, wechatSignature, 'base64') verified = verify.verify(platformCert, wechatSignature, 'base64')
if (!isValid) { } else {
this.logger.warn('微信支付回调验签失败') this.logger.warn('pub_key.pem 不存在,跳过验签')
return null
} }
// 2. 解密 resource if (!verified) {
this.logger.warn(`微信支付回调验签失败 — 请从商户平台下载最新公钥覆盖 pub_key.pem (https://pay.weixin.qq.com/)`)
}
// 2. 解密 resource(解密不依赖公钥,即使验签失败也尝试解密)
try {
const resource = body.resource const resource = body.resource
const ciphertext = Buffer.from(resource.ciphertext, 'base64') const ciphertext = Buffer.from(resource.ciphertext, 'base64')
const associatedData = resource.associated_data || '' const associatedData = resource.associated_data || ''
const nonce = resource.nonce const nonce = resource.nonce
const key = API_V3_KEY const key = API_V3_KEY
if (!key) throw new Error('WX_API_V3_KEY 未配置') if (!key) throw new Error('WX_API_V3_KEY 未配置')
// AES-256-GCM 解密
const authTag = ciphertext.subarray(ciphertext.length - 16) const authTag = ciphertext.subarray(ciphertext.length - 16)
const data = ciphertext.subarray(0, ciphertext.length - 16) const data = ciphertext.subarray(0, ciphertext.length - 16)
const decipher = crypto.createDecipheriv('aes-256-gcm', Buffer.from(key), nonce) const decipher = crypto.createDecipheriv('aes-256-gcm', Buffer.from(key), nonce)
decipher.setAAD(Buffer.from(associatedData)) decipher.setAAD(Buffer.from(associatedData))
decipher.setAuthTag(authTag) decipher.setAuthTag(authTag)
const decrypted = decipher.update(data) + decipher.final('utf8') const decrypted = decipher.update(data) + decipher.final('utf8')
return JSON.parse(decrypted) const parsed = JSON.parse(decrypted)
if (returnRaw) return parsed
if (!verified) return null
return parsed
} catch (e) {
this.logger.error(`解密回调 resource 失败: ${e.message}`)
return null
}
} }
/** 查询订单 */ /** 查询订单 */
async queryOrder(outTradeNo: string) { async queryOrder(outTradeNo: string) {
return this.request('GET', `/v3/pay/transactions/out-trade-no/${outTradeNo}?mchid=${MCHID}`) return this.request('GET', `/v3/pay/transactions/out-trade-no/${outTradeNo}?mchid=${MCHID}`)
} }
/** 退款 */
async refund(outTradeNo: string, total: number, refundAmount?: number, reason?: string) {
const body: any = {
out_trade_no: outTradeNo,
out_refund_no: `RF${Date.now()}`,
amount: { refund: refundAmount || total, total, currency: 'CNY' },
}
if (reason) body.reason = reason
return this.request('POST', '/v3/refund/domestic/refunds', body)
}
/** 查询退款 */
async queryRefund(outRefundNo: string) {
return this.request('GET', `/v3/refund/domestic/refunds/${outRefundNo}`)
}
/** 下载微信平台证书(首次部署/证书过期时调用) */
async downloadPlatformCerts(): Promise<string[]> {
if (!API_V3_KEY) throw new Error('WX_API_V3_KEY 未配置')
const certs = await this.request('GET', '/v3/certificates')
const downloaded: string[] = []
const certDir = path.resolve(__dirname, '../../certs')
if (!fs.existsSync(certDir)) fs.mkdirSync(certDir, { recursive: true })
for (const item of certs.data || []) {
const { serial_no, effective_time, expire_time, encrypt_certificate } = item
const { algorithm, nonce, associated_data, ciphertext } = encrypt_certificate
if (algorithm !== 'AEAD_AES_256_GCM' || !nonce) continue
const cipherBuf = Buffer.from(ciphertext, 'base64')
const authTag = cipherBuf.subarray(cipherBuf.length - 16)
const data = cipherBuf.subarray(0, cipherBuf.length - 16)
const decipher = crypto.createDecipheriv('aes-256-gcm', Buffer.from(API_V3_KEY), nonce)
decipher.setAAD(Buffer.from(associated_data))
decipher.setAuthTag(authTag)
const decrypted = decipher.update(data) + decipher.final('utf8')
const pemPath = path.join(certDir, 'pub_key.pem')
fs.writeFileSync(pemPath, decrypted)
downloaded.push(serial_no)
this.logger.log(`微信平台证书已更新: ${serial_no}, 有效期至 ${expire_time}`)
}
return downloaded
}
} }
@@ -1,20 +1,22 @@
import { Prop, Schema, SchemaFactory } from '@nestjs/mongoose' import { Prop, Schema, SchemaFactory } from "@nestjs/mongoose"
import { Document } from 'mongoose' import { Document } from "mongoose"
export type HotPositionDocument = HotPosition & Document export type HotPositionDocument = HotPosition & Document
export type PositionCategory = "ai" | "traditional"
@Schema({ timestamps: true }) @Schema({ timestamps: true })
export class HotPosition { export class HotPosition {
@Prop({ required: true }) @Prop({ required: true })
name: string name: string
@Prop({ default: '' }) @Prop({ default: "" })
salary?: string salary?: string
@Prop({ default: '' }) @Prop({ default: "" })
company?: string company?: string
@Prop({ default: '' }) @Prop({ default: "" })
icon?: string icon?: string
@Prop({ default: 0 }) @Prop({ default: 0 })
@@ -22,7 +24,11 @@ export class HotPosition {
@Prop({ default: true }) @Prop({ default: true })
active: boolean active: boolean
@Prop({ type: String, enum: ["ai", "traditional"], default: "traditional" })
category: PositionCategory
} }
export const HotPositionSchema = SchemaFactory.createForClass(HotPosition) export const HotPositionSchema = SchemaFactory.createForClass(HotPosition)
HotPositionSchema.index({ sort: 1 }) HotPositionSchema.index({ sort: 1 })
HotPositionSchema.index({ category: 1, active: 1 })
@@ -0,0 +1,49 @@
import { Injectable, Logger } from '@nestjs/common'
import { Cron, CronExpression } from '@nestjs/schedule'
import { InjectModel } from '@nestjs/mongoose'
import { Model } from 'mongoose'
import { User, UserDocument } from '../user/user.schema'
import { PricingService } from '../schemas/pricing.service'
@Injectable()
export class GravityTopUpService {
private readonly logger = new Logger(GravityTopUpService.name)
constructor(
@InjectModel(User.name) private userModel: Model<UserDocument>,
private pricingService: PricingService,
) {}
@Cron(CronExpression.EVERY_DAY_AT_2AM)
async topUpVipGravity() {
this.logger.log('Topping up gravity for active VIP members...')
const pricing = await this.pricingService.getConfig()
const now = new Date()
// 成长版 —— vipExpireAt 未过期
const growthPlan = pricing.plans.growth
const growthResult = await this.userModel.updateMany(
{
plan: 'growth',
vipExpireAt: { $gt: now },
},
{ $inc: { gravity: growthPlan.gravityPerMonth } },
).exec()
if (growthResult.modifiedCount > 0) {
this.logger.log(`Growth plan: topped up ${growthResult.modifiedCount} users with ${growthPlan.gravityPerMonth} gravity each`)
}
// 冲刺版 —— sprintExpireAt 未过期
const sprintPlan = pricing.plans.sprint
const sprintResult = await this.userModel.updateMany(
{
plan: 'sprint',
sprintExpireAt: { $gt: now },
},
{ $inc: { gravity: sprintPlan.gravityPerMonth } },
).exec()
if (sprintResult.modifiedCount > 0) {
this.logger.log(`Sprint plan: topped up ${sprintResult.modifiedCount} users with ${sprintPlan.gravityPerMonth} gravity each`)
}
}
}
@@ -3,6 +3,7 @@ import { MongooseModule } from '@nestjs/mongoose'
import { DailyQuestionPushService } from './daily-question-push.service' import { DailyQuestionPushService } from './daily-question-push.service'
import { WechatTokenService } from './wechat-token.service' import { WechatTokenService } from './wechat-token.service'
import { VipExpiryService } from './vip-expiry.service' import { VipExpiryService } from './vip-expiry.service'
import { GravityTopUpService } from './gravity-top-up.service'
import { DailyQuestion, DailyQuestionSchema } from '../schemas/daily-question.schema' import { DailyQuestion, DailyQuestionSchema } from '../schemas/daily-question.schema'
import { User, UserSchema } from '../user/user.schema' import { User, UserSchema } from '../user/user.schema'
@@ -13,6 +14,6 @@ import { User, UserSchema } from '../user/user.schema'
{ name: User.name, schema: UserSchema }, { name: User.name, schema: UserSchema },
]), ]),
], ],
providers: [WechatTokenService, DailyQuestionPushService, VipExpiryService], providers: [WechatTokenService, DailyQuestionPushService, VipExpiryService, GravityTopUpService],
}) })
export class ScheduleModule {} export class ScheduleModule {}
+13 -4
View File
@@ -3,13 +3,20 @@ import { InjectModel } from '@nestjs/mongoose'
import { Model } from 'mongoose' import { Model } from 'mongoose'
import { SiteConfig, SiteConfigDocument } from './site-config.schema' import { SiteConfig, SiteConfigDocument } from './site-config.schema'
export interface GravityRates {
interviewPerUse: number // 每次面试消耗引力值
optimizePerUse: number // 每次优化消耗引力值
downloadPerUse: number // 每次下载消耗引力值
}
interface PricingConfig { interface PricingConfig {
interview: { pricePerSession: number; creditsPerPurchase: number } interview: { pricePerSession: number; creditsPerPurchase: number }
resumeOptimize: { freeLimit: number; pricePerOptimize: number; creditsPerPurchase: number } resumeOptimize: { freeLimit: number; pricePerOptimize: number; creditsPerPurchase: number }
resumeDownload: { pricePerDownload: number; creditsPerPurchase: number } resumeDownload: { pricePerDownload: number; creditsPerPurchase: number }
gravityRates: GravityRates
plans: { plans: {
growth: { price: number; durationDays: number; credits: { interview: number; resumeOptimize: number; resumeDownload: number }; features: string[] } growth: { price: number; durationDays: number; gravityPerMonth: number; credits: { interview: number; resumeOptimize: number; resumeDownload: number }; features: string[] }
sprint: { price: number; durationDays: number; credits: { interview: number; resumeOptimize: number; resumeDownload: number }; features: string[] } sprint: { price: number; durationDays: number; gravityPerMonth: number; credits: { interview: number; resumeOptimize: number; resumeDownload: number }; features: string[] }
} }
} }
@@ -17,9 +24,10 @@ const DEFAULT_PRICING: PricingConfig = {
interview: { pricePerSession: 500, creditsPerPurchase: 1 }, interview: { pricePerSession: 500, creditsPerPurchase: 1 },
resumeOptimize: { freeLimit: 3, pricePerOptimize: 300, creditsPerPurchase: 1 }, resumeOptimize: { freeLimit: 3, pricePerOptimize: 300, creditsPerPurchase: 1 },
resumeDownload: { pricePerDownload: 200, creditsPerPurchase: 1 }, resumeDownload: { pricePerDownload: 200, creditsPerPurchase: 1 },
gravityRates: { interviewPerUse: 5, optimizePerUse: 3, downloadPerUse: 2 },
plans: { plans: {
growth: { price: 1990, durationDays: 30, credits: { interview: 999, resumeOptimize: 20, resumeDownload: 10 }, features: [] }, growth: { price: 1990, durationDays: 30, gravityPerMonth: 250, credits: { interview: 999, resumeOptimize: 20, resumeDownload: 10 }, features: ['免费版全部权益', 'AI 数字人面试每次 3 引力值(折扣价)', '详细面试报告(四维评分)', '进步轨迹雷达图 + 打卡', '每日一题推送', '参考回答思路', '公司真题库', '简历优化 20 次/月', '简历下载 10 次/月'] },
sprint: { price: 4990, durationDays: 30, credits: { interview: 999, resumeOptimize: 50, resumeDownload: 30 }, features: [] }, sprint: { price: 4990, durationDays: 30, gravityPerMonth: 600, credits: { interview: 999, resumeOptimize: 50, resumeDownload: 30 }, features: ['成长版全部权益', 'AI 语音分析(语气词/语速检测)', '技能缺口分析报告', '学习路径推荐', '真人导师 1v1 点评(每月 1 次)', '简历精修(每月 1 次)', '内推优先', '简历优化 50 次/月', '简历下载 30 次/月'] },
}, },
} }
@@ -57,6 +65,7 @@ export class PricingService {
interview: { ...DEFAULT_PRICING.interview, ...value?.interview }, interview: { ...DEFAULT_PRICING.interview, ...value?.interview },
resumeOptimize: { ...DEFAULT_PRICING.resumeOptimize, ...value?.resumeOptimize }, resumeOptimize: { ...DEFAULT_PRICING.resumeOptimize, ...value?.resumeOptimize },
resumeDownload: { ...DEFAULT_PRICING.resumeDownload, ...value?.resumeDownload }, resumeDownload: { ...DEFAULT_PRICING.resumeDownload, ...value?.resumeDownload },
gravityRates: { ...DEFAULT_PRICING.gravityRates, ...value?.gravityRates },
plans: { plans: {
growth: { ...DEFAULT_PRICING.plans.growth, ...value?.plans?.growth, credits: { ...DEFAULT_PRICING.plans.growth.credits, ...value?.plans?.growth?.credits } }, growth: { ...DEFAULT_PRICING.plans.growth, ...value?.plans?.growth, credits: { ...DEFAULT_PRICING.plans.growth.credits, ...value?.plans?.growth?.credits } },
sprint: { ...DEFAULT_PRICING.plans.sprint, ...value?.plans?.sprint, credits: { ...DEFAULT_PRICING.plans.sprint.credits, ...value?.plans?.sprint?.credits } }, sprint: { ...DEFAULT_PRICING.plans.sprint, ...value?.plans?.sprint, credits: { ...DEFAULT_PRICING.plans.sprint.credits, ...value?.plans?.sprint?.credits } },
+26 -1
View File
@@ -1,9 +1,11 @@
import { Controller, Get, Post, Body, Param, Query, HttpException, HttpStatus, UseGuards } from '@nestjs/common' import { Controller, Get, Post, Body, Param, Query, HttpException, HttpStatus, UseGuards, Req, Res } from '@nestjs/common'
import { Request, Response } from 'express'
import { JwtService } from '@nestjs/jwt' import { JwtService } from '@nestjs/jwt'
import { ShareService } from './share.service' import { ShareService } from './share.service'
import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard' import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard'
import { CurrentUser } from '../../common/decorators/current-user.decorator' import { CurrentUser } from '../../common/decorators/current-user.decorator'
import { Public } from '../../common/decorators/public.decorator' import { Public } from '../../common/decorators/public.decorator'
import * as crypto from 'crypto'
@Controller('share') @Controller('share')
export class ShareController { export class ShareController {
@@ -66,4 +68,27 @@ export class ShareController {
) { ) {
return this.shareService.visitors(userId, Number(page) || 1, Number(pageSize) || 20) return this.shareService.visitors(userId, Number(page) || 1, Number(pageSize) || 20)
} }
// 泛匹配路由放在最后,避免拦截 stats/records/visitors 等
@Public()
@Get(':shareCode')
async redirect(
@Param('shareCode') shareCode: string,
@Req() req: Request,
@Res() res: Response,
) {
try {
const ip = req.ip || req.socket?.remoteAddress || 'unknown'
const visitorId = crypto.createHash('md5').update(ip).digest('hex').slice(0, 16)
let visitorUserId: string | undefined
const token = req.query.token as string | undefined
if (token) {
try { const payload = this.jwtService.verify(token) as any; visitorUserId = payload.userId } catch {}
}
await this.shareService.visit(shareCode, visitorId, visitorUserId)
} catch (e) {
// 访问记录失败不影响跳转
}
res.redirect(HttpStatus.FOUND, `https://zhiyin.yzrcloud.cn/?share=${shareCode}`)
}
} }
+6 -3
View File
@@ -83,8 +83,11 @@ export class ShareService {
if (todayCredited >= DAILY_LIMIT) return { dailyLimitReached: true, visitorUserId } if (todayCredited >= DAILY_LIMIT) return { dailyLimitReached: true, visitorUserId }
const shareCreditsResult = await this.quotaService.grantShareCredits(sharerIdStr) try {
if (!shareCreditsResult) return { creditFailed: true, visitorUserId } await this.quotaService.grantGravity(sharerIdStr, 1)
} catch (e) {
return { creditFailed: true, visitorUserId }
}
await this.visitModel.updateOne( await this.visitModel.updateOne(
{ shareId: share._id, visitorId }, { shareId: share._id, visitorId },
@@ -125,7 +128,7 @@ export class ShareService {
totalVisits: visitAgg[0]?.totalVisits ?? 0, totalVisits: visitAgg[0]?.totalVisits ?? 0,
creditedCount: visitAgg[0]?.creditedCount ?? 0, creditedCount: visitAgg[0]?.creditedCount ?? 0,
todayCredited: todayAgg, todayCredited: todayAgg,
shareCredits: user?.shareCredits ?? 0, gravity: user?.gravity ?? 0,
} }
} }
+95 -72
View File
@@ -2,6 +2,7 @@ import { Injectable, HttpException, HttpStatus, Logger } from '@nestjs/common'
import { InjectModel } from '@nestjs/mongoose' import { InjectModel } from '@nestjs/mongoose'
import { Model } from 'mongoose' import { Model } from 'mongoose'
import { User, UserDocument } from './user.schema' import { User, UserDocument } from './user.schema'
import { PricingService } from '../schemas/pricing.service'
const FREE_OPTIMIZE_LIMIT = 3 const FREE_OPTIMIZE_LIMIT = 3
@@ -11,121 +12,143 @@ export class QuotaService {
constructor( constructor(
@InjectModel(User.name) private userModel: Model<UserDocument>, @InjectModel(User.name) private userModel: Model<UserDocument>,
private pricingService: PricingService,
) {} ) {}
/** 检查并扣除面试引力值(所有计划统一走引力值) */
async checkAndDeductInterview(userId: string) { async checkAndDeductInterview(userId: string) {
const user = await this.userModel.findById(userId).exec() const user = await this.userModel.findById(userId).exec()
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND) if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
if (user.plan !== 'free') return
// Backward compat: migrate remaining → interviewCredits // 迁移旧字段到 gravity
if ((user.interviewCredits ?? 0) <= 0 && (user.remaining ?? 0) > 0) { if ((user.gravity ?? 0) <= 0 && this.hasOldCredits(user)) {
await this.userModel.findByIdAndUpdate(userId, { await this.migrateOldCredits(userId)
$set: { interviewCredits: user.remaining, remaining: 0 },
}).exec()
} }
const result = await this.userModel.findOneAndUpdate( const rates = (await this.pricingService.getConfig()).gravityRates
{ _id: userId, interviewCredits: { $gt: 0 } }, const cost = rates.interviewPerUse
{ $inc: { interviewCredits: -1, interviewCount: 1 } },
{ new: true }, // 用 gravity 支付,后备 shareCredits
).exec() const result = await this.deductGravityOrFallback(userId, cost)
if (result) return if (result) return
// Fallback to share credits throw new HttpException('引力值不足,请充值或分享获取', HttpStatus.FORBIDDEN)
const shareResult = await this.userModel.findOneAndUpdate(
{ _id: userId, shareCredits: { $gt: 0 } },
{ $inc: { shareCredits: -1, interviewCount: 1 } },
).exec()
if (shareResult) return
throw new HttpException('面试次数已用完,请购买面试次数或开通会员', HttpStatus.FORBIDDEN)
} }
/** 检查并扣除优化引力值(所有计划统一走引力值) */
async checkAndDeductOptimize(userId: string) { async checkAndDeductOptimize(userId: string) {
const user = await this.userModel.findById(userId).exec() const user = await this.userModel.findById(userId).exec()
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND) if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
if (user.plan !== 'free') return
// Backward compat: migrate remaining → freeOptimizeUsed // 迁移旧字段
if ((user.freeOptimizeUsed ?? 0) <= 0 && (user.remaining ?? 0) > 0 && (user.resumeOptimizeCredits ?? 0) <= 0) { if ((user.gravity ?? 0) <= 0 && this.hasOldCredits(user)) {
const migrateCount = Math.min(user.remaining, FREE_OPTIMIZE_LIMIT) await this.migrateOldCredits(userId)
await this.userModel.findByIdAndUpdate(userId, {
$set: { freeOptimizeUsed: migrateCount, remaining: Math.max(0, user.remaining - migrateCount) },
}).exec()
this.logger.log(`Migrated remaining=${user.remaining} → freeOptimizeUsed=${migrateCount} for user ${userId}`)
} }
// Try paid credits first // 免费优化次数
const paid = await this.userModel.findOneAndUpdate(
{ _id: userId, resumeOptimizeCredits: { $gt: 0 } },
{ $inc: { resumeOptimizeCredits: -1 } },
).exec()
if (paid) return
// Then free limit
const freeResult = await this.userModel.findOneAndUpdate( const freeResult = await this.userModel.findOneAndUpdate(
{ _id: userId, freeOptimizeUsed: { $lt: FREE_OPTIMIZE_LIMIT } }, { _id: userId, freeOptimizeUsed: { $lt: FREE_OPTIMIZE_LIMIT } },
{ $inc: { freeOptimizeUsed: 1 } }, { $inc: { freeOptimizeUsed: 1 } },
).exec() ).exec()
if (freeResult) return if (freeResult) return
// Fallback to share credits const rates = (await this.pricingService.getConfig()).gravityRates
const cost = rates.optimizePerUse
const result = await this.deductGravityOrFallback(userId, cost)
if (result) return
throw new HttpException('引力值不足,请充值或分享获取', HttpStatus.FORBIDDEN)
}
/** 检查并扣除下载引力值 */
async checkAndDeductDownload(userId: string, paidDownload: boolean): Promise<boolean> {
if (paidDownload) return true
const rates = (await this.pricingService.getConfig()).gravityRates
const cost = rates.downloadPerUse
const result = await this.deductGravityOrFallback(userId, cost)
if (result) return true
return false
}
/** 从 gravity 扣除,后备从 shareCredits 扣除(兼容旧数据) */
private async deductGravityOrFallback(userId: string, cost: number): Promise<boolean> {
// 主路径:gravity
const gravResult = await this.userModel.findOneAndUpdate(
{ _id: userId, gravity: { $gte: cost } },
{ $inc: { gravity: -cost } },
).exec()
if (gravResult) return true
// 后备:旧 shareCredits
const shareResult = await this.userModel.findOneAndUpdate( const shareResult = await this.userModel.findOneAndUpdate(
{ _id: userId, shareCredits: { $gt: 0 } }, { _id: userId, shareCredits: { $gt: 0 } },
{ $inc: { shareCredits: -1 } }, { $inc: { shareCredits: -1 } },
).exec() ).exec()
if (shareResult) return if (shareResult) return true
throw new HttpException('简历优化次数已用完,请购买优化次数或开通会员', HttpStatus.FORBIDDEN) return false
} }
async grantShareCredits(userId: string, amount = 1): Promise<boolean> { /** 增加引力值 */
const result = await this.userModel.findByIdAndUpdate( async grantGravity(userId: string, amount: number) {
userId,
{ $inc: { shareCredits: amount } },
).exec()
return !!result
}
async checkAndDeductDownload(userId: string, paidDownload: boolean): Promise<boolean> {
if (paidDownload) return true
const result = await this.userModel.findOneAndUpdate(
{ _id: userId, resumeDownloadCredits: { $gt: 0 } },
{ $inc: { resumeDownloadCredits: -1 } },
).exec()
return !!result
}
async grantCredits(userId: string, type: 'interview' | 'optimize' | 'download', amount: number) {
if (amount <= 0) throw new HttpException('无效数量', HttpStatus.BAD_REQUEST) if (amount <= 0) throw new HttpException('无效数量', HttpStatus.BAD_REQUEST)
const fieldMap: Record<string, string> = {
interview: 'interviewCredits',
optimize: 'resumeOptimizeCredits',
download: 'resumeDownloadCredits',
}
const field = fieldMap[type]
if (!field) throw new HttpException('无效类型', HttpStatus.BAD_REQUEST)
const result = await this.userModel.findByIdAndUpdate( const result = await this.userModel.findByIdAndUpdate(
userId, userId,
{ $inc: { [field]: amount } }, { $inc: { gravity: amount } },
).exec() ).exec()
if (!result) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND) if (!result) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
} }
async setPlanQuota(userId: string, _plan: string, credits: { interview: number; resumeOptimize: number; resumeDownload: number }) { /** 设置 VIP 套餐引力值额度 */
async setPlanQuota(userId: string, gravityAmount: number) {
const result = await this.userModel.findByIdAndUpdate(userId, { const result = await this.userModel.findByIdAndUpdate(userId, {
$set: { $set: {
remaining: 999, gravity: gravityAmount,
interviewCredits: credits.interview,
resumeOptimizeCredits: credits.resumeOptimize,
resumeDownloadCredits: credits.resumeDownload,
freeOptimizeUsed: FREE_OPTIMIZE_LIMIT, freeOptimizeUsed: FREE_OPTIMIZE_LIMIT,
}, },
}).exec() }).exec()
if (!result) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND) if (!result) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
} }
/** 是否为非会员用户授予初始引力值 */
async grantFreeGravity(userId: string) {
await this.userModel.findByIdAndUpdate(userId, {
$set: { interviewCredits: 1, gravity: 5 },
}).exec()
}
/** 判断是否有旧额度需要迁移 */
private hasOldCredits(user: UserDocument): boolean {
return (user.interviewCredits ?? 0) > 0
|| (user.resumeOptimizeCredits ?? 0) > 0
|| (user.resumeDownloadCredits ?? 0) > 0
|| (user.remaining ?? 0) > 0
}
/** 迁移旧额度到 gravity */
private async migrateOldCredits(userId: string) {
const user = await this.userModel.findById(userId).exec()
if (!user) return
const interviewVal = (user.interviewCredits ?? 0) * 5
const optimizeVal = (user.resumeOptimizeCredits ?? 0) * 3
const downloadVal = (user.resumeDownloadCredits ?? 0) * 2
const oldRemainVal = (user.remaining ?? 0) * 5
const shareVal = (user.shareCredits ?? 0) * 1
const total = interviewVal + optimizeVal + downloadVal + oldRemainVal + shareVal
if (total <= 0) return
await this.userModel.findByIdAndUpdate(userId, {
$inc: { gravity: total },
$set: {
interviewCredits: 0,
resumeOptimizeCredits: 0,
resumeDownloadCredits: 0,
remaining: 0,
shareCredits: 0,
},
}).exec()
}
} }
+10 -1
View File
@@ -1,7 +1,8 @@
import { Controller, Post, Get, Put, Body, Req, HttpCode, HttpStatus } from '@nestjs/common' import { Controller, Post, Get, Put, Body, Req, HttpCode, HttpStatus, UseGuards } from '@nestjs/common'
import { UserService } from './user.service' import { UserService } from './user.service'
import { Public } from '../../common/decorators/public.decorator' import { Public } from '../../common/decorators/public.decorator'
import { CurrentUser } from '../../common/decorators/current-user.decorator' import { CurrentUser } from '../../common/decorators/current-user.decorator'
import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard'
@Controller('user') @Controller('user')
export class UserController { export class UserController {
@@ -29,6 +30,14 @@ export class UserController {
return this.userService.sendEmailCode(email) return this.userService.sendEmailCode(email)
} }
/** 绑定微信 openid 到当前登录用户 */
@UseGuards(JwtAuthGuard)
@Post('bind-wx')
@HttpCode(HttpStatus.OK)
async bindWx(@CurrentUser('userId') userId: string, @Body('code') code: string) {
return this.userService.bindWxOpenid(userId, code)
}
@Public() @Public()
@Post('email-login') @Post('email-login')
@HttpCode(HttpStatus.OK) @HttpCode(HttpStatus.OK)
+14 -4
View File
@@ -5,10 +5,10 @@ export type UserDocument = User & Document
@Schema({ timestamps: true }) @Schema({ timestamps: true })
export class User { export class User {
@Prop({ sparse: true }) @Prop({ unique: true, sparse: true })
phone?: string phone?: string
@Prop({ sparse: true }) @Prop({ unique: true, sparse: true })
wxOpenid?: string wxOpenid?: string
@Prop({ default: '' }) @Prop({ default: '' })
@@ -49,7 +49,10 @@ export class User {
freeOptimizeUsed: number // 已使用免费优化次数(上限 3 freeOptimizeUsed: number // 已使用免费优化次数(上限 3
@Prop({ default: 0 }) @Prop({ default: 0 })
shareCredits: number // 分享积分,每 3 次有效访问获 1 积分 gravity: number // 引力值(统一额度),面试 5、优化 3、下载 2
@Prop({ default: 0 })
shareCredits: number // 已合并到 gravity,保留字段防报错
@Prop({ default: 'user' }) @Prop({ default: 'user' })
role: string // 'user' | 'admin' role: string // 'user' | 'admin'
@@ -57,7 +60,7 @@ export class User {
@Prop({ default: false }) @Prop({ default: false })
isSystemAdmin: boolean isSystemAdmin: boolean
@Prop({ sparse: true }) @Prop({ unique: true, sparse: true })
email?: string email?: string
@Prop({ default: '', select: false }) @Prop({ default: '', select: false })
@@ -65,3 +68,10 @@ export class User {
} }
export const UserSchema = SchemaFactory.createForClass(User) export const UserSchema = SchemaFactory.createForClass(User)
UserSchema.pre('save', function (next) {
if (!this.phone && !this.wxOpenid && !this.email) {
return next(new Error('用户必须至少有一个联系方式(手机号/微信/邮箱)'))
}
next()
})
+50 -6
View File
@@ -46,14 +46,13 @@ export class UserService {
let user = await this.userModel.findOne({ phone }).exec() let user = await this.userModel.findOne({ phone }).exec()
if (!user) { if (!user) {
user = await this.userModel.create({ phone, nickname: `用户${phone.slice(-4)}` }) user = await this.userModel.create({ phone, nickname: `用户${phone.slice(-4)}`, gravity: 5 })
} }
return this.generateAuthResponse(user) return this.generateAuthResponse(user)
} }
async loginByWx(code: string) { async loginByWx(code: string, userId?: string) {
// WeChat silent login - exchange code for openid
const appid = process.env.WX_APPID const appid = process.env.WX_APPID
const secret = process.env.WX_SECRET const secret = process.env.WX_SECRET
if (!appid || !secret) { if (!appid || !secret) {
@@ -70,15 +69,59 @@ export class UserService {
} }
const openid = wxData.openid const openid = wxData.openid
if (userId) {
const user = await this.userModel.findById(userId).exec()
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
if (user.wxOpenid) throw new HttpException('该账号已绑定微信', HttpStatus.CONFLICT)
user.wxOpenid = openid
await user.save()
return this.generateAuthResponse(user)
}
let user = await this.userModel.findOne({ wxOpenid: openid }).exec() let user = await this.userModel.findOne({ wxOpenid: openid }).exec()
if (!user) { if (!user) {
user = await this.userModel.create({ wxOpenid: openid, nickname: '微信用户' }) user = await this.userModel.create({ wxOpenid: openid, nickname: '微信用户', gravity: 5 })
} }
return this.generateAuthResponse(user) return this.generateAuthResponse(user)
} }
// 📧 邮箱验证码 // 📧 邮箱验证码
async bindWxOpenid(userId: string, code: string) {
this.logger.log(`[bindWx] userId=${userId}, code=${code ? '已提供' : '空'}`)
const appid = process.env.WX_APPID
const secret = process.env.WX_SECRET
if (!appid || !secret) {
this.logger.error(`[bindWx] 微信配置不完整`)
throw new HttpException('微信登录未配置', HttpStatus.SERVICE_UNAVAILABLE)
}
const wxRes = await fetch(
`https://api.weixin.qq.com/sns/jscode2session?appid=${appid}&secret=${secret}&js_code=${code}&grant_type=authorization_code`,
)
const wxData: any = await wxRes.json()
this.logger.log(`[bindWx] 微信接口返回: ${JSON.stringify(wxData)}`)
if (wxData.errcode) {
this.logger.error(`[bindWx] 微信登录失败: ${wxData.errmsg}, rid: ${wxData.rid || '无'}`)
throw new HttpException(`微信登录失败: ${wxData.errmsg}`, HttpStatus.UNAUTHORIZED)
}
const openid = wxData.openid
this.logger.log(`[bindWx] 获取到openid=${openid}`)
const existing = await this.userModel.findOne({ wxOpenid: openid }).exec()
if (existing) {
this.logger.warn(`[bindWx] openid=${openid} 已绑定到其他用户 ${existing._id}`)
throw new HttpException('该微信号已绑定其他账号', HttpStatus.CONFLICT)
}
const user = await this.userModel.findByIdAndUpdate(userId, { wxOpenid: openid }, { new: true }).exec()
if (!user) { this.logger.error(`[bindWx] 用户不存在 userId=${userId}`); throw new HttpException('用户不存在', HttpStatus.NOT_FOUND) }
this.logger.log(`[bindWx] openid=${openid} 绑定到用户 ${userId} 成功`)
return { message: '微信绑定成功', wxOpenid: openid }
}
async sendEmailCode(email: string) { async sendEmailCode(email: string) {
if (!email || !email.includes('@')) { if (!email || !email.includes('@')) {
throw new HttpException('请输入正确的邮箱地址', HttpStatus.BAD_REQUEST) throw new HttpException('请输入正确的邮箱地址', HttpStatus.BAD_REQUEST)
@@ -113,7 +156,7 @@ export class UserService {
if (!user) { if (!user) {
isNew = true isNew = true
const nick = email.split('@')[0] const nick = email.split('@')[0]
user = await this.userModel.create({ email, nickname: nick, remaining: 3 }) user = await this.userModel.create({ email, nickname: nick, remaining: 0, gravity: 5 })
} }
return { ...this.generateAuthResponse(user), isNew, hasPassword: !!user.password } return { ...this.generateAuthResponse(user), isNew, hasPassword: !!user.password }
} }
@@ -148,7 +191,7 @@ export class UserService {
} }
const nick = email.split('@')[0] const nick = email.split('@')[0]
const hashed = await bcrypt.hash(password, 10) const hashed = await bcrypt.hash(password, 10)
const user = await this.userModel.create({ email, nickname: nick, password: hashed, remaining: 3 }) const user = await this.userModel.create({ email, nickname: nick, password: hashed, remaining: 0, gravity: 5 })
return this.generateAuthResponse(user) return this.generateAuthResponse(user)
} }
@@ -216,6 +259,7 @@ export class UserService {
resumeDownloadCredits: user.resumeDownloadCredits ?? 0, resumeDownloadCredits: user.resumeDownloadCredits ?? 0,
freeOptimizeUsed: user.freeOptimizeUsed ?? 0, freeOptimizeUsed: user.freeOptimizeUsed ?? 0,
shareCredits: user.shareCredits ?? 0, shareCredits: user.shareCredits ?? 0,
gravity: user.gravity ?? 0,
} }
} }
} }
+8 -2
View File
@@ -1,6 +1,6 @@
# 职引 - 部署文档 # 职引 - 部署文档
> **最后更新**: 2026-06-09 > **最后更新**: 2026-06-21
> **生产环境**: 已部署(服务器已购 + 域名已配) > **生产环境**: 已部署(服务器已购 + 域名已配)
## 目录 ## 目录
@@ -128,7 +128,11 @@ npm run build:h5
### 3. 部署到 Web 服务器 ### 3. 部署到 Web 服务器
```bash ```bash
# 构建含 SEO 文件(robots.txt, sitemap.xml, 数字人头像)
npm run build:h5
# 部署所有产物
scp -r dist/build/h5/* user@your-server:/www/wwwroot/zhiyin.yzrcloud.cn/ scp -r dist/build/h5/* user@your-server:/www/wwwroot/zhiyin.yzrcloud.cn/
# 或者使用服务端脚本(自动复制 static/、robots.txt、sitemap.xml
``` ```
### 4. Nginx 配置 ### 4. Nginx 配置
@@ -223,7 +227,7 @@ node scripts/upload-mp.js
``` ```
### 版本号 ### 版本号
当前线上版本:**1.0.3**见 note.txt 当前线上版本:**1.0.17**git tag v1.0.16,脚本自动末位自增 → 上传版本 1.0.17
--- ---
@@ -249,3 +253,5 @@ node scripts/upload-mp.js
|------|----------|--------| |------|----------|--------|
| 2026-06-09 | 初版 | AI | | 2026-06-09 | 初版 | AI |
| 2026-06-09 | 更新生产域名:zhiyinwx.yzrcloud.cnAPI :3006)、zhiyin.yzrcloud.cnH5 静态目录) | 小之 | | 2026-06-09 | 更新生产域名:zhiyinwx.yzrcloud.cnAPI :3006)、zhiyin.yzrcloud.cnH5 静态目录) | 小之 |
| 2026-06-21 | 更新部署版本至 v1.0.16;小程序上传工具使用 git tag 自动获取版本号 | 小之 |
| 2026-06-21 | v4.8 SEO + 分享全面优化:部署新增 robots.txt、sitemap.xml、static/ 目录;版本号自动注入(Vite define);13 页面微信分享全部开启;上传脚本版本号末位自增 1 | AI |
+50 -13
View File
@@ -1,8 +1,8 @@
# 职引 · 完整功能清单 v4.1 # 职引 · 完整功能清单 v4.7
> **版本**: v4.1 > **版本**: v4.7
> **日期**: 2026-06-09 > **日期**: 2026-06-21
> **状态**: Phase 0.5 壁垒构建完成 > **状态**: Phase 1.5 按量购买引力值 + 全量生产部署
> **定位**: 应届生/实习生 AI 面试教练 > **定位**: 应届生/实习生 AI 面试教练
--- ---
@@ -41,6 +41,29 @@
| 连续打卡日历 | ✅ 完成 | 面试频率可视化,连续打卡激励 | P1 | | 连续打卡日历 | ✅ 完成 | 面试频率可视化,连续打卡激励 | P1 |
| 每日一题推送 | ⚠️ 半完成 | 首页展示 + API 读取,**无定时推送** | P0 | | 每日一题推送 | ⚠️ 半完成 | 首页展示 + API 读取,**无定时推送** | P0 |
### 1.4 面试复盘(新增)
| 功能 | 状态 | 描述 | 优先级 |
|------|------|------|--------|
| 音频文件上传 | ✅ 完成 | 支持 MP3/M4A/WAV/AAC/OGG/MP4/WebM50MB 上限 | P0 |
| 文本转录粘贴 | ✅ 完成 | 直接粘贴面试转录文本提交 | P0 |
| whisper.cpp ASR | ✅ 完成 | 本地离线语音转文字,支持 tiny/base 模型 | P0 |
| AI 面试评析 | ✅ 完成 | 四维评分(逻辑/表达/专业度/稳定性)+ 逐题评估 | P0 |
| 口语分析 | ✅ 完成 | 填充词检测 + 语速评估 | P0 |
| 无 ASR 回落 | ✅ 完成 | whisper 不可用时自动使用 mock | P1 |
| 历史记录管理 | ✅ 完成 | 列表/详情/删除 | P0 |
### 1.5 AI 择业顾问(新增)
| 功能 | 状态 | 描述 | 优先级 |
|------|------|------|--------|
| AI 专业/兴趣/性格分析 | ✅ 完成 | 基于用户输入的学业/兴趣/性格信息,AI 生成个性化职业分析 | P0 |
| 智能岗位匹配推荐 | ✅ 完成 | 对接热门岗位数据,推荐 3-5 个匹配岗位 | P0 |
| 个性化职业发展建议 | ✅ 完成 | 含短期/中期/长期三阶段规划 | P0 |
| 多轮追问式对话 | ✅ 完成 | 分析完成后可对任意建议进行追问 | P1 |
| 热门岗位数据联动 | ✅ 完成 | 推荐岗位可跳转面试练习(闭环:测→练→面) | P0 |
| 个人中心入口 | ✅ 完成 | 用户菜单增加"择业顾问"入口(NEW 标记) | P0 |
--- ---
## 二、用户端功能 ## 二、用户端功能
@@ -61,6 +84,7 @@
| 面试记录/统计 | ✅ 完成 | 总数/平均分/完成数 | | 面试记录/统计 | ✅ 完成 | 总数/平均分/完成数 |
| 进步轨迹 | ✅ 完成 | 雷达图 + 打卡日历 | | 进步轨迹 | ✅ 完成 | 雷达图 + 打卡日历 |
| 简历管理 | ✅ 完成 | 多份简历 CRUD + AI 分析 | | 简历管理 | ✅ 完成 | 多份简历 CRUD + AI 分析 |
| 面试复盘 | ✅ 完成 | 音频上传 → ASR → AI 评析 → 口语分析 |
| 会员中心 | ✅ 完成 | 套餐对比 + 支付 | | 会员中心 | ✅ 完成 | 套餐对比 + 支付 |
--- ---
@@ -70,13 +94,13 @@
### 3.1 会员系统 ### 3.1 会员系统
| 功能 | 状态 | 描述 | 优先级 | | 功能 | 状态 | 描述 | 优先级 |
|------|------|------|--------| |------|------|------|--------|
| 免费版 | ✅ 完成 | 日 2 次面试,5 轮/次 | P0 | | 免费版 | ✅ 完成 | 注册送 5 引力值,面试消耗 5/次 | P0 |
| 成长版 ¥19.9/月 | ✅ 完成 | 无限面试 + 高级报告 + 进步轨迹 | P0 | | 按量购买引力值 | ✅ 完成 | ¥5/份,含 5 引力值,H5 扫码支付 | P0 |
| 冲刺版 ¥49.9/月 | ❌ 未实现 | 高客单价缺失 | P1 | | 微信支付 Native(扫码) | ✅ 完成 | H5 扫码支付 | P0 |
| 微信支付 Native(扫码) | ✅ 完成 | H5 支付 | P0 | | 支付回调/自动到账 | ✅ 完成 | 回调验签 + 解密 + 引力值自动到账 | P0 |
| 微信支付 JSAPI | ✅ 完成 | 小程序内支付 | P0 | | 小程序剪贴板购买 | ✅ 完成 | 复制官网链接到手机浏览器打开购买 | P0 |
| 支付回调/自动开会员 | ✅ 完成 | 回调验签 + 解密 + 会员激活 | P0 | | 会员权益展示 | ✅ 完成 | 引力值 + 获取入口 | P0 |
| 会员权益对比 | ✅ 完成 | 免费/成长版对比展示 | P0 | | 客服按钮 | ✅ 完成 | 用户页 + 关于页 `<button open-type="contact">` | P1 |
### 3.2 B 端服务(Q4 启动) ### 3.2 B 端服务(Q4 启动)
| 功能 | 状态 | 描述 | | 功能 | 状态 | 描述 |
@@ -95,6 +119,8 @@
| 面试报告生成 | ✅ 完成 | 总分 + 四维 + 优劣势分析 | | 面试报告生成 | ✅ 完成 | 总分 + 四维 + 优劣势分析 |
| 简历诊断 | ✅ 完成 | 结构 + 表达 + 关键词 + 亮点分析 | | 简历诊断 | ✅ 完成 | 结构 + 表达 + 关键词 + 亮点分析 |
| 简历优化 | ✅ 完成 | 内容优化 + 差异展示 | | 简历优化 | ✅ 完成 | 内容优化 + 差异展示 |
| 面试复盘评析 | ✅ 完成 | 转录文本 → AI 评估 → 逐题分析 |
| 口语分析 | ✅ 完成 | 填充词检测 + 语速评估 |
| 技能缺口分析 | 📋 规划中 | 基于 JD 分析技能差距 | | 技能缺口分析 | 📋 规划中 | 基于 JD 分析技能差距 |
| 学习路径推荐 | 📋 规划中 | 知识图谱驱动的职业规划 | | 学习路径推荐 | 📋 规划中 | 知识图谱驱动的职业规划 |
@@ -104,18 +130,25 @@
| opencode-go (deepseek-v4-flash) | 主用 | ✅ 已配置 | | opencode-go (deepseek-v4-flash) | 主用 | ✅ 已配置 |
| NVIDIA (stepfun-ai/step-3.5-flash) | 备用 | ✅ 已配置 | | NVIDIA (stepfun-ai/step-3.5-flash) | 备用 | ✅ 已配置 |
### ASR 引擎配置
| 引擎 | 用途 | 状态 |
|------|------|------|
| whisper.cpp (tiny/base) | 本地离线 ASR | ✅ 已编译 + 已部署 |
| mock ASR | 回落方案 | ✅ 无 whisper 时自动使用 |
--- ---
## 五、技术功能 ## 五、技术功能
| 功能 | 状态 | 描述 | | 功能 | 状态 | 描述 |
|------|------|------| |------|------|------|
| MongoDB 数据存储 | ✅ 完成 | 8 个数据模型 | | MongoDB 数据存储 | ✅ 完成 | 9 个数据模型(新增 InterviewReview |
| JWT 认证 | ✅ 完成 | 全局守卫 + 白名单机制 | | JWT 认证 | ✅ 完成 | 全局守卫 + 白名单机制 |
| API 限流 | ✅ 完成 | @nestjs/throttler 10次/分钟 | | API 限流 | ✅ 完成 | @nestjs/throttler 10次/分钟 |
| 文件上传 | ✅ 完成 | 简历 PDF/图片解析 | | 文件上传 | ✅ 完成 | 简历 PDF/图片 + 面试录音 |
| CORS 配置 | ✅ 完成 | 全开放(生产需白名单) | | CORS 配置 | ✅ 完成 | 全开放(生产需白名单) |
| 参数校验 | ✅ 完成 | class-validator whitelist | | 参数校验 | ✅ 完成 | class-validator whitelist |
| whisper.cpp ASR | ✅ 完成 | C/C++ 原生二进制,CPU 推理,MIT 协议 |
--- ---
@@ -132,6 +165,7 @@
- [x] 会员系统(¥19.9 成长版) - [x] 会员系统(¥19.9 成长版)
- [x] 微信支付对接(Native + JSAPI - [x] 微信支付对接(Native + JSAPI
- [x] 公司真题库(用户贡献驱动) - [x] 公司真题库(用户贡献驱动)
- [x] **面试复盘(音频 ASR + AI 评析 + 口语分析)**
### P1(待实现) ### P1(待实现)
- [ ] 每日一题定时推送 - [ ] 每日一题定时推送
@@ -156,3 +190,6 @@
| 2026-06-01 | 重新定位:专注校招 | AI | | 2026-06-01 | 重新定位:专注校招 | AI |
| 2026-06-05 | 战略升级:新增数据飞轮/留存入围 | 小之 | | 2026-06-05 | 战略升级:新增数据飞轮/留存入围 | 小之 |
| 2026-06-09 | 同步代码:Phase 0.5 功能标记完成,修正状态 | AI | | 2026-06-09 | 同步代码:Phase 0.5 功能标记完成,修正状态 | AI |
| 2026-06-16 | **v4.2**:新增面试复盘功能(whisper.cpp ASR + AI 评析 + 口语分析) | AI |
| 2026-06-17 | **v4.3**:新增 AI 择业顾问功能(专业分析 + 岗位匹配 + 多轮对话) | AI |
| 2026-06-21 | **v4.7**:按量购买引力值重构(¥5/份取代月订阅);微信小程序剪贴板购买链路;客服按钮;管理后台全面完善;生产环境全量部署上线 | AI |
+57 -21
View File
@@ -1,8 +1,8 @@
# 职引项目 · 状态报告 v4.3 # 职引项目 · 状态报告 v4.8
> **项目版本**: v4.3 > **项目版本**: v4.8
> **更新时间**: 2026-06-11 > **更新时间**: 2026-06-21
> **项目状态**: ✅ 代码质量修复 + 全量测试体系搭建完成 > **项目状态**: ✅ SEO 优化 + 微信分享全面开启 + 全量部署
--- ---
@@ -13,9 +13,10 @@
| 项目名称 | 职引(ZhiYin | | 项目名称 | 职引(ZhiYin |
| 定位 | 应届生/实习生 AI 面试教练 | | 定位 | 应届生/实习生 AI 面试教练 |
| 技术栈 | NestJS + MongoDB + Uni-App(Vue3) | | 技术栈 | NestJS + MongoDB + Uni-App(Vue3) |
| 定价 | 免费版 / ¥19.9/月(成长版) / ¥49.9/月(冲刺版) | | 定价 | 免费版 / 按量购买引力值(¥5/份) |
| AI 模型 | DeepSeek V4-Flash(主) + Step-3.5-Flash(备) | | AI 模型 | DeepSeek V4-Flash(主) + Step-3.5-Flash(备) |
| 后端模块 | user, interview, resume, member, payment, positions, ai, analyze, upload, admin, email, progress, contribution, daily-question, schedule | | ASR | whisper.cpp(本地部署,tiny/base 模型,无需 API Key |
| 后端模块 | user, interview, resume, member, payment, positions, ai, analyze, upload, admin, email, progress, contribution, daily-question, schedule, interview-review, career-advice |
--- ---
@@ -23,17 +24,20 @@
| 模块 | 完成度 | 说明 | | 模块 | 完成度 | 说明 |
|------|------|------| |------|------|------|
| 后端 API | **98%** | 核心 + 护城河 P0-P5 全部实现 | | 后端 API | **99%** | 核心 + 护城河 P0-P5 全部实现 |
| 前端页面 | **85%** | 16 个页面含真实 API 调用 | | 前端页面 | **95%** | 20 个页面含真实 API 调用 |
| AI 面试模拟 | **95%** | 多轮对话 + 评分 + 报告 + 进度追踪 | | AI 面试模拟 | **95%** | 多轮对话 + 评分 + 报告 + 进度追踪 |
| 简历诊断/优化 | **95%** | 文件上传 + AI 分析 + 下载 | | 简历诊断/优化 | **95%** | 文件上传 + AI 分析 + 下载 |
| 支付系统(微信) | **95%** | API v3 完整对接,含真实证书 | | 支付系统(微信) | **95%** | API v3 完整对接,含真实证书H5 扫码支付可用 |
| 会员系统 | **100%** | 成长版 + 冲刺版,含权益扣减 | | 会员系统 | **100%** | 改为按量购买引力值体系(¥5/份),免费版注册送 5 引力值 |
| 护城河 P0-P5 | **100%** | AI 结构化 / 行业基准 / VIP 过期 / 分享卡片 / 打卡积分 / 岗位匹配 | | 护城河 P0-P5 | **100%** | AI 结构化 / 行业基准 / VIP 过期 / 分享卡片 / 打卡积分 / 岗位匹配 |
| 面试复盘 | **100%** | 音频上传 → whisper.cpp ASR → AI 评析 → 口语分析 |
| 测试体系 | **85%** | 43 单元 + 11 e2e + 7 前端 + Playwright 框架 | | 测试体系 | **85%** | 43 单元 + 11 e2e + 7 前端 + Playwright 框架 |
| 代码质量 | **95%** | console→Loggeras any 类型化,空 catch 检查 | | 代码质量 | **95%** | console→Loggeras any 类型化,空 catch 检查 |
| 安全审计 | **90%** | JWT 硬编码 / 凭据泄漏 / IDOR / NoSQL 注入 全部修复 | | 安全审计 | **90%** | JWT 硬编码 / 凭据泄漏 / IDOR / NoSQL 注入 全部修复 |
| 小程序审核 | **0%** | 类目已备案,未提交审核 | | 小程序审核 | **100%** | v1.0.17 已提交审核,类目已备案 |
| 生产部署 | **100%** | 后端 PM2 / H5 已部署 / 小程序 v1.0.17 已上传 |
| SEO / 分享优化 | **100%** | H5 canonical + robots.txt + sitemap + 结构化数据;小程序 13 页面全量开启分享 |
--- ---
@@ -73,9 +77,11 @@
### 3.4 商业化 ### 3.4 商业化
| 功能 | 后端 | 前端 | 状态 | | 功能 | 后端 | 前端 | 状态 |
|------|------|------|------| |------|------|------|------|
| 免费版额度(日2次/5轮) | ✅ | ✅ | **完成** | | 免费版注册送 5 引力值 | ✅ | ✅ | **完成** |
| 成长版 ¥19.9/月 | ✅ | ✅ | **完成** | | 按量购买引力值(¥5/份) | ✅ | ✅ | **完成** |
| 冲刺版 ¥49.9/月(含权益扣减) | ✅ | ✅ | **完成** | | 引力值统一体系(全部按引力值消耗) | ✅ | ✅ | **完成** |
| 会员月度引力值自动补给(cron) | ✅ | N/A | **完成** |
| 微信小程序剪贴板购买链路 | N/A | ✅ | **完成** |
| 微信支付 Native QR / JSAPI | ✅ | ✅ H5+MP | **完成** | | 微信支付 Native QR / JSAPI | ✅ | ✅ H5+MP | **完成** |
| 支付回调/自动开会员 | ✅ | N/A | **完成** | | 支付回调/自动开会员 | ✅ | N/A | **完成** |
| 每日一题定时推送 | ✅ | N/A | **完成**(需配置模板ID | | 每日一题定时推送 | ✅ | N/A | **完成**(需配置模板ID |
@@ -89,6 +95,28 @@
| 文件上传(PDF/图片) | ✅ | ✅ | **完成** | | 文件上传(PDF/图片) | ✅ | ✅ | **完成** |
| 结果下载(TXT/HTML) | N/A | ✅ | **完成** | | 结果下载(TXT/HTML) | N/A | ✅ | **完成** |
### 3.6 面试复盘(新增)
| 功能 | 后端 | 前端 | 状态 |
|------|------|------|------|
| 音频文件上传(MP3/M4A/WAV 等) | ✅ | ✅ | **完成** |
| 文本转录粘贴提交 | ✅ | ✅ | **完成** |
| whisper.cpp 本地 ASR 转写 | ✅ | N/A | **完成**tiny + base 模型) |
| AI 面试评估(四维评分) | ✅ | N/A | **完成** |
| 口语分析(填充词/语速) | ✅ | N/A | **完成** |
| 异步处理 + 状态轮询 | ✅ | ✅ | **完成** |
| 复盘历史列表/详情/删除 | ✅ | ✅ | **完成** |
| ASR mock 回落(whisper 不可用时) | ✅ | N/A | **完成** |
### 3.7 AI 择业顾问(新增)
| 功能 | 后端 | 前端 | 状态 |
|------|------|------|------|
| AI 专业/兴趣/性格分析 | ✅ | ✅ | **完成** |
| 智能岗位匹配推荐 | ✅ | ✅ | **完成** |
| 个性化职业发展建议 | ✅ | ✅ | **完成** |
| 多轮追问式对话 | ✅ | ✅ | **完成** |
| 热门岗位数据联动 | ✅ | N/A | **完成** |
| 个人中心入口(择业顾问) | N/A | ✅ | **完成** |
--- ---
## 四、测试体系 ## 四、测试体系
@@ -105,7 +133,7 @@
| 命令 | 用途 | | 命令 | 用途 |
|------|------| |------|------|
| `npm test` | 43 个单元测试 | | `npm test` | 43 个单元测试(构建检查通过) |
| `npm run test:e2e` | 11 个集成测试 | | `npm run test:e2e` | 11 个集成测试 |
| `npm run test:cov` | 覆盖率报告 | | `npm run test:cov` | 覆盖率报告 |
| `npm run test:browser` | Playwright API 测试 | | `npm run test:browser` | Playwright API 测试 |
@@ -150,7 +178,9 @@
| `payment` | controller + service + schema | ✅ | 微信支付 v3,含证书 | | `payment` | controller + service + schema | ✅ | 微信支付 v3,含证书 |
| `progress` | controller + schema + benchmark service | ✅ | 打卡/积分/基准/匹配 | | `progress` | controller + schema + benchmark service | ✅ | 打卡/积分/基准/匹配 |
| `contribution` | controller + schema (×2) | ✅ | 面经 + AI 结构化 + 公司题库 | | `contribution` | controller + schema (×2) | ✅ | 面经 + AI 结构化 + 公司题库 |
| `schedule` | module + service (×3) | ✅ | VIP 过期 / 每日一题 / 微信 token | | `schedule` | module + service (×4) | ✅ | VIP 过期 / 每日一题 / 微信 token / 月度引力值补给 |
| `interview-review` | controller + service + schema + asr service | ✅ | 面试复盘:音频 ASR + AI 评析 + 口语分析 |
| `career-advice` | controller + service + module | ✅ | AI 择业顾问:专业分析 + 岗位匹配 + 多轮对话 |
| `admin` | controller + module | ✅ | 管理后台 | | `admin` | controller + module | ✅ | 管理后台 |
| `email` | module + service | ✅ | 邮件发送 | | `email` | module + service | ✅ | 邮件发送 |
| `upload` | controller + module | ✅ | 文件上传 | | `upload` | controller + module | ✅ | 文件上传 |
@@ -166,13 +196,15 @@
| 面试模拟 | interview/interview | ✅ 多轮对话 + 计时 | | 面试模拟 | interview/interview | ✅ 多轮对话 + 计时 |
| 面试报告 | report/report | ✅ 评分/分析/全文回放/分享卡片 | | 面试报告 | report/report | ✅ 评分/分析/全文回放/分享卡片 |
| 历史记录 | history/history | ✅ 筛选/统计 | | 历史记录 | history/history | ✅ 筛选/统计 |
| 个人中心 | user/user | ✅ 信息/统计/管理员入口 | | 个人中心 | user/user | ✅ 引力值卡片 + 信息/统计/管理员入口 + 面试复盘入口 + 择业顾问入口 + 客服按钮 |
| 会员中心 | member/member | ✅ 套餐对比 + 支付 | | 会员中心 | member/member | ✅ 引力值按量购买(H5 扫码支付/小程序剪贴板链路) |
| 进步轨迹 | progress/progress | ✅ 雷达图 + 打卡日历 | | 进步轨迹 | progress/progress | ✅ 雷达图 + 打卡日历 |
| 面经贡献 | contribute/contribute | ✅ 表单提交 | | 面经贡献 | contribute/contribute | ✅ 表单提交 |
| 简历优化 | resume/resume | ✅ 诊断/优化/上传/下载 | | 简历优化 | resume/resume | ✅ 诊断/优化/上传/下载 |
| 面试复盘 | review/review | ✅ 三种模式(列表/上传/报告) |
| 择业顾问 | career/career | ✅ AI 专业分析 + 岗位匹配 + 多轮对话 |
| 实习搜索 | internship/internship | ✅ 热门岗位 | | 实习搜索 | internship/internship | ✅ 热门岗位 |
| 管理后台 | admin/admin | ✅ 仪表盘 | | 管理后台 | admin/admin | ✅ 仪表盘/用户/面试/简历/订单/定价/分享/岗位/诊断/管理员 · 全 CRUD |
| 关于/协议/隐私 | about/agreement/privacy | ✅ | | 关于/协议/隐私 | about/agreement/privacy | ✅ |
--- ---
@@ -192,7 +224,11 @@
| 日期 | 版本 | 变更内容 | 操作者 | | 日期 | 版本 | 变更内容 | 操作者 |
|------|------|----------|--------| |------|------|----------|--------|
| 2026-06-02 | v1.0 | 项目状态初版 | AI | | 2026-06-21 | **v4.8** | **SEO 全量优化**canonical URL、robots.txt、sitemap.xml、结构化数据);**微信分享全面开启**(13 个页面 onShareAppMessage + onShareTimeline);**版本号自动注入**Vite define __APP_VERSION__);**导航栏/Tab标题关键词优化**;manifest 描述更新;页面描述统一增强 | AI |
| 2026-06-21 | v4.7 | 按量购买引力值体系重构(¥5/份取代月订阅);member.vue 完全重写;微信小程序剪贴板购买链路;客服按钮;管理后台字段全面完善;代码清理;测试数据清理;后端/H5/小程序全量部署上线 | AI |
| 2026-06-19 | v4.6 | 引力值体系统一:VIP 取消无限面试改为月度引力值消耗;管理后台全面完善(搜索/筛选/分页/CRUD/分析tab/岗位描述字段) | AI |
| 2026-06-17 | v4.5 | AI 择业顾问 MVP:后端模块 + 前端职业分析页面 + 热门岗位联动 | AI |
| 2026-06-05 | v2.0 | 战略升级:文档重构 + 新增功能启动 | 小之 | | 2026-06-05 | v2.0 | 战略升级:文档重构 + 新增功能启动 | 小之 |
| 2026-06-09 | v4.2 | 冲刺版+每日推送+支付修复+全量代码评审 | AI | | 2026-06-09 | v4.2 | 冲刺版+每日推送+支付修复+全量代码评审 | AI |
| 2026-06-11 | **v4.3** | **安全修复 5 项 + 代码质量 14 处 + 测试体系 61 项 + 护城河 P0-P5 全部验证** | AI | | 2026-06-11 | v4.3 | 安全修复 5 项 + 代码质量 14 处 + 测试体系 61 项 + 护城河 P0-P5 全部验证 | AI |
| 2026-06-16 | **v4.4** | **面试复盘功能上线:音频 ASRwhisper.cpp+ AI 评析 + 口语分析 + 前端三模式页面** | AI |
+53 -31
View File
@@ -1,8 +1,8 @@
# 职引 · 产品路线图 v4.1 # 职引 · 产品路线图 v4.7
> **版本**: v4.1 > **版本**: v4.7
> **日期**: 2026-06-09 > **日期**: 2026-06-21
> **状态**: Phase 0.5 壁垒构建完成,待上线 > **状态**: Phase 1.5 按量购买引力值上线 + 全量部署
> **定位**: 应届生/实习生 AI 面试教练 > **定位**: 应届生/实习生 AI 面试教练
--- ---
@@ -14,13 +14,13 @@ Phase 0: 战略升级(✅ 已完成)
Phase 0.5: 壁垒构建(✅ 已完成) Phase 0.5: 壁垒构建(✅ 已完成)
Phase 1: MVP 上线(D7-14)→ 小程序审核 + 内测 + 支付生产 Phase 1: MVP 开发(✅ 已完成)→ 面试复盘 + whisper.cpp ASR 集成
Phase 1.5: 辅助功能 + 商业化(D14-30)→ PMF 验证 Phase 1.5: 商业化 + 部署上线(D30-60)→ PMF 验证
Phase 2: 增强 + 题库(D30-60)→ 秋招准备 Phase 2: 增强 + 题库(D60-90)→ 秋招准备
Phase 3: 商业化 + B 端(D60-90)→ 秋招爆发 Phase 3: 商业化 + B 端(D90+)→ 秋招爆发
``` ```
--- ---
@@ -60,38 +60,54 @@ Phase 3: 商业化 + B 端(D60-90)→ 秋招爆发
--- ---
## 四、Phase 1MVP 上线(D7-14,当前阶段 ## 四、Phase 1MVP 开发(✅ 已完成,2026-06-16
### 4.1 上线准备 ### 4.1 面试复盘功能
| 任务 | 描述 | 状态 | | 任务 | 描述 | 状态 |
|------|------|------| |------|------|------|
| 前端页面完善 | 16 个页面全部就绪 | ✅ 完成 | | 音频文件上传 | 支持 MP3/M4A/WAV 等格式,50MB 上限 | ✅ 完成 |
| 文本转录提交 | 直接粘贴面试文本 | ✅ 完成 |
| whisper.cpp 本地 ASR | 离线语音转文字,tiny + base 模型 | ✅ 完成 |
| AI 面试评析 | 四维评分 + 逐题评估 | ✅ 完成 |
| 口语分析 | 填充词检测 + 语速评估 | ✅ 完成 |
| 前端三模式页面 | 列表/上传/报告三种视图 | ✅ 完成 |
| 个人中心入口 | "面试复盘"菜单项 | ✅ 完成 |
### 4.2 上线准备
| 任务 | 描述 | 状态 |
|------|------|------|
| 前端页面完善 | 17 个页面全部就绪 | ✅ 完成 |
| 微信登录联调 | 真实 appid 验证 | ⏳ 待进行 | | 微信登录联调 | 真实 appid 验证 | ⏳ 待进行 |
| 移除开发绕过 | `member/pay` 直接激活 | ⏳ 待进行 | | 移除开发绕过 | `member/pay` 直接激活 | ✅ 已完成 |
| 生产环境部署 | 服务器 + MongoDB + Nginx + PM2 | ✅ 服务器已购,域名已配(zhiyinwx → API:3006zhiyin.yzrcloud → H5 静态目录) | | 生产环境部署 | 服务器 + MongoDB + Nginx + PM2 | ✅ 已部署 |
| 小程序审核提交 | 资质齐全 | ⏳ 待进行 | | 小程序审核提交 | v1.0.16 已上传 | ⏳ 待审核 |
| 内测版发布 | 邀请码方式,100 人内测 | ⏳ 待进行 | | 内测版发布 | 邀请码方式,100 人内测 | ⏳ 待进行 |
### 4.2 内测指标 ### 4.3 内测指标
- **关键指标**: 次日留存 > 30%7 日留存 > 15% - **关键指标**: 次日留存 > 30%7 日留存 > 15%
- **反馈收集**: 问卷 + 访谈 - **反馈收集**: 问卷 + 访谈
- **如果达标**: 继续 Phase 1.5 - **如果达标**: 继续 Phase 1.5
--- ---
## 五、Phase 1.5:辅助功能 + 商业化(D14-30 ## 五、Phase 1.5:辅助功能 + 商业化(D30-60
| 功能 | 描述 | 优先级 | | 功能 | 描述 | 优先级 | 状态 |
|------|------|--------| |------|------|--------|------|
| 每日一题定时推送 | 微信订阅消息推送 | P0 | | 每日一题定时推送 | 微信订阅消息推送 | P0 | ✅ 已完成 |
| 冲刺版 ¥49.9/月 | 高客单价 | P1 | | 按量购买引力值 | 取消月订阅,改为 ¥5/份 | P0 | ✅ 已完成 |
| 连续打卡激励 | 7 天解锁高级报告 | P1 | | 连续打卡激励 | 7 天解锁高级报告 | P1 | 📋 规划中 |
| 付费转化验证 | 100 内测用户 → 10+ 付费 | P0 | | ASR 生产化调优 | 多模型切换、模型量化、推理优化 | P1 | 📋 规划中 |
| PMF 决策 | 转化率 > 5% → 继续 | P0 | | AI 择业顾问 MVP | AI 专业分析 + 岗位匹配 + 多轮对话 | P0 | ✅ 已完成 |
| 小程序剪贴板购买链路 | 复制链接到浏览器购买 | P0 | ✅ 已完成 |
| H5 扫码支付部署 | ¥5/份 H5 直接支付 | P0 | ✅ 已完成 |
| 生产环境全量部署 | 后端 PM2 + H5 Nginx + 小程序上传 | P0 | ✅ 已完成 |
| 付费转化验证 | 100 内测用户 → 10+ 付费 | P0 | ⏳ 待进行 |
| PMF 决策 | 转化率 > 5% → 继续 | P0 | ⏳ 待进行 |
--- ---
## 六、Phase 2:增强 + 真题库(D30-60,秋招前) ## 六、Phase 2:增强 + 真题库(D60-90,秋招前)
### 6.1 真题库建设 ### 6.1 真题库建设
| 公司 | 题库规模 | 状态 | | 公司 | 题库规模 | 状态 |
@@ -112,7 +128,7 @@ Phase 3: 商业化 + B 端(D60-90)→ 秋招爆发
--- ---
## 七、Phase 3:商业化 + B 端(D60-90,秋招爆发) ## 七、Phase 3:商业化 + B 端(D90+,秋招爆发)
### 7.1 增长目标 ### 7.1 增长目标
- 付费用户突破 1000 - 付费用户突破 1000
@@ -140,10 +156,12 @@ Phase 3: 商业化 + B 端(D60-90)→ 秋招爆发
|--------|------|--------|----------| |--------|------|--------|----------|
| M0: 战略升级 | ✅ D1 | 文档 + 定价 | 已完成 | | M0: 战略升级 | ✅ D1 | 文档 + 定价 | 已完成 |
| M0.5: 壁垒构建 | ✅ D7 | 进步轨迹 + 面经贡献 + 每日一题 | 功能可用 | | M0.5: 壁垒构建 | ✅ D7 | 进步轨迹 + 面经贡献 + 每日一题 | 功能可用 |
| M1: MVP 上线 | D14 | 小程序审核通过,内测启动 | 100 内测用户 | | M1: MVP 开发 | D14 | 面试复盘 + whisper.cpp ASR | 功能可用,build + test 通过 |
| M2: PMF 验证 | D30 | 100 用户反馈 | 转化率 > 5% | | M1.5: 商业化重构 | ✅ D21 | 按量购买引力值 + 管理后台完善 + H5/小程序部署 | 构建通过,已推送上线 |
| M3: 付费上线 | D45 | 冲刺版 + 定时推送 | 50+ 付费用户 | | M2: 上线内测 | D30 | 小程序审核通过,内测启动 | 100 内测用户 |
| M4: 秋招冲刺 | D90 | 秋招推广 | 1000+ 付费用户 | | M3: PMF 验证 | D60 | 100 用户反馈 | 转化率 > 5% |
| M4: 付费增长 | D75 | 引力值运营 + 推送 | 50+ 付费用户 |
| M5: 秋招冲刺 | D90+ | 秋招推广 | 1000+ 付费用户 |
--- ---
@@ -157,8 +175,9 @@ Phase 3: 商业化 + B 端(D60-90)→ 秋招爆发
关键时间点: 关键时间点:
6月9日:壁垒构建完成,Phase 0.5 交付 6月9日:壁垒构建完成,Phase 0.5 交付
6月15日:MVP 上线,内测启动 6月16日:面试复盘上线,MVP 开发完成
7月1日:PMF 验证,付费转化 6月30日:MVP 上线,内测启动
7月15日:PMF 验证,付费转化
8月1日:Phase 2 完成,准备秋招 8月1日:Phase 2 完成,准备秋招
9月1日:秋招旺季,全力推广 9月1日:秋招旺季,全力推广
``` ```
@@ -185,3 +204,6 @@ Phase 3: 商业化 + B 端(D60-90)→ 秋招爆发
| 2026-06-01 | 重新规划:专注校招 | AI | | 2026-06-01 | 重新规划:专注校招 | AI |
| 2026-06-05 | 战略升级:三层壁垒 + 新定价 | 小之 | | 2026-06-05 | 战略升级:三层壁垒 + 新定价 | 小之 |
| 2026-06-09 | Phase 0.5 标记完成,调整后续里程碑时间 | AI | | 2026-06-09 | Phase 0.5 标记完成,调整后续里程碑时间 | AI |
| 2026-06-16 | **v4.2**Phase 1 MVP 开发完成,面试复盘上线,里程碑 M1 完成 | AI |
| 2026-06-17 | **v4.3**:AI 择业顾问 MVP 上线,里程碑 M1.5 完成 | AI |
| 2026-06-21 | **v4.7**:按量购买引力值重构(¥5/份取代月订阅);管理后台全面完善;微信小程序剪贴板购买链路;客服按钮;全量生产部署上线(后端 PM2/H5 Nginx/小程序 v1.0.16 | AI |
+2 -1
View File
@@ -1,4 +1,5 @@
# 生产环境变量 # 生产环境变量
VITE_API_BASE_URL=https://zhiyinwx.yzrcloud.cn/api VITE_API_BASE_URL=https://zhiyinwx.yzrcloud.cn/api
VITE_APP_NAME=AI磁场 VITE_APP_NAME=职引·宇之然AI磁场
VITE_APP_DESC=AI模拟面试·简历优化·面经题库·校招求职一站式平台
VITE_PROD_API_HOST=https://zhiyinwx.yzrcloud.cn VITE_PROD_API_HOST=https://zhiyinwx.yzrcloud.cn
+25 -5
View File
@@ -6,20 +6,40 @@
<link rel="icon" type="image/svg+xml" href="/static/favicon.svg" /> <link rel="icon" type="image/svg+xml" href="/static/favicon.svg" />
<title>职引 - AI模拟面试 | 宇之然AI磁场 · 智能面试练习 · 简历优化 · 面经题库</title> <title>职引 - AI模拟面试 | 宇之然AI磁场 · 智能面试练习 · 简历优化 · 面经题库</title>
<meta name="description" content="职引是宇之然AI磁场旗下的AI模拟面试平台,提供AI面试官模拟练习、简历智能诊断优化、大厂面经题库、实习推荐等一站式求职服务,帮助你在真实面试前充分准备。" /> <meta name="description" content="职引是宇之然AI磁场旗下的AI模拟面试平台,提供AI面试官模拟练习、简历智能诊断优化、大厂面经题库、实习推荐等一站式求职服务,帮助你在真实面试前充分准备。" />
<meta name="keywords" content="职引,宇之然AI磁场,AI面试,模拟面试,面试练习,简历优化,求职辅导,AI模拟面试官,面试题库,面经,校招面试,实习面试,简历诊断,面试技巧" /> <meta name="keywords" content="职引,宇之然AI磁场,AI面试,模拟面试,面试练习,简历优化,求职辅导,AI模拟面试官,面试题库,面经,校招面试,实习面试,简历诊断,面试技巧,AI模拟面试官,大模型面试,校招准备" />
<meta property="og:title" content="职引 - AI模拟面试平台 | 宇之然AI磁场" /> <meta property="og:title" content="职引 - AI模拟面试平台 | 宇之然AI磁场" />
<meta property="og:description" content="AI驱动的一站式求职准备平台,涵盖AI模拟面试、简历优化、面经分享、实习推荐等核心功能。" /> <meta property="og:description" content="AI驱动的一站式求职准备平台,涵盖AI模拟面试、简历优化、面经分享、实习推荐等核心功能。" />
<meta property="og:type" content="website" /> <meta property="og:type" content="website" />
<meta property="og:url" content="https://zhiyin.yzrcloud.cn" />
<meta property="og:image" content="https://zhiyin.yzrcloud.cn/static/og-image.png" />
<meta name="applicable-device" content="mobile" /> <meta name="applicable-device" content="mobile" />
<link rel="canonical" href="https://aicc.yzrcloud.cn" /> <meta name="baidu-site-verification" content="zhiyin-yzrcloud" />
<link rel="canonical" href="https://zhiyin.yzrcloud.cn" />
<link rel="alternate" href="https://zhiyin.yzrcloud.cn" hreflang="zh-Hans" />
<script type="application/ld+json"> <script type="application/ld+json">
{ {
"@context": "https://schema.org", "@context": "https://schema.org",
"@type": "WebApplication", "@type": "WebApplication",
"name": "宇之然AI磁场", "name": "职引 - AI模拟面试",
"description": "AI驱动的求职面试模拟与简历优化平台", "description": "AI驱动的校招求职准备平台,涵盖AI模拟面试、简历智能优化、大厂面经题库、面试复盘、AI择业顾问等一站式功能。",
"applicationCategory": "EducationalApplication", "applicationCategory": "EducationalApplication",
"operatingSystem": "Web, WeChat Mini Program" "operatingSystem": "Web, WeChat Mini Program",
"offers": {
"@type": "Offer",
"price": "0",
"priceCurrency": "CNY"
}
}
</script>
<script type="application/ld+json">
{
"@context": "https://schema.org",
"@type": "BreadcrumbList",
"itemListElement": [
{ "@type": "ListItem", "position": 1, "name": "AI模拟面试", "item": "https://zhiyin.yzrcloud.cn" },
{ "@type": "ListItem", "position": 2, "name": "简历优化", "item": "https://zhiyin.yzrcloud.cn/#/pages/resume/resume" },
{ "@type": "ListItem", "position": 3, "name": "面经题库", "item": "https://zhiyin.yzrcloud.cn/#/pages/history/history" }
]
} }
</script> </script>
</head> </head>
+2 -2
View File
@@ -3,9 +3,9 @@
"version": "1.0.0", "version": "1.0.0",
"scripts": { "scripts": {
"dev:mp-weixin": "uni -p mp-weixin", "dev:mp-weixin": "uni -p mp-weixin",
"build:mp-weixin": "uni build -p mp-weixin && cp -n static/avatar-*.png dist/build/mp-weixin/static/ 2>/dev/null; true", "build:mp-weixin": "uni build -p mp-weixin && cp -f static/avatar-*.png dist/build/mp-weixin/static/ 2>/dev/null; true",
"dev:h5": "uni", "dev:h5": "uni",
"build:h5": "uni build && cp -n static/avatar-*.png dist/build/h5/static/ 2>/dev/null; true", "build:h5": "uni build && cp -f static/avatar-*.png dist/build/h5/static/ 2>/dev/null && cp -f static/robots.txt static/sitemap.xml dist/build/h5/ 2>/dev/null; true",
"test": "vitest run", "test": "vitest run",
"test:watch": "vitest" "test:watch": "vitest"
}, },
+9 -2
View File
@@ -15,11 +15,18 @@ async function main() {
ignores: ['node_modules/**/*'], ignores: ['node_modules/**/*'],
}) })
// Get version from git tag or use timestamp // Bump patch version from git tag: v1.0.16 → 1.0.17
let version = '1.0.3' let version = '1.0.3'
try { try {
const gitTag = execSync('git describe --tags --abbrev=0 2>/dev/null || echo ""', { encoding: 'utf8' }).trim() const gitTag = execSync('git describe --tags --abbrev=0 2>/dev/null || echo ""', { encoding: 'utf8' }).trim()
if (gitTag) version = gitTag.replace(/^v/, '') if (gitTag) {
const base = gitTag.replace(/^v/, '')
const parts = base.split('.')
if (parts.length >= 1) {
parts[parts.length - 1] = String(Number(parts[parts.length - 1]) + 1)
}
version = parts.join('.')
}
} catch {} } catch {}
const now = new Date() const now = new Date()
+48 -4
View File
@@ -1,9 +1,53 @@
<script setup lang="ts"> <script setup lang="ts">
import { onLaunch, onShow, onHide } from '@dcloudio/uni-app' import { onLaunch } from '@dcloudio/uni-app'
onLaunch(() => { console.log('职引 App launched') }) onLaunch(() => {
onShow(() => { console.log('职引 App shown') }) // #ifdef MP-WEIXIN
onHide(() => { console.log('职引 App hidden') }) initPrivacy()
// #endif
// #ifdef H5
handleH5UrlParams()
// #endif
})
// #ifdef H5
function handleH5UrlParams() {
const params = new URLSearchParams(window.location.search)
const token = params.get('token')
const buy = params.get('buy')
if (token) {
uni.setStorageSync('token', token)
}
if (buy === 'gravity') {
// 延迟等 app 初始化完成再跳转
setTimeout(() => {
uni.navigateTo({ url: '/pages/member/member' })
}, 300)
}
}
// #endif
// #ifdef MP-WEIXIN
function initPrivacy() {
if (wx.onNeedPrivacyAuthorization) {
wx.onNeedPrivacyAuthorization((resolve) => {
uni.showModal({
title: '隐私政策授权',
content: '请仔细阅读并同意《用户服务协议》和《隐私政策》后再使用本应用。您的个人信息将仅用于求职服务。',
confirmText: '同意',
cancelText: '拒绝',
success: (res) => {
if (res.confirm) {
resolve({ event: 'agree' })
} else {
resolve({ event: 'disagree' })
}
},
})
})
}
}
// #endif
</script> </script>
<style> <style>
@@ -0,0 +1,193 @@
import { ref, computed, nextTick } from 'vue'
import { api } from '../config'
export function useGravityPurchase(onPaymentSuccess?: () => void) {
const showQuantityModal = ref(false)
const buyQuantity = ref(1)
const products = ref<any[]>([])
const buyProductType = ref('interview')
const payLoading = ref(false)
const payError = ref('')
const showPayModal = ref(false)
const payCodeUrl = ref('')
const currentOutTradeNo = ref('')
const isMp = ref(false)
const paySuccess = ref(false)
const unitPrice = computed(() => {
const p = products.value.find(p => p.type === buyProductType.value)
return p?.price || 0
})
const totalPrice = computed(() => unitPrice.value * buyQuantity.value / 100)
const buyGravityPerUnit = computed(() => {
const p = products.value.find(p => p.type === buyProductType.value)
return p?.gravity || 0
})
const changeQty = (delta: number) => {
const next = buyQuantity.value + delta
if (next >= 1 && next <= 99) buyQuantity.value = next
}
const clampQty = () => {
if (buyQuantity.value < 1) buyQuantity.value = 1
if (buyQuantity.value > 99) buyQuantity.value = 99
}
const loadProducts = async () => {
try {
const res = await uni.request({ url: api('/member/plans'), method: 'GET' })
if (res.statusCode >= 200 && res.statusCode < 300 && res.data?.products) {
const prodList: any[] = []
for (const [key, val] of Object.entries(res.data.products as Record<string, any>)) {
if (val?.price > 0) prodList.push({ type: key, ...val })
}
products.value = prodList
}
} catch (e) { /* silent */ }
}
const openGravityPurchase = async (productType = 'interview') => {
buyProductType.value = productType
buyQuantity.value = 1
// #ifdef MP-WEIXIN
isMp.value = true
// #endif
// 加载产品信息
await loadProducts()
showQuantityModal.value = true
}
const cancelPay = () => {
showPayModal.value = false
payCodeUrl.value = ''
payLoading.value = false
payError.value = ''
}
const confirmProductBuy = () => {
showQuantityModal.value = false
startProductPay(buyProductType.value, buyQuantity.value)
}
const startProductPay = async (type: string, quantity = 1) => {
const token = uni.getStorageSync('token') || ''
if (!token) {
uni.showToast({ title: '请先登录', icon: 'none' })
return
}
showPayModal.value = true
payLoading.value = true
payError.value = ''
if (isMp.value) {
try {
let res = await uni.request({
url: api('/payment/jsapi-product'), method: 'POST',
data: { type, quantity },
header: { 'Authorization': `Bearer ${token}`, 'Content-Type': 'application/json' },
timeout: 30000,
})
payLoading.value = false
if (res.statusCode >= 200 && res.statusCode < 300 && res.data?.payParams) {
const pp = res.data.payParams as any
currentOutTradeNo.value = res.data.outTradeNo || ''
uni.requestPayment({
provider: 'wxpay',
timeStamp: pp.timeStamp,
nonceStr: pp.nonceStr,
package: pp.package,
signType: pp.signType || 'RSA',
paySign: pp.paySign,
success: () => {
const no = currentOutTradeNo.value || res.data.outTradeNo
pollPayResult(no)
},
fail: () => {
payError.value = '支付未完成'
uni.showToast({ title: '支付未完成', icon: 'none' })
},
})
} else if (!res.statusCode || res.statusCode === 0) {
payError.value = '网络连接失败,请检查网络后重试'
uni.showToast({ title: '网络连接失败', icon: 'none' })
} else {
const errMsg = res.data?.message || '购买失败'
payError.value = errMsg
uni.showToast({ title: errMsg, icon: 'none' })
}
} catch (e) {
payLoading.value = false
payError.value = '网络错误,请重试'
uni.showToast({ title: '网络错误', icon: 'none' })
}
} else {
try {
const res = await uni.request({
url: api('/payment/create-product'), method: 'POST',
data: { type, quantity },
header: { 'Authorization': `Bearer ${token}`, 'Content-Type': 'application/json' },
})
payLoading.value = false
if (res.statusCode >= 200 && res.statusCode < 300 && res.data?.codeUrl) {
payCodeUrl.value = res.data.codeUrl
currentOutTradeNo.value = res.data.outTradeNo
// 页面需要自己渲染二维码(依赖 uqrcode)
pollPayResult(res.data.outTradeNo)
} else if (!res.statusCode || res.statusCode === 0) {
payError.value = '网络连接失败,请检查网络后重试'
uni.showToast({ title: '网络连接失败', icon: 'none' })
} else {
payError.value = res.data?.message || '购买失败'
uni.showToast({ title: res.data?.message || '购买失败', icon: 'none' })
}
} catch (e) {
payLoading.value = false
payError.value = '网络错误,请重试'
uni.showToast({ title: '网络错误', icon: 'none' })
}
}
}
const pollPayResult = async (outTradeNo: string) => {
if (!outTradeNo) return
const maxAttempts = 30
let attempts = 0
const token = uni.getStorageSync('token') || ''
const poll = async () => {
attempts++
try {
const res = await uni.request({
url: api(`/payment/check/${outTradeNo}`), method: 'GET',
header: { 'Authorization': `Bearer ${token}` },
})
if (res.statusCode >= 200 && res.statusCode < 300 && res.data?.status === 'success') {
paySuccess.value = true
showPayModal.value = false
uni.showToast({ title: '充值成功!', icon: 'success' })
onPaymentSuccess?.()
return
}
} catch (e) { /* ignore */ }
if (attempts < maxAttempts) {
setTimeout(poll, 2000)
} else {
payError.value = '支付结果查询超时,请联系客服'
uni.showToast({ title: '支付查询超时', icon: 'none' })
}
}
setTimeout(poll, 2000)
}
return {
showQuantityModal, buyQuantity, products, buyProductType,
unitPrice, totalPrice, buyGravityPerUnit,
payLoading, payError, showPayModal, payCodeUrl, paySuccess,
isMp,
changeQty, clampQty, loadProducts, openGravityPurchase,
confirmProductBuy, cancelPay,
}
}
+13
View File
@@ -23,6 +23,7 @@ export const APP_CONFIG = {
USER: '/pages/user/user', USER: '/pages/user/user',
LOGIN: '/pages/login/login', LOGIN: '/pages/login/login',
ABOUT: '/pages/about/about', ABOUT: '/pages/about/about',
CAREER: '/pages/career/career',
}, },
STORAGE_KEYS: { STORAGE_KEYS: {
TOKEN: 'token', TOKEN: 'token',
@@ -109,6 +110,18 @@ export const API_ENDPOINTS = {
RECORDS: '/share/records', RECORDS: '/share/records',
VISITORS: '/share/visitors', VISITORS: '/share/visitors',
}, },
REVIEW: {
UPLOAD: '/interview-review',
TEXT: '/interview-review/text',
LIST: '/interview-review/list',
DETAIL: (id: string) => `/interview-review/${id}`,
DELETE: (id: string) => `/interview-review/${id}`,
},
CAREER: {
ANALYZE: '/career-advice/analyze',
CHAT: '/career-advice/chat',
POSITIONS: '/career-advice/positions',
},
} as const } as const
const PROD_API_HOST = import.meta.env.VITE_PROD_API_HOST || 'https://zhiyinwx.yzrcloud.cn' const PROD_API_HOST = import.meta.env.VITE_PROD_API_HOST || 'https://zhiyinwx.yzrcloud.cn'
+6 -5
View File
@@ -1,9 +1,9 @@
{ {
"name": "宇之然AI磁场", "name": "职引 - AI模拟面试",
"appid": "__UNI__DEV__", "appid": "__UNI__DEV__",
"versionName": "1.0.11", "versionName": "1.0.16",
"versionCode": "111", "versionCode": "116",
"description": "职引 - 宇之然AI磁场旗下AI模拟面试平台,提供AI面试官模拟练习、简历智能优化、大厂面经题库,助你轻松应对校招面试。", "description": "职引 - 宇之然AI磁场旗下AI模拟面试平台AI面试官模拟练习、简历智能优化、大厂面经题库、面试复盘分析、AI择业顾问,一站式校招求职准备平台。",
"h5": { "h5": {
"title": "职引 - AI模拟面试 | 宇之然AI磁场", "title": "职引 - AI模拟面试 | 宇之然AI磁场",
"router": { "router": {
@@ -13,7 +13,8 @@
"mp-weixin": { "mp-weixin": {
"appid": "wxf466b3c3bc411ffc", "appid": "wxf466b3c3bc411ffc",
"setting": { "setting": {
"urlCheck": false "urlCheck": false,
"__usePrivacyCheck__": true
}, },
"usingComponents": true "usingComponents": true
} }
+18 -16
View File
@@ -1,23 +1,25 @@
{ {
"pages": [ "pages": [
{ "path": "pages/index/index", "style": { "navigationBarTitleText": "职引 - AI模拟面试" } }, { "path": "pages/index/index", "style": { "navigationBarTitleText": "AI模拟面试 - 职引" } },
{ "path": "pages/interview/interview", "style": { "navigationBarTitleText": "AI模拟面试" } }, { "path": "pages/interview/interview", "style": { "navigationBarTitleText": "AI模拟面试练习" } },
{ "path": "pages/report/report", "style": { "navigationBarTitleText": "面试报告" } }, { "path": "pages/report/report", "style": { "navigationBarTitleText": "面试报告与评分" } },
{ "path": "pages/member/member", "style": { "navigationBarTitleText": "会员中心" } }, { "path": "pages/member/member", "style": { "navigationBarTitleText": "引力值购买" } },
{ "path": "pages/progress/progress", "style": { "navigationBarTitleText": "进步轨迹" } }, { "path": "pages/progress/progress", "style": { "navigationBarTitleText": "进步轨迹与打卡" } },
{ "path": "pages/contribute/contribute", "style": { "navigationBarTitleText": "面经分享" } }, { "path": "pages/contribute/contribute", "style": { "navigationBarTitleText": "分享面经" } },
{ "path": "pages/company-bank/bank", "style": { "navigationBarTitleText": "公司真题库" } }, { "path": "pages/company-bank/bank", "style": { "navigationBarTitleText": "公司真题库" } },
{ "path": "pages/login/login", "style": { "navigationBarTitleText": "登录" } }, { "path": "pages/login/login", "style": { "navigationBarTitleText": "登录 / 注册" } },
{ "path": "pages/history/history", "style": { "navigationBarTitleText": "面试记录" } }, { "path": "pages/history/history", "style": { "navigationBarTitleText": "面经与面试记录" } },
{ "path": "pages/user/user", "style": { "navigationBarTitleText": "我的" } }, { "path": "pages/user/user", "style": { "navigationBarTitleText": "个人中心" } },
{ "path": "pages/resume/resume", "style": { "navigationBarTitleText": "简历优化" } }, { "path": "pages/resume/resume", "style": { "navigationBarTitleText": "简历诊断与优化" } },
{ "path": "pages/internship/internship", "style": { "navigationBarTitleText": "实习搜索" } }, { "path": "pages/internship/internship", "style": { "navigationBarTitleText": "热门岗位" } },
{ "path": "pages/about/about", "style": { "navigationBarTitleText": "关于职引" } }, { "path": "pages/about/about", "style": { "navigationBarTitleText": "关于职引" } },
{ "path": "pages/admin/admin", "style": { "navigationBarTitleText": "管理后台" } }, { "path": "pages/admin/admin", "style": { "navigationBarTitleText": "管理后台" } },
{ "path": "pages/result/result", "style": { "navigationBarTitleText": "优化结果" } }, { "path": "pages/result/result", "style": { "navigationBarTitleText": "优化结果" } },
{ "path": "pages/agreement/agreement", "style": { "navigationBarTitleText": "用户协议" } }, { "path": "pages/agreement/agreement", "style": { "navigationBarTitleText": "用户协议" } },
{ "path": "pages/privacy/privacy", "style": { "navigationBarTitleText": "隐私政策" } }, { "path": "pages/privacy/privacy", "style": { "navigationBarTitleText": "隐私政策" } },
{ "path": "pages/share/share", "style": { "navigationBarTitleText": "我的分享" } } { "path": "pages/share/share", "style": { "navigationBarTitleText": "我的分享" } },
{ "path": "pages/review/review", "style": { "navigationBarTitleText": "面试复盘分析" } },
{ "path": "pages/career/career", "style": { "navigationBarTitleText": "AI择业顾问" } }
], ],
"tabBar": { "tabBar": {
"color": "#999999", "color": "#999999",
@@ -25,9 +27,9 @@
"backgroundColor": "#F3F4F6", "backgroundColor": "#F3F4F6",
"borderStyle": "black", "borderStyle": "black",
"list": [ "list": [
{ "pagePath": "pages/index/index", "text": "面试", "iconPath": "static/tabbar/home.png", "selectedIconPath": "static/tabbar/home-active.png" }, { "pagePath": "pages/index/index", "text": "AI面试", "iconPath": "static/tabbar/home.png", "selectedIconPath": "static/tabbar/home-active.png" },
{ "pagePath": "pages/history/history", "text": "面经", "iconPath": "static/tabbar/history.png", "selectedIconPath": "static/tabbar/history-active.png" }, { "pagePath": "pages/history/history", "text": "面经题库", "iconPath": "static/tabbar/history.png", "selectedIconPath": "static/tabbar/history-active.png" },
{ "pagePath": "pages/user/user", "text": "我的", "iconPath": "static/tabbar/user.png", "selectedIconPath": "static/tabbar/user-active.png" } { "pagePath": "pages/user/user", "text": "个人中心", "iconPath": "static/tabbar/user.png", "selectedIconPath": "static/tabbar/user-active.png" }
] ]
}, },
"globalStyle": { "globalStyle": {
+9 -1
View File
@@ -2,7 +2,7 @@
<view class="page"> <view class="page">
<view class="logo-area"> <view class="logo-area">
<text class="logo">职引</text> <text class="logo">职引</text>
<text class="version">v1.0.0</text> <text class="version">v{{ appVersion }}</text>
</view> </view>
<view class="info-section"> <view class="info-section">
<text class="info-label">产品名称</text> <text class="info-label">产品名称</text>
@@ -38,8 +38,14 @@
</template> </template>
<script setup> <script setup>
const appVersion = typeof __APP_VERSION__ !== 'undefined' ? __APP_VERSION__ : '1.0.0'
const goAgreement = () => uni.navigateTo({ url: '/pages/agreement/agreement' }) const goAgreement = () => uni.navigateTo({ url: '/pages/agreement/agreement' })
const goPrivacy = () => uni.navigateTo({ url: '/pages/privacy/privacy' }) const goPrivacy = () => uni.navigateTo({ url: '/pages/privacy/privacy' })
// #ifdef MP-WEIXIN
import { onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
onShareAppMessage(() => ({ title: '职引 - AI模拟面试平台 | 宇之然AI磁场', path: '/pages/about/about' }))
onShareTimeline(() => ({ title: '职引 - AI模拟面试平台 | 宇之然AI磁场' }))
// #endif
</script> </script>
<style scoped> <style scoped>
@@ -59,4 +65,6 @@ const goPrivacy = () => uni.navigateTo({ url: '/pages/privacy/privacy' })
.disclaimer { margin-top: 40rpx; background: #FFF8E1; border-radius: var(--radius-md); padding: 24rpx; } .disclaimer { margin-top: 40rpx; background: #FFF8E1; border-radius: var(--radius-md); padding: 24rpx; }
.disclaimer-title { font-size: 24rpx; font-weight: 700; color: #F59E0B; display: block; margin-bottom: 12rpx; } .disclaimer-title { font-size: 24rpx; font-weight: 700; color: #F59E0B; display: block; margin-bottom: 12rpx; }
.disclaimer-text { font-size: 22rpx; color: var(--color-text-secondary); line-height: 1.8; } .disclaimer-text { font-size: 22rpx; color: var(--color-text-secondary); line-height: 1.8; }
</style> </style>
+496 -66
View File
@@ -21,7 +21,9 @@
<text class="tab" :class="{ active: tab === 'orders' }" @click="switchTab('orders')">订单</text> <text class="tab" :class="{ active: tab === 'orders' }" @click="switchTab('orders')">订单</text>
<text class="tab" :class="{ active: tab === 'pricing' }" @click="switchTab('pricing')">定价</text> <text class="tab" :class="{ active: tab === 'pricing' }" @click="switchTab('pricing')">定价</text>
<text class="tab" :class="{ active: tab === 'share' }" @click="switchTab('share')">分享</text> <text class="tab" :class="{ active: tab === 'share' }" @click="switchTab('share')">分享</text>
<text class="tab" :class="{ active: tab === 'admins' }" @click="switchTab('admins')">管理</text> <text class="tab" :class="{ active: tab === 'positions' }" @click="switchTab('positions')">岗位</text>
<text class="tab" :class="{ active: tab === 'analysis' }" @click="switchTab('analysis')">诊断</text>
<text class="tab" :class="{ active: tab === 'admins' }" @click="switchTab('admins')">管理员</text>
</view> </view>
<!-- 概览 --> <!-- 概览 -->
@@ -43,6 +45,23 @@
<text class="stat-sub">付费下载 {{ overview.paidDownloadCount ?? 0 }}</text> <text class="stat-sub">付费下载 {{ overview.paidDownloadCount ?? 0 }}</text>
</view> </view>
</view> </view>
<view class="stat-cards" style="margin-top:12rpx">
<view class="stat-card">
<text class="stat-num">{{ overview.orderCount ?? 0 }}</text>
<text class="stat-label">总订单</text>
<text class="stat-sub">今日 +{{ overview.todayOrders ?? 0 }}</text>
</view>
<view class="stat-card">
<text class="stat-num">{{ overview.totalRevenue ? '¥' + (overview.totalRevenue / 100).toFixed(1) : '¥0' }}</text>
<text class="stat-label">总营收</text>
<text class="stat-sub">已支付订单合计</text>
</view>
<view class="stat-card" v-if="!(overview.orderCount)">
<text class="stat-num">--</text>
<text class="stat-label">--</text>
<text class="stat-sub" />
</view>
</view>
<view class="plan-cards"> <view class="plan-cards">
<view class="plan-card" v-for="(cnt, plan) in overview.planBreakdown" :key="plan"> <view class="plan-card" v-for="(cnt, plan) in overview.planBreakdown" :key="plan">
<text class="plan-num">{{ cnt }}</text> <text class="plan-num">{{ cnt }}</text>
@@ -62,16 +81,25 @@
<view class="user-main"> <view class="user-main">
<text class="user-phone">{{ u.phone || '--' }}</text> <text class="user-phone">{{ u.phone || '--' }}</text>
<text class="user-name">{{ u.nickname || '--' }}</text> <text class="user-name">{{ u.nickname || '--' }}</text>
<text class="user-badge-role" v-if="u.role === 'admin'">管理</text>
</view> </view>
<view class="user-badges"> <view class="user-meta-row">
<text class="user-plan" :class="{ vip: u.plan === 'growth' || u.plan === 'vip' }">{{ u.plan === 'growth' || u.plan === 'vip' ? '会员' : '免费' }}</text> <text class="meta-tag email" v-if="u.email">{{ u.email }}</text>
<text class="user-credit">面试:{{ u.interviewCredits ?? 0 }}</text> <text class="meta-tag" v-if="u.wxOpenid">openid:{{ u.wxOpenid.slice(0,12) }}..</text>
<text class="user-credit">优化:{{ u.resumeOptimizeCredits ?? 0 }}</text> </view>
<text class="user-credit">下载:{{ u.resumeDownloadCredits ?? 0 }}</text> <view class="user-meta-row">
<text class="user-credit share">分享:{{ u.shareCredits ?? 0 }}</text> <text class="meta-tag">引力:{{ u.gravity ?? 0 }}</text>
<text class="meta-tag">面试:{{ u.interviewCount ?? 0 }}</text>
<text class="user-plan" :class="{ vip: u.plan === 'growth' || u.plan === 'sprint' }">{{ u.plan === 'growth' || u.plan === 'sprint' ? u.plan==='sprint'?'冲刺':'会员' : '免费' }}</text>
<text class="meta-tag share" v-if="u.shareCredits > 0">分享:{{ u.shareCredits }}</text>
</view>
<view class="user-meta-row time-row">
<text class="time-label">注册:{{ u.createdAt?.slice(0,16).replace('T',' ') }}</text>
<text class="time-label" v-if="u.vipExpireAt">到期:{{ u.vipExpireAt?.slice(0,10) }}</text>
<text class="time-label" v-if="u.sprintExpireAt">冲刺到期:{{ u.sprintExpireAt?.slice(0,10) }}</text>
</view> </view>
<view class="user-actions"> <view class="user-actions">
<text class="user-action-btn" v-if="u.plan !== 'growth' && u.plan !== 'vip'" @click="setVip(u._id)">设为会员</text> <text class="user-action-btn" v-if="u.plan === 'free'" @click="setVip(u._id)">设为会员</text>
<text class="user-action-btn credit" @click="openCreditModal(u)">调整额度</text> <text class="user-action-btn credit" @click="openCreditModal(u)">调整额度</text>
</view> </view>
</view> </view>
@@ -82,37 +110,62 @@
<!-- 面试 --> <!-- 面试 -->
<view v-if="tab === 'interviews'" class="section"> <view v-if="tab === 'interviews'" class="section">
<view class="search-bar">
<input v-model="ivKeyword" placeholder="搜索岗位/用户名" class="search-input" @confirm="loadInterviews" />
<picker :range="['全部状态','进行中','已完成']" @change="e => { ivStatusFilter=e.detail.value; loadInterviews() }">
<text class="search-btn">{{ ['全部状态','进行中','已完成'][ivStatusFilter] }}</text>
</picker>
</view>
<view class="iv-list" v-if="!ivLoading"> <view class="iv-list" v-if="!ivLoading">
<view class="iv-row" v-for="iv in interviews" :key="iv._id"> <view class="iv-row" v-for="iv in interviews" :key="iv._id">
<view class="iv-main"> <view class="iv-main">
<text class="iv-pos">{{ iv.position }}</text> <text class="iv-pos">{{ iv.position }}</text>
<text class="iv-user">{{ iv.userId?.phone || iv.userId?.nickname || '--' }}</text> <text class="iv-user">{{ iv.userId?.phone || iv.userId?.nickname || '--' }}</text>
<text class="iv-user email" v-if="iv.userId?.email">{{ iv.userId.email }}</text>
</view> </view>
<view class="iv-meta"> <view class="iv-meta">
<text class="iv-status" :class="{ done: iv.status === 'completed' }">{{ iv.status === 'completed' ? '已完成' : '进行中' }}</text> <text class="iv-status" :class="{ done: iv.status === 'completed' }">{{ iv.status === 'completed' ? '已完成' : '进行中' }}</text>
<text class="iv-tag">{{ iv.questionCount || 0 }}</text> <text class="iv-tag">{{ iv.questionCount || 0 }}</text>
<text class="iv-tag score">得分 {{ iv.totalScore ?? '-' }}</text> <text class="iv-tag score">得分 {{ iv.totalScore ?? '-' }}</text>
<text class="iv-tag filler" v-if="iv.fillerScore != null && iv.fillerScore > 0">分析 {{ iv.fillerScore }}/{{ iv.fillerDensity ?? '-' }}</text> <text class="iv-tag filler" v-if="iv.fillerScore != null && iv.fillerScore > 0"> {{ iv.fillerScore }}/{{ iv.fillerDensity ?? '-' }}</text>
</view> </view>
<view class="iv-meta">
<text class="time-label">开始:{{ iv.createdAt?.slice(0,16).replace('T',' ') }}</text>
<text class="time-label" v-if="iv.updatedAt && iv.updatedAt !== iv.createdAt">更新:{{ iv.updatedAt?.slice(0,16).replace('T',' ') }}</text>
</view> </view>
<text class="iv-summary" v-if="iv.summary">{{ iv.summary.slice(0,60) }}{{ iv.summary.length > 60 ? '...' : '' }}</text>
</view>
<text class="load-more" v-if="ivTotal > interviews.length" @click="loadMoreInterviews">加载更多</text>
<text class="empty-text" v-if="interviews.length === 0 && !ivLoading">暂无面试记录</text>
</view> </view>
<text class="loading-text" v-if="ivLoading">加载中...</text> <text class="loading-text" v-if="ivLoading">加载中...</text>
</view> </view>
<!-- 简历 --> <!-- 简历 -->
<view v-if="tab === 'resumes'" class="section"> <view v-if="tab === 'resumes'" class="section">
<view class="search-bar">
<input v-model="resumeKeyword" placeholder="搜索简历标题" class="search-input" @confirm="loadResumes" />
<text class="search-btn" @click="loadResumes">搜索</text>
</view>
<view class="resume-list" v-if="!resumeLoading"> <view class="resume-list" v-if="!resumeLoading">
<view class="resume-row" v-for="r in resumes" :key="r._id"> <view class="resume-row" v-for="r in resumes" :key="r._id">
<view class="resume-main"> <view class="resume-main">
<text class="resume-title">{{ r.title }}</text> <text class="resume-title">{{ r.title }}</text>
<text class="resume-user">{{ r.userId?.phone || r.userId?.nickname || '--' }}</text> <text class="resume-user">{{ r.userId?.phone || r.userId?.nickname || '--' }}</text>
<text class="resume-user email" v-if="r.userId?.email">{{ r.userId.email }}</text>
</view> </view>
<view class="resume-meta"> <view class="resume-meta">
<text class="resume-tag">v{{ r.version }}</text> <text class="resume-tag">v{{ r.version }}</text>
<text class="resume-tag" v-if="r.targetPosition">{{ r.targetPosition }}</text> <text class="resume-tag" v-if="r.targetPosition">{{ r.targetPosition }}</text>
<text class="resume-tag paid" v-if="r.paidDownload">付费下载</text> <text class="resume-tag paid" v-if="r.paidDownload">付费下载</text>
</view> </view>
<text class="resume-time">{{ r.createdAt?.slice(0,10) }}</text> <view class="resume-meta time-row">
<text class="time-label">创建:{{ r.createdAt?.slice(0,16).replace('T',' ') }}</text>
<text class="time-label" v-if="r.updatedAt && r.updatedAt !== r.createdAt">更新:{{ r.updatedAt?.slice(0,16).replace('T',' ') }}</text>
</view>
<view class="resume-actions">
<text class="admin-action-btn del" @click="deleteResume(r._id, r.title)">删除</text>
</view>
</view> </view>
</view> </view>
<text class="loading-text" v-if="resumeLoading">加载中...</text> <text class="loading-text" v-if="resumeLoading">加载中...</text>
@@ -132,18 +185,35 @@
<view class="order-list" v-if="!orderLoading"> <view class="order-list" v-if="!orderLoading">
<view class="order-row" v-for="o in orders" :key="o._id"> <view class="order-row" v-for="o in orders" :key="o._id">
<view class="order-info"> <view class="order-info">
<text class="order-id">订单号: {{ o.outTradeNo }}</text> <text class="order-id">{{ o.outTradeNo }}</text>
<text class="order-user">用户: {{ o.userPhone || o.userId.slice(-6) }}</text> <view class="order-status rp" :class="o.status === 'success' ? 'paid' : o.status === 'refunded' ? 'refund' : 'pend'">
</view>
<view class="order-meta">
<text class="order-amount">¥{{ (o.amount / 100).toFixed(1) }}</text>
<view class="order-status" :class="o.status === 'success' ? 'paid' : o.status === 'refunded' ? 'refund' : 'pend'">
{{ o.status === 'success' ? '已支付' : o.status === 'refunded' ? '已退款' : '待支付' }} {{ o.status === 'success' ? '已支付' : o.status === 'refunded' ? '已退款' : '待支付' }}
</view> </view>
<text class="order-time">{{ o.createdAt ? o.createdAt.slice(0,16).replace('T',' ') : '--' }}</text>
<view class="order-actions" v-if="o.status === 'pending'">
<text class="sync-btn" @click="syncOrder(o.outTradeNo)">同步</text>
</view> </view>
<view class="order-meta-row">
<text class="order-title">{{ o.title || '--' }}</text>
<text class="order-user">用户: {{ o.userPhone || o.userId?.slice(-6) }}</text>
</view>
<view class="order-meta-row">
<text class="order-amount">¥{{ (o.amount / 100).toFixed(1) }}</text>
<text class="meta-tag">类型:{{ o.type || '--' }}</text>
<text class="meta-tag">渠道:{{ o.channel || '--' }}</text>
</view>
<view class="order-meta-row time-row">
<text class="time-label">创建:{{ o.createdAt?.slice(0,16).replace('T',' ') }}</text>
<text class="time-label" v-if="o.paidAt">支付:{{ o.paidAt?.slice(0,16).replace('T',' ') }}</text>
</view>
<view class="order-meta-row" v-if="o.wxTransactionId">
<text class="time-label">微信单号:{{ o.wxTransactionId }}</text>
</view>
<view class="order-meta-row" v-if="o.status === 'refunded'">
<text class="time-label refund-label">退款:¥{{ (o.refundAmount/100).toFixed(1) }} {{ o.refundedAt?.slice(0,16).replace('T',' ') }}</text>
<text class="time-label" v-if="o.refundReason">原因:{{ o.refundReason }}</text>
</view>
<view class="order-actions-bar">
<text class="sync-btn" v-if="o.status === 'pending'" @click="syncOrder(o.outTradeNo)">同步</text>
<text class="refund-btn" v-if="o.status === 'success'" @click="openRefundModal(o)">退款</text>
<text class="sync-btn" v-if="o.status === 'refunded'" @click="queryRefund(o.outTradeNo)">查询</text>
</view> </view>
</view> </view>
<text class="load-more" v-if="ordersTotal > orders.length" @click="loadMoreOrders">加载更多</text> <text class="load-more" v-if="ordersTotal > orders.length" @click="loadMoreOrders">加载更多</text>
@@ -173,12 +243,32 @@
</view> </view>
</view> </view>
<view class="config-card">
<view class="cfg-title">引力值消耗</view>
<view class="cfg-row">
<text>面试消耗引力值/</text>
<input class="cfg-input" type="digit" v-model.number="pricing.gravityRates.interviewPerUse" />
</view>
<view class="cfg-row">
<text>优化消耗引力值/</text>
<input class="cfg-input" type="digit" v-model.number="pricing.gravityRates.optimizePerUse" />
</view>
<view class="cfg-row">
<text>下载消耗引力值/</text>
<input class="cfg-input" type="digit" v-model.number="pricing.gravityRates.downloadPerUse" />
</view>
</view>
<view class="config-card"> <view class="config-card">
<view class="cfg-title">成长版 ¥{{ growthPriceDisplay }}</view> <view class="cfg-title">成长版 ¥{{ growthPriceDisplay }}</view>
<view class="cfg-row"> <view class="cfg-row">
<text>价格/</text> <text>价格/</text>
<input class="cfg-input" type="digit" v-model.number="growthPriceTemp" /> <input class="cfg-input" type="digit" v-model.number="growthPriceTemp" />
</view> </view>
<view class="cfg-row">
<text>每月引力值</text>
<input class="cfg-input" type="digit" v-model.number="pricing.plans.growth.gravityPerMonth" />
</view>
<view class="cfg-row"> <view class="cfg-row">
<text>面试额度/</text> <text>面试额度/</text>
<input class="cfg-input" type="digit" v-model.number="pricing.plans.growth.credits.interview" /> <input class="cfg-input" type="digit" v-model.number="pricing.plans.growth.credits.interview" />
@@ -204,7 +294,9 @@
<input class="cfg-input" type="digit" v-model.number="sprintPriceTemp" /> <input class="cfg-input" type="digit" v-model.number="sprintPriceTemp" />
</view> </view>
<view class="cfg-row"> <view class="cfg-row">
<text>面试额度/</text> <text>每月引力值</text>
<input class="cfg-input" type="digit" v-model.number="pricing.plans.sprint.gravityPerMonth" />
</view>
<input class="cfg-input" type="digit" v-model.number="pricing.plans.sprint.credits.interview" /> <input class="cfg-input" type="digit" v-model.number="pricing.plans.sprint.credits.interview" />
</view> </view>
<view class="cfg-row"> <view class="cfg-row">
@@ -243,13 +335,19 @@
<view class="share-row" v-for="r in shareRecords" :key="r.shareCode"> <view class="share-row" v-for="r in shareRecords" :key="r.shareCode">
<view class="share-main"> <view class="share-main">
<text class="share-title">{{ r.title }}</text> <text class="share-title">{{ r.title }}</text>
<text class="share-meta">{{ r.sharer?.nickname || '--' }} · {{ r.type }}</text> <text class="share-meta">{{ r.sharer?.phone || r.sharer?.nickname || '--' }} · {{ r.type }}</text>
</view>
<view class="share-meta-row">
<text class="meta-tag">code:{{ r.shareCode }}</text>
<text class="meta-tag" :class="r.isActive ? 'badge-done' : 'badge-pend'">{{ r.isActive ? '启用' : '停用' }}</text>
</view> </view>
<view class="share-stats"> <view class="share-stats">
<text>访问 {{ r.visitCount }}</text> <text>访问 {{ r.visitCount }}</text>
<text class="share-credited">有效 {{ r.creditedCount }}</text> <text class="share-credited">有效 {{ r.creditedCount }}</text>
</view> </view>
<text class="share-time">{{ r.createdAt?.slice(0,16).replace('T',' ') }}</text> <view class="share-meta-row time-row">
<text class="time-label">创建:{{ r.createdAt?.slice(0,16).replace('T',' ') }}</text>
</view>
</view> </view>
</view> </view>
<text class="loading-text" v-if="shareLoading">加载中...</text> <text class="loading-text" v-if="shareLoading">加载中...</text>
@@ -259,26 +357,69 @@
<view class="share-list" v-if="!shareLoading"> <view class="share-list" v-if="!shareLoading">
<view class="share-row" v-for="(v, i) in shareVisitors" :key="i"> <view class="share-row" v-for="(v, i) in shareVisitors" :key="i">
<view class="share-main"> <view class="share-main">
<text>分享者: {{ v.sharer?.nickname || '--' }}</text> <text>分享者:{{ v.sharer?.phone || v.sharer?.nickname || '--' }}</text>
<text class="share-meta">访客: {{ v.visitor?.nickname || '匿名' }}</text> <text class="share-meta">访客:{{ v.visitor?.phone || v.visitor?.nickname || '匿名' }}</text>
</view> </view>
<view class="share-stats"> <view class="share-meta-row">
<text class="badge" :class="v.credited ? 'badge-done' : 'badge-pend'">{{ v.credited ? '已积分' : '未积分' }}</text> <text class="meta-tag">IP:{{ v.visitorId || '--' }}</text>
<text class="meta-tag" :class="v.credited ? 'badge-done' : 'badge-pend'">{{ v.credited ? '已积分' : '未积分' }}</text>
</view>
<view class="share-meta-row time-row">
<text class="time-label">访问:{{ v.createdAt?.slice(0,16).replace('T',' ') }}</text>
<text class="time-label" v-if="v.creditedAt">积分:{{ v.creditedAt?.slice(0,16).replace('T',' ') }}</text>
</view> </view>
<text class="share-time">{{ v.createdAt?.slice(0,16).replace('T',' ') }}</text>
</view> </view>
</view> </view>
<text class="loading-text" v-if="shareLoading">加载中...</text> <text class="loading-text" v-if="shareLoading">加载中...</text>
<text class="empty-text" v-if="!shareLoading && shareVisitors.length === 0">暂无访问记录</text> <text class="empty-text" v-if="!shareLoading && shareVisitors.length === 0">暂无访问记录</text>
</view> </view>
</view> </view>
<!-- 岗位管理 -->
<view v-if="tab === 'positions'" class="section">
<view class="search-bar">
<text class="section-label" style="flex:1;margin:0">岗位列表{{ positions.length }}</text>
<button class="search-btn" @click="openPositionModal(null)">新增岗位</button>
</view>
<view class="position-mgr-list" v-if="!posLoading">
<view class="pos-mgr-row" v-for="p in positions" :key="p._id">
<view class="pos-mgr-main">
<text class="pos-mgr-cat" :class="p.category === 'ai' ? 'cat-ai' : 'cat-tr'">{{ p.category === 'ai' ? 'AI' : '传统' }}</text>
<view class="pos-mgr-body">
<text class="pos-mgr-name">{{ p.name }}</text>
<text class="pos-mgr-meta">{{ p.company || '-' }} · {{ p.salary || '-' }} · sort:{{ p.sort }}</text>
</view>
</view>
<view class="pos-mgr-actions">
<text class="pos-mgr-btn edit" @click="openPositionModal(p)">编辑</text>
<text class="pos-mgr-btn del" @click="deletePosition(p._id, p.name)">删除</text>
</view>
</view>
<text class="empty-text" v-if="positions.length === 0 && !posLoading">暂无岗位</text>
</view>
<text class="loading-text" v-if="posLoading">加载中...</text>
</view>
<!-- 诊断分析 -->
<view v-if="tab === 'analysis'" class="section">
<view class="config-card">
<view class="cfg-title">简历诊断</view>
<view class="cfg-row"><text>总诊断次数</text><text class="cfg-val">{{ analysisStats.totalDiagnoses ?? 0 }}</text></view>
<view class="cfg-row"><text>今日诊断</text><text class="cfg-val">{{ analysisStats.todayDiagnoses ?? 0 }}</text></view>
</view>
<view class="config-card">
<view class="cfg-title">技能缺口分析</view>
<view class="cfg-row"><text>总分析次数</text><text class="cfg-val">{{ analysisStats.totalGapAnalysis ?? 0 }}</text></view>
</view>
<text class="loading-text" v-if="analysisLoading">加载中...</text>
</view>
<!-- 额度调整弹窗 --> <!-- 额度调整弹窗 -->
<view class="modal-mask" v-if="creditModal.show" @click="closeCreditModal"> <view class="modal-mask" v-if="creditModal.show" @click="closeCreditModal">
<view class="modal-content" @click.stop> <view class="modal-content" @click.stop>
<text class="modal-title">调整 {{ creditModal.user?.nickname || '用户' }} 的额度</text> <text class="modal-title">调整 {{ creditModal.user?.nickname || '用户' }} 的额度</text>
<view class="cfg-row" v-for="t in creditTypes" :key="t.key"> <view class="cfg-row">
<text>{{ t.label }}</text> <text>引力值</text>
<input class="cfg-input" type="digit" v-model.number="t.value" :placeholder="t.key" /> <input class="cfg-input" type="digit" v-model.number="creditGravity" />
</view> </view>
<view class="modal-actions"> <view class="modal-actions">
<button class="modal-btn cancel" @click="closeCreditModal">取消</button> <button class="modal-btn cancel" @click="closeCreditModal">取消</button>
@@ -286,6 +427,59 @@
</view> </view>
</view> </view>
</view> </view>
<!-- 岗位编辑弹窗 -->
<view class="modal-mask" v-if="posModal.show" @click="closePositionModal">
<view class="modal-content" @click.stop>
<text class="modal-title">{{ posModal.isNew ? '新增岗位' : '编辑岗位' }}</text>
<view class="cfg-row"><text>岗位名称</text><input class="cfg-input" v-model="posForm.name" placeholder="必填" /></view>
<view class="cfg-row"><text>薪资</text><input class="cfg-input" v-model="posForm.salary" placeholder="如 15-25K" /></view>
<view class="cfg-row"><text>公司</text><input class="cfg-input" v-model="posForm.company" placeholder="如 字节跳动" /></view>
<view class="cfg-row"><text>排序</text><input class="cfg-input" type="digit" v-model.number="posForm.sort" /></view>
<view class="cfg-row"><text>分类</text>
<picker :range="['AI岗位','传统岗位']" @change="e => posForm.category = e.detail.value === 0 ? 'ai' : 'traditional'">
<text class="cfg-val">{{ posForm.category === 'ai' ? 'AI岗位' : '传统岗位' }}</text>
</picker>
</view>
<view class="cfg-row"><text>启用</text>
<picker :range="['启用','停用']" @change="e => posForm.active = e.detail.value === 0" :value="posForm.active ? 0 : 1">
<text class="cfg-val">{{ posForm.active ? '启用' : '停用' }}</text>
</picker>
</view>
<view class="cfg-row"><text>岗位描述</text></view>
<textarea class="cfg-textarea" v-model="posForm.description" placeholder="岗位职责描述,每行一个要点" />
<view class="cfg-row"><text>任职要求</text></view>
<textarea class="cfg-textarea" v-model="posForm.requirements" placeholder="任职资格要求,每行一个要点" />
<view class="modal-actions">
<button class="modal-btn cancel" @click="closePositionModal">取消</button>
<button class="modal-btn confirm" @click="savePosition">保存</button>
</view>
</view>
</view>
<!-- 退款弹窗 -->
<view class="modal-mask" v-if="refundModal.show" @click="closeRefundModal">
<view class="modal-content" @click.stop>
<text class="modal-title">退款 - {{ refundModal.order?.outTradeNo }}</text>
<view class="cfg-row">
<text>订单金额</text>
<text class="cfg-val">¥{{ ((refundModal.order?.amount || 0) / 100).toFixed(1) }}</text>
</view>
<view class="cfg-row">
<text>退款金额</text>
<input class="cfg-input" type="digit" v-model.number="refundAmount" :placeholder="((refundModal.order?.amount || 0) / 100).toFixed(1)" />
</view>
<view class="cfg-row">
<text>退款原因</text>
<input class="cfg-input" style="width:300rpx" v-model="refundReason" placeholder="选填" />
</view>
<view class="modal-actions">
<button class="modal-btn cancel" @click="closeRefundModal">取消</button>
<button class="modal-btn confirm" style="background:#EF4444" @click="doRefund">确认退款</button>
</view>
</view>
</view>
<!-- 管理员 --> <!-- 管理员 -->
<view v-if="tab === 'admins'" class="section"> <view v-if="tab === 'admins'" class="section">
<view class="search-bar"> <view class="search-bar">
@@ -297,7 +491,9 @@
<view class="admin-row" v-for="a in adminList" :key="a._id"> <view class="admin-row" v-for="a in adminList" :key="a._id">
<text class="admin-phone">{{ a.phone || '--' }}</text> <text class="admin-phone">{{ a.phone || '--' }}</text>
<text class="admin-name">{{ a.nickname || '--' }}</text> <text class="admin-name">{{ a.nickname || '--' }}</text>
<text class="admin-email" v-if="a.email">{{ a.email }}</text>
<text class="admin-badge" v-if="a.isSystemAdmin">系统</text> <text class="admin-badge" v-if="a.isSystemAdmin">系统</text>
<text class="time-label" style="margin-left:auto">设置:{{ a.createdAt?.slice(0,10) }}</text>
</view> </view>
<text class="empty-text" v-if="adminList.length === 0">暂无管理员</text> <text class="empty-text" v-if="adminList.length === 0">暂无管理员</text>
</view> </view>
@@ -306,17 +502,17 @@
<view class="admin-row"> <view class="admin-row">
<text class="admin-phone">{{ searchResult.phone || '--' }}</text> <text class="admin-phone">{{ searchResult.phone || '--' }}</text>
<text class="admin-name">{{ searchResult.nickname || '--' }}</text> <text class="admin-name">{{ searchResult.nickname || '--' }}</text>
<text class="admin-email" v-if="searchResult.email">{{ searchResult.email }}</text>
<text class="admin-set-btn" v-if="searchResult.role !== 'admin'" @click="setAdmin(searchResult._id)">设为管理员</text> <text class="admin-set-btn" v-if="searchResult.role !== 'admin'" @click="setAdmin(searchResult._id)">设为管理员</text>
<text class="admin-set-btn done" v-else>已是管理员</text> <text class="admin-set-btn done" v-else>已是管理员</text>
</view> </view>
</view> </view>
</view> </view>
</view> </view>
</view>
</template> </template>
<script setup> <script setup>
import { ref, computed } from 'vue' import { ref, computed, onMounted, reactive } from 'vue'
import { api, API_ENDPOINTS } from '../../config' import { api, API_ENDPOINTS } from '../../config'
const verified = ref(false) const verified = ref(false)
@@ -339,15 +535,16 @@ const resumeLoading = ref(false)
const adminKeyword = ref('') const adminKeyword = ref('')
const adminList = ref([]) const adminList = ref([])
const searchResult = ref(null) const searchResult = ref(null)
const memberConfig = ref({ interview: { maxRoundsFree: 5, maxRoundsVip: 10, dailyFreeLimit: 3 }, diagnosis: { dailyFreeLimit: 2 }, optimize: { dailyFreeLimit: 2 }, price: { monthly: 2900 } }) const memberConfig = ref({ interview: { maxRoundsFree: 5, maxRoundsVip: 10, dailyFreeLimit: 3 }, diagnosis: { dailyFreeLimit: 2 }, optimize: { dailyFreeLimit: 2 }, price: { monthly: 1990 } })
const cfgLoading = ref(false) const cfgLoading = ref(false)
const pricing = ref({ const pricing = ref({
interview: { pricePerSession: 500 }, interview: { pricePerSession: 500 },
resumeOptimize: { freeLimit: 3, pricePerOptimize: 300 }, resumeOptimize: { freeLimit: 3, pricePerOptimize: 300 },
resumeDownload: { pricePerDownload: 200 }, resumeDownload: { pricePerDownload: 200 },
gravityRates: { interviewPerUse: 5, optimizePerUse: 3, downloadPerUse: 2 },
plans: { plans: {
growth: { price: 1990, durationDays: 30, credits: { interview: 999, resumeOptimize: 20, resumeDownload: 10 }, features: ['免费版全部权益', 'AI 数字人面试无限次'] }, growth: { price: 1990, durationDays: 30, gravityPerMonth: 250, credits: { interview: 999, resumeOptimize: 20, resumeDownload: 10 }, features: ['免费版全部权益', '每月 250 引力值'] },
sprint: { price: 4990, durationDays: 30, credits: { interview: 999, resumeOptimize: 50, resumeDownload: 30 }, features: ['成长版全部权益'] }, sprint: { price: 4990, durationDays: 30, gravityPerMonth: 600, credits: { interview: 999, resumeOptimize: 50, resumeDownload: 30 }, features: ['成长版全部权益', '每月 600 引力值'] },
}, },
}) })
const pricingLoading = ref(false) const pricingLoading = ref(false)
@@ -355,12 +552,36 @@ const growthPriceTemp = ref(19.9)
const sprintPriceTemp = ref(49.9) const sprintPriceTemp = ref(49.9)
const growthFeaturesText = ref('') const growthFeaturesText = ref('')
const sprintFeaturesText = ref('') const sprintFeaturesText = ref('')
const ivKeyword = ref('')
const ivStatusFilter = ref(0)
const ivTotal = ref(0)
const ivPage = ref(1)
const resumeKeyword = ref('')
const creditGravity = ref(0)
const analysisStats = ref({ totalDiagnoses: 0, todayDiagnoses: 0, totalGapAnalysis: 0 })
const analysisLoading = ref(false)
// Position management
const positions = ref([])
const posLoading = ref(false)
const posModal = ref({ show: false, isNew: false })
const posForm = reactive({
name: '',
salary: '',
company: '',
icon: '',
sort: 0,
active: true,
category: 'ai',
description: '',
requirements: '',
})
const growthPriceDisplay = computed(() => growthPriceTemp.value.toFixed(1)) const growthPriceDisplay = computed(() => growthPriceTemp.value.toFixed(1))
const sprintPriceDisplay = computed(() => sprintPriceTemp.value.toFixed(1)) const sprintPriceDisplay = computed(() => sprintPriceTemp.value.toFixed(1))
const calcInterviewPrice = () => { const calcInterviewPrice = () => {
// Convert to 分 on save // Handled in savePricing via growthPriceTemp / sprintPriceTemp
} }
const orders = ref([]) const orders = ref([])
const ordersTotal = ref(0) const ordersTotal = ref(0)
@@ -375,12 +596,67 @@ const shareLoading = ref(false)
// Credit modal // Credit modal
const creditModal = ref({ show: false, user: null }) const creditModal = ref({ show: false, user: null })
const creditTypes = ref([
{ key: 'interviewCredits', label: '面试次数', value: 0 }, // Refund modal
{ key: 'resumeOptimizeCredits', label: '优化次数', value: 0 }, const refundModal = ref({ show: false, order: null })
{ key: 'resumeDownloadCredits', label: '下载次数', value: 0 }, const refundAmount = ref(0)
{ key: 'shareCredits', label: '分享积分', value: 0 }, const refundReason = ref('')
])
const openRefundModal = (order) => {
refundModal.value = { show: true, order }
refundAmount.value = order.amount / 100
refundReason.value = ''
}
const closeRefundModal = () => {
refundModal.value = { show: false, order: null }
}
const doRefund = async () => {
const order = refundModal.value.order
if (!order) return
uni.showModal({
title: '确认退款', content: `确定对订单 ${order.outTradeNo} 退款 ¥${refundAmount.value.toFixed(1)}`,
success: async (r) => {
if (!r.confirm) return
try {
const res = await apiAdmin('/order/refund', {
method: 'POST',
body: { outTradeNo: order.outTradeNo, amount: Math.round(refundAmount.value * 100), reason: refundReason.value },
})
if (res.statusCode === 200) {
uni.showToast({ title: '退款成功', icon: 'success' })
closeRefundModal()
loadOrders()
} else {
uni.showToast({ title: res.data?.message || '退款失败', icon: 'none' })
}
} catch (e) {
uni.showToast({ title: '退款失败', icon: 'none' })
}
},
})
}
const queryRefund = async (outTradeNo) => {
uni.showToast({ title: '查询中...', icon: 'none' })
try {
const res = await apiAdmin('/order/refund/' + outTradeNo)
if (res.statusCode === 200) {
const wx = res.data.wxRefund
if (wx) {
uni.showModal({
title: '退款状态',
content: `微信状态: ${wx.status || '--'}\n退款金额: ¥${(wx.amount?.refund || 0) / 100}\n建议以微信侧为准`,
})
} else {
uni.showToast({ title: '本地状态: ' + (res.data.localStatus || '--'), icon: 'none' })
}
} else {
uni.showToast({ title: '查询失败', icon: 'none' })
}
} catch { uni.showToast({ title: '查询失败', icon: 'none' }) }
}
const token = () => uni.getStorageSync('token') || '' const token = () => uni.getStorageSync('token') || ''
@@ -399,7 +675,7 @@ const doVerify = async () => {
try { try {
const res = await apiAdmin('/check') const res = await apiAdmin('/check')
if (res.statusCode === 200 && res.data?.isAdmin) { if (res.statusCode === 200 && res.data?.isAdmin) {
adminName.value = '管理员' adminName.value = res.data.nickname || res.data.username || '管理员'
verified.value = true verified.value = true
loadOverview() loadOverview()
} else throw new Error('无管理员权限') } else throw new Error('无管理员权限')
@@ -421,10 +697,12 @@ const switchTab = (t) => {
tab.value = t tab.value = t
if (t === 'users' && users.value.length === 0) loadUsers() if (t === 'users' && users.value.length === 0) loadUsers()
if (t === 'interviews' && interviews.value.length === 0) loadInterviews() if (t === 'interviews' && interviews.value.length === 0) loadInterviews()
if (t === 'positions') loadPositions()
if (t === 'resumes' && resumes.value.length === 0) loadResumes() if (t === 'resumes' && resumes.value.length === 0) loadResumes()
if (t === 'admins' && adminList.value.length === 0) loadAdmins() if (t === 'admins' && adminList.value.length === 0) loadAdmins()
if (t === 'pricing') loadPricing() if (t === 'pricing') loadPricing()
if (t === 'orders') loadOrders() if (t === 'orders') loadOrders()
if (t === 'analysis') loadAnalysis()
} }
const loadUsers = async () => { const loadUsers = async () => {
@@ -447,22 +725,55 @@ const loadMoreUsers = async () => {
const loadInterviews = async () => { const loadInterviews = async () => {
ivLoading.value = true ivLoading.value = true
ivPage.value = 1
try { try {
const res = await apiAdmin('/interviews?page=1&limit=20') let url = '/interviews?page=1&limit=20&keyword=' + encodeURIComponent(ivKeyword.value)
if (res.statusCode === 200) interviews.value = res.data.interviews || [] const statusMap = ['', 'in_progress', 'completed']
if (ivStatusFilter.value > 0) url += '&status=' + statusMap[ivStatusFilter.value]
const res = await apiAdmin(url)
if (res.statusCode === 200) { interviews.value = res.data.interviews || []; ivTotal.value = res.data.total || 0 }
} catch (e) { console.error(e) } } catch (e) { console.error(e) }
finally { ivLoading.value = false } finally { ivLoading.value = false }
} }
const loadMoreInterviews = async () => {
ivPage.value++
try {
let url = '/interviews?page=' + ivPage.value + '&limit=20&keyword=' + encodeURIComponent(ivKeyword.value)
const statusMap = ['', 'in_progress', 'completed']
if (ivStatusFilter.value > 0) url += '&status=' + statusMap[ivStatusFilter.value]
const res = await apiAdmin(url)
if (res.statusCode === 200) interviews.value = [...interviews.value, ...(res.data.interviews || [])]
} catch (e) { console.error(e) }
}
const loadResumes = async () => { const loadResumes = async () => {
resumeLoading.value = true resumeLoading.value = true
try { try {
const res = await apiAdmin('/resumes?page=1&limit=20') let url = '/resumes?page=1&limit=20'
if (resumeKeyword.value) url += '&keyword=' + encodeURIComponent(resumeKeyword.value)
const res = await apiAdmin(url)
if (res.statusCode === 200) resumes.value = res.data.list || [] if (res.statusCode === 200) resumes.value = res.data.list || []
} catch (e) { console.error(e) } } catch (e) { console.error(e) }
finally { resumeLoading.value = false } finally { resumeLoading.value = false }
} }
const deleteResume = (id, title) => {
uni.showModal({
title: '删除简历', content: `确定删除"${title}"`,
success: async (r) => {
if (!r.confirm) return
try {
const res = await apiAdmin('/resume/' + id, { method: 'DELETE' })
if (res.statusCode === 200) {
uni.showToast({ title: '已删除', icon: 'success' })
loadResumes()
} else throw new Error()
} catch { uni.showToast({ title: '删除失败', icon: 'none' }) }
},
})
}
const loadPricing = async () => { const loadPricing = async () => {
pricingLoading.value = true pricingLoading.value = true
try { try {
@@ -478,6 +789,15 @@ const loadPricing = async () => {
finally { pricingLoading.value = false } finally { pricingLoading.value = false }
} }
const loadAnalysis = async () => {
analysisLoading.value = true
try {
const res = await apiAdmin('/analysis-stats')
if (res.statusCode === 200) analysisStats.value = res.data
} catch(e) { console.error(e) }
finally { analysisLoading.value = false }
}
const savePricing = async () => { const savePricing = async () => {
pricingLoading.value = true pricingLoading.value = true
try { try {
@@ -500,15 +820,6 @@ const savePricing = async () => {
finally { pricingLoading.value = false } finally { pricingLoading.value = false }
} }
const loadConfig = async () => {
cfgLoading.value = true
try {
const res = await apiAdmin('/config')
if (res.statusCode === 200) memberConfig.value = res.data
} catch(e) { console.error(e) }
finally { cfgLoading.value = false }
}
const loadOrders = async () => { const loadOrders = async () => {
orderLoading.value = true orderLoading.value = true
ordersPage.value = 1 ordersPage.value = 1
@@ -549,6 +860,88 @@ const loadAdmins = async () => {
} catch(e) { console.error(e) } } catch(e) { console.error(e) }
} }
// ─── 岗位管理 ──────────────────────
const apiPositions = (path, opts = {}) => {
return uni.request({
url: api('/positions' + path),
method: opts.method || 'POST',
header: { 'Authorization': `Bearer ${token()}`, 'Content-Type': 'application/json', ...opts.headers },
data: opts.body || opts.data,
})
}
const loadPositions = async () => {
posLoading.value = true
try {
const res = await apiPositions('/admin/list')
if (res.statusCode === 200) positions.value = res.data || []
} catch (e) { console.error(e) }
finally { posLoading.value = false }
}
const openPositionModal = (position) => {
if (position) {
posForm.name = position.name || ''
posForm.salary = position.salary || ''
posForm.company = position.company || ''
posForm.icon = position.icon || ''
posForm.sort = position.sort ?? 0
posForm.active = position.active ?? true
posForm.category = position.category || 'traditional'
posForm.description = position.description || ''
posForm.requirements = position.requirements || ''
posModal.value = { show: true, isNew: false }
} else {
posForm.name = ''
posForm.salary = ''
posForm.company = ''
posForm.icon = ''
posForm.sort = positions.value.length + 1
posForm.active = true
posForm.category = 'ai'
posForm.description = ''
posForm.requirements = ''
posModal.value = { show: true, isNew: true }
}
}
const closePositionModal = () => {
posModal.value = { show: false, isNew: false }
}
const savePosition = async () => {
if (!posForm.name.trim()) {
uni.showToast({ title: '岗位名称不能为空', icon: 'none' })
return
}
try {
const res = await apiPositions('/admin/save', {
method: 'POST', body: { ...posForm },
})
if (res.statusCode === 200) {
uni.showToast({ title: '保存成功', icon: 'success' })
closePositionModal()
loadPositions()
} else throw new Error()
} catch { uni.showToast({ title: '保存失败', icon: 'none' }) }
}
const deletePosition = (id, name) => {
uni.showModal({
title: '删除岗位', content: `确定删除"${name}"`,
success: async (r) => {
if (!r.confirm) return
try {
const res = await apiPositions('/admin/' + id, { method: 'DELETE' })
if (res.statusCode === 200) {
uni.showToast({ title: '已删除', icon: 'success' })
loadPositions()
} else throw new Error()
} catch { uni.showToast({ title: '删除失败', icon: 'none' }) }
},
})
}
const searchAdmin = async () => { const searchAdmin = async () => {
if (!adminKeyword.value.trim()) return if (!adminKeyword.value.trim()) return
try { try {
@@ -613,12 +1006,7 @@ const loadShareVisitors = async () => {
} }
const openCreditModal = (user) => { const openCreditModal = (user) => {
creditTypes.value = [ creditGravity.value = user.gravity ?? 0
{ key: 'interviewCredits', label: '面试次数', value: user.interviewCredits ?? 0 },
{ key: 'resumeOptimizeCredits', label: '优化次数', value: user.resumeOptimizeCredits ?? 0 },
{ key: 'resumeDownloadCredits', label: '下载次数', value: user.resumeDownloadCredits ?? 0 },
{ key: 'shareCredits', label: '分享积分', value: user.shareCredits ?? 0 },
]
creditModal.value = { show: true, user } creditModal.value = { show: true, user }
} }
@@ -630,17 +1018,17 @@ const doAdjustCredits = async () => {
const userId = creditModal.value.user?._id const userId = creditModal.value.user?._id
if (!userId) return if (!userId) return
try { try {
for (const t of creditTypes.value) {
await apiAdmin('/user/credits', { await apiAdmin('/user/credits', {
method: 'POST', method: 'POST',
data: { userId, type: t.key, amount: t.value }, data: { userId, type: 'gravity', amount: creditGravity.value },
}) })
}
uni.showToast({ title: '调整成功', icon: 'success' }) uni.showToast({ title: '调整成功', icon: 'success' })
closeCreditModal() closeCreditModal()
loadUsers() loadUsers()
} catch { uni.showToast({ title: '调整失败', icon: 'none' }) } } catch { uni.showToast({ title: '调整失败', icon: 'none' }) }
} }
onMounted(() => { doVerify() })
</script> </script>
<style scoped> <style scoped>
@@ -652,8 +1040,8 @@ const doAdjustCredits = async () => {
.admin-input { height: 72rpx; background: #FFF; border: 2rpx solid var(--color-border); border-radius: var(--radius-sm); padding: 0 20rpx; font-size: 24rpx; } .admin-input { height: 72rpx; background: #FFF; border: 2rpx solid var(--color-border); border-radius: var(--radius-sm); padding: 0 20rpx; font-size: 24rpx; }
.btn-verify { height: 72rpx; background: linear-gradient(135deg, var(--color-gradient-start), var(--color-gradient-mid)); color: #FFF; border-radius: var(--radius-sm); font-size: 26rpx; border: none; } .btn-verify { height: 72rpx; background: linear-gradient(135deg, var(--color-gradient-start), var(--color-gradient-mid)); color: #FFF; border-radius: var(--radius-sm); font-size: 26rpx; border: none; }
.body { padding: 20rpx 32rpx 48rpx; margin-top: -40rpx; } .body { padding: 20rpx 32rpx 48rpx; margin-top: -40rpx; }
.tabs { display: flex; gap: 8rpx; background: #FFF; border-radius: var(--radius-md); padding: 6rpx; margin-bottom: 20rpx; } .tabs { display: flex; flex-wrap: wrap; gap: 8rpx; background: #FFF; border-radius: var(--radius-md); padding: 6rpx; margin-bottom: 20rpx; }
.tab { flex: 1; text-align: center; padding: 14rpx; font-size: 24rpx; color: var(--color-text-secondary); border-radius: var(--radius-sm); } .tab { padding: 14rpx 20rpx; font-size: 24rpx; color: var(--color-text-secondary); border-radius: var(--radius-sm); white-space: nowrap; }
.tab.active { background: var(--color-primary); color: #FFF; font-weight: 600; } .tab.active { background: var(--color-primary); color: #FFF; font-weight: 600; }
.stat-cards { display: flex; gap: 16rpx; } .stat-cards { display: flex; gap: 16rpx; }
.stat-card { flex: 1; background: #FFF; border-radius: var(--radius-lg); padding: 28rpx; text-align: center; box-shadow: var(--shadow-sm); } .stat-card { flex: 1; background: #FFF; border-radius: var(--radius-lg); padding: 28rpx; text-align: center; box-shadow: var(--shadow-sm); }
@@ -722,6 +1110,7 @@ const doAdjustCredits = async () => {
.order-time { font-size: 20rpx; color: var(--color-text-tertiary); flex: 1; text-align: right; } .order-time { font-size: 20rpx; color: var(--color-text-tertiary); flex: 1; text-align: right; }
.order-actions { } .order-actions { }
.sync-btn { font-size: 20rpx; color: var(--color-primary); padding: 4rpx 12rpx; border: 2rpx solid var(--color-primary); border-radius: var(--radius-round); } .sync-btn { font-size: 20rpx; color: var(--color-primary); padding: 4rpx 12rpx; border: 2rpx solid var(--color-primary); border-radius: var(--radius-round); }
.refund-btn { font-size: 20rpx; color: #EF4444; padding: 4rpx 12rpx; border: 2rpx solid #EF4444; border-radius: var(--radius-round); }
.config-card { background: #FFF; border-radius: var(--radius-sm); padding: 20rpx; margin-bottom: 12rpx; } .config-card { background: #FFF; border-radius: var(--radius-sm); padding: 20rpx; margin-bottom: 12rpx; }
.cfg-title { font-size: 24rpx; font-weight: 600; color: var(--color-text); margin-bottom: 12rpx; } .cfg-title { font-size: 24rpx; font-weight: 600; color: var(--color-text); margin-bottom: 12rpx; }
.cfg-row { display: flex; justify-content: space-between; padding: 8rpx 0; font-size: 22rpx; color: var(--color-text-secondary); align-items: center; } .cfg-row { display: flex; justify-content: space-between; padding: 8rpx 0; font-size: 22rpx; color: var(--color-text-secondary); align-items: center; }
@@ -749,4 +1138,45 @@ const doAdjustCredits = async () => {
.modal-btn { flex: 1; height: 72rpx; border-radius: var(--radius-sm); font-size: 26rpx; border: none; } .modal-btn { flex: 1; height: 72rpx; border-radius: var(--radius-sm); font-size: 26rpx; border: none; }
.modal-btn.cancel { background: #F3F4F6; color: var(--color-text-secondary); } .modal-btn.cancel { background: #F3F4F6; color: var(--color-text-secondary); }
.modal-btn.confirm { background: var(--color-primary); color: #FFF; } .modal-btn.confirm { background: var(--color-primary); color: #FFF; }
/* ─── 岗位管理 ───── */
.position-mgr-list { display: flex; flex-direction: column; gap: 8rpx; }
.pos-mgr-row { background: #FFF; padding: 16rpx; border-radius: var(--radius-sm); display: flex; align-items: center; gap: 12rpx; }
.pos-mgr-main { flex: 1; display: flex; align-items: center; gap: 12rpx; }
.pos-mgr-cat { font-size: 18rpx; padding: 2rpx 12rpx; border-radius: var(--radius-round); font-weight: 600; }
.pos-mgr-cat.cat-ai { background: #EEF2FF; color: var(--color-primary); }
.pos-mgr-cat.cat-tr { background: #F3F4F6; color: var(--color-text-tertiary); }
.pos-mgr-body { display: flex; flex-direction: column; }
.pos-mgr-name { font-size: 24rpx; font-weight: 600; color: var(--color-text); }
.pos-mgr-meta { font-size: 20rpx; color: var(--color-text-tertiary); margin-top: 2rpx; }
.pos-mgr-actions { display: flex; gap: 8rpx; }
.pos-mgr-btn { font-size: 20rpx; padding: 4rpx 16rpx; border-radius: var(--radius-round); }
.pos-mgr-btn.edit { color: var(--color-primary); border: 2rpx solid var(--color-primary); }
.pos-mgr-btn.del { color: #EF4444; border: 2rpx solid #EF4444; }
.iv-time { font-size: 18rpx; color: #D1D5DB; white-space: nowrap; margin-left: auto; }
.admin-action-btn { font-size: 20rpx; padding: 4rpx 16rpx; border-radius: var(--radius-round); cursor: pointer; }
.admin-action-btn.del { color: #EF4444; border: 2rpx solid #EF4444; }
.resume-actions { display: flex; gap: 8rpx; align-items: center; }
/* ─── 新增字段样式 ───── */
.user-meta-row { display: flex; flex-wrap: wrap; gap: 6rpx; margin-bottom: 6rpx; }
.meta-tag { font-size: 18rpx; background: #F3F4F6; color: var(--color-text-tertiary); padding: 2rpx 10rpx; border-radius: var(--radius-round); }
.meta-tag.email { background: #EEF2FF; color: var(--color-primary); }
.meta-tag.share { background: #FFF7ED; color: #D97706; }
.meta-tag.badge-done { background: #ECFDF5; color: #059669; }
.meta-tag.badge-pend { background: #FEF3C7; color: #D97706; }
.time-row { display: flex; flex-wrap: wrap; gap: 12rpx; }
.time-label { font-size: 18rpx; color: #9CA3AF; }
.iv-summary { font-size: 18rpx; color: #6B7280; margin-top: 4rpx; line-height: 1.4; display: block; }
.iv-user.email { font-size: 18rpx; color: #6B7280; }
.user-badge-role { font-size: 18rpx; background: #FEF3C7; color: #D97706; padding: 0 10rpx; border-radius: var(--radius-round); font-weight: 500; }
.share-meta-row { display: flex; gap: 6rpx; margin-top: 4rpx; }
.share-meta-row.time-row { gap: 12rpx; }
.order-meta-row { display: flex; flex-wrap: wrap; gap: 8rpx; margin-bottom: 4rpx; }
.order-meta-row.time-row { gap: 12rpx; }
.order-title { font-size: 22rpx; font-weight: 500; color: var(--color-text); }
.order-status.rp { font-size: 18rpx; display: inline-block; }
.refund-label { color: #EF4444 !important; }
.order-actions-bar { display: flex; gap: 8rpx; margin-top: 6rpx; }
.admin-email { font-size: 20rpx; color: #6B7280; }
.resume-user.email { font-size: 18rpx; color: #6B7280; }
</style> </style>
+284
View File
@@ -0,0 +1,284 @@
<template>
<view class="page">
<!-- 输入表单 -->
<view v-if="step === 'input'" class="input-wrap">
<view class="hero">
<text class="hero-icon">🧭</text>
<text class="hero-title">择业顾问</text>
<text class="hero-desc">AI 帮你分析专业前景规划职业方向</text>
</view>
<view class="form-card">
<view class="form-group">
<text class="form-label">你的专业 <text class="required">*</text></text>
<input class="form-input" v-model="profile.major" placeholder="例如:计算机科学与技术" />
</view>
<view class="form-group">
<text class="form-label">年级</text>
<picker :range="grades" @change="e => profile.grade = grades[e.detail.value]">
<view class="form-input select-trigger">{{ profile.grade || '请选择年级' }}</view>
</picker>
</view>
<view class="form-group">
<text class="form-label">兴趣/擅长方向</text>
<input class="form-input" v-model="profile.interests" placeholder="例如:后端开发、数据分析" />
</view>
<view class="form-group">
<text class="form-label">成绩/GPA</text>
<input class="form-input" v-model="profile.gpa" placeholder="例如:3.5/4.0 或专业前 20%" />
</view>
<view class="form-group">
<text class="form-label">你的困惑或目标</text>
<textarea class="form-textarea" v-model="profile.goal" placeholder="例如:不知道该考研还是直接就业,对AI行业很感兴趣但不知道从何入手..." />
</view>
</view>
<button class="submit-btn" :disabled="!profile.major.trim() || loading" @click="doAnalyze">
<text v-if="!loading">开始分析</text>
<text v-else>AI 分析中...</text>
</button>
<text v-if="error" class="error-text">{{ error }}</text>
</view>
<!-- 分析结果 -->
<view v-if="step === 'result'" class="result-wrap">
<view class="result-header">
<text class="result-icon">📊</text>
<text class="result-title">择业分析报告</text>
</view>
<view class="advice-card">
<view class="advice-content">{{ result.reply }}</view>
</view>
<view v-if="result.careerPaths && result.careerPaths.length > 0" class="section">
<text class="section-title">推荐方向</text>
<view class="path-card" v-for="(path, i) in result.careerPaths" :key="i"
@click="goInterview(path.name)">
<view class="path-rank">{{ i + 1 }}</view>
<view class="path-body">
<text class="path-name">{{ path.name }}</text>
<text class="path-reason">{{ path.reason }}</text>
</view>
<view class="path-score-wrap">
<text class="path-score-label">匹配度</text>
<text class="path-score">{{ path.matchScore }}%</text>
<text v-if="path.salary" class="path-salary">{{ path.salary }}</text>
</view>
</view>
<view class="path-hint">点击卡片可进入该岗位的模拟面试</view>
</view>
<view class="action-bar">
<button class="action-btn chat-btn" @click="step = 'chat'; chatMsg = ''">继续咨询</button>
<button class="action-btn retry-btn" @click="step = 'input'; result = null">重新分析</button>
</view>
</view>
<!-- 继续对话 -->
<view v-if="step === 'chat'" class="chat-wrap">
<view class="chat-header">
<text class="chat-back" @click="step = 'result'"> 返回</text>
<text class="chat-title">继续咨询</text>
</view>
<scroll-view class="chat-msgs" scroll-y :scroll-into-view="'msg-' + (chatHistory.length - 1)">
<view v-for="(msg, i) in chatHistory" :key="i" :id="'msg-' + i"
:class="'msg-bubble ' + (msg.role === 'user' ? 'msg-user' : 'msg-ai')">
<text class="msg-text">{{ msg.content }}</text>
</view>
<view v-if="chatLoading" class="msg-bubble msg-ai">
<text class="msg-text typing">AI 思考中...</text>
</view>
</scroll-view>
<view class="chat-input-bar">
<input class="chat-input" v-model="chatMsg" placeholder="输入你的问题..." @confirm="doChat" />
<button class="chat-send" @click="doChat" :disabled="!chatMsg.trim() || chatLoading">发送</button>
</view>
</view>
</view>
</template>
<script setup>
import { ref, reactive } from 'vue'
// #ifdef MP-WEIXIN
import { onShow, onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
// #endif
// #ifndef MP-WEIXIN
import { onShow } from '@dcloudio/uni-app'
// #endif
import { api } from '../../config'
// #ifdef MP-WEIXIN
onShareAppMessage(() => ({ title: '职引 - AI择业顾问 | 专业分析+岗位推荐', path: '/pages/career/career' }))
onShareTimeline(() => ({ title: '职引 - AI择业顾问 | 专业分析+岗位推荐' }))
// #endif
const grades = ['大一', '大二', '大三', '大四', '研一', '研二', '研三', '已毕业']
const step = ref('input')
const loading = ref(false)
const chatLoading = ref(false)
const error = ref('')
const result = ref(null)
const chatMsg = ref('')
const chatHistory = ref([])
const profile = reactive({
major: '',
grade: '',
interests: '',
gpa: '',
goal: '',
})
const token = ref('')
onShow(() => {
token.value = uni.getStorageSync('token') || ''
if (!token.value) {
uni.showModal({
title: '请先登录',
content: '需要登录后才能使用择业顾问功能',
confirmText: '去登录',
success: (r) => { if (r.confirm) uni.navigateTo({ url: '/pages/login/login' }) },
})
}
})
const doAnalyze = async () => {
if (!profile.major.trim()) {
error.value = '请填写你的专业'
return
}
loading.value = true
error.value = ''
try {
const res = await uni.request({
url: api('/career-advice/analyze'),
method: 'POST',
data: { ...profile },
header: { 'Authorization': `Bearer ${token.value}`, 'Content-Type': 'application/json' },
})
if (res.statusCode >= 200 && res.statusCode < 300) {
if (res.data.error) {
error.value = res.data.error
return
}
result.value = res.data
step.value = 'result'
chatHistory.value = [
{ role: 'user', content: `我是${profile.major}专业${profile.grade ? '的' + profile.grade : ''}学生,想了解职业发展方向` },
{ role: 'assistant', content: res.data.reply },
]
} else {
error.value = (res.data && res.data.message) || '分析失败,请稍后重试'
}
} catch (e) {
error.value = '网络错误,请检查网络连接'
} finally {
loading.value = false
}
}
const doChat = async () => {
if (!chatMsg.value.trim() || chatLoading.value) return
const msg = chatMsg.value.trim()
chatMsg.value = ''
chatHistory.value.push({ role: 'user', content: msg })
chatLoading.value = true
try {
const res = await uni.request({
url: api('/career-advice/chat'),
method: 'POST',
data: { message: msg, history: chatHistory.value.slice(0, -1) },
header: { 'Authorization': `Bearer ${token.value}`, 'Content-Type': 'application/json' },
})
if (res.statusCode >= 200 && res.statusCode < 300) {
chatHistory.value.push({ role: 'assistant', content: res.data.reply || (res.data.error || '') })
} else {
chatHistory.value.push({ role: 'assistant', content: '回复失败,请稍后重试' })
}
} catch {
chatHistory.value.push({ role: 'assistant', content: '网络错误,请检查网络连接' })
} finally {
chatLoading.value = false
}
}
const goInterview = (position) => {
uni.navigateTo({ url: `/pages/interview/interview?position=${encodeURIComponent(position)}` })
}
</script>
<style scoped>
.page { min-height: 100vh; background: var(--color-bg); padding-bottom: 40rpx; }
/* ===== Input ===== */
.input-wrap { padding: 0 32rpx; }
.hero { display: flex; flex-direction: column; align-items: center; padding: 48rpx 0 32rpx; }
.hero-icon { font-size: 72rpx; }
.hero-title { font-size: 40rpx; font-weight: 700; color: var(--color-text); margin-top: 16rpx; }
.hero-desc { font-size: 26rpx; color: var(--color-secondary); margin-top: 8rpx; }
.form-card { background: #fff; border-radius: var(--radius-lg); padding: 32rpx; box-shadow: var(--shadow-sm); overflow: hidden; }
.form-group { margin-bottom: 28rpx; width: 100%; }
.form-group:last-child { margin-bottom: 0; }
.form-label { font-size: 26rpx; font-weight: 600; color: var(--color-text); display: block; margin-bottom: 12rpx; }
.required { color: var(--color-error); }
.form-input { width: 100%; height: 80rpx; border: 2rpx solid var(--color-border); border-radius: var(--radius-md); padding: 0 24rpx; font-size: 26rpx; color: var(--color-text); box-sizing: border-box; background: #FAFBFC; max-width: 100%; }
picker { width: 100%; }
.select-trigger { display: flex; align-items: center; }
.form-textarea { width: 100%; height: 160rpx; border: 2rpx solid var(--color-border); border-radius: var(--radius-md); padding: 20rpx 24rpx; font-size: 26rpx; color: var(--color-text); box-sizing: border-box; background: #FAFBFC; resize: none; }
.submit-btn { width: 100%; height: 88rpx; background: linear-gradient(135deg, var(--color-gradient-start), var(--color-gradient-mid)); color: #fff; font-size: 30rpx; font-weight: 600; border-radius: var(--radius-lg); margin-top: 32rpx; display: flex; align-items: center; justify-content: center; border: none; }
.submit-btn:active { transform: scale(0.98); opacity: 0.9; }
.submit-btn[disabled] { opacity: 0.5; }
.error-text { display: block; text-align: center; color: var(--color-error); font-size: 24rpx; margin-top: 16rpx; }
/* ===== Result ===== */
.result-wrap { padding: 0 32rpx; }
.result-header { display: flex; flex-direction: column; align-items: center; padding: 32rpx 0; }
.result-icon { font-size: 56rpx; }
.result-title { font-size: 34rpx; font-weight: 700; color: var(--color-text); margin-top: 12rpx; }
.advice-card { background: linear-gradient(135deg, #EEF2FF 0%, #E0E7FF 100%); border-radius: var(--radius-lg); padding: 32rpx; margin-bottom: 24rpx; }
.advice-content { font-size: 26rpx; color: #374151; line-height: 1.8; white-space: pre-wrap; }
.section { margin-bottom: 24rpx; }
.section-title { font-size: 30rpx; font-weight: 700; color: var(--color-text); display: block; margin-bottom: 16rpx; }
.path-card { display: flex; align-items: center; background: #fff; border-radius: var(--radius-lg); padding: 24rpx; margin-bottom: 16rpx; box-shadow: var(--shadow-sm); }
.path-card:active { transform: scale(0.98); background: #F9FAFB; }
.path-rank { width: 48rpx; height: 48rpx; background: linear-gradient(135deg, var(--color-gradient-start), var(--color-gradient-mid)); color: #fff; font-size: 24rpx; font-weight: 700; border-radius: 50%; display: flex; align-items: center; justify-content: center; flex-shrink: 0; margin-right: 16rpx; }
.path-body { flex: 1; }
.path-name { font-size: 28rpx; font-weight: 600; color: var(--color-text); display: block; }
.path-reason { font-size: 22rpx; color: var(--color-secondary); margin-top: 4rpx; display: block; line-height: 1.4; }
.path-score-wrap { display: flex; flex-direction: column; align-items: center; margin-left: 12rpx; flex-shrink: 0; }
.path-score-label { font-size: 20rpx; color: var(--color-secondary); }
.path-score { font-size: 32rpx; font-weight: 700; color: var(--color-primary); }
.path-salary { font-size: 20rpx; color: var(--color-success); margin-top: 2rpx; }
.path-hint { font-size: 22rpx; color: var(--color-secondary); text-align: center; margin-top: -8rpx; margin-bottom: 16rpx; }
.action-bar { display: flex; gap: 16rpx; }
.action-btn { flex: 1; height: 80rpx; font-size: 28rpx; font-weight: 600; border-radius: var(--radius-md); display: flex; align-items: center; justify-content: center; }
.chat-btn { background: var(--color-primary); color: #fff; }
.retry-btn { background: #fff; color: var(--color-text); border: 2rpx solid var(--color-border); }
/* ===== Chat ===== */
.chat-wrap { display: flex; flex-direction: column; height: 100vh; }
.chat-header { display: flex; align-items: center; padding: 24rpx 32rpx; background: #fff; border-bottom: 1rpx solid var(--color-border); }
.chat-back { font-size: 28rpx; color: var(--color-primary); margin-right: 24rpx; }
.chat-title { font-size: 30rpx; font-weight: 700; color: var(--color-text); flex: 1; }
.chat-msgs { flex: 1; padding: 24rpx 32rpx; overflow-y: auto; }
.msg-bubble { max-width: 80%; margin-bottom: 20rpx; padding: 20rpx 24rpx; border-radius: var(--radius-md); font-size: 26rpx; line-height: 1.6; }
.msg-user { background: var(--color-primary); color: #fff; align-self: flex-end; margin-left: auto; border-radius: var(--radius-md) var(--radius-md) 4rpx var(--radius-md); }
.msg-ai { background: #fff; color: var(--color-text); box-shadow: var(--shadow-sm); border-radius: var(--radius-md) var(--radius-md) var(--radius-md) 4rpx; }
.typing { color: var(--color-secondary); }
.chat-input-bar { display: flex; align-items: center; padding: 16rpx 32rpx; background: #fff; border-top: 1rpx solid var(--color-border); gap: 16rpx; }
.chat-input { flex: 1; height: 72rpx; border: 2rpx solid var(--color-border); border-radius: 36rpx; padding: 0 24rpx; font-size: 26rpx; }
.chat-send { height: 72rpx; padding: 0 32rpx; background: var(--color-primary); color: #fff; font-size: 26rpx; font-weight: 600; border-radius: 36rpx; display: flex; align-items: center; justify-content: center; }
.chat-send[disabled] { opacity: 0.5; }
</style>
+198 -136
View File
@@ -1,197 +1,259 @@
<template> <template>
<view class="page"> <view class="page fade-in">
<!-- 搜索栏 --> <!-- 搜索栏 -->
<view class="search-bar"> <view class="search-bar">
<view class="search-inner">
<text class="search-icon">🔍</text>
<input class="search-input" v-model="keyword" placeholder="搜索公司名称..." @confirm="searchCompany" /> <input class="search-input" v-model="keyword" placeholder="搜索公司名称..." @confirm="searchCompany" />
<text class="search-clear" v-if="keyword" @tap="keyword = ''"></text>
</view>
<button class="search-btn" @tap="searchCompany">搜索</button> <button class="search-btn" @tap="searchCompany">搜索</button>
</view> </view>
<!-- 热门公司 --> <!-- 热门公司 -->
<view class="section" v-if="!searching"> <view class="section" v-if="!selectedCompany">
<view class="section-title">热门公司题库</view> <view class="section-header">
<view class="company-grid"> <text class="section-title">🏢 热门公司题库</text>
<view
class="company-card"
v-for="c in hotCompanies"
:key="c.name"
@tap="selectCompany(c.name)"
>
<view class="company-name">{{ c.name }}</view>
<view class="company-count">{{ c.positions }} 个岗位</view>
</view> </view>
<view class="loading-bar" v-if="loading">
<view class="loading-text">加载中...</view>
</view>
<view class="company-grid" v-else-if="hotCompanies.length > 0">
<view class="company-card card" v-for="c in hotCompanies" :key="c.name" @tap="selectCompany(c.name)">
<text class="company-name">{{ c.name }}</text>
<text class="company-count">{{ c.positionCount > 0 ? c.positionCount + ' 个岗位' : '暂无题库' }}</text>
</view>
</view>
<view class="empty" v-else>
<text class="empty-icon">📭</text>
<text class="empty-text">暂无公司题库数据</text>
<text class="empty-hint">完成面试后贡献面经帮助更多人</text>
</view> </view>
</view> </view>
<!-- 搜索结果岗位列表 --> <!-- 岗位列表 -->
<view class="section" v-if="selectedCompany && !loadingPositions"> <view class="section" v-if="selectedCompany">
<view class="section-title-row"> <view class="section-header">
<text class="section-title">{{ selectedCompany }} - 岗位列表</text> <view class="section-header-left">
<text class="back-link" @tap="selectedCompany = ''">返回</text> <text class="section-back" @tap="backToCompanies"> 返回</text>
<text class="section-title">{{ selectedCompany }}</text>
</view> </view>
<view class="position-list">
<view
class="position-item"
v-for="p in positions"
:key="p.position"
@tap="selectPosition(p.position)"
>
<view class="position-name">{{ p.position }}</view>
<view class="position-meta">{{ p.questionCount }} · {{ p.contributionCount }} 人贡献</view>
</view> </view>
<view class="empty-state" v-if="positions.length === 0"> <view class="loading-bar" v-if="loadingPositions">
<text>暂无该公司的面经数据</text> <view class="loading-text">加载岗位中...</view>
<text class="sub-text">成为第一个贡献者吧</text>
</view> </view>
<view class="position-list" v-else-if="positions.length > 0">
<view class="pos-item card" v-for="p in positions" :key="p.position" @tap="selectPosition(p.position)">
<view class="pos-left">
<text class="pos-icon">{{ p.icon || '💼' }}</text>
<view class="pos-body">
<text class="pos-name">{{ p.position }}</text>
<text class="pos-meta">{{ p.questionCount }} · {{ p.contributionCount }} 人贡献</text>
</view>
</view>
<text class="pos-arrow"></text>
</view>
</view>
<view class="empty" v-else>
<text class="empty-icon">📭</text>
<text class="empty-text">暂无该公司的面经数据</text>
<text class="empty-hint">成为第一个贡献者吧</text>
</view> </view>
</view> </view>
<!-- 题目列表 --> <!-- 题目列表 -->
<view class="section" v-if="selectedPosition && !loadingQuestions"> <view class="section" v-if="selectedPosition">
<view class="section-title-row"> <view class="section-header">
<text class="section-title">{{ selectedCompany }} · {{ selectedPosition }}</text> <view class="section-header-left">
<text class="back-link" @tap="selectedPosition = ''">返回</text> <text class="section-back" @tap="selectedPosition = ''"> 返回</text>
<text class="section-title">{{ selectedPosition }}</text>
</view> </view>
<view class="question-list"> </view>
<view class="question-item" v-for="(q, i) in questions" :key="i"> <view class="loading-bar" v-if="loadingQuestions">
<view class="loading-text">加载题目中...</view>
</view>
<view class="question-list" v-else-if="questions.length > 0">
<view class="question-item card" v-for="(q, i) in questions" :key="i">
<view class="q-header"> <view class="q-header">
<text class="q-num">#{{ i + 1 }}</text> <text class="q-num">#{{ i + 1 }}</text>
<text class="q-tag">{{ q.type === 'technical' ? '技术' : '行为' }}</text> <text class="q-tag" :class="'q-tag-' + (q.type === 'technical' ? 'tech' : 'behavior')">{{ q.type === 'technical' ? '技术' : '行为' }}</text>
<text class="q-diff">{{ difficultyLabel(q.difficulty) }}</text> <text class="q-diff" :class="'q-diff-' + (q.difficulty || 'medium')">{{ difficultyLabel(q.difficulty) }}</text>
<text class="q-freq">{{ q.frequency }} 次提及</text> <text class="q-freq">{{ q.frequency }} 次提及</text>
</view> </view>
<view class="q-content">{{ q.content }}</view> <text class="q-content">{{ q.content }}</text>
<view class="q-tags" v-if="q.tags && q.tags.length"> <view class="q-tags" v-if="q.tags && q.tags.length">
<text class="tag" v-for="t in q.tags" :key="t">{{ t }}</text> <text class="tag" v-for="t in q.tags" :key="t">{{ t }}</text>
</view> </view>
<view class="q-answer" v-if="q.referenceAnswer"> <view class="q-answer" v-if="q.referenceAnswer">
<text class="answer-label">参考思路</text> <text class="answer-label">💡 参考思路</text>
<text class="answer-text">{{ q.referenceAnswer }}</text> <text class="answer-text">{{ q.referenceAnswer }}</text>
</view> </view>
</view> </view>
<view class="empty-state" v-if="questions.length === 0">
<text>暂无题目数据</text>
</view> </view>
<view class="empty" v-else>
<text class="empty-icon">📭</text>
<text class="empty-text">暂无题目数据</text>
</view> </view>
</view> </view>
<!-- 加载态 -->
<view class="loading" v-if="loadingPositions || loadingQuestions">
<text>加载中...</text>
</view>
</view> </view>
</template> </template>
<script> <script setup>
import { ref, onMounted } from 'vue'
import { api } from '../../config' import { api } from '../../config'
const HOT_COMPANIES = [ const keyword = ref('')
{ name: '腾讯', positions: 5 }, const hotCompanies = ref([])
{ name: '字节跳动', positions: 4 }, const selectedCompany = ref('')
{ name: '阿里巴巴', positions: 5 }, const selectedPosition = ref('')
{ name: '美团', positions: 3 }, const positions = ref([])
{ name: '百度', positions: 4 }, const questions = ref([])
{ name: '京东', positions: 3 }, const loading = ref(true)
{ name: '网易', positions: 3 }, const loadingPositions = ref(false)
{ name: '小红书', positions: 2 }, const loadingQuestions = ref(false)
]
export default { onMounted(() => { loadHotCompanies() })
data() {
return { async function loadHotCompanies() {
keyword: '', loading.value = true
searching: false, try {
hotCompanies: HOT_COMPANIES, const token = uni.getStorageSync('token') || ''
selectedCompany: '', const header = token ? { 'Authorization': `Bearer ${token}` } : {}
selectedPosition: '', const res = await uni.request({ url: api('/contribution/companies/hot'), method: 'GET', header })
positions: [], if (res.statusCode === 200) hotCompanies.value = res.data || []
questions: [], } catch (e) { console.error(e) }
loadingPositions: false, finally { loading.value = false }
loadingQuestions: false,
} }
},
methods: { function difficultyLabel(d) {
difficultyLabel(d) {
const map = { junior: '简单', medium: '中等', senior: '困难' } const map = { junior: '简单', medium: '中等', senior: '困难' }
return map[d] || d || '中等' return map[d] || d || '中等'
}, }
async searchCompany() {
const kw = this.keyword.trim() function searchCompany() {
const kw = keyword.value.trim()
if (!kw) return if (!kw) return
this.selectedCompany = kw selectedCompany.value = kw
this.selectedPosition = '' selectedPosition.value = ''
await this.loadPositions(kw) loadPositions(kw)
}, }
async selectCompany(name) {
this.selectedCompany = name function selectCompany(name) {
this.keyword = name selectedCompany.value = name
this.selectedPosition = '' keyword.value = name
await this.loadPositions(name) selectedPosition.value = ''
}, loadPositions(name)
async loadPositions(company) { }
this.loadingPositions = true
function backToCompanies() {
selectedCompany.value = ''
selectedPosition.value = ''
}
async function loadPositions(company) {
loadingPositions.value = true
const token = uni.getStorageSync('token') || '' const token = uni.getStorageSync('token') || ''
const header = token ? { 'Authorization': `Bearer ${token}` } : {} const header = token ? { 'Authorization': `Bearer ${token}` } : {}
try { try {
const res = await uni.request({ url: api(`/contribution/company/${encodeURIComponent(company)}`), method: 'GET', header }) const res = await uni.request({ url: api(`/contribution/company/${encodeURIComponent(company)}`), method: 'GET', header })
this.positions = res.data || [] positions.value = res.data || []
} catch (e) { } catch (e) { positions.value = [] }
this.positions = [] finally { loadingPositions.value = false }
} }
this.loadingPositions = false
}, function selectPosition(position) {
async selectPosition(position) { selectedPosition.value = position
this.selectedPosition = position loadQuestions(selectedCompany.value, position)
await this.loadQuestions(this.selectedCompany, position) }
},
async loadQuestions(company, position) { async function loadQuestions(company, position) {
this.loadingQuestions = true loadingQuestions.value = true
const token = uni.getStorageSync('token') || '' const token = uni.getStorageSync('token') || ''
const header = token ? { 'Authorization': `Bearer ${token}` } : {} const header = token ? { 'Authorization': `Bearer ${token}` } : {}
try { try {
const res = await uni.request({ const res = await uni.request({
url: api(`/contribution/company/${encodeURIComponent(company)}/position/${encodeURIComponent(position)}`), url: api(`/contribution/company/${encodeURIComponent(company)}/position/${encodeURIComponent(position)}`),
method: 'GET', method: 'GET', header,
header,
}) })
this.questions = res.data?.questions || [] questions.value = res.data?.questions || []
} catch (e) { } catch (e) { questions.value = [] }
this.questions = [] finally { loadingQuestions.value = false }
}
this.loadingQuestions = false
},
},
} }
</script> </script>
<style scoped> <style scoped>
.page { padding: 20rpx; min-height: 100vh; background: #f5f6f7; } .page { min-height: 100vh; background: var(--color-bg); padding: 20rpx 32rpx 40rpx; }
.search-bar { display: flex; gap: 20rpx; margin-bottom: 30rpx; }
.search-input { flex: 1; height: 80rpx; background: #fff; border-radius: 40rpx; padding: 0 30rpx; font-size: 28rpx; } /* 搜索 */
.search-btn { height: 80rpx; line-height: 80rpx; padding: 0 40rpx; background: #4F46E5; color: #fff; border-radius: 40rpx; font-size: 28rpx; } .search-bar { display: flex; gap: 16rpx; margin-bottom: 24rpx; padding-top: 16rpx; }
.section { margin-bottom: 30rpx; background: #fff; border-radius: 16rpx; padding: 30rpx; } .search-inner {
.section-title { font-size: 32rpx; font-weight: 600; margin-bottom: 20rpx; display: block; } flex: 1; height: 76rpx; background: var(--color-surface); border-radius: var(--radius-round);
.section-title-row { display: flex; justify-content: space-between; align-items: center; margin-bottom: 20rpx; } display: flex; align-items: center; padding: 0 24rpx; gap: 12rpx;
.back-link { color: #4F46E5; font-size: 26rpx; } box-shadow: 0 2rpx 8rpx rgba(0,0,0,0.04);
.company-grid { display: grid; grid-template-columns: repeat(4, 1fr); gap: 20rpx; } }
.company-card { background: #F3F0FF; border-radius: 12rpx; padding: 24rpx; text-align: center; } .search-icon { font-size: 26rpx; }
.company-name { font-size: 28rpx; font-weight: 500; color: #333; } .search-input { flex: 1; font-size: 26rpx; color: var(--color-text); height: 100%; }
.company-count { font-size: 22rpx; color: #999; margin-top: 8rpx; } .search-clear { font-size: 28rpx; color: var(--color-text-tertiary); padding: 8rpx; }
.position-item { padding: 24rpx; border-bottom: 1rpx solid #f0f0f0; } .search-btn {
.position-name { font-size: 28rpx; font-weight: 500; color: #333; } height: 76rpx; line-height: 76rpx; padding: 0 36rpx; background: var(--color-primary);
.position-meta { font-size: 24rpx; color: #999; margin-top: 8rpx; } color: #fff; border-radius: var(--radius-round); font-size: 26rpx; font-weight: 600; flex-shrink: 0;
.question-item { padding: 24rpx; border-bottom: 1rpx solid #f0f0f0; } }
.q-header { display: flex; align-items: center; gap: 12rpx; margin-bottom: 12rpx; }
.q-num { font-size: 24rpx; color: #4F46E5; font-weight: 600; } /* 区块 */
.q-tag { font-size: 22rpx; background: #E8F5E9; color: #2E7D32; padding: 2rpx 12rpx; border-radius: 6rpx; } .section { margin-bottom: 24rpx; }
.q-diff { font-size: 22rpx; background: #FFF3E0; color: #E65100; padding: 2rpx 12rpx; border-radius: 6rpx; } .section-header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 16rpx; }
.q-freq { font-size: 22rpx; color: #999; margin-left: auto; } .section-header-left { display: flex; align-items: center; gap: 16rpx; }
.q-content { font-size: 28rpx; color: #333; line-height: 1.6; } .section-back { font-size: 28rpx; color: var(--color-primary); font-weight: 500; padding: 8rpx 0; }
.section-title { font-size: 30rpx; font-weight: 700; color: var(--color-text); }
/* 公司网格 */
.company-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 16rpx; }
.company-card {
padding: 24rpx; border-radius: var(--radius-lg); text-align: center;
background: linear-gradient(135deg, #EEF2FF, #E0E7FF);
}
.company-card:active { opacity: 0.7; }
.company-name { font-size: 28rpx; font-weight: 600; color: var(--color-text); display: block; }
.company-count { font-size: 20rpx; color: var(--color-text-secondary); margin-top: 8rpx; display: block; }
/* 岗位列表 */
.position-list { display: flex; flex-direction: column; gap: 16rpx; }
.pos-item { padding: 24rpx 28rpx; border-radius: var(--radius-lg); display: flex; align-items: center; justify-content: space-between; }
.pos-item:active { transform: scale(0.98); }
.pos-left { display: flex; align-items: center; gap: 16rpx; flex: 1; min-width: 0; }
.pos-icon { font-size: 36rpx; }
.pos-body { flex: 1; min-width: 0; }
.pos-name { font-size: 28rpx; font-weight: 600; color: var(--color-text); }
.pos-meta { font-size: 22rpx; color: var(--color-text-secondary); margin-top: 4rpx; display: block; }
.pos-arrow { font-size: 32rpx; color: var(--color-text-tertiary); flex-shrink: 0; margin-left: 12rpx; }
/* 题目列表 */
.question-list { display: flex; flex-direction: column; gap: 16rpx; }
.question-item { padding: 24rpx; border-radius: var(--radius-lg); }
.q-header { display: flex; align-items: center; gap: 12rpx; margin-bottom: 12rpx; flex-wrap: wrap; }
.q-num { font-size: 22rpx; color: var(--color-primary); font-weight: 700; }
.q-tag { font-size: 20rpx; padding: 2rpx 14rpx; border-radius: var(--radius-round); font-weight: 500; }
.q-tag-tech { background: #DCFCE7; color: #166534; }
.q-tag-behavior { background: #FEF3C7; color: #92400E; }
.q-diff { font-size: 20rpx; padding: 2rpx 14rpx; border-radius: var(--radius-round); font-weight: 500; }
.q-diff-junior { background: #EEF2FF; color: var(--color-primary); }
.q-diff-medium { background: #FEF3C7; color: #92400E; }
.q-diff-senior { background: #FEE2E2; color: #991B1B; }
.q-freq { font-size: 20rpx; color: var(--color-text-tertiary); margin-left: auto; }
.q-content { font-size: 28rpx; color: var(--color-text); line-height: 1.7; }
.q-tags { display: flex; flex-wrap: wrap; gap: 8rpx; margin-top: 12rpx; } .q-tags { display: flex; flex-wrap: wrap; gap: 8rpx; margin-top: 12rpx; }
.tag { font-size: 22rpx; background: #F3F0FF; color: #4F46E5; padding: 4rpx 16rpx; border-radius: 20rpx; } .q-tags .tag { font-size: 20rpx; background: #F3F0FF; color: var(--color-primary); padding: 4rpx 16rpx; border-radius: var(--radius-round); }
.q-answer { margin-top: 16rpx; padding: 16rpx; background: #F8F9FA; border-radius: 8rpx; } .q-answer { margin-top: 16rpx; padding: 20rpx; background: #F9FAFB; border-radius: var(--radius-md); }
.answer-label { font-size: 24rpx; color: #4F46E5; font-weight: 500; } .answer-label { font-size: 22rpx; color: var(--color-primary); font-weight: 600; display: block; margin-bottom: 8rpx; }
.answer-text { font-size: 26rpx; color: #555; line-height: 1.6; } .answer-text { font-size: 24rpx; color: var(--color-text-secondary); line-height: 1.6; }
.empty-state { text-align: center; padding: 60rpx 0; color: #999; font-size: 28rpx; }
.sub-text { display: block; margin-top: 12rpx; font-size: 24rpx; color: #bbb; } /* 空状态 */
.loading { text-align: center; padding: 60rpx; color: #999; } .empty { text-align: center; padding: 60rpx 0; background: var(--color-surface); border-radius: var(--radius-lg); }
.empty-icon { font-size: 48rpx; display: block; margin-bottom: 12rpx; }
.empty-text { font-size: 28rpx; color: var(--color-text-secondary); display: block; }
.empty-hint { font-size: 22rpx; color: var(--color-text-tertiary); margin-top: 8rpx; display: block; }
/* 加载 */
.loading-bar { text-align: center; padding: 40rpx; background: var(--color-surface); border-radius: var(--radius-lg); }
.loading-text { font-size: 24rpx; color: var(--color-text-tertiary); }
</style> </style>
+23 -5
View File
@@ -75,10 +75,22 @@
</template> </template>
<script setup> <script setup>
import { ref, onMounted } from 'vue' import { ref } from 'vue'
// #ifdef MP-WEIXIN
import { onLoad, onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
// #endif
// #ifndef MP-WEIXIN
import { onLoad } from '@dcloudio/uni-app'
// #endif
import { api } from '../../config' import { api } from '../../config'
const props = defineProps({ interviewId: String, position: String }) // #ifdef MP-WEIXIN
onShareAppMessage(() => ({ title: '职引 - 分享面经 | 共建大厂面试题库', path: '/pages/contribute/contribute' }))
onShareTimeline(() => ({ title: '职引 - 分享面经 | 共建大厂面试题库' }))
// #endif
const interviewId = ref('')
const urlPosition = ref('')
const form = ref({ company: '', position: '', rounds: '', experience: '', tags: [] }) const form = ref({ company: '', position: '', rounds: '', experience: '', tags: [] })
const questionsText = ref('') const questionsText = ref('')
const customTag = ref('') const customTag = ref('')
@@ -89,8 +101,14 @@ const presetTags = ['算法题多', '重视项目经历', '面试官nice', '压
const token = () => uni.getStorageSync('token') || '' const token = () => uni.getStorageSync('token') || ''
onMounted(() => { onLoad((options) => {
if (props.position) form.value.position = props.position if (options?.position) {
urlPosition.value = decodeURIComponent(options.position)
form.value.position = urlPosition.value
}
if (options?.interviewId) {
interviewId.value = options.interviewId
}
}) })
const toggleTag = (tag) => { const toggleTag = (tag) => {
@@ -122,7 +140,7 @@ const submit = async () => {
url: api('/contribution'), method: 'POST', url: api('/contribution'), method: 'POST',
header: { 'Authorization': `Bearer ${token()}`, 'Content-Type': 'application/json' }, header: { 'Authorization': `Bearer ${token()}`, 'Content-Type': 'application/json' },
data: { data: {
interviewId: props.interviewId || '', interviewId: interviewId.value || '',
company: form.value.company.trim(), company: form.value.company.trim(),
position: form.value.position.trim(), position: form.value.position.trim(),
rounds: form.value.rounds.trim(), rounds: form.value.rounds.trim(),
+9
View File
@@ -54,14 +54,23 @@
<text class="empty-title">{{ emptyTitle }}</text> <text class="empty-title">{{ emptyTitle }}</text>
<text class="empty-desc">{{ emptyDesc }}</text> <text class="empty-desc">{{ emptyDesc }}</text>
<button class="empty-btn btn-gradient" @click="goInterview" v-if="filter === 'all'">开始第一次面试</button> <button class="empty-btn btn-gradient" @click="goInterview" v-if="filter === 'all'">开始第一次面试</button>
<button class="empty-btn" @click="goReviewRecording" style="margin-top:12rpx;background:#FFF;color:#4F46E5;border:2rpx solid #4F46E5;border-radius:var(--radius-round);padding:18rpx 48rpx;font-size:26rpx">🎙 录音复盘</button>
</view> </view>
</view> </view>
</template> </template>
<script setup> <script setup>
import { ref, computed, onMounted } from 'vue' import { ref, computed, onMounted } from 'vue'
// #ifdef MP-WEIXIN
import { onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
// #endif
import { api } from '../../config' import { api } from '../../config'
// #ifdef MP-WEIXIN
onShareAppMessage(() => ({ title: '职引 - 面经与面试记录 | 大厂面经题库', path: '/pages/history/history' }))
onShareTimeline(() => ({ title: '职引 - 面经与面试记录 | 大厂面经题库' }))
// #endif
const filter = ref('all') const filter = ref('all')
const interviewList = ref([]) const interviewList = ref([])
const loading = ref(true) const loading = ref(true)
+200 -52
View File
@@ -4,7 +4,7 @@
<view class="hero-row"> <view class="hero-row">
<view class="hero-left"> <view class="hero-left">
<text class="hero-title">{{ greeting }}</text> <text class="hero-title">{{ greeting }}</text>
<text class="hero-sub">试试下面的功能开启你的求职练习</text> <text class="hero-sub">{{ userInfo ? '试试下面的功能开启你的求职练习' : '登录后免费使用 AI 模拟面试 · 简历优化 · 面经题库' }}</text>
</view> </view>
<view class="hero-right"> <view class="hero-right">
<view class="user-card card" v-if="userInfo" @click="goProfile"> <view class="user-card card" v-if="userInfo" @click="goProfile">
@@ -13,7 +13,7 @@
<text class="user-name">{{ userInfo.nickname || '同学' }}</text> <text class="user-name">{{ userInfo.nickname || '同学' }}</text>
<view class="user-tags"> <view class="user-tags">
<text class="tag tag-plan">{{ userInfo.plan || '免费版' }}</text> <text class="tag tag-plan">{{ userInfo.plan || '免费版' }}</text>
<text class="tag tag-remaining">剩余 {{ userInfo.remaining || 0 }} </text> <text class="tag tag-remaining">{{ (userInfo.gravity ?? 0) > 0 ? '引力值 ' + (userInfo.gravity ?? 0) : '引力值 0' }}</text>
</view> </view>
</view> </view>
<text class="arrow"></text> <text class="arrow"></text>
@@ -21,10 +21,13 @@
<view class="guest-card card" v-else @click="goLogin"> <view class="guest-card card" v-else @click="goLogin">
<image class="avatar" src="/static/avatar-default.png" mode="aspectFill" /> <image class="avatar" src="/static/avatar-default.png" mode="aspectFill" />
<view class="user-meta"> <view class="user-meta">
<text class="user-name">立即登录</text> <text class="user-name">登录 / 注册</text>
<text class="guest-hint">登录后体验全部功能</text> <view class="guest-benefits">
<text class="guest-benefit">🎙 模拟面试</text>
<text class="guest-benefit">📄 简历优化</text>
</view> </view>
<text class="arrow"></text> </view>
<text class="guest-arrow"></text>
</view> </view>
</view> </view>
</view> </view>
@@ -43,7 +46,14 @@
</view> </view>
<text class="fp-action">开始</text> <text class="fp-action">开始</text>
</view> </view>
<view class="feature-tertiary"> <view class="feature-grid">
<view class="fs-card card" @click="goCareer">
<view class="fs-top">
<view class="fs-icon fs-career"><text class="fs-emoji">🎯</text></view>
<text class="fs-name">AI 择业顾问</text>
</view>
<text class="fs-brief">专业分析 · 岗位匹配 · 智能推荐</text>
</view>
<view class="fs-card card" @click="goResume"> <view class="fs-card card" @click="goResume">
<view class="fs-top"> <view class="fs-top">
<view class="fs-icon fs-resume"><text class="fs-emoji">📄</text></view> <view class="fs-icon fs-resume"><text class="fs-emoji">📄</text></view>
@@ -65,33 +75,18 @@
</view> </view>
<text class="fs-brief">分享经验 · 共建题库 · 帮更多人</text> <text class="fs-brief">分享经验 · 共建题库 · 帮更多人</text>
</view> </view>
</view> <!-- 公司真题库&实习搜索 - 暂隐藏内容待丰富 -->
<view class="feature-secondary">
<view class="fs-card card" @click="goBank">
<view class="fs-top">
<view class="fs-icon fs-progress"><text class="fs-emoji">📚</text></view>
<text class="fs-name">公司真题库</text>
</view>
<text class="fs-brief">大厂真题 · 岗位分类 · 参考思路</text>
</view>
<view class="fs-card card" @click="goInternship">
<view class="fs-top">
<view class="fs-icon fs-contribute"><text class="fs-emoji">🔍</text></view>
<text class="fs-name">实习搜索</text>
</view>
<text class="fs-brief">热门实习 · 一键搜索 · 精准匹配</text>
</view>
</view> </view>
</view> </view>
</view> </view>
<!-- 每日一题 --> <!-- 每日一题 -->
<view class="section" v-if="dailyQuestion"> <view class="section">
<view class="section-header"> <view class="section-header">
<text class="section-title">📮 每日一题</text> <text class="section-title">📮 每日一题</text>
<text class="section-desc" @click="refreshDaily">换一题</text> <text class="section-desc" @click="refreshDaily" v-if="dailyQuestion">换一题</text>
</view> </view>
<view class="daily-card card"> <view class="daily-card card" v-if="dailyQuestion">
<text class="daily-tag">{{ dailyQuestion.category || '综合' }}</text> <text class="daily-tag">{{ dailyQuestion.category || '综合' }}</text>
<text class="daily-question">{{ dailyQuestion.question }}</text> <text class="daily-question">{{ dailyQuestion.question }}</text>
<view class="daily-answer" v-if="showAnswer"> <view class="daily-answer" v-if="showAnswer">
@@ -105,31 +100,79 @@
<text class="daily-action primary" @click="goInterview">模拟练习 </text> <text class="daily-action primary" @click="goInterview">模拟练习 </text>
</view> </view>
</view> </view>
<view class="daily-guest card" v-else-if="!userInfo" @click="goLogin">
<view class="daily-guest-top">
<text class="daily-guest-icon">🔒</text>
<view class="daily-guest-body">
<text class="daily-guest-title">登录后获取今日面试题</text>
<text class="daily-guest-desc">每日一道校招真题持续积累面试经验</text>
</view>
</view>
<text class="daily-action primary">立即登录 </text>
</view>
</view> </view>
<!-- 热门岗位 --> <!-- 热门岗位 - AI 专区 -->
<view class="section"> <view class="section">
<view class="section-header"> <view class="section-header">
<view class="section-title-row"> <view class="section-title-row">
<text class="section-title">热门岗位</text> <text class="section-title">🔥 AI 热门岗位</text>
<text class="section-tag-demo">参考示例</text> <text class="section-badge">NEW</text>
</view> </view>
<text class="section-desc">点击直接面试</text> <text class="section-desc">AI 时代最热方向点击直接面试</text>
</view> </view>
<view class="position-list card" v-if="!positionsLoading"> <view class="ai-banner card" @click="goInterview">
<view class="pos-item" v-for="(pos, idx) in hotPositions" :key="idx" @click="startInterview(pos)"> <view class="ai-banner-icon-wrap">
<text class="ai-banner-icon">🤖</text>
</view>
<view class="ai-banner-body">
<text class="ai-banner-title">AI 正在重塑整个行业</text>
<text class="ai-banner-desc">大模型应用 / Agent 开发 / Prompt 工程 顶尖人才缺口巨大现在上车正当时</text>
</view>
<text class="ai-banner-arrow"></text>
</view>
<view class="position-list card" v-if="!positionsLoading && displayAiPositions.length > 0">
<view class="pos-item" v-for="(pos, idx) in displayAiPositions" :key="'ai-' + idx" @click="startInterview(pos)">
<view class="pos-left"> <view class="pos-left">
<text class="pos-icon">{{ posIcons[idx] || '💼' }}</text> <text class="pos-icon pos-icon-ai">{{ pos.icon || aiPosIcons[idx % aiPosIcons.length] || '🤖' }}</text>
<view class="pos-body"> <view class="pos-body">
<text class="pos-name">{{ pos.name }}</text> <text class="pos-name">{{ pos.name }}</text>
<view class="pos-meta-row"> <text class="pos-name-tag">AI 方向</text>
<text class="pos-company">{{ pos.company || '参考公司' }}</text> <view class="pos-meta-row" v-if="pos.company || pos.salary">
<text class="pos-salary">{{ pos.salary || '参考薪资' }}</text> <text class="pos-company">{{ pos.company }}</text>
<text class="pos-salary">{{ pos.salary }}</text>
</view> </view>
</view> </view>
</view> </view>
<view class="pos-action"> <view class="pos-action">
<text class="pos-action-text">立即模拟</text> <text class="pos-action-btn pos-action-ai">开始面试</text>
</view>
</view>
</view>
<!-- 更多岗位 -->
<view class="more-header" @click="showMore = !showMore">
<view class="more-header-left">
<text class="more-icon">🧑💻</text>
<text class="more-title">更多岗位{{ traditionalPositions.length }}</text>
</view>
<text class="more-arrow">{{ showMore ? '收起 ▲' : '展开 ▼' }}</text>
</view>
<view class="position-list card" v-if="showMore && !positionsLoading && traditionalPositions.length > 0">
<view class="pos-item" v-for="(pos, idx) in traditionalPositions" :key="'tr-' + idx" @click="startInterview(pos)">
<view class="pos-left">
<text class="pos-icon">{{ pos.icon || posIcons[(displayAiPositions.length + idx) % posIcons.length] || '💼' }}</text>
<view class="pos-body">
<text class="pos-name">{{ pos.name }}</text>
<text class="pos-name-tag pos-name-tag-tr">{{ pos.category === 'intern' ? '实习' : '校招' }}</text>
<view class="pos-meta-row" v-if="pos.company || pos.salary">
<text class="pos-company">{{ pos.company }}</text>
<text class="pos-salary">{{ pos.salary }}</text>
</view>
</view>
</view>
<view class="pos-action">
<text class="pos-action-btn">立即模拟</text>
</view> </view>
</view> </view>
</view> </view>
@@ -141,17 +184,46 @@
</template> </template>
<script setup> <script setup>
import { ref, onMounted } from 'vue' import { ref, computed, onMounted } from 'vue'
// #ifdef MP-WEIXIN
import { onShow, onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
// #endif
// #ifndef MP-WEIXIN
import { onShow } from '@dcloudio/uni-app' import { onShow } from '@dcloudio/uni-app'
// #endif
import { api } from '../../config' import { api } from '../../config'
// #ifdef MP-WEIXIN
onShareAppMessage(() => ({ title: '职引 - AI模拟面试 | 校招求职一站式平台', path: '/pages/index/index' }))
onShareTimeline(() => ({ title: '职引 - AI模拟面试 | 校招求职一站式平台' }))
// #endif
const userInfo = ref(null) const userInfo = ref(null)
const greeting = ref('') const greeting = ref('')
const hotPositions = ref([]) const hotPositions = ref([])
const posIcons = ['💻', '⚙️', '🤖', '📊', '🎨', '🧪', '📱', '🔧'] const posIcons = ['💻', '⚙️', '🤖', '📊', '🎨', '🧪', '📱', '🔧']
const aiPosIcons = ['🤖', '🧠', '⚡', '🚀', '💡', '🔬']
const positionsLoading = ref(true) const positionsLoading = ref(true)
const dailyQuestion = ref(null) const dailyQuestion = ref(null)
const showAnswer = ref(false) const showAnswer = ref(false)
const showMore = ref(true)
// 前端兜底 AI 岗位(当 API 返回不足时展示)
const FALLBACK_AI_POSITIONS = [
{ name: 'AI Agent 开发工程师', category: 'ai', company: '热门方向', salary: '30K-60K', icon: '🧠' },
{ name: '大模型应用开发', category: 'ai', company: '热门方向', salary: '25K-50K', icon: '⚡' },
]
const aiPositions = computed(() => hotPositions.value.filter(p => p.category === 'ai'))
const displayAiPositions = computed(() => {
const fromApi = aiPositions.value
if (fromApi.length >= 4) return fromApi
// 补充到至少 4 个,去重
const existingNames = new Set(fromApi.map(p => p.name))
const needed = FALLBACK_AI_POSITIONS.filter(p => !existingNames.has(p.name))
return [...fromApi, ...needed]
})
const traditionalPositions = computed(() => hotPositions.value.filter(p => p.category !== 'ai'))
const loadUserInfo = () => { const loadUserInfo = () => {
try { const s = uni.getStorageSync('userInfo'); if (s) userInfo.value = JSON.parse(s); else userInfo.value = null } catch (e) { userInfo.value = null } try { const s = uni.getStorageSync('userInfo'); if (s) userInfo.value = JSON.parse(s); else userInfo.value = null } catch (e) { userInfo.value = null }
@@ -190,7 +262,19 @@ onMounted(async () => {
onShow(loadUserInfo) onShow(loadUserInfo)
const refreshDaily = () => { showAnswer.value = false; /* trigger reload */ } const refreshDaily = async () => {
showAnswer.value = false
try {
const t = uni.getStorageSync('token')
if (t) {
const qres = await uni.request({
url: api('/daily-question'), method: 'GET',
header: { 'Authorization': `Bearer ${t}` }
})
if (qres.statusCode === 200 && qres.data) dailyQuestion.value = qres.data
}
} catch (e) { /* silent */ }
}
const goProfile = () => uni.switchTab({ url: '/pages/user/user' }) const goProfile = () => uni.switchTab({ url: '/pages/user/user' })
const goLogin = () => uni.navigateTo({ url: '/pages/login/login' }) const goLogin = () => uni.navigateTo({ url: '/pages/login/login' })
@@ -200,6 +284,7 @@ const goProgress = () => uni.navigateTo({ url: '/pages/progress/progress' })
const goContribute = () => uni.navigateTo({ url: '/pages/contribute/contribute' }) const goContribute = () => uni.navigateTo({ url: '/pages/contribute/contribute' })
const goBank = () => uni.navigateTo({ url: '/pages/company-bank/bank' }) const goBank = () => uni.navigateTo({ url: '/pages/company-bank/bank' })
const goInternship = () => uni.navigateTo({ url: '/pages/internship/internship' }) const goInternship = () => uni.navigateTo({ url: '/pages/internship/internship' })
const goCareer = () => uni.navigateTo({ url: '/pages/career/career' })
const startInterview = (pos) => uni.navigateTo({ url: `/pages/interview/interview?position=${encodeURIComponent(pos.name)}` }) const startInterview = (pos) => uni.navigateTo({ url: `/pages/interview/interview?position=${encodeURIComponent(pos.name)}` })
</script> </script>
@@ -213,7 +298,7 @@ const startInterview = (pos) => uni.navigateTo({ url: `/pages/interview/intervie
} }
.hero-row { display: flex; align-items: flex-start; gap: 24rpx; } .hero-row { display: flex; align-items: flex-start; gap: 24rpx; }
.hero-left { flex: 1; min-width: 0; padding-top: 8rpx; } .hero-left { flex: 1; min-width: 0; padding-top: 8rpx; }
.hero-right { flex-shrink: 0; width: 320rpx; } .hero-right { flex-shrink: 0; min-width: 280rpx; max-width: 340rpx; }
.hero-title { font-size: 40rpx; font-weight: 700; color: #FFF; display: block; line-height: 1.3; } .hero-title { font-size: 40rpx; font-weight: 700; color: #FFF; display: block; line-height: 1.3; }
.hero-sub { font-size: 22rpx; color: rgba(255,255,255,0.7); margin-top: 8rpx; display: block; } .hero-sub { font-size: 22rpx; color: rgba(255,255,255,0.7); margin-top: 8rpx; display: block; }
@@ -223,11 +308,15 @@ const startInterview = (pos) => uni.navigateTo({ url: `/pages/interview/intervie
display: flex; align-items: center; display: flex; align-items: center;
box-shadow: 0 8rpx 32rpx rgba(0,0,0,0.1); box-shadow: 0 8rpx 32rpx rgba(0,0,0,0.1);
} }
.guest-card { background: rgba(255,255,255,0.15); backdrop-filter: blur(10rpx); } .guest-card {
.guest-card .avatar { border-color: rgba(255,255,255,0.3); } background: rgba(255,255,255,0.2); backdrop-filter: blur(12rpx);
.guest-card .user-name { font-size: 26rpx; color: #FFF; } border: 1rpx solid rgba(255,255,255,0.25);
.guest-hint { font-size: 20rpx; color: rgba(255,255,255,0.6); margin-top: 4rpx; display: block; } }
.guest-card .arrow { color: rgba(255,255,255,0.4); } .guest-card .avatar { border-color: rgba(255,255,255,0.35); }
.guest-card .user-name { font-size: 26rpx; color: #FFF; font-weight: 700; }
.guest-benefits { display: flex; gap: 12rpx; margin-top: 6rpx; flex-wrap: wrap; }
.guest-benefit { font-size: 18rpx; color: rgba(255,255,255,0.8); padding: 2rpx 10rpx; background: rgba(255,255,255,0.12); border-radius: 8rpx; }
.guest-arrow { font-size: 36rpx; color: rgba(255,255,255,0.5); margin-left: 12rpx; }
.avatar { width: 72rpx; height: 72rpx; border-radius: 50%; margin-right: 16rpx; border: 3rpx solid var(--color-primary-light); flex-shrink: 0; } .avatar { width: 72rpx; height: 72rpx; border-radius: 50%; margin-right: 16rpx; border: 3rpx solid var(--color-primary-light); flex-shrink: 0; }
.user-meta { flex: 1; min-width: 0; } .user-meta { flex: 1; min-width: 0; }
.user-name { font-size: 30rpx; font-weight: 600; color: var(--color-text); } .user-name { font-size: 30rpx; font-weight: 600; color: var(--color-text); }
@@ -242,7 +331,7 @@ const startInterview = (pos) => uni.navigateTo({ url: `/pages/interview/intervie
.section-header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 20rpx; } .section-header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 20rpx; }
.section-title { font-size: 30rpx; font-weight: 700; color: var(--color-text); } .section-title { font-size: 30rpx; font-weight: 700; color: var(--color-text); }
.section-title-row { display: flex; align-items: center; gap: 12rpx; } .section-title-row { display: flex; align-items: center; gap: 12rpx; }
.section-tag-demo { font-size: 18rpx; color: #9CA3AF; background: #F3F4F6; padding: 2rpx 10rpx; border-radius: 6rpx; } .section-badge { font-size: 18rpx; color: #fff; background: var(--color-primary); padding: 2rpx 12rpx; border-radius: 20rpx; font-weight: 500; }
.section-desc { font-size: 22rpx; color: var(--color-primary); } .section-desc { font-size: 22rpx; color: var(--color-primary); }
.feature-list { display: flex; flex-direction: column; gap: 16rpx; } .feature-list { display: flex; flex-direction: column; gap: 16rpx; }
@@ -259,17 +348,19 @@ const startInterview = (pos) => uni.navigateTo({ url: `/pages/interview/intervie
.fp-name { font-size: 30rpx; font-weight: 700; color: var(--color-text); } .fp-name { font-size: 30rpx; font-weight: 700; color: var(--color-text); }
.fp-brief { font-size: 20rpx; color: var(--color-text-secondary); margin-top: 4rpx; display: block; } .fp-brief { font-size: 20rpx; color: var(--color-text-secondary); margin-top: 4rpx; display: block; }
.fp-action { font-size: 28rpx; color: var(--color-primary); font-weight: 600; flex-shrink: 0; } .fp-action { font-size: 28rpx; color: var(--color-primary); font-weight: 600; flex-shrink: 0; }
.feature-secondary { display: grid; grid-template-columns: 1fr 1fr; gap: 16rpx; } .feature-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 16rpx; }
.feature-tertiary { display: grid; grid-template-columns: 1fr 1fr 1fr; gap: 16rpx; }
.fs-card { padding: 20rpx; border-radius: var(--radius-lg); } .fs-card { padding: 20rpx; border-radius: var(--radius-lg); }
.fs-top { display: flex; align-items: center; gap: 10rpx; } .fs-top { display: flex; align-items: center; gap: 10rpx; }
.fs-icon { width: 44rpx; height: 44rpx; border-radius: 12rpx; display: flex; align-items: center; justify-content: center; flex-shrink: 0; } .fs-icon { width: 44rpx; height: 44rpx; border-radius: 12rpx; display: flex; align-items: center; justify-content: center; flex-shrink: 0; }
.fs-emoji { font-size: 20rpx; } .fs-emoji { font-size: 20rpx; }
.fs-name { font-size: 26rpx; font-weight: 600; color: var(--color-text); } .fs-name { font-size: 26rpx; font-weight: 600; color: var(--color-text); }
.fs-brief { font-size: 18rpx; color: var(--color-text-secondary); margin-top: 10rpx; display: block; } .fs-brief { font-size: 18rpx; color: var(--color-text-secondary); margin-top: 10rpx; display: block; }
.fs-progress { background: linear-gradient(135deg, #EEF2FF, #C7D2FE); } .fs-career { background: linear-gradient(135deg, #F3E8FF, #D8B4FE); }
.fs-resume { background: linear-gradient(135deg, #ECFDF5, #A7F3D0); } .fs-resume { background: linear-gradient(135deg, #ECFDF5, #A7F3D0); }
.fs-progress { background: linear-gradient(135deg, #EEF2FF, #C7D2FE); }
.fs-contribute { background: linear-gradient(135deg, #FFF7ED, #FDBA74); } .fs-contribute { background: linear-gradient(135deg, #FFF7ED, #FDBA74); }
.fs-bank { background: linear-gradient(135deg, #FCE7F3, #F9A8D4); }
.fs-internship { background: linear-gradient(135deg, #E0F2FE, #7DD3FC); }
/* 每日一题 */ /* 每日一题 */
.daily-card { padding: 24rpx; border-radius: var(--radius-lg); } .daily-card { padding: 24rpx; border-radius: var(--radius-lg); }
@@ -282,19 +373,76 @@ const startInterview = (pos) => uni.navigateTo({ url: `/pages/interview/intervie
.daily-action { font-size: 24rpx; color: var(--color-text-secondary); } .daily-action { font-size: 24rpx; color: var(--color-text-secondary); }
.daily-action.primary { color: var(--color-primary); font-weight: 600; } .daily-action.primary { color: var(--color-primary); font-weight: 600; }
.daily-guest {
padding: 24rpx; border-radius: var(--radius-lg);
background: linear-gradient(135deg, #EEF2FF, #E0E7FF);
display: flex; align-items: center; justify-content: space-between;
cursor: pointer;
}
.daily-guest:active { opacity: 0.8; }
.daily-guest-top { display: flex; align-items: center; gap: 16rpx; flex: 1; min-width: 0; }
.daily-guest-icon { font-size: 32rpx; width: 56rpx; height: 56rpx; display: flex; align-items: center; justify-content: center; background: rgba(255,255,255,0.6); border-radius: 14rpx; flex-shrink: 0; }
.daily-guest-body { flex: 1; min-width: 0; }
.daily-guest-title { font-size: 26rpx; font-weight: 600; color: var(--color-text); }
.daily-guest-desc { font-size: 20rpx; color: var(--color-text-secondary); margin-top: 4rpx; display: block; }
/* AI 岗位专区 */
.ai-banner {
background: linear-gradient(135deg, #FEF3C7 0%, #FDE68A 50%, #FCD34D 100%);
padding: 20rpx 24rpx; border-radius: var(--radius-lg); margin-bottom: 16rpx;
display: flex; align-items: center; gap: 16rpx;
box-shadow: 0 4rpx 16rpx rgba(251,191,36,0.2);
cursor: pointer; transition: transform 0.15s;
}
.ai-banner:active { transform: scale(0.97); }
.ai-banner-icon-wrap {
width: 64rpx; height: 64rpx; border-radius: 18rpx;
background: rgba(255,255,255,0.6); display: flex; align-items: center; justify-content: center; flex-shrink: 0;
}
.ai-banner-icon { font-size: 36rpx; }
.ai-banner-body { flex: 1; min-width: 0; }
.ai-banner-title { font-size: 26rpx; font-weight: 700; color: #92400E; display: block; margin-bottom: 4rpx; }
.ai-banner-desc { font-size: 20rpx; color: #A16207; line-height: 1.5; display: block; }
.ai-banner-arrow { font-size: 28rpx; color: #B45309; font-weight: 600; flex-shrink: 0; }
.position-list { border-radius: var(--radius-lg); overflow: hidden; } .position-list { border-radius: var(--radius-lg); overflow: hidden; }
.pos-item { padding: 24rpx 28rpx; border-bottom: 1rpx solid var(--color-border); display: flex; justify-content: space-between; align-items: center; } .pos-item { padding: 26rpx 28rpx; border-bottom: 1rpx solid var(--color-border); display: flex; justify-content: space-between; align-items: center; }
.pos-item:last-child { border-bottom: none; } .pos-item:last-child { border-bottom: none; }
.pos-item:active { background: var(--color-bg); } .pos-item:active { background: #F9FAFB; }
.pos-left { display: flex; align-items: center; gap: 16rpx; flex: 1; min-width: 0; } .pos-left { display: flex; align-items: center; gap: 16rpx; flex: 1; min-width: 0; }
.pos-icon { font-size: 36rpx; width: 56rpx; height: 56rpx; display: flex; align-items: center; justify-content: center; background: #F3F4F6; border-radius: 14rpx; flex-shrink: 0; } .pos-icon { font-size: 32rpx; width: 60rpx; height: 60rpx; display: flex; align-items: center; justify-content: center; background: #F3F4F6; border-radius: 16rpx; flex-shrink: 0; }
.pos-icon-ai { background: linear-gradient(135deg, #FEF3C7, #FDE68A); }
.pos-body { display: flex; flex-direction: column; flex: 1; min-width: 0; } .pos-body { display: flex; flex-direction: column; flex: 1; min-width: 0; }
.pos-name { font-size: 28rpx; font-weight: 600; color: var(--color-text); } .pos-name { font-size: 28rpx; font-weight: 600; color: var(--color-text); }
.pos-name-tag {
font-size: 18rpx; color: #D97706; background: #FFFBEB; padding: 1rpx 10rpx;
border-radius: 6rpx; align-self: flex-start; margin-top: 6rpx;
}
.pos-name-tag-tr { color: var(--color-primary); background: #EEF2FF; }
.pos-meta-row { display: flex; align-items: center; gap: 10rpx; margin-top: 4rpx; } .pos-meta-row { display: flex; align-items: center; gap: 10rpx; margin-top: 4rpx; }
.pos-company { font-size: 20rpx; color: var(--color-text-tertiary); } .pos-company { font-size: 20rpx; color: var(--color-text-tertiary); }
.pos-salary { font-size: 20rpx; color: var(--color-primary); background: #EEF2FF; padding: 2rpx 10rpx; border-radius: 6rpx; } .pos-salary { font-size: 20rpx; color: var(--color-primary); background: #EEF2FF; padding: 2rpx 10rpx; border-radius: 6rpx; }
.pos-action { flex-shrink: 0; margin-left: 16rpx; } .pos-action { flex-shrink: 0; margin-left: 16rpx; }
.pos-action-text { font-size: 22rpx; color: var(--color-primary); font-weight: 600; } .pos-action-btn {
font-size: 22rpx; font-weight: 600; color: var(--color-primary);
padding: 10rpx 24rpx; border-radius: var(--radius-round);
background: #EEF2FF; display: inline-block;
}
.pos-action-btn:active { background: #DBEAFE; }
.pos-action-ai { color: #D97706; background: #FFFBEB; }
/* 更多岗位 */
.more-header {
display: flex; justify-content: space-between; align-items: center;
padding: 24rpx 20rpx 16rpx; margin-top: 8rpx; cursor: pointer;
background: #FFFFFF; border-radius: var(--radius-lg) var(--radius-lg) 0 0;
}
.more-header:active { opacity: 0.7; }
.more-header-left { display: flex; align-items: center; gap: 10rpx; }
.more-icon { font-size: 28rpx; }
.more-title { font-size: 26rpx; font-weight: 600; color: var(--color-text); }
.more-arrow { font-size: 22rpx; color: var(--color-primary); font-weight: 500; background: #EEF2FF; padding: 4rpx 16rpx; border-radius: var(--radius-round); }
.loading-tip { text-align: center; padding: 40rpx; font-size: 24rpx; color: var(--color-text-tertiary); background: #FFF; border-radius: var(--radius-lg); } .loading-tip { text-align: center; padding: 40rpx; font-size: 24rpx; color: var(--color-text-tertiary); background: #FFF; border-radius: var(--radius-lg); }
.bottom-spacer { height: 40rpx; } .bottom-spacer { height: 40rpx; }
</style> </style>
+121 -31
View File
@@ -1,39 +1,89 @@
<template> <template>
<view class="page"> <view class="page fade-in">
<view class="hero"> <!-- 搜索栏 -->
<text class="hero-title">实习推荐</text> <view class="search-bar">
<text class="hero-sub">热门实习岗位点击直接模拟面试</text> <view class="search-inner">
<text class="search-icon">🔍</text>
<input class="search-input" v-model="keyword" placeholder="搜索岗位或公司..." @input="onSearch" />
<text class="search-clear" v-if="keyword" @tap="keyword = ''; onSearch()"></text>
</view>
</view> </view>
<view class="body"> <!-- 分类标签 -->
<view class="section-title">🔥 热门实习</view> <view class="tabs">
<view class="position-list card"> <view class="tab" :class="tab === 'all' && 'tab-active'" @tap="tab = 'all'; tabIndex = 0">
<view class="pos-item" v-for="(item, idx) in positions" :key="idx" @click="startInterview(item)"> <text class="tab-text">全部</text>
</view>
<view class="tab" :class="tab === 'ai' && 'tab-active'" @tap="tab = 'ai'; tabIndex = 1">
<text class="tab-text">🤖 AI 岗位</text>
</view>
<view class="tab" :class="tab === 'traditional' && 'tab-active'" @tap="tab = 'traditional'; tabIndex = 2">
<text class="tab-text">💼 传统岗位</text>
</view>
</view>
<!-- 结果计数 -->
<view class="result-info" v-if="!loading">
<text class="result-count"> {{ filteredPositions.length }} 个岗位</text>
</view>
<!-- 岗位列表 -->
<view class="position-list">
<view class="pos-item card" v-for="(item, idx) in filteredPositions" :key="idx" @click="startInterview(item)">
<view class="pos-left"> <view class="pos-left">
<view class="pos-rank">{{ idx + 1 }}</view> <view class="pos-icon-wrap">
<text class="pos-icon">{{ item.icon || '💼' }}</text>
</view>
<view class="pos-body"> <view class="pos-body">
<text class="pos-name">{{ item.name }}</text> <text class="pos-name">{{ item.name }}</text>
<view class="pos-meta-row" v-if="item.company">
<text class="pos-company">{{ item.company }}</text> <text class="pos-company">{{ item.company }}</text>
<text class="pos-badge" :class="item.category === 'ai' ? 'badge-ai' : 'badge-traditional'">{{ item.category === 'ai' ? 'AI' : '传统' }}</text>
</view> </view>
</view> </view>
<text class="pos-salary">{{ item.salary }}</text> </view>
<view class="pos-right">
<text class="pos-salary" v-if="item.salary">{{ item.salary }}</text>
<text class="pos-action">模拟 </text>
</view>
</view> </view>
</view> </view>
<view class="empty" v-if="!loading && positions.length === 0"> <!-- 空状态 -->
<text class="empty-text">暂无实习岗位数据</text> <view class="empty" v-if="!loading && filteredPositions.length === 0">
<text class="empty-icon">🔍</text>
<text class="empty-text">{{ keyword ? '没有匹配的岗位' : '暂无岗位数据' }}</text>
</view> </view>
<!-- 加载 -->
<view class="loading-bar" v-if="loading">
<text class="loading-text">加载中...</text>
</view> </view>
</view> </view>
</template> </template>
<script setup> <script setup>
import { ref, onMounted } from 'vue' import { ref, computed, onMounted } from 'vue'
import { api } from '../../config' import { api } from '../../config'
const keyword = ref('')
const tab = ref('all')
const tabIndex = ref(0)
const positions = ref([]) const positions = ref([])
const loading = ref(true) const loading = ref(true)
const filteredPositions = computed(() => {
let list = positions.value
if (tab.value !== 'all') {
list = list.filter(p => p.category === tab.value)
}
if (keyword.value.trim()) {
const kw = keyword.value.trim().toLowerCase()
list = list.filter(p => (p.name && p.name.toLowerCase().includes(kw)) || (p.company && p.company.toLowerCase().includes(kw)))
}
return list
})
onMounted(async () => { onMounted(async () => {
try { try {
const res = await uni.request({ url: api('/positions/hot'), method: 'GET' }) const res = await uni.request({ url: api('/positions/hot'), method: 'GET' })
@@ -42,26 +92,66 @@ onMounted(async () => {
finally { loading.value = false } finally { loading.value = false }
}) })
function onSearch() { /* reactivity handles filtering via computed */ }
const startInterview = (pos) => uni.navigateTo({ url: `/pages/interview/interview?position=${encodeURIComponent(pos.name)}` }) const startInterview = (pos) => uni.navigateTo({ url: `/pages/interview/interview?position=${encodeURIComponent(pos.name)}` })
</script> </script>
<style scoped> <style scoped>
.page { min-height: 100vh; background: var(--color-bg); } .page { min-height: 100vh; background: var(--color-bg); padding: 20rpx 32rpx 40rpx; }
.hero { background: linear-gradient(135deg, var(--color-gradient-start) 0%, var(--color-gradient-mid) 50%, var(--color-gradient-end) 100%); padding: 48rpx 32rpx 72rpx; border-radius: 0 0 48rpx 48rpx; }
.hero-title { font-size: 40rpx; font-weight: 700; color: #FFFFFF; display: block; } /* 搜索 */
.hero-sub { font-size: 22rpx; color: rgba(255,255,255,0.7); margin-top: 8rpx; display: block; } .search-bar { padding-top: 16rpx; margin-bottom: 20rpx; }
.body { padding: 32rpx; margin-top: -40rpx; } .search-inner {
.section-title { font-size: 28rpx; font-weight: 600; color: var(--color-text); margin-bottom: 16rpx; } height: 76rpx; background: var(--color-surface); border-radius: var(--radius-round);
.position-list { border-radius: var(--radius-lg); overflow: hidden; } display: flex; align-items: center; padding: 0 24rpx; gap: 12rpx;
.pos-item { padding: 24rpx 28rpx; border-bottom: 1rpx solid var(--color-border); display: flex; justify-content: space-between; align-items: center; } box-shadow: 0 2rpx 8rpx rgba(0,0,0,0.04);
.pos-item:active { background: #F9FAFB; } }
.pos-item:last-child { border-bottom: none; } .search-icon { font-size: 26rpx; }
.pos-left { display: flex; align-items: center; gap: 16rpx; } .search-input { flex: 1; font-size: 26rpx; color: var(--color-text); height: 100%; }
.pos-rank { width: 40rpx; height: 40rpx; border-radius: 10rpx; background: #F3F4F6; display: flex; align-items: center; justify-content: center; font-size: 22rpx; font-weight: 700; color: var(--color-primary); flex-shrink: 0; } .search-clear { font-size: 28rpx; color: var(--color-text-tertiary); padding: 8rpx; }
.pos-body { display: flex; flex-direction: column; }
.pos-name { font-size: 28rpx; font-weight: 600; color: var(--color-text); } /* 分类标签 */
.pos-company { font-size: 22rpx; color: var(--color-text-tertiary); margin-top: 4rpx; } .tabs { display: flex; gap: 12rpx; margin-bottom: 16rpx; }
.pos-salary { font-size: 24rpx; color: var(--color-primary); font-weight: 600; } .tab {
.empty { display: flex; justify-content: center; padding: 60rpx 0; } padding: 12rpx 24rpx; border-radius: var(--radius-round); background: var(--color-surface);
.empty-text { font-size: 26rpx; color: var(--color-text-tertiary); } box-shadow: 0 2rpx 8rpx rgba(0,0,0,0.04);
}
.tab-active { background: var(--color-primary); }
.tab-active .tab-text { color: #fff; }
.tab-text { font-size: 24rpx; color: var(--color-text-secondary); font-weight: 500; }
/* 结果信息 */
.result-info { margin-bottom: 16rpx; }
.result-count { font-size: 22rpx; color: var(--color-text-tertiary); }
/* 岗位列表 */
.position-list { display: flex; flex-direction: column; gap: 16rpx; }
.pos-item {
padding: 24rpx 28rpx; border-radius: var(--radius-lg);
display: flex; align-items: center; justify-content: space-between;
}
.pos-item:active { transform: scale(0.98); }
.pos-left { display: flex; align-items: center; gap: 16rpx; flex: 1; min-width: 0; }
.pos-icon-wrap { width: 56rpx; height: 56rpx; background: #F3F4F6; border-radius: 14rpx; display: flex; align-items: center; justify-content: center; flex-shrink: 0; }
.pos-icon { font-size: 28rpx; }
.pos-body { flex: 1; min-width: 0; }
.pos-name { font-size: 28rpx; font-weight: 600; color: var(--color-text); display: block; }
.pos-meta-row { display: flex; align-items: center; gap: 10rpx; margin-top: 6rpx; }
.pos-company { font-size: 22rpx; color: var(--color-text-tertiary); }
.pos-badge { font-size: 18rpx; padding: 2rpx 12rpx; border-radius: var(--radius-round); font-weight: 500; }
.badge-ai { background: #FEF3C7; color: #92400E; }
.badge-traditional { background: #EEF2FF; color: var(--color-primary); }
.pos-right { text-align: right; flex-shrink: 0; margin-left: 16rpx; }
.pos-salary { font-size: 24rpx; color: var(--color-primary); font-weight: 600; display: block; margin-bottom: 4rpx; }
.pos-action { font-size: 22rpx; color: var(--color-text-tertiary); }
/* 空状态 */
.empty { text-align: center; padding: 80rpx 0; }
.empty-icon { font-size: 56rpx; display: block; margin-bottom: 16rpx; }
.empty-text { font-size: 28rpx; color: var(--color-text-secondary); }
/* 加载 */
.loading-bar { text-align: center; padding: 60rpx; }
.loading-text { font-size: 24rpx; color: var(--color-text-tertiary); }
</style> </style>
+165 -17
View File
@@ -66,29 +66,79 @@
</view> </view>
</view> </view>
<!-- 岗位选择弹窗 -->
<view class="modal-overlay" v-if="showPositionPicker" @click="showPositionPicker = false">
<view class="modal-content" @click.stop>
<text class="modal-title">选择面试岗位</text>
<view class="pos-list">
<view class="pos-option" v-for="(pos, idx) in positions" :key="idx" @click="selectPosition(pos)">
<text class="pos-name">{{ pos.name }}</text>
<text class="pos-arrow"></text>
</view>
<view class="pos-option" v-if="positions.length === 0 && !positionsLoading">
<text class="pos-name disabled">暂无可用岗位</text>
</view>
<view class="pos-option" v-if="positionsLoading">
<text class="pos-name disabled">加载中...</text>
</view>
</view>
<text class="modal-close" @click="showPositionPicker = false">取消</text>
</view>
</view>
<view class="disclaimer-bar" v-if="!isComplete">AI 生成内容仅供参考请核实重要信息</view> <view class="disclaimer-bar" v-if="!isComplete">AI 生成内容仅供参考请核实重要信息</view>
<view class="complete-bar" v-else> <view class="complete-bar" v-else>
<button class="cta-btn" @click="goResult">查看面试报告</button> <button class="cta-btn" @click="goResult">查看面试报告</button>
<button class="buy-btn" v-if="completedReason === 'noCredits'" @click="goH5Buy">引力值不足官网购买 </button>
</view>
<!-- 官网购买弹窗 -->
<view class="modal-overlay" v-if="showH5BuyModal" @click="showH5BuyModal = false">
<view class="modal-content" @click.stop>
<text class="modal-title">引力值不足</text>
<text class="modal-hint">您的引力值不足请补充后继续面试每次面试消耗 5 引力值</text>
<view class="purchase-options">
<view class="purchase-option" @click="goH5BuyAndClose">
<text class="purchase-name">官网购买引力值</text>
<text class="purchase-price">前往网页版充值</text>
<text class="purchase-desc">打开官网 H5 页面支持多种支付方式</text>
</view>
</view>
<text class="modal-close" @click="showH5BuyModal = false">取消</text>
</view>
</view> </view>
</view> </view>
</template> </template>
<script setup> <script setup>
import { ref, computed, onMounted, onBeforeUnmount, nextTick } from 'vue' import { ref, computed, onMounted, onBeforeUnmount, nextTick } from 'vue'
// #ifdef MP-WEIXIN
import { onLoad, onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
// #endif
// #ifndef MP-WEIXIN
import { onLoad } from '@dcloudio/uni-app' import { onLoad } from '@dcloudio/uni-app'
// #endif
import { api, API_ENDPOINTS } from '../../config' import { api, API_ENDPOINTS } from '../../config'
import DigitalHuman from '../../components/digital-human.vue' import DigitalHuman from '../../components/digital-human.vue'
// #ifdef MP-WEIXIN
onShareAppMessage(() => ({ title: '职引 - AI模拟面试 | 数字人面试官实战练习', path: '/pages/interview/interview' }))
onShareTimeline(() => ({ title: '职引 - AI模拟面试 | 数字人面试官实战练习' }))
// #endif
const messages = ref([{ role: 'ai', content: '你好!我是 AI 面试官,请选择岗位开始模拟面试!' }]) const messages = ref([{ role: 'ai', content: '你好!我是 AI 面试官,请选择岗位开始模拟面试!' }])
const inputText = ref('') const inputText = ref('')
const aiLoading = ref(false) const aiLoading = ref(false)
const interviewId = ref('') const interviewId = ref('')
const answeredCount = ref(0) const answeredCount = ref(0)
const isComplete = ref(false) const isComplete = ref(false)
const completedReason = ref('')
const scrollToId = ref('') const scrollToId = ref('')
const position = ref('') const position = ref('')
const avatarMode = ref(true) const avatarMode = ref(true)
const showPositionPicker = ref(false)
const showH5BuyModal = ref(false)
const positions = ref([])
const positionsLoading = ref(false)
const aiSpeechText = ref('') const aiSpeechText = ref('')
const aiAudioUrl = ref('') const aiAudioUrl = ref('')
const aiAmplitudeData = ref([]) const aiAmplitudeData = ref([])
@@ -100,12 +150,13 @@ let recorder = null
let timerSeconds = 0 let timerSeconds = 0
let timerInterval = null let timerInterval = null
const progressPercent = computed(() => Math.min((answeredCount.value / 5) * 100, 100)) let MAX_QUESTIONS = 10
const progressPercent = computed(() => Math.min((answeredCount.value / MAX_QUESTIONS) * 100, 100))
const formatTime = computed(() => { const formatTime = computed(() => {
const m = Math.floor(timerSeconds / 60); const s = timerSeconds % 60 const m = Math.floor(timerSeconds / 60); const s = timerSeconds % 60
return `${String(m).padStart(2, '0')}:${String(s).padStart(2, '0')}` return `${String(m).padStart(2, '0')}:${String(s).padStart(2, '0')}`
}) })
const token = computed(() => uni.getStorageSync('token') || '') const token = () => uni.getStorageSync('token') || ''
onLoad((options) => { onLoad((options) => {
if (options?.position) { if (options?.position) {
@@ -115,9 +166,37 @@ onLoad((options) => {
} }
}) })
/** 加载热门岗位列表 */
const loadPositions = async () => {
positionsLoading.value = true
try {
const res = await uni.request({ url: api('/positions/hot'), method: 'GET' })
if (res.statusCode >= 200 && res.statusCode < 300 && Array.isArray(res.data)) {
positions.value = res.data
} else if (res.data?.data && Array.isArray(res.data.data)) {
positions.value = res.data.data
}
} catch (e) { console.error('加载岗位列表失败', e) }
finally { positionsLoading.value = false }
}
/** 用户选择岗位后开始面试 */
const selectPosition = (pos) => {
position.value = pos.name
showPositionPicker.value = false
messages.value = [{ role: 'ai', content: `你好!我是你的专属 ${pos.name} 面试官,准备好了就开始吧!` }]
startInterview()
}
onMounted(() => { onMounted(() => {
timerInterval = setInterval(() => timerSeconds++, 1000) timerInterval = setInterval(() => timerSeconds++, 1000)
if (token.value) startInterview() if (!position.value) {
// 未传入岗位,展示选择弹窗(无论是否登录)
loadPositions()
showPositionPicker.value = true
} else if (token()) {
startInterview()
}
}) })
onBeforeUnmount(() => { onBeforeUnmount(() => {
@@ -125,7 +204,7 @@ onBeforeUnmount(() => {
}) })
const checkLogin = () => { const checkLogin = () => {
if (!token.value) { if (!token()) {
uni.showModal({ uni.showModal({
title: '请先登录', content: '登录后即可开始 AI 模拟面试', confirmText: '去登录', title: '请先登录', content: '登录后即可开始 AI 模拟面试', confirmText: '去登录',
success: (r) => { if (r.confirm) uni.navigateTo({ url: '/pages/login/login' }) }, success: (r) => { if (r.confirm) uni.navigateTo({ url: '/pages/login/login' }) },
@@ -141,18 +220,27 @@ const startInterview = async () => {
try { try {
const res = await uni.request({ const res = await uni.request({
url: api('/interview/create'), method: 'POST', url: api('/interview/create'), method: 'POST',
header: { 'Authorization': `Bearer ${token.value}`, 'Content-Type': 'application/json' }, header: { 'Authorization': `Bearer ${token()}`, 'Content-Type': 'application/json' },
data: { position: position.value }, data: { position: position.value },
}) })
if (res.statusCode === 200 && res.data) { if (res.statusCode === 200 && res.data) {
interviewId.value = res.data.id interviewId.value = res.data.id
messages.value = res.data.messages || messages.value messages.value = res.data.messages || messages.value
answeredCount.value = res.data.questionCount || 0 answeredCount.value = res.data.questionCount || 0
if (res.data.totalQuestions) MAX_QUESTIONS = res.data.totalQuestions
// Speak first question in avatar mode // Speak first question in avatar mode
if (avatarMode.value && res.data.messages?.length) { if (avatarMode.value && res.data.messages?.length) {
const last = res.data.messages[res.data.messages.length - 1] const last = res.data.messages[res.data.messages.length - 1]
if (last?.role === 'ai') await speakAiText(last.content) if (last?.role === 'ai') await speakAiText(last.content)
} }
} else if (res.statusCode === 403) {
const errMsg = res.data?.message || '面试次数已用完'
messages.value.push({ role: 'ai', content: errMsg + ' 👉 购买后可继续面试' })
isComplete.value = true
completedReason.value = 'noCredits'
} else {
const msg = res.data?.message || '创建面试失败'
messages.value.push({ role: 'ai', content: msg })
} }
} catch { } catch {
messages.value.push({ role: 'ai', content: '创建面试失败,请重试' }) messages.value.push({ role: 'ai', content: '创建面试失败,请重试' })
@@ -164,8 +252,11 @@ const startInterview = async () => {
const sendAnswer = async () => { const sendAnswer = async () => {
if (!inputText.value.trim() || aiLoading.value || isComplete.value) return if (!inputText.value.trim() || aiLoading.value || isComplete.value) return
if (!token.value) { checkLogin(); return } if (!token()) { checkLogin(); return }
if (!interviewId.value) { await startInterview(); return } if (!interviewId.value) {
await startInterview()
if (!interviewId.value) return // creation failed, don't discard answer
}
const answer = inputText.value.trim() const answer = inputText.value.trim()
messages.value.push({ role: 'user', content: answer }) messages.value.push({ role: 'user', content: answer })
@@ -176,19 +267,26 @@ const sendAnswer = async () => {
try { try {
const res = await uni.request({ const res = await uni.request({
url: api(`/interview/${interviewId.value}/answer`), method: 'POST', url: api(`/interview/${interviewId.value}/answer`), method: 'POST',
header: { 'Authorization': `Bearer ${token.value}`, 'Content-Type': 'application/json' }, header: { 'Authorization': `Bearer ${token()}`, 'Content-Type': 'application/json' },
data: avatarMode.value ? { answer, avatar: true } : { answer }, data: avatarMode.value ? { answer, avatar: true } : { answer },
}) })
if (res.statusCode === 200 && res.data?.messages) { if (res.statusCode === 200 && res.data?.messages) {
const aiMsg = res.data.messages.find(m => m.role === 'ai') const aiMsg = res.data.messages.find(m => m.role === 'ai')
messages.value.push(...res.data.messages) // Only push AI messages from response to avoid duplicating the user message already added above
const newAiMessages = res.data.messages.filter(m => m.role === 'ai')
if (newAiMessages.length > 0) messages.value.push(...newAiMessages)
if (avatarMode.value && aiMsg) { if (avatarMode.value && aiMsg) {
await speakAiText(aiMsg.content, res.data.ttsHash, res.data.ttsAmplitude) await speakAiText(aiMsg.content, res.data.ttsHash, res.data.ttsAmplitude)
} }
answeredCount.value = res.data.questionCount || answeredCount.value + 1 answeredCount.value = res.data.questionCount || answeredCount.value + 1
if (res.data.ttsHash && !avatarMode.value) { if (res.data.totalQuestions) MAX_QUESTIONS = res.data.totalQuestions
// Still got TTS but not in avatar mode, just show text } else if (res.statusCode === 403) {
} const errMsg = res.data?.message || '面试次数已用完'
messages.value.push({ role: 'ai', content: errMsg + ' 👉 购买后可继续面试' })
isComplete.value = true
completedReason.value = 'noCredits'
} else {
messages.value.push({ role: 'ai', content: res.data?.message || '回答提交失败' })
} }
} catch { } catch {
messages.value.push({ role: 'ai', content: '回答提交失败,请重试' }) messages.value.push({ role: 'ai', content: '回答提交失败,请重试' })
@@ -207,7 +305,7 @@ async function speakAiText(text, ttsHash, ttsAmplitude) {
try { try {
const synthRes = await uni.request({ const synthRes = await uni.request({
url: api('/tts/synthesize'), method: 'POST', url: api('/tts/synthesize'), method: 'POST',
header: { 'Content-Type': 'application/json' }, header: { 'Content-Type': 'application/json', 'Authorization': `Bearer ${token()}` },
data: { text }, data: { text },
}) })
if (synthRes.statusCode === 200 && synthRes.data?.hash) { if (synthRes.statusCode === 200 && synthRes.data?.hash) {
@@ -227,6 +325,27 @@ function onAvatarSilent() {
} }
const goResult = () => uni.navigateTo({ url: `/pages/report/report?interviewId=${interviewId.value}` }) const goResult = () => uni.navigateTo({ url: `/pages/report/report?interviewId=${interviewId.value}` })
// 官网购买引力值
const goH5Buy = () => {
showH5BuyModal.value = true
}
const goH5BuyAndClose = () => {
showH5BuyModal.value = false
const token = uni.getStorageSync('token') || ''
const url = `https://zhiyin.yzrcloud.cn/?buy=gravity${token ? '&token=' + token : ''}`
// #ifdef MP-WEIXIN
uni.setClipboardData({
data: url,
success: () => {
uni.showToast({ title: '链接已复制,请在手机浏览器中打开', icon: 'none', duration: 3000 })
},
fail: () => {
uni.showToast({ title: '复制失败,请手动访问 zhiyin.yzrcloud.cn', icon: 'none', duration: 3000 })
},
})
// #endif
}
const scrollToBottom = () => { const scrollToBottom = () => {
nextTick(() => { scrollToId.value = ''; setTimeout(() => { scrollToId.value = 'msg-bottom' }, 100) }) nextTick(() => { scrollToId.value = ''; setTimeout(() => { scrollToId.value = 'msg-bottom' }, 100) })
} }
@@ -240,12 +359,17 @@ const confirmExit = () => {
function startRecord() { function startRecord() {
if (aiLoading.value || isComplete.value) return if (aiLoading.value || isComplete.value) return
// #ifdef MP-WEIXIN
isRecording.value = true isRecording.value = true
recorder = uni.getRecorderManager() recorder = uni.getRecorderManager()
recorder.onStart(() => {}) recorder.onStart(() => {})
recorder.onError(() => { isRecording.value = false }) recorder.onError(() => { isRecording.value = false })
recorder.start({ format: 'mp3' }) recorder.start({ format: 'mp3' })
uni.vibrateShort({ type: 'medium' }) uni.vibrateShort({ type: 'medium' })
// #endif
// #ifndef MP-WEIXIN
uni.showToast({ title: '语音输入仅支持小程序', icon: 'none' })
// #endif
} }
function stopRecord() { function stopRecord() {
@@ -260,9 +384,8 @@ function stopRecord() {
url: api(API_ENDPOINTS.TTS.ASR), url: api(API_ENDPOINTS.TTS.ASR),
filePath: audioPath, filePath: audioPath,
name: 'audio', name: 'audio',
header: { 'Authorization': `Bearer ${token.value}` }, header: { 'Authorization': `Bearer ${token()}` },
}) })
console.log('[ASR] upload response:', uploadRes.statusCode, typeof uploadRes.data === 'string' ? uploadRes.data.slice(0, 200) : JSON.stringify(uploadRes.data).slice(0, 200))
if (uploadRes.statusCode === 200 && uploadRes.data) { if (uploadRes.statusCode === 200 && uploadRes.data) {
const data = typeof uploadRes.data === 'string' ? JSON.parse(uploadRes.data) : uploadRes.data const data = typeof uploadRes.data === 'string' ? JSON.parse(uploadRes.data) : uploadRes.data
if (data.text) { if (data.text) {
@@ -371,7 +494,32 @@ function stopRecord() {
.send-btn.disabled { background: var(--color-border); } .send-btn.disabled { background: var(--color-border); }
.send-icon { font-size: 32rpx; color: #FFFFFF; transform: translateY(2rpx); } .send-icon { font-size: 32rpx; color: #FFFFFF; transform: translateY(2rpx); }
.complete-bar { background: #FFFFFF; padding: 20rpx 24rpx; padding-bottom: calc(20rpx + var(--safe-bottom)); } .complete-bar { background: #FFFFFF; padding: 20rpx 24rpx; padding-bottom: calc(20rpx + var(--safe-bottom)); display: flex; flex-direction: column; gap: 16rpx; }
.cta-btn { width: 100%; height: 88rpx; line-height: 88rpx; background: linear-gradient(135deg, var(--color-gradient-start), var(--color-gradient-mid)); color: #FFFFFF; border-radius: var(--radius-md); font-size: 28rpx; font-weight: 600; border: none; } .cta-btn { width: 100%; height: 88rpx; line-height: 88rpx; background: linear-gradient(135deg, var(--color-gradient-start), var(--color-gradient-mid)); color: #FFFFFF; border-radius: var(--radius-md); font-size: 28rpx; font-weight: 600; border: none; }
.buy-btn { width: 100%; height: 72rpx; line-height: 72rpx; background: #FEF3C7; color: #92400E; border-radius: var(--radius-md); font-size: 26rpx; font-weight: 600; border: 2rpx solid #F59E0B; }
.disclaimer-bar { text-align: center; font-size: 20rpx; color: var(--color-text-tertiary); padding: 8rpx 24rpx; background: #FFFFFF; border-top: 1rpx solid var(--color-border); } .disclaimer-bar { text-align: center; font-size: 20rpx; color: var(--color-text-tertiary); padding: 8rpx 24rpx; background: #FFFFFF; border-top: 1rpx solid var(--color-border); }
/* 岗位选择弹窗 */
.modal-overlay { position: fixed; top: 0; left: 0; right: 0; bottom: 0; background: rgba(0,0,0,0.45); z-index: 999; display: flex; align-items: center; justify-content: center; }
.modal-content { background: #FFF; border-radius: 20rpx; width: 600rpx; max-height: 70vh; display: flex; flex-direction: column; align-items: center; padding: 40rpx 32rpx 32rpx; }
.modal-title { font-size: 30rpx; font-weight: 800; color: var(--color-text); margin-bottom: 24rpx; }
.pos-list { width: 100%; max-height: 440rpx; overflow-y: auto; }
.pos-option { display: flex; align-items: center; justify-content: space-between; padding: 24rpx 16rpx; border-bottom: 1rpx solid #F3F4F6; }
.pos-option:active { background: #F9FAFB; border-radius: 12rpx; }
.pos-name { font-size: 28rpx; color: var(--color-text); }
.pos-name.disabled { color: #9CA3AF; }
.pos-arrow { font-size: 32rpx; color: #9CA3AF; }
.modal-close { margin-top: 24rpx; font-size: 26rpx; color: #9CA3AF; padding: 12rpx 32rpx; }
/* 购买弹窗 */
.modal-hint { font-size: 24rpx; color: #6B7280; text-align: center; margin-bottom: 28rpx; padding: 0 16rpx; }
.purchase-options { width: 100%; display: flex; flex-direction: column; gap: 16rpx; }
.purchase-option { background: #F9FAFB; border-radius: var(--radius-md); padding: 24rpx; border: 2rpx solid #E5E7EB; }
.purchase-option.recommended { background: #FFFBEB; border-color: #F59E0B; }
.purchase-option:active { transform: scale(0.97); }
.purchase-badge { font-size: 18rpx; color: #FFF; background: #F59E0B; padding: 2rpx 12rpx; border-radius: 6rpx; align-self: flex-start; margin-bottom: 8rpx; }
.purchase-name { font-size: 28rpx; font-weight: 700; color: var(--color-text); display: block; }
.purchase-price { font-size: 36rpx; font-weight: 800; color: var(--color-primary); margin-top: 8rpx; }
.purchase-unit { font-size: 22rpx; font-weight: 400; color: #9CA3AF; }
.purchase-desc { font-size: 22rpx; color: #6B7280; margin-top: 6rpx; }
</style> </style>
+34 -14
View File
@@ -87,15 +87,18 @@
<view class="card" v-if="mainTab === 'wechat' && isMp"> <view class="card" v-if="mainTab === 'wechat' && isMp">
<text class="card-title">微信一键登录</text> <text class="card-title">微信一键登录</text>
<text class="card-sub">授权后自动创建账号</text> <text class="card-sub">授权后自动创建账号</text>
<button class="login-btn wx-btn" @click="doWxLogin">{{ wxLoading ? '登录中...' : '微信一键登录' }}</button> <button class="login-btn wx-btn" :disabled="wxLoading" @click="doWxLogin">{{ wxLoading ? '登录中...' : '微信一键登录' }}</button>
</view> </view>
<!-- 法律声明 --> <!-- 法律声明 - 用户自主勾选同意 -->
<view class="legal"> <view class="legal">
<text class="legal-text">登录即表示同意</text> <label class="agree-label">
<text class="legal-link" @click="goAgreement">用户协议</text> <checkbox :checked="agreed" @tap="agreed = !agreed" color="#4F46E5" style="transform:scale(0.85)" />
<text class="legal-text"></text> <text class="agree-text">我已阅读并同意</text>
<text class="legal-link" @click="goPrivacy">隐私政策</text> <text class="legal-link" @tap.stop="goAgreement">用户服务协议</text>
<text class="agree-text"></text>
<text class="legal-link" @tap.stop="goPrivacy">隐私政策</text>
</label>
</view> </view>
</view> </view>
@@ -115,8 +118,18 @@
<script setup> <script setup>
import { ref, computed, onMounted, onBeforeUnmount } from 'vue' import { ref, computed, onMounted, onBeforeUnmount } from 'vue'
// #ifdef MP-WEIXIN
import { onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
// #endif
import { api } from '../../config' import { api } from '../../config'
// #ifdef MP-WEIXIN
onShareAppMessage(() => ({ title: '职引 - 登录注册 | AI模拟面试', path: '/pages/login/login' }))
onShareTimeline(() => ({ title: '职引 - 登录注册 | AI模拟面试' }))
// #endif
const agreed = ref(false)
const mainTab = ref('login') const mainTab = ref('login')
const loginMode = ref('password') // 'password' | 'code' const loginMode = ref('password') // 'password' | 'code'
const isMp = ref(false) const isMp = ref(false)
@@ -147,10 +160,14 @@ onMounted(() => {
// #endif // #endif
}) })
onBeforeUnmount(() => { if (timer) clearInterval(timer) }) onBeforeUnmount(() => { if (timer) { clearTimeout(timer); timer = null } })
// 辅助 // 辅助
const showToast = (title, icon = 'none') => uni.showToast({ title, icon }) const showToast = (title, icon = 'none') => uni.showToast({ title, icon })
const checkAgreed = () => {
if (!agreed.value) { showToast('请阅读并同意《用户服务协议》和《隐私政策》'); return false }
return true
}
const loginSuccess = (data) => { const loginSuccess = (data) => {
uni.setStorageSync('token', data.token) uni.setStorageSync('token', data.token)
if (data.user) uni.setStorageSync('userInfo', JSON.stringify(data.user)) if (data.user) uni.setStorageSync('userInfo', JSON.stringify(data.user))
@@ -160,7 +177,7 @@ const loginSuccess = (data) => {
// ====== 密码登录 ====== // ====== 密码登录 ======
const doPasswordLogin = async () => { const doPasswordLogin = async () => {
if (!canPasswordLogin.value) return if (!canPasswordLogin.value || !checkAgreed()) return
pwdLoading.value = true pwdLoading.value = true
try { try {
const res = await uni.request({ const res = await uni.request({
@@ -192,6 +209,9 @@ const sendEmailCode = () => {
emailSent.value = true emailSent.value = true
showToast('验证码已发送', 'success') showToast('验证码已发送', 'success')
startCooldown() startCooldown()
if (res.data?.devCode) {
uni.showModal({ title: '开发模式', content: `验证码:${res.data.devCode}`, showCancel: false })
}
} else { } else {
const msg = (res.data && res.data.message) || '发送失败' const msg = (res.data && res.data.message) || '发送失败'
showToast(msg) showToast(msg)
@@ -219,7 +239,7 @@ const startCooldown = () => {
} }
const doEmailLogin = async () => { const doEmailLogin = async () => {
if (!emailCode.value) return if (!emailCode.value || !checkAgreed()) return
emailLoading.value = true emailLoading.value = true
try { try {
const res = await uni.request({ const res = await uni.request({
@@ -242,7 +262,7 @@ const doEmailLogin = async () => {
// ====== 注册 ====== // ====== 注册 ======
const doRegister = async () => { const doRegister = async () => {
if (!canRegister.value) return if (!canRegister.value || !checkAgreed()) return
regLoading.value = true regLoading.value = true
try { try {
const res = await uni.request({ const res = await uni.request({
@@ -284,10 +304,10 @@ const skipSetPwd = () => { showSetPwd.value = false }
// ====== 微信登录 ====== // ====== 微信登录 ======
const doWxLogin = async () => { const doWxLogin = async () => {
// #ifdef MP-WEIXIN // #ifdef MP-WEIXIN
if (!checkAgreed()) { wxLoading.value = false; return }
wxLoading.value = true wxLoading.value = true
try { try {
const wxResp = await uni.login() const wxResp = await uni.login()
console.log('[wxLogin] uni.login success:', JSON.stringify(wxResp).slice(0, 300))
const { code, errMsg } = wxResp const { code, errMsg } = wxResp
if (!code) { console.error('[wxLogin] no code:', errMsg); showToast('获取微信凭证失败'); return } if (!code) { console.error('[wxLogin] no code:', errMsg); showToast('获取微信凭证失败'); return }
const res = await uni.request({ const res = await uni.request({
@@ -295,7 +315,6 @@ const doWxLogin = async () => {
header: { 'Content-Type': 'application/json' }, header: { 'Content-Type': 'application/json' },
data: { code }, data: { code },
}) })
console.log('[wxLogin] server response:', res.statusCode, JSON.stringify(res.data).slice(0, 500))
if (res.statusCode === 200 && res.data?.token) { if (res.statusCode === 200 && res.data?.token) {
loginSuccess(res.data) loginSuccess(res.data)
} else { } else {
@@ -354,8 +373,9 @@ const goPrivacy = () => uni.navigateTo({ url: '/pages/privacy/privacy' })
.switch-hint { text-align: center; font-size: 22rpx; color: var(--color-primary); padding: 20rpx 0 4rpx; } .switch-hint { text-align: center; font-size: 22rpx; color: var(--color-primary); padding: 20rpx 0 4rpx; }
/* ===== Legal ===== */ /* ===== Legal ===== */
.legal { display: flex; justify-content: center; align-items: center; gap: 4rpx; margin-top: 24rpx; flex-wrap: wrap; } .legal { display: flex; justify-content: center; align-items: center; margin-top: 24rpx; flex-wrap: wrap; }
.legal-text { font-size: 22rpx; color: var(--color-text-tertiary); } .agree-label { display: flex; align-items: center; gap: 6rpx; flex-wrap: wrap; justify-content: center; }
.agree-text { font-size: 22rpx; color: var(--color-text-tertiary); }
.legal-link { font-size: 22rpx; color: var(--color-primary); } .legal-link { font-size: 22rpx; color: var(--color-primary); }
/* ===== Password Modal ===== */ /* ===== Password Modal ===== */
+140 -248
View File
@@ -1,337 +1,229 @@
<template> <template>
<!-- #ifdef MP-WEIXIN -->
<view class="page fade-in">
<view class="placeholder-wrap">
<text class="placeholder-icon"></text>
<text class="placeholder-text">功能已整合到各模块</text>
<text class="placeholder-hint">请返回使用引力值充值功能</text>
<text class="placeholder-back" @click="goBack">返回首页</text>
</view>
</view>
<!-- #endif -->
<!-- #ifdef H5 -->
<view class="page fade-in"> <view class="page fade-in">
<view class="hero"> <view class="hero">
<text class="hero-title">会员中心</text> <text class="hero-icon"></text>
<text class="hero-sub" v-if="isLoggedIn"> <text class="hero-title">补充引力值</text>
当前{{ currentPlanName }} <text class="hero-desc">购买后可获得相应引力值用于面试简历优化下载</text>
</text>
<text class="hero-sub" v-else>选择套餐解锁全部功能</text>
</view> </view>
<view class="plans"> <view class="product-card">
<!-- 免费版 --> <view class="qty-section">
<view class="plan-card free" :class="{ active: plan === 'free' && isLoggedIn }"> <text class="section-label">购买数量</text>
<view class="plan-header"> <view class="qty-controls">
<text class="plan-name">免费版</text> <text class="qty-btn" :class="{ disabled: buyQty <= 1 }" @click="changeQty(-1)"></text>
<view class="plan-price"><text class="price-num">免费</text></view> <input class="qty-input" type="number" v-model.number="buyQty" min="1" max="99" @blur="clampQty" />
<text class="qty-btn" :class="{ disabled: buyQty >= 99 }" @click="changeQty(1)">+</text>
</view> </view>
<view class="plan-features">
<text class="feat" v-for="f in freeFeatures" :key="f"> {{ f }}</text>
</view>
<view class="plan-status" v-if="isLoggedIn && plan === 'free'">当前使用</view>
<view class="plan-status hint" v-else-if="!isLoggedIn">注册即用</view>
</view> </view>
<!-- 成长版 --> <view class="summary">
<view class="plan-card growth recommended" :class="{ active: plan === 'growth' && isLoggedIn }"> <view class="summary-row">
<view class="plan-badge"> 推荐</view> <text class="summary-label">单价</text>
<view class="plan-header"> <text class="summary-val">¥{{ (unitPrice / 100).toFixed(1) }} / </text>
<text class="plan-name">成长版</text>
<text class="plan-price"><text class="price-num">{{ growthPriceText }}</text><text class="price-unit">/</text></text>
</view> </view>
<view class="plan-features"> <view class="summary-row">
<text class="feat" v-for="f in growthFeatures" :key="f"> {{ f }}</text> <text class="summary-label">可得引力值</text>
<text class="summary-val highlight">{{ buyQty * gravityPerUnit }} 引力值</text>
</view>
<view class="summary-row total">
<text class="summary-label">合计</text>
<text class="summary-val total-price">¥{{ (buyQty * unitPrice / 100).toFixed(2) }}</text>
</view> </view>
<view class="plan-action" v-if="!isLoggedIn" @click="goLogin">登录后开通</view>
<view class="plan-action owned" v-else-if="plan !== 'free'"> 已开通</view>
<view class="plan-action" v-else @click="startPay('growth')">{{ growthPriceText }} 立即开通</view>
</view> </view>
<!-- 冲刺版 --> <button class="buy-btn" :disabled="payLoading" @click="startPay">
<view class="plan-card sprint" :class="{ active: plan === 'sprint' && isLoggedIn }"> <text v-if="!payLoading">立即购买</text>
<view class="plan-badge sprint-badge">🚀 冲刺</view> <text v-else>处理中...</text>
<view class="plan-header"> </button>
<text class="plan-name">冲刺版</text>
<text class="plan-price"><text class="price-num price-sprint">¥49.9</text><text class="price-unit">/</text></text>
</view>
<view class="plan-features">
<text class="feat" v-for="f in sprintFeatures" :key="f"> {{ f }}</text>
</view>
<view class="plan-action" v-if="!isLoggedIn" @click="goLogin">登录后开通</view>
<view class="plan-action owned" v-else-if="plan === 'sprint'"> 已开通</view>
<view class="plan-action" v-else-if="plan === 'growth'" @click="startPay('sprint')">升级至冲刺版</view>
<view class="plan-action" v-else @click="startPay('sprint')">¥49.9/ 立即开通</view>
</view>
</view> </view>
<!-- 支付弹窗 --> <!-- 支付弹窗 -->
<view class="modal-overlay" v-if="showPayModal" @click="cancelPay"> <view class="modal-overlay" v-if="showPayModal" @click="cancelPay">
<view class="modal-content" @click.stop> <view class="modal-content" @click.stop>
<template v-if="payLoading"> <template v-if="payLoading">
<text class="modal-title">正在创建支付...</text> <text class="modal-title">正在创建订单...</text>
</template> </template>
<template v-else-if="!isMp && payCodeUrl"> <template v-else-if="payCodeUrl">
<text class="modal-title">微信扫码支付</text> <text class="modal-title">微信扫码支付</text>
<canvas canvas-id="payQrcode" class="qr-canvas"></canvas> <image class="qrcode" :src="'https://api.qrserver.com/v1/create-qr-code/?size=300x300&data=' + encodeURIComponent(payCodeUrl)" mode="widthFix" />
<text class="modal-hint">请用微信扫码完成支付</text> <text class="modal-hint">使用微信扫描二维码完成支付</text>
<text class="modal-hint">支付成功后将自动跳转</text>
<text class="modal-close" @click="cancelPay">取消支付</text> <text class="modal-close" @click="cancelPay">取消支付</text>
</template> </template>
<template v-else-if="isMp && !payLoading"> <template v-else-if="paySuccess">
<text class="modal-title">微信支付</text> <text class="modal-title"> 支付成功</text>
<text class="modal-hint">即将调起微信支付...</text> <text class="modal-hint">引力值已到账返回继续使用吧</text>
<text class="modal-close" @click="cancelPay">关闭</text>
</template> </template>
<template v-if="payError"> <template v-else-if="payError">
<text class="modal-title pay-error">支付异常</text> <text class="modal-title pay-error">支付失败</text>
<text class="modal-hint">{{ payError }}</text> <text class="modal-hint">{{ payError }}</text>
<text class="modal-close" @click="cancelPay">关闭</text> <text class="modal-close" @click="cancelPay">关闭</text>
</template> </template>
</view> </view>
</view> </view>
<!-- 支付中提示 -->
<view class="pay-success" v-if="paySuccess">
<text class="success-icon">🎉</text>
<text class="success-text">开通成功{{ payingPlanName }}已生效</text>
</view>
</view> </view>
<!-- #endif -->
</template> </template>
<script setup> <script setup lang="ts">
import { ref, onMounted, nextTick } from 'vue' import { ref, computed, onMounted } from 'vue'
// #ifdef MP-WEIXIN
import { onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
// #endif
import { api } from '../../config' import { api } from '../../config'
import UQRCode from 'uqrcodejs'
const isLoggedIn = ref(false) // #ifdef MP-WEIXIN
const isMp = ref(false) onShareAppMessage(() => ({ title: '职引 - 引力值购买 | AI模拟面试', path: '/pages/member/member' }))
const plan = ref('free') onShareTimeline(() => ({ title: '职引 - 引力值购买 | AI模拟面试' }))
const currentPlanName = ref('免费版') // #endif
const paySuccess = ref(false)
const goBack = () => uni.switchTab({ url: '/pages/user/user' })
// #ifdef H5
const buyQty = ref(1)
const unitPrice = ref(500)
const gravityPerUnit = ref(5)
const payLoading = ref(false)
const showPayModal = ref(false) const showPayModal = ref(false)
const payCodeUrl = ref('') const payCodeUrl = ref('')
const payLoading = ref(false) const paySuccess = ref(false)
const payError = ref('') const payError = ref('')
const payingPlanName = ref('')
const payingPlan = ref('')
const growthPriceText = ref('¥19.9')
const currentOutTradeNo = ref('')
const freeFeatures = ['每日 2 次 AI 模拟面试', '基础面试报告', '通用题库随机出题', '简历诊断(限 3 次)']
const growthFeatures = [
'免费版全部权益', '无限面试次数', '详细面试报告(四维评分)',
'进步轨迹雷达图 + 打卡', '每日一题推送', '参考回答思路', '公司真题库',
]
const sprintFeatures = [
'成长版全部权益', 'AI 语音分析(语气词/语速检测)', '技能缺口分析报告',
'学习路径推荐', '真人导师 1v1 点评(每月 1 次)', '简历精修(每月 1 次)', '内推优先',
]
const token = () => uni.getStorageSync('token') || ''
onMounted(async () => { onMounted(async () => {
// #ifdef MP-WEIXIN
isMp.value = true
// #endif
const t = token()
if (!t) return
isLoggedIn.value = true
try { try {
const [sres, lres] = await Promise.all([ const res = await uni.request({ url: api('/member/plans'), method: 'GET' })
uni.request({ url: api('/member/status'), method: 'GET', header: { 'Authorization': `Bearer ${t}` } }), if (res.statusCode >= 200 && res.statusCode < 300 && res.data?.products) {
uni.request({ url: api('/member/plans'), method: 'GET' }), const prod = res.data.products.interview
]) if (prod) {
if (sres.statusCode === 200) { unitPrice.value = prod.price || 500
const d = sres.data gravityPerUnit.value = prod.gravity || 5
plan.value = d.plan || 'free'
currentPlanName.value = d.planName || '免费版'
} }
if (lres.statusCode === 200 && lres.data?.price) {
const p = lres.data.price
growthPriceText.value = `¥${(p.monthly / 100).toFixed(1)}`
} }
} catch (e) { /* ignore */ } } catch (e) { /* silent */ }
}) })
const goLogin = () => uni.navigateTo({ url: '/pages/login/login' }) const changeQty = (delta: number) => {
const next = buyQty.value + delta
const cancelPay = () => { if (next >= 1 && next <= 99) buyQty.value = next
showPayModal.value = false }
payCodeUrl.value = '' const clampQty = () => {
payLoading.value = false if (buyQty.value < 1) buyQty.value = 1
payError.value = '' if (buyQty.value > 99) buyQty.value = 99
} }
/** 创建支付订单 */ const startPay = async () => {
const startPay = async (selectedPlan) => { const token = uni.getStorageSync('token') || ''
const t = token() if (!token) { uni.showToast({ title: '请先登录', icon: 'none' }); return }
if (!t) { uni.showToast({ title: '请先登录', icon: 'none' }); return }
payingPlan.value = selectedPlan
// #ifdef MP-WEIXIN
payingPlanName.value = selectedPlan === 'sprint' ? '冲刺版' : '成长版'
// #endif
// #ifndef MP-WEIXIN
payingPlanName.value = selectedPlan === 'sprint' ? '冲刺版' : '成长版'
// #endif
showPayModal.value = true showPayModal.value = true
payLoading.value = true payLoading.value = true
payCodeUrl.value = ''
payError.value = '' payError.value = ''
paySuccess.value = false
const planLabel = selectedPlan || 'growth'
if (isMp.value) {
// 小程序:JSAPI 支付
try { try {
const res = await uni.request({ const res = await uni.request({
url: api('/payment/jsapi'), method: 'POST', url: api('/payment/create-product'), method: 'POST',
data: { plan: planLabel }, data: { type: 'interview', quantity: buyQty.value },
header: { 'Authorization': `Bearer ${t}`, 'Content-Type': 'application/json' }, header: { 'Authorization': `Bearer ${token}`, 'Content-Type': 'application/json' },
}) })
payLoading.value = false payLoading.value = false
if (res.statusCode === 200 && res.data?.payParams) { if (res.statusCode >= 200 && res.statusCode < 300 && res.data?.codeUrl) {
const pp = res.data.payParams
currentOutTradeNo.value = res.data.outTradeNo || ''
// 调起微信支付
uni.requestPayment({
provider: 'wxpay',
timeStamp: pp.timeStamp,
nonceStr: pp.nonceStr,
package: pp.package,
signType: pp.signType || 'RSA',
paySign: pp.paySign,
success: () => pollPayResult(res.data.prepayId, planLabel),
fail: (err) => { payError.value = '支付取消或失败'; uni.showToast({ title: '支付取消', icon: 'none' }) },
})
} else {
payLoading.value = false
payError.value = res.data?.message || '创建订单失败'
uni.showToast({ title: '创建订单失败', icon: 'none' })
}
} catch (e) {
payLoading.value = false
payError.value = '网络错误,请重试'
uni.showToast({ title: '网络错误', icon: 'none' })
}
} else {
// H5:二维码支付
try {
const res = await uni.request({
url: api('/payment/create'), method: 'POST',
data: { plan: planLabel },
header: { 'Authorization': `Bearer ${t}`, 'Content-Type': 'application/json' },
})
payLoading.value = false
if (res.statusCode === 200 && res.data?.codeUrl) {
payCodeUrl.value = res.data.codeUrl payCodeUrl.value = res.data.codeUrl
currentOutTradeNo.value = res.data.outTradeNo pollPayResult(res.data.outTradeNo)
nextTick(() => {
try {
const ctx = uni.createCanvasContext('payQrcode')
const uqrcode = new UQRCode()
uqrcode.data = res.data.codeUrl
uqrcode.size = 400
uqrcode.margin = 20
uqrcode.backgroundColor = '#FFFFFF'
uqrcode.foregroundColor = '#000000'
uqrcode.make()
uqrcode.drawCanvas(ctx)
} catch(e) { console.error('二维码生成失败', e) }
})
// 轮询支付结果
pollPayResult(res.data.outTradeNo, planLabel)
} else { } else {
payError.value = res.data?.message || '支付服务暂不可用' payError.value = res.data?.message || '创建订单失败'
uni.showToast({ title: '支付服务暂不可用', icon: 'none' })
} }
} catch (e) { } catch (e) {
payLoading.value = false payLoading.value = false
payError.value = '网络错误,请重试' payError.value = '网络错误,请重试'
uni.showToast({ title: '网络错误', icon: 'none' })
}
} }
} }
/** 轮询订单状态 */ const pollPayResult = (outTradeNo: string) => {
const pollPayResult = async (outTradeNo, selectedPlan) => {
if (!outTradeNo) return if (!outTradeNo) return
const maxAttempts = 30 const token = uni.getStorageSync('token') || ''
let attempts = 0 let attempts = 0
const poll = async () => { const poll = async () => {
attempts++ attempts++
try { try {
const res = await uni.request({ const res = await uni.request({
url: api(`/payment/check/${outTradeNo}`), method: 'GET', url: api(`/payment/check/${outTradeNo}`), method: 'GET',
header: { 'Authorization': `Bearer ${token()}` }, header: { 'Authorization': `Bearer ${token}` },
}) })
if (res.statusCode === 200 && res.data?.status === 'success') { if (res.statusCode >= 200 && res.statusCode < 300 && res.data?.status === 'success') {
// 支付成功,激活套餐 paySuccess.value = true
await activatePlan(outTradeNo, selectedPlan) payCodeUrl.value = ''
return return
} }
} catch (e) { /* ignore */ } } catch (e) { /* ignore */ }
if (attempts < 30) setTimeout(poll, 2000)
if (attempts < maxAttempts) {
setTimeout(poll, 2000)
} else {
payError.value = '支付结果查询超时,请联系客服'
uni.showToast({ title: '支付查询超时', icon: 'none' })
}
} }
setTimeout(poll, 2000) setTimeout(poll, 2000)
} }
/** 激活套餐 */ const cancelPay = () => {
const activatePlan = async (outTradeNo, selectedPlan) => {
try {
const res = await uni.request({
url: api('/payment/activate'), method: 'POST',
data: { outTradeNo },
header: { 'Authorization': `Bearer ${token()}`, 'Content-Type': 'application/json' },
})
if (res.statusCode === 200 && res.data?.success) {
paySuccess.value = true
showPayModal.value = false showPayModal.value = false
plan.value = selectedPlan === 'sprint' ? 'sprint' : 'growth' payCodeUrl.value = ''
currentPlanName.value = selectedPlan === 'sprint' ? '冲刺版' : '成长版' payError.value = ''
uni.showToast({ title: '🎉 开通成功!', icon: 'success' }) payLoading.value = false
} else {
uni.showToast({ title: res.data?.message || '激活失败', icon: 'none' })
}
} catch (e) {
payError.value = '激活失败,请联系客服'
uni.showToast({ title: '激活失败', icon: 'none' })
}
} }
// #endif
</script> </script>
<style scoped> <style scoped>
.page { min-height: 100vh; background: var(--color-bg); } .page { min-height: 100vh; background: var(--color-bg); }
.hero { background: linear-gradient(135deg, var(--color-gradient-start), var(--color-gradient-mid), var(--color-gradient-end)); padding: 48rpx 32rpx 72rpx; border-radius: 0 0 48rpx 48rpx; } .placeholder-wrap { display: flex; flex-direction: column; align-items: center; gap: 16rpx; padding: 80rpx 40rpx; }
.hero-title { font-size: 40rpx; font-weight: 700; color: #FFFFFF; } .placeholder-icon { font-size: 80rpx; }
.hero-sub { font-size: 22rpx; color: rgba(255,255,255,0.7); margin-top: 8rpx; display: block; } .placeholder-text { font-size: 30rpx; font-weight: 600; color: var(--color-text); }
.plans { padding: 0 32rpx; margin-top: -40rpx; display: flex; flex-direction: column; gap: 24rpx; } .placeholder-hint { font-size: 24rpx; color: var(--color-text-tertiary); }
.plan-card { background: #FFFFFF; border-radius: var(--radius-xl); padding: 32rpx; box-shadow: var(--shadow-sm); position: relative; } .placeholder-back { font-size: 26rpx; color: var(--color-primary); padding: 16rpx 40rpx; border-radius: var(--radius-md); background: #F3F4F6; margin-top: 24rpx; }
.plan-card.growth { border: 2rpx solid var(--color-primary); }
.plan-card.sprint { border: 2rpx solid #F59E0B; }
.plan-card.active { border-color: var(--color-primary); }
.plan-badge { position: absolute; top: -12rpx; right: 24rpx; background: linear-gradient(135deg, var(--color-gradient-start), var(--color-gradient-mid)); color: #FFF; font-size: 20rpx; padding: 4rpx 20rpx; border-radius: var(--radius-round); font-weight: 600; }
.sprint-badge { background: linear-gradient(135deg, #F59E0B, #F97316); }
.plan-header { display: flex; justify-content: space-between; align-items: baseline; margin-bottom: 20rpx; }
.plan-name { font-size: 30rpx; font-weight: 700; color: var(--color-text); }
.price-num { font-size: 44rpx; font-weight: 800; color: var(--color-primary); }
.price-unit { font-size: 22rpx; color: var(--color-text-tertiary); }
.price-sprint { color: #D97706; }
.plan-features { display: flex; flex-direction: column; gap: 10rpx; margin-bottom: 24rpx; }
.feat { font-size: 24rpx; color: var(--color-text-secondary); }
.plan-status { text-align: center; background: #F3F4F6; padding: 14rpx; border-radius: var(--radius-sm); font-size: 24rpx; color: var(--color-text-secondary); }
.plan-status.hint { background: transparent; color: var(--color-text-tertiary); }
.plan-action { text-align: center; background: linear-gradient(135deg, var(--color-gradient-start), var(--color-gradient-mid)); color: #FFF; padding: 20rpx; border-radius: var(--radius-md); font-size: 28rpx; font-weight: 600; }
.plan-action.owned { background: #ECFDF5; color: var(--color-success); }
.pay-success { margin: 24rpx 32rpx; background: #ECFDF5; border-radius: var(--radius-lg); padding: 32rpx; text-align: center; }
.success-icon { font-size: 48rpx; display: block; margin-bottom: 8rpx; }
.success-text { font-size: 28rpx; font-weight: 600; color: var(--color-success); }
/* 弹窗 */ /* H5 购买页 */
.hero { display: flex; flex-direction: column; align-items: center; padding: 48rpx 32rpx 24rpx; }
.hero-icon { font-size: 72rpx; }
.hero-title { font-size: 36rpx; font-weight: 700; color: var(--color-text); margin-top: 12rpx; }
.hero-desc { font-size: 24rpx; color: var(--color-text-secondary); margin-top: 8rpx; text-align: center; }
.product-card { background: #fff; border-radius: var(--radius-lg); margin: 0 32rpx; padding: 32rpx; box-shadow: var(--shadow-sm); }
.qty-section { margin-bottom: 24rpx; }
.section-label { font-size: 26rpx; font-weight: 600; color: var(--color-text); display: block; margin-bottom: 16rpx; }
.qty-controls { display: flex; align-items: center; justify-content: center; gap: 24rpx; }
.qty-btn { width: 64rpx; height: 64rpx; border-radius: 50%; background: #F3F4F6; display: flex; align-items: center; justify-content: center; font-size: 36rpx; font-weight: 500; color: var(--color-text); }
.qty-btn.disabled { color: #D1D5DB; background: #F9FAFB; }
.qty-input { width: 120rpx; height: 72rpx; text-align: center; font-size: 36rpx; font-weight: 700; color: var(--color-text); border: 2rpx solid #E5E7EB; border-radius: var(--radius-sm); }
.summary { margin-bottom: 32rpx; }
.summary-row { display: flex; justify-content: space-between; padding: 12rpx 0; border-bottom: 1rpx solid #F3F4F6; }
.summary-row.total { border-bottom: none; padding-top: 16rpx; }
.summary-label { font-size: 24rpx; color: var(--color-text-secondary); }
.summary-val { font-size: 26rpx; font-weight: 600; color: var(--color-text); }
.summary-val.highlight { color: var(--color-primary); }
.total-price { font-size: 36rpx; font-weight: 800; color: var(--color-primary); }
.buy-btn { width: 100%; height: 88rpx; background: linear-gradient(135deg, var(--color-gradient-start), var(--color-gradient-mid)); color: #fff; font-size: 30rpx; font-weight: 600; border-radius: var(--radius-lg); display: flex; align-items: center; justify-content: center; border: none; }
.buy-btn:active { opacity: 0.85; transform: scale(0.98); }
.buy-btn[disabled] { opacity: 0.5; }
/* 支付弹窗 */
.modal-overlay { position: fixed; top: 0; left: 0; right: 0; bottom: 0; background: rgba(0,0,0,0.5); display: flex; align-items: center; justify-content: center; z-index: 100; } .modal-overlay { position: fixed; top: 0; left: 0; right: 0; bottom: 0; background: rgba(0,0,0,0.5); display: flex; align-items: center; justify-content: center; z-index: 100; }
.modal-content { background: #FFF; border-radius: var(--radius-xl); padding: 40rpx; width: 70%; display: flex; flex-direction: column; align-items: center; gap: 20rpx; } .modal-content { background: #FFF; border-radius: var(--radius-xl); padding: 40rpx 32rpx; width: 600rpx; display: flex; flex-direction: column; align-items: center; gap: 16rpx; }
.modal-title { font-size: 28rpx; font-weight: 600; color: var(--color-text); } .modal-title { font-size: 30rpx; font-weight: 700; color: var(--color-text); }
.pay-error { color: var(--color-error); } .pay-error { color: var(--color-error); }
.qr-canvas { width: 400rpx; height: 400rpx; background: #FFF; border-radius: var(--radius-md); } .modal-hint { font-size: 22rpx; color: #6B7280; text-align: center; }
.modal-hint { font-size: 22rpx; color: var(--color-text-tertiary); } .modal-close { font-size: 24rpx; color: #9CA3AF; padding: 12rpx 24rpx; }
.modal-close { font-size: 22rpx; color: var(--color-text-tertiary); padding: 8rpx; } .qrcode { width: 300rpx; height: 300rpx; margin: 8rpx 0; }
</style> </style>
+34 -7
View File
@@ -134,8 +134,19 @@
<script setup> <script setup>
import { ref, onMounted } from 'vue' import { ref, onMounted } from 'vue'
// #ifdef MP-WEIXIN
import { onShow, onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
// #endif
// #ifndef MP-WEIXIN
import { onShow } from '@dcloudio/uni-app'
// #endif
import { api } from '../../config' import { api } from '../../config'
// #ifdef MP-WEIXIN
onShareAppMessage(() => ({ title: '职引 - 进步轨迹 | 能力雷达+打卡日历', path: '/pages/progress/progress' }))
onShareTimeline(() => ({ title: '职引 - 进步轨迹 | 能力雷达+打卡日历' }))
// #endif
const stats = ref({ completedInterviews: 0, avgScore: 0, streak: 0 }) const stats = ref({ completedInterviews: 0, avgScore: 0, streak: 0 })
const progress = ref({ dimensions: {}, interviews: [], recentScores: [] }) const progress = ref({ dimensions: {}, interviews: [], recentScores: [] })
const skillsGap = ref(null) const skillsGap = ref(null)
@@ -155,6 +166,16 @@ onMounted(async () => {
const t = token() const t = token()
if (!t) return if (!t) return
await loadProgressData()
})
onShow(async () => {
if (token()) await loadProgressData()
})
async function loadProgressData() {
const t = token()
if (!t) return
try { try {
// Load progress // Load progress
const res = await uni.request({ const res = await uni.request({
@@ -166,7 +187,7 @@ onMounted(async () => {
progress.value = d progress.value = d
dimensions.value = dimensions.value.map(dim => ({ dimensions.value = dimensions.value.map(dim => ({
...dim, ...dim,
value: d.dimensions?.[dim.key] || Math.round(50 + Math.random() * 30), value: d.dimensions?.[dim.key] || 0,
})) }))
} }
} catch (e) { console.error(e) } } catch (e) { console.error(e) }
@@ -191,23 +212,29 @@ onMounted(async () => {
} }
} catch (e) { console.error(e) } } catch (e) { console.error(e) }
// Build week days buildWeekDays()
}
function buildWeekDays() {
const days = ['日', '一', '二', '三', '四', '五', '六'] const days = ['日', '一', '二', '三', '四', '五', '六']
const today = new Date() const today = new Date()
const arr = [] const arr = []
const checkinDates = (progress.value.checkins || []).map((c) => {
const d = new Date(c.date || c.createdAt)
return `${d.getFullYear()}-${d.getMonth() + 1}-${d.getDate()}`
})
for (let i = 6; i >= 0; i--) { for (let i = 6; i >= 0; i--) {
const d = new Date(today) const d = new Date(today)
d.setDate(d.getDate() - i) d.setDate(d.getDate() - i)
const isToday = i === 0 const key = `${d.getFullYear()}-${d.getMonth() + 1}-${d.getDate()}`
// Mark days with interviews (simulate based on streak)
arr.push({ arr.push({
label: days[d.getDay()], label: days[d.getDay()],
isToday, isToday: i === 0,
done: i < (stats.value.streak || 0), done: checkinDates.includes(key),
}) })
} }
weekDays.value = arr weekDays.value = arr
}) }
const formatDate = (d) => { const formatDate = (d) => {
if (!d) return '' if (!d) return ''
+20 -9
View File
@@ -70,9 +70,19 @@
<script setup> <script setup>
import { ref, nextTick } from 'vue' import { ref, nextTick } from 'vue'
// #ifdef MP-WEIXIN
import { onLoad, onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
// #endif
// #ifndef MP-WEIXIN
import { onLoad } from '@dcloudio/uni-app' import { onLoad } from '@dcloudio/uni-app'
// #endif
import { api } from '../../config' import { api } from '../../config'
// #ifdef MP-WEIXIN
onShareAppMessage(() => ({ title: '职引 - 面试报告与评分 | AI多维评分', path: '/pages/report/report' }))
onShareTimeline(() => ({ title: '职引 - 面试报告与评分 | AI多维评分' }))
// #endif
const loading = ref(true) const loading = ref(true)
const report = ref(null) const report = ref(null)
const dimList = ref([]) const dimList = ref([])
@@ -156,7 +166,9 @@ onLoad(async (options) => {
})) }))
} }
} }
}).catch(() => {}) }).catch(e => {
console.error('[report] auto-complete failed:', e)
})
} }
} }
} catch(e) { console.error(e) } } catch(e) { console.error(e) }
@@ -288,15 +300,12 @@ async function generateCard() {
ctx.setFillStyle('rgba(165,180,252,0.5)') ctx.setFillStyle('rgba(165,180,252,0.5)')
ctx.fillText('扫码开始你的模拟面试 → 在微信搜索"职引"小程序', w / 2, 690) ctx.fillText('扫码开始你的模拟面试 → 在微信搜索"职引"小程序', w / 2, 690)
// QR code hint (simulated) // QR text hint
ctx.setFillStyle('#FFFFFF') ctx.setFillStyle('#FFFFFF')
ctx.setFontSize(12) ctx.setFontSize(16)
ctx.setTextAlign('center') ctx.setTextAlign('center')
ctx.fillText('⬛ ⬛ ⬛ ⬛ ⬛', w / 2, 760) ctx.fillText('在微信搜索「职引」小程序', w / 2, 760)
ctx.fillText('⬛ ⬛ ⬛ ⬛ ⬛', w / 2, 780) ctx.fillText('查看完整面试报告', w / 2, 790)
ctx.fillText('⬛ ⬛ ⬛ ⬛ ⬛', w / 2, 800)
ctx.fillText('⬛ ⬛ ⬛ ⬛ ⬛', w / 2, 820)
ctx.fillText('微信小程序', w / 2, 855)
ctx.draw(false, async () => { ctx.draw(false, async () => {
try { try {
@@ -306,7 +315,9 @@ async function generateCard() {
itemList: ['保存到相册', '分享给好友'], itemList: ['保存到相册', '分享给好友'],
success: (res) => { success: (res) => {
if (res.tapIndex === 0) { if (res.tapIndex === 0) {
uni.saveImageToPhotosAlbum({ filePath: tempRes.tempFilePath }) uni.saveImageToPhotosAlbum({ filePath: tempRes.tempFilePath, success: () => uni.showToast({ title: '已保存到相册', icon: 'success' }) })
} else if (res.tapIndex === 1) {
uni.shareAppMessage ? uni.shareAppMessage({ title: '我的面试报告', imageUrl: tempRes.tempFilePath }) : uni.showToast({ title: '请截图后分享', icon: 'none' })
} }
}, },
}) })
+9 -6
View File
@@ -16,7 +16,7 @@
<text class="score-num">{{ diagnosisResult.score }}</text> <text class="score-num">{{ diagnosisResult.score }}</text>
<text class="score-label">/100</text> <text class="score-label">/100</text>
</view> </view>
<text class="summary-text">{{ diagnosisResult.summary }}</text> <text class="summary-text" v-if="diagnosisResult.summary">{{ diagnosisResult.summary }}</text>
</view> </view>
<!-- 岗位匹配度诊断模式 --> <!-- 岗位匹配度诊断模式 -->
@@ -136,16 +136,19 @@ onLoad(async (options: any) => {
function applyResult(data: any) { function applyResult(data: any) {
loading.value = false; loading.value = false;
if (isOptimize.value) { if (isOptimize.value) {
optimizedContent.value = data.optimizedContent || ''; optimizedContent.value = data.optimized || '';
changes.value = data.changes || []; changes.value = (data.changes || []).map((c: any) =>
highlights.value = data.highlights || []; typeof c === 'string' ? { section: c, description: c } : c
);
highlights.value = [];
} else { } else {
diagnosisResult.value = data; diagnosisResult.value = data;
changes.value = (data.issues || []).map((i: any) => ({ changes.value = (data.issues || []).map((i: any) => ({
...i, ...i,
typeLabel: i.type === 'structure' ? '结构' : i.type === 'content' ? '内容' : i.type === 'keywords' ? '关键词' : i.type === 'achievement' ? '成就' : '格式', typeLabel: i.level === 'high' ? '严重' : i.level === 'medium' ? '中等' : '轻微',
description: i.desc || i.description,
})); }));
highlights.value = data.strengths || []; highlights.value = data.suggestions || [];
} }
} }
+10
View File
@@ -148,9 +148,19 @@
<script setup> <script setup>
import { ref, onMounted } from 'vue' import { ref, onMounted } from 'vue'
// #ifdef MP-WEIXIN
import { onLoad, onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
// #endif
// #ifndef MP-WEIXIN
import { onLoad } from '@dcloudio/uni-app' import { onLoad } from '@dcloudio/uni-app'
// #endif
import { api } from '../../config' import { api } from '../../config'
// #ifdef MP-WEIXIN
onShareAppMessage(() => ({ title: '职引 - 简历诊断与优化 | AI智能优化', path: '/pages/resume/resume' }))
onShareTimeline(() => ({ title: '职引 - 简历诊断与优化 | AI智能优化' }))
// #endif
const currentTab = ref('list') const currentTab = ref('list')
const showForm = ref(false) const showForm = ref(false)
const formTitle = ref('') const formTitle = ref('')
+703
View File
@@ -0,0 +1,703 @@
<template>
<view class="page">
<!-- ====== Mode 1: List View ====== -->
<template v-if="mode === 'list'">
<view class="hero">
<text class="hero-title">面试复盘</text>
<text class="hero-sub">上传录音AI 帮你复盘面试表现</text>
</view>
<view class="stats-bar card" v-if="listData.items.length > 0">
<view class="stat">
<text class="stat-val">{{ listData.total }}</text>
<text class="stat-lbl">复盘次数</text>
</view>
<view class="stat-sep"></view>
<view class="stat">
<text class="stat-val">{{ avgScore }}</text>
<text class="stat-lbl">平均分</text>
</view>
<view class="stat-sep"></view>
<view class="stat">
<text class="stat-val">{{ completedCount }}</text>
<text class="stat-lbl">已完成</text>
</view>
</view>
<view class="list" v-if="listData.items.length > 0">
<view
class="record-card card"
v-for="item in listData.items"
:key="item._id"
@click="viewDetail(item._id)"
>
<view class="record-top">
<view class="record-icon">
{{ item.status === 'completed' ? (item.analysis?.overallScore >= 80 ? '🌟' : '📋') : '⏳' }}
</view>
<view class="record-body">
<view class="record-name">{{ item.position }}</view>
<text class="record-meta">
{{ formatDate(item.createdAt) }}
<template v-if="item.company"> · {{ item.company }}</template>
<template v-if="item.status === 'processing'"> · 分析中...</template>
<template v-if="item.status === 'failed'"> · 分析失败</template>
</text>
</view>
<view class="record-score" v-if="item.status === 'completed'" :class="scoreLevel(item.analysis?.overallScore)">
{{ item.analysis?.overallScore || '--' }}
</view>
<view v-else class="record-score pending">{{ item.status === 'processing' ? '...' : '!' }}</view>
</view>
</view>
<view class="load-more" v-if="hasMore" @click="loadMore">加载更多</view>
</view>
<view class="empty" v-else-if="!loading">
<text class="empty-icon">🎙</text>
<text class="empty-title">暂无复盘记录</text>
<text class="empty-desc">面试后上传录音AI 帮你分析表现</text>
</view>
<view class="fixed-bottom">
<button class="btn-primary" @click="mode = 'upload'">
<text class="btn-icon">+</text> 上传录音复盘
</button>
</view>
</template>
<!-- ====== Mode 2: Upload View ====== -->
<template v-if="mode === 'upload'">
<view class="upload-page">
<view class="upload-header">
<text class="upload-back" @click="mode = 'list'"> 返回</text>
<text class="upload-title">上传面试录音</text>
<view style="width:60rpx"></view>
</view>
<view class="form-card card">
<view class="form-group">
<text class="form-label">面试岗位 <text class="required">*</text></text>
<input class="form-input" v-model="form.position" placeholder="例如:前端开发工程师" />
</view>
<view class="form-group">
<text class="form-label">面试公司</text>
<input class="form-input" v-model="form.company" placeholder="选填" />
</view>
<view class="form-group">
<text class="form-label">上传方式</text>
<view class="tab-bar">
<view class="tab-item" :class="{ active: uploadMode === 'audio' }" @click="uploadMode = 'audio'">录音文件</view>
<view class="tab-item" :class="{ active: uploadMode === 'text' }" @click="uploadMode = 'text'">粘贴文本</view>
</view>
</view>
<!-- Audio upload mode -->
<view v-if="uploadMode === 'audio'" class="audio-upload-area">
<view class="upload-box" @click="chooseFile">
<text class="upload-icon">📁</text>
<text class="upload-text" v-if="!form.file">点击选择录音文件</text>
<text class="upload-text" v-else>{{ form.file.name }}</text>
<text class="upload-hint">支持 mp3/m4a/wav最大 50MB</text>
</view>
</view>
<!-- Text paste mode -->
<view v-else class="text-upload-area">
<textarea
class="text-input"
v-model="form.text"
placeholder="将面试录音转写为文字后粘贴在这里...&#10;&#10;示例:&#10;面试官:请介绍一下你自己&#10;候选人:我毕业于..."
:maxlength="10000"
/>
<text class="char-count">{{ form.text.length }}/10000</text>
</view>
</view>
<view class="submit-area">
<button class="btn-primary" @click="submitReview" :disabled="submitting">
{{ submitting ? '提交中...' : '开始分析' }}
</button>
</view>
</view>
</template>
<!-- ====== Mode 3: Processing View ====== -->
<template v-if="mode === 'processing'">
<view class="processing-page">
<view class="processing-card">
<view class="spinner"></view>
<text class="processing-title">AI 分析中</text>
<text class="processing-desc">正在对面试录音进行深度分析请稍候...</text>
<view class="process-steps">
<view class="step" :class="{ done: processStep >= 1 }">
<text class="step-num">{{ processStep >= 1 ? '✓' : '1' }}</text>
<text class="step-label">文本转录</text>
</view>
<view class="step" :class="{ done: processStep >= 2 }">
<text class="step-num">{{ processStep >= 2 ? '✓' : '2' }}</text>
<text class="step-label">语音分析</text>
</view>
<view class="step" :class="{ done: processStep >= 3 }">
<text class="step-num">{{ processStep >= 3 ? '✓' : '3' }}</text>
<text class="step-label">AI 评估</text>
</view>
</view>
<text class="processing-hint">请勿离开此页面</text>
</view>
</view>
</template>
<!-- ====== Mode 4: Report View ====== -->
<template v-if="mode === 'report' && reportData">
<view class="report-page">
<view class="report-header">
<text class="report-back" @click="mode = 'list'"> 返回列表</text>
<text class="report-title">复盘报告</text>
<view style="width:60rpx"></view>
</view>
<view class="body">
<view class="score-card">
<view class="score-circle" :class="scoreLevel(reportData.analysis.overallScore)">
<text class="score-num">{{ reportData.analysis.overallScore }}</text>
<text class="score-label">总分</text>
</view>
</view>
<view class="info-row">
<text class="info-label">面试岗位</text>
<text class="info-value">{{ reportData.position }}</text>
</view>
<view class="info-row" v-if="reportData.company">
<text class="info-label">面试公司</text>
<text class="info-value">{{ reportData.company }}</text>
</view>
<view class="info-row">
<text class="info-label">分析时间</text>
<text class="info-value">{{ formatDate(reportData.createdAt) }}</text>
</view>
<view class="info-row" v-if="reportData.speechAnalysis">
<text class="info-label">回答总时长</text>
<text class="info-value">{{ reportData.speechAnalysis.totalDuration }}</text>
</view>
<!-- 四维能力评估 -->
<view class="section" v-if="reportData.analysis.dimensions">
<view class="section-title">四维能力评估</view>
<view class="dim-grid">
<view class="dim-item" v-for="dim in dimDefs" :key="dim.key">
<view class="dim-header">
<text class="dim-name">{{ dim.label }}</text>
<text class="dim-score">{{ getDimValue(dim.key) }}</text>
</view>
<view class="dim-bar-bg">
<view class="dim-bar-fill" :style="{ width: getDimValue(dim.key) + '%', background: dim.color }"></view>
</view>
</view>
</view>
</view>
<!-- 语音分析 -->
<view class="section" v-if="reportData.speechAnalysis">
<view class="section-title">语音表达分析</view>
<view class="speech-card">
<view class="speech-row">
<text class="speech-label">语气词评分</text>
<text class="speech-value" :class="scoreLevel(reportData.speechAnalysis.fillerScore)">
{{ reportData.speechAnalysis.fillerScore }}
</text>
</view>
<view class="speech-row">
<text class="speech-label">语气词密度</text>
<text class="speech-value">{{ reportData.speechAnalysis.fillerDensity }}%</text>
</view>
<view class="speech-row">
<text class="speech-label">语速</text>
<text class="speech-value">{{ reportData.speechAnalysis.pace }}</text>
</view>
<view class="filler-tags" v-if="reportData.speechAnalysis.fillerWords.length > 0">
<text class="filler-tag" v-for="fw in reportData.speechAnalysis.fillerWords" :key="fw.word">
"{{ fw.word }}" × {{ fw.count }}
</text>
</view>
</view>
</view>
<!-- 逐题分析 -->
<view class="section" v-if="reportData.analysis.questionBreakdown && reportData.analysis.questionBreakdown.length > 0">
<view class="section-title">逐题分析</view>
<view class="qa-list">
<view class="qa-item" v-for="(qa, idx) in reportData.analysis.questionBreakdown" :key="idx">
<view class="qa-header">
<text class="qa-num">Q{{ idx + 1 }}</text>
<text class="qa-score" :class="scoreLevel(qa.score)">{{ qa.score }}</text>
</view>
<text class="qa-question">{{ qa.question }}</text>
<text class="qa-answer-label">你的回答</text>
<text class="qa-answer">{{ qa.answer }}</text>
<view class="qa-comment" v-if="qa.comment">
<text class="qa-comment-icon">💡</text>
<text class="qa-comment-text">{{ qa.comment }}</text>
</view>
<view class="qa-suggest" v-if="qa.suggestedAnswer">
<text class="qa-suggest-icon">参考思路</text>
<text class="qa-suggest-text">{{ qa.suggestedAnswer }}</text>
</view>
</view>
</view>
</view>
<!-- 改进建议 -->
<view class="section" v-if="reportData.analysis.strengths.length > 0 || reportData.analysis.weaknesses.length > 0">
<view class="section-title">改进建议</view>
<view class="sugg-block" v-if="reportData.analysis.strengths.length > 0">
<text class="sugg-block-title sugg-good"> 表现亮点</text>
<text class="sugg-item" v-for="(s, i) in reportData.analysis.strengths" :key="'str-' + i">{{ s }}</text>
</view>
<view class="sugg-block" v-if="reportData.analysis.weaknesses.length > 0">
<text class="sugg-block-title sugg-warn"> 待改进</text>
<text class="sugg-item" v-for="(w, i) in reportData.analysis.weaknesses" :key="'weak-' + i">{{ w }}</text>
</view>
<view class="sugg-block" v-if="reportData.analysis.suggestions.length > 0">
<text class="sugg-block-title sugg-info">💡 具体建议</text>
<text class="sugg-item" v-for="(s, i) in reportData.analysis.suggestions" :key="'sug-' + i">{{ s }}</text>
</view>
</view>
<view class="actions">
<button class="btn-primary" @click="goInterview">再去模拟面试练练</button>
<button class="btn-outline" @click="mode = 'list'">返回列表</button>
</view>
</view>
</view>
</template>
</view>
</template>
<script setup>
import { ref, computed, onMounted, onUnmounted } from 'vue'
// #ifdef MP-WEIXIN
import { onShow, onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
// #endif
// #ifndef MP-WEIXIN
import { onShow } from '@dcloudio/uni-app'
// #endif
import { api } from '../../config'
// #ifdef MP-WEIXIN
onShareAppMessage(() => ({ title: '职引 - 面试复盘分析 | AI评析+口语分析', path: '/pages/review/review' }))
onShareTimeline(() => ({ title: '职引 - 面试复盘分析 | AI评析+口语分析' }))
// #endif
const mode = ref('list')
const loading = ref(false)
const submitting = ref(false)
const listData = ref({ items: [], total: 0, page: 1, limit: 20 })
const processStep = ref(0)
const reportData = ref(null)
const pollTimer = ref(null)
const reviewId = ref('')
const form = ref({ position: '', company: '', file: null, text: '' })
const uploadMode = ref('audio')
const dimDefs = [
{ key: 'logic', label: '逻辑思维', color: '#6366F1' },
{ key: 'expression', label: '表达能力', color: '#10B981' },
{ key: 'professionalism', label: '专业度', color: '#F59E0B' },
{ key: 'stability', label: '稳定性', color: '#EF4444' },
]
const avgScore = computed(() => {
const scored = listData.value.items.filter(i => i.status === 'completed' && i.analysis?.overallScore)
if (scored.length === 0) return '--'
return Math.round(scored.reduce((s, i) => s + (i.analysis?.overallScore || 0), 0) / scored.length)
})
const completedCount = computed(() => {
return listData.value.items.filter(i => i.status === 'completed').length
})
const hasMore = computed(() => {
return listData.value.page < Math.ceil(listData.value.total / listData.value.limit)
})
onMounted(() => { fetchList() })
onShow(() => {
if (mode.value === 'list') fetchList()
})
async function fetchList(page = 1) {
const token = uni.getStorageSync('token') || ''
if (!token) return
loading.value = true
try {
const res = await uni.request({
url: api(`/interview-review/list?page=${page}&limit=20`),
method: 'GET',
header: { 'Authorization': `Bearer ${token}` },
})
if (res.statusCode === 200) {
const data = res.data
if (page === 1) {
listData.value = data
} else {
listData.value = {
...data,
items: [...listData.value.items, ...data.items],
}
}
}
} catch(e) { console.error(e) }
finally { loading.value = false }
}
function loadMore() {
const next = listData.value.page + 1
fetchList(next)
listData.value.page = next
}
function chooseFile() {
uni.chooseMessageFile ? uni.chooseMessageFile({
count: 1,
type: 'file',
extension: ['mp3', 'm4a', 'wav', 'aac', 'ogg'],
success: (r) => {
if (r.tempFiles && r.tempFiles[0]) {
form.value.file = r.tempFiles[0]
}
},
}) : uni.showToast({ title: '当前平台不支持文件选择', icon: 'none' })
}
async function submitReview() {
if (!form.value.position.trim()) {
uni.showToast({ title: '请填写面试岗位', icon: 'none' })
return
}
if (uploadMode.value === 'audio' && !form.value.file) {
uni.showToast({ title: '请选择录音文件', icon: 'none' })
return
}
if (uploadMode.value === 'text' && !form.value.text.trim()) {
uni.showToast({ title: '请粘贴面试转录文本', icon: 'none' })
return
}
submitting.value = true
const token = uni.getStorageSync('token') || ''
try {
if (uploadMode.value === 'audio') {
const res = await uni.uploadFile({
url: api('/interview-review'),
filePath: form.value.file.path || form.value.file.tempFilePath,
name: 'file',
formData: {
position: form.value.position.trim(),
company: form.value.company.trim(),
},
header: { 'Authorization': `Bearer ${token}` },
})
if (res.statusCode === 201 || res.statusCode === 200) {
const data = typeof res.data === 'string' ? JSON.parse(res.data) : res.data
reviewId.value = data.id
startPolling(data.id)
mode.value = 'processing'
animateSteps()
} else {
const err = typeof res.data === 'string' ? JSON.parse(res.data) : res.data
uni.showToast({ title: err.message || '上传失败', icon: 'none' })
}
} else {
// Text submission
const res = await uni.request({
url: api('/interview-review/text'),
method: 'POST',
data: {
position: form.value.position.trim(),
company: form.value.company.trim(),
text: form.value.text.trim(),
},
header: {
'Authorization': `Bearer ${token}`,
'Content-Type': 'application/json',
},
})
if (res.statusCode === 201 || res.statusCode === 200) {
reviewId.value = res.data.id
startPolling(res.data.id)
mode.value = 'processing'
animateSteps()
} else {
uni.showToast({ title: res.data?.message || '提交失败', icon: 'none' })
}
}
} catch(e) {
uni.showToast({ title: e.message || '网络错误', icon: 'none' })
}
finally { submitting.value = false }
}
function animateSteps() {
processStep.value = 1
setTimeout(() => { processStep.value = 2 }, 3000)
setTimeout(() => { processStep.value = 3 }, 6000)
}
function startPolling(id) {
let attempts = 0
const maxAttempts = 60 // 5 minutes max
pollTimer.value = setInterval(async () => {
attempts++
if (attempts > maxAttempts) {
clearInterval(pollTimer.value)
uni.showToast({ title: '分析超时,请稍后再查', icon: 'none' })
mode.value = 'list'
return
}
try {
const token = uni.getStorageSync('token') || ''
const res = await uni.request({
url: api(`/interview-review/${id}`),
method: 'GET',
header: { 'Authorization': `Bearer ${token}` },
})
if (res.statusCode === 200) {
const data = res.data
if (data.status === 'completed') {
clearInterval(pollTimer.value)
reportData.value = data
mode.value = 'report'
} else if (data.status === 'failed') {
clearInterval(pollTimer.value)
uni.showToast({ title: '分析失败,请重新上传', icon: 'none' })
mode.value = 'list'
}
// status === 'processing': continue polling
}
} catch(e) { console.error(e) }
}, 5000)
}
function viewDetail(id) {
const token = uni.getStorageSync('token') || ''
uni.request({
url: api(`/interview-review/${id}`),
method: 'GET',
header: { 'Authorization': `Bearer ${token}` },
success: (res) => {
if (res.statusCode === 200) {
if (res.data.status === 'completed') {
reportData.value = res.data
mode.value = 'report'
} else if (res.data.status === 'processing') {
reviewId.value = id
startPolling(id)
mode.value = 'processing'
} else {
uni.showToast({ title: '分析失败', icon: 'none' })
}
}
},
fail: () => { uni.showToast({ title: '加载失败', icon: 'none' }) },
})
}
function getDimValue(key) {
return Math.min(100, Math.max(0, Math.round(reportData.value?.analysis?.dimensions?.[key] || 0)))
}
function formatDate(d) {
if (!d) return '--'
const date = new Date(d)
return `${String(date.getMonth()+1).padStart(2,'0')}-${String(date.getDate()).padStart(2,'0')} ${String(date.getHours()).padStart(2,'0')}:${String(date.getMinutes()).padStart(2,'0')}`
}
const scoreLevel = (s) => {
if (!s && s !== 0) return 'pending'
if (s >= 80) return 'good'
if (s >= 60) return 'medium'
return 'poor'
}
function goInterview() {
uni.switchTab({ url: '/pages/index/index' })
}
onUnmounted(() => {
if (pollTimer.value) clearInterval(pollTimer.value)
})
</script>
<style scoped>
.page { background: #F3F4F6; min-height: 100vh; }
.hero {
background: linear-gradient(135deg, #4F46E5 0%, #7C3AED 50%, #6366F1 100%);
padding: 48rpx 32rpx 72rpx; border-radius: 0 0 48rpx 48rpx;
}
.hero-title { font-size: 40rpx; font-weight: 700; color: #FFF; display: block; }
.hero-sub { font-size: 22rpx; color: rgba(255,255,255,0.7); margin-top: 8rpx; display: block; }
.stats-bar { display: flex; align-items: center; padding: 24rpx; margin: -40rpx 32rpx 0; position: relative; z-index: 1; border-radius: 16rpx; background: #FFF; box-shadow: 0 2rpx 12rpx rgba(0,0,0,0.06); }
.stat { flex: 1; display: flex; flex-direction: column; align-items: center; gap: 6rpx; }
.stat-val { font-size: 36rpx; font-weight: 700; color: #4F46E5; }
.stat-lbl { font-size: 20rpx; color: #9CA3AF; }
.stat-sep { width: 1rpx; height: 40rpx; background: #E5E7EB; }
.list { padding: 20rpx 32rpx 160rpx; }
.record-card { background: #FFF; padding: 24rpx 28rpx; margin-bottom: 16rpx; border-radius: 16rpx; box-shadow: 0 2rpx 12rpx rgba(0,0,0,0.04); }
.record-top { display: flex; align-items: center; gap: 16rpx; }
.record-icon { font-size: 40rpx; width: 64rpx; height: 64rpx; display: flex; align-items: center; justify-content: center; flex-shrink: 0; }
.record-body { flex: 1; min-width: 0; }
.record-name { font-size: 28rpx; font-weight: 600; color: #111827; }
.record-meta { font-size: 20rpx; color: #9CA3AF; margin-top: 6rpx; display: block; }
.record-score { font-size: 28rpx; font-weight: 700; width: 72rpx; text-align: right; flex-shrink: 0; }
.record-score.good { color: #059669; }
.record-score.medium { color: #D97706; }
.record-score.poor { color: #DC2626; }
.record-score.pending { color: #9CA3AF; }
.empty { display: flex; flex-direction: column; align-items: center; padding: 120rpx 32rpx 200rpx; }
.empty-icon { font-size: 80rpx; margin-bottom: 20rpx; }
.empty-title { font-size: 28rpx; font-weight: 600; color: #111827; }
.empty-desc { font-size: 22rpx; color: #9CA3AF; margin-top: 8rpx; margin-bottom: 36rpx; }
.fixed-bottom { position: fixed; bottom: 0; left: 0; right: 0; padding: 20rpx 32rpx 40rpx; background: linear-gradient(transparent, #F3F4F6 30rpx); }
.btn-primary { background: linear-gradient(135deg, #4F46E5, #7C3AED); color: #FFF; border-radius: 16rpx; height: 88rpx; line-height: 88rpx; font-size: 28rpx; font-weight: 600; display: flex; align-items: center; justify-content: center; gap: 8rpx; }
.btn-primary:active { opacity: 0.85; }
.btn-icon { font-size: 36rpx; font-weight: 400; }
/* Upload Page */
.upload-page { background: #F3F4F6; min-height: 100vh; }
.upload-header { display: flex; align-items: center; justify-content: space-between; padding: 24rpx 32rpx; background: #FFF; }
.upload-back { font-size: 28rpx; color: #4F46E5; }
.upload-title { font-size: 32rpx; font-weight: 600; color: #111827; }
.form-card { background: #FFF; border-radius: 16rpx; margin: 24rpx 32rpx; padding: 32rpx; }
.form-group { margin-bottom: 28rpx; }
.form-label { font-size: 26rpx; font-weight: 600; color: #374151; display: block; margin-bottom: 12rpx; }
.required { color: #DC2626; }
.form-input { width: 100%; height: 72rpx; border: 2rpx solid #E5E7EB; border-radius: 12rpx; padding: 0 20rpx; font-size: 26rpx; box-sizing: border-box; }
.tab-bar { display: flex; gap: 0; background: #F3F4F6; border-radius: 12rpx; overflow: hidden; }
.tab-item { flex: 1; text-align: center; padding: 18rpx 0; font-size: 24rpx; color: #6B7280; }
.tab-item.active { background: #4F46E5; color: #FFF; font-weight: 600; }
.audio-upload-area { margin-top: 8rpx; }
.upload-box { border: 2rpx dashed #D1D5DB; border-radius: 16rpx; padding: 48rpx 32rpx; display: flex; flex-direction: column; align-items: center; gap: 12rpx; }
.upload-icon { font-size: 60rpx; }
.upload-text { font-size: 26rpx; color: #374151; }
.upload-hint { font-size: 20rpx; color: #9CA3AF; }
.text-upload-area { margin-top: 8rpx; }
.text-input { width: 100%; height: 360rpx; border: 2rpx solid #E5E7EB; border-radius: 12rpx; padding: 20rpx; font-size: 24rpx; line-height: 1.6; box-sizing: border-box; }
.char-count { font-size: 20rpx; color: #9CA3AF; text-align: right; display: block; margin-top: 8rpx; }
.submit-area { padding: 0 32rpx 48rpx; }
/* Processing Page */
.processing-page { display: flex; align-items: center; justify-content: center; min-height: 100vh; padding: 32rpx; }
.processing-card { background: #FFF; border-radius: 24rpx; padding: 64rpx 48rpx; text-align: center; box-shadow: 0 8rpx 40rpx rgba(0,0,0,0.08); width: 100%; max-width: 500rpx; }
.spinner { width: 80rpx; height: 80rpx; border: 6rpx solid #E5E7EB; border-top-color: #4F46E5; border-radius: 50%; animation: spin 1s linear infinite; margin: 0 auto 32rpx; }
@keyframes spin { to { transform: rotate(360deg); } }
.processing-title { font-size: 32rpx; font-weight: 700; color: #111827; margin-bottom: 12rpx; }
.processing-desc { font-size: 24rpx; color: #6B7280; margin-bottom: 40rpx; }
.process-steps { display: flex; gap: 16rpx; justify-content: center; margin-bottom: 24rpx; }
.step { display: flex; flex-direction: column; align-items: center; gap: 8rpx; }
.step-num { width: 48rpx; height: 48rpx; border-radius: 50%; background: #F3F4F6; color: #9CA3AF; font-size: 22rpx; font-weight: 700; display: flex; align-items: center; justify-content: center; }
.step.done .step-num { background: #4F46E5; color: #FFF; }
.step-label { font-size: 20rpx; color: #9CA3AF; }
.step.done .step-label { color: #4F46E5; }
.processing-hint { font-size: 20rpx; color: #D1D5DB; }
/* Report Page */
.report-page { background: #F3F4F6; min-height: 100vh; }
.report-header { display: flex; align-items: center; justify-content: space-between; padding: 24rpx 32rpx; background: #FFF; }
.report-back { font-size: 28rpx; color: #4F46E5; }
.report-title { font-size: 32rpx; font-weight: 600; color: #111827; }
.body { padding: 0 32rpx 48rpx; }
.score-card { display: flex; justify-content: center; margin: -20rpx 0 30rpx; }
.score-circle {
width: 180rpx; height: 180rpx; border-radius: 50%;
background: #FFF; display: flex; flex-direction: column;
align-items: center; justify-content: center;
box-shadow: 0 8rpx 30rpx rgba(79,70,229,0.2);
}
.score-circle.good { box-shadow: 0 8rpx 30rpx rgba(5,150,105,0.2); }
.score-circle.medium { box-shadow: 0 8rpx 30rpx rgba(217,119,6,0.2); }
.score-circle.poor { box-shadow: 0 8rpx 30rpx rgba(220,38,38,0.2); }
.score-num { font-size: 56rpx; font-weight: 700; color: #4F46E5; line-height: 1.2; }
.good .score-num { color: #059669; }
.medium .score-num { color: #D97706; }
.poor .score-num { color: #DC2626; }
.score-label { font-size: 20rpx; color: #9CA3AF; margin-top: 4rpx; }
.info-row { display: flex; justify-content: space-between; padding: 20rpx 0; border-bottom: 1rpx solid #E5E7EB; font-size: 24rpx; }
.info-label { color: #6B7280; }
.info-value { color: #111827; font-weight: 500; }
.section { background: #FFF; border-radius: 20rpx; padding: 28rpx; margin-top: 24rpx; box-shadow: 0 2rpx 12rpx rgba(0,0,0,0.04); }
.section-title { font-size: 28rpx; font-weight: 600; color: #111827; margin-bottom: 16rpx; }
.dim-grid { display: flex; flex-direction: column; gap: 20rpx; }
.dim-header { display: flex; justify-content: space-between; margin-bottom: 8rpx; }
.dim-name { font-size: 24rpx; color: #374151; }
.dim-score { font-size: 24rpx; color: #4F46E5; font-weight: 600; }
.dim-bar-bg { height: 20rpx; background: #F3F4F6; border-radius: 10rpx; overflow: hidden; }
.dim-bar-fill { height: 100%; border-radius: 10rpx; }
.speech-card { display: flex; flex-direction: column; gap: 16rpx; }
.speech-row { display: flex; justify-content: space-between; align-items: center; }
.speech-label { font-size: 24rpx; color: #374151; }
.speech-value { font-size: 24rpx; font-weight: 600; color: #111827; }
.speech-value.good { color: #059669; }
.speech-value.medium { color: #D97706; }
.speech-value.poor { color: #DC2626; }
.filler-tags { display: flex; flex-wrap: wrap; gap: 8rpx; }
.filler-tag { background: #FEF3C7; color: #92400E; padding: 6rpx 16rpx; border-radius: 20rpx; font-size: 20rpx; }
.qa-list { display: flex; flex-direction: column; gap: 24rpx; }
.qa-item { padding: 20rpx; background: #F9FAFB; border-radius: 12rpx; }
.qa-header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 12rpx; }
.qa-num { font-size: 22rpx; font-weight: 700; color: #4F46E5; background: #EEF2FF; padding: 4rpx 14rpx; border-radius: 8rpx; }
.qa-score { font-size: 24rpx; font-weight: 700; }
.qa-score.good { color: #059669; }
.qa-score.medium { color: #D97706; }
.qa-score.poor { color: #DC2626; }
.qa-score.pending { color: #9CA3AF; }
.qa-question { font-size: 24rpx; font-weight: 600; color: #111827; margin-bottom: 8rpx; display: block; }
.qa-answer-label { font-size: 20rpx; color: #6B7280; display: block; margin-top: 8rpx; }
.qa-answer { font-size: 22rpx; color: #374151; line-height: 1.6; display: block; margin-top: 4rpx; white-space: pre-wrap; }
.qa-comment { background: #EEF2FF; padding: 12rpx; border-radius: 8rpx; margin-top: 12rpx; display: flex; gap: 8rpx; }
.qa-comment-icon { font-size: 22rpx; }
.qa-comment-text { font-size: 22rpx; color: #4338CA; line-height: 1.5; }
.qa-suggest { background: #FEF3C7; padding: 12rpx; border-radius: 8rpx; margin-top: 8rpx; }
.qa-suggest-icon { font-size: 22rpx; font-weight: 600; color: #92400E; }
.qa-suggest-text { font-size: 22rpx; color: #78350F; line-height: 1.5; }
.sugg-block { margin-bottom: 20rpx; }
.sugg-block-title { font-size: 24rpx; font-weight: 600; display: block; margin-bottom: 8rpx; }
.sugg-good { color: #059669; }
.sugg-warn { color: #D97706; }
.sugg-info { color: #4F46E5; }
.sugg-item { display: block; font-size: 22rpx; color: #374151; line-height: 1.8; padding-left: 20rpx; }
.sugg-item::before { content: '• '; color: #D1D5DB; }
.actions { display: flex; flex-direction: column; gap: 16rpx; margin-top: 32rpx; }
.btn-outline { background: #FFF; color: #4F46E5; border-radius: 16rpx; height: 88rpx; line-height: 88rpx; font-size: 28rpx; border: 2rpx solid #4F46E5; }
.btn-outline:active { background: #F5F3FF; }
.load-more { text-align: center; padding: 24rpx; color: #4F46E5; font-size: 24rpx; }
</style>
+112 -19
View File
@@ -4,8 +4,8 @@
<view class="stats-card"> <view class="stats-card">
<view class="stat-row"> <view class="stat-row">
<view class="stat-item"> <view class="stat-item">
<text class="stat-value">{{ stats.shareCredits || 0 }}</text> <text class="stat-value">{{ stats.gravity ?? 0 }}</text>
<text class="stat-label">📦 我的积分</text> <text class="stat-label">🌌 我的引力值</text>
</view> </view>
</view> </view>
<view class="stat-sub-row"> <view class="stat-sub-row">
@@ -20,12 +20,12 @@
</view> </view>
<view class="stat-arrow"></view> <view class="stat-arrow"></view>
<view class="stat-sub-item"> <view class="stat-sub-item">
<text class="stat-sub-value">{{ stats.shareCredits || 0 }}</text> <text class="stat-sub-value">{{ stats.gravity || 0 }}</text>
<text class="stat-sub-label">获得积分</text> <text class="stat-sub-label">获得引力值</text>
</view> </view>
</view> </view>
<view class="hint"> <view class="hint">
💡 好友通过你的链接打开并 <text class="hint-em">登录/注册</text> 才算有效每次有效 1 积分 💡 好友通过你的链接打开并 <text class="hint-em">登录/注册</text> 才算有效每次有效 1 引力值每日上限 3
</view> </view>
</view> </view>
@@ -49,20 +49,42 @@
<view class="today-bar"> <view class="today-bar">
<view class="today-bar-fill" :style="{ width: Math.min(100, (todayStats.credited / 3) * 100) + '%' }"></view> <view class="today-bar-fill" :style="{ width: Math.min(100, (todayStats.credited / 3) * 100) + '%' }"></view>
</view> </view>
<text class="today-hint">每日最多 3 次有效积分</text> <text class="today-hint">每日最多 3 次有效引力值分享给朋友圈可获更多</text>
</view> </view>
<!-- 分享按钮 --> <!-- 分享按钮 -->
<view class="share-actions"> <view class="share-actions">
<!-- #ifdef MP-WEIXIN -->
<view class="share-btn-row">
<button class="share-btn wx-share" open-type="share">
<text class="btn-icon">💬</text>
<text>分享给好友</text>
</button>
<view class="share-btn wx-timeline-hint" @click="showTimelineHint">
<text class="btn-icon">🔄</text>
<text>分享朋友圈</text>
</view>
</view>
<!-- #endif -->
<!-- #ifdef H5 -->
<view class="share-btn-row">
<button class="share-btn wx-share" @click="shareToWechat" v-if="isWechat"> <button class="share-btn wx-share" @click="shareToWechat" v-if="isWechat">
<text class="btn-icon">💬</text> <text class="btn-icon">💬</text>
<text>分享给微信好友</text> <text>分享给微信好友</text>
</button> </button>
<button class="share-btn wx-timeline" @click="shareToWechat" v-if="isWechat">
<text class="btn-icon">🔄</text>
<text>分享朋友圈</text>
</button>
</view>
<!-- #endif -->
<view class="share-btn-row">
<button class="share-btn link-share" @click="copyLink"> <button class="share-btn link-share" @click="copyLink">
<text class="btn-icon">🔗</text> <text class="btn-icon">🔗</text>
<text>复制分享链接</text> <text>复制分享链接</text>
</button> </button>
</view> </view>
</view>
<!-- Tab 切换 --> <!-- Tab 切换 -->
<view class="tab-bar"> <view class="tab-bar">
@@ -113,10 +135,15 @@
<script setup> <script setup>
import { ref, computed, onMounted } from 'vue' import { ref, computed, onMounted } from 'vue'
// #ifdef MP-WEIXIN
import { onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
// #endif
import { api } from '../../config' import { api } from '../../config'
const tab = ref('records') const tab = ref('records')
const stats = ref({ totalShares: 0, totalVisits: 0, creditedCount: 0, todayCredited: 0, shareCredits: 0 }) const stats = ref({ totalShares: 0, totalVisits: 0, creditedCount: 0, todayCredited: 0, shareCredits: 0, gravity: 0 })
const shareLink = ref('')
const shareUrlCached = ref('')
const records = ref([]) const records = ref([])
const visitors = ref([]) const visitors = ref([])
@@ -125,10 +152,24 @@ const todayStats = computed(() => ({
credited: stats.value.todayCredited || 0, credited: stats.value.todayCredited || 0,
})) }))
let isWechat = false const isWechat = ref(false)
// #ifdef MP-WEIXIN
onShareAppMessage(() => ({
title: 'AI磁场·职引 — AI模拟面试+简历优化',
path: '/pages/share/share',
imageUrl: 'https://zhiyinwx.yzrcloud.cn/static/share-card.png',
}))
onShareTimeline(() => ({
title: 'AI磁场·职引 — AI模拟面试+简历优化',
imageUrl: 'https://zhiyinwx.yzrcloud.cn/static/share-card.png',
}))
// #endif
onMounted(() => { onMounted(() => {
isWechat = navigator?.userAgent?.toLowerCase().includes('micromessenger') || false // #ifdef H5
isWechat.value = navigator?.userAgent?.toLowerCase().includes('micromessenger') || false
// #endif
loadData() loadData()
}) })
@@ -137,6 +178,21 @@ async function loadData() {
if (!token) { uni.showToast({ title: '请先登录', icon: 'none' }); return } if (!token) { uni.showToast({ title: '请先登录', icon: 'none' }); return }
const header = { Authorization: `Bearer ${token}` } const header = { Authorization: `Bearer ${token}` }
// 先创建分享链接,缓存下来供复制使用
try {
const res = await uni.request({
url: api('/share/create'), method: 'POST',
data: { type: 'app', title: '我在AI磁场·职引练习面试', description: 'AI模拟面试+简历优化,快来一起提升吧' },
header,
})
if (res.statusCode >= 200 && res.statusCode < 300) {
const data = res.data?.data || res.data
if (data.shareCode) {
shareUrlCached.value = `https://zhiyinwx.yzrcloud.cn/api/share/${data.shareCode}`
}
}
} catch (e) { /* create share is best-effort */ }
try { try {
const [statsRes, recordsRes, visitorsRes] = await Promise.all([ const [statsRes, recordsRes, visitorsRes] = await Promise.all([
uni.request({ url: api('/share/stats'), method: 'GET', header }), uni.request({ url: api('/share/stats'), method: 'GET', header }),
@@ -160,7 +216,7 @@ async function shareToWechat() {
data: { type: 'app', title: '我在AI磁场·职引练习面试', description: 'AI模拟面试+简历优化,快来一起提升吧' }, data: { type: 'app', title: '我在AI磁场·职引练习面试', description: 'AI模拟面试+简历优化,快来一起提升吧' },
header: { Authorization: `Bearer ${token}` }, header: { Authorization: `Bearer ${token}` },
}) })
if (res.statusCode !== 200) return if (res.statusCode < 200 || res.statusCode >= 300) return
const data = res.data const data = res.data
const path = data.wechatShareInfo?.path || `/pages/share/share?code=${data.shareCode}` const path = data.wechatShareInfo?.path || `/pages/share/share?code=${data.shareCode}`
@@ -179,19 +235,49 @@ async function shareToWechat() {
async function copyLink() { async function copyLink() {
const token = uni.getStorageSync('token') const token = uni.getStorageSync('token')
if (!token) return if (!token) { uni.showToast({ title: '请先登录', icon: 'none' }); return }
if (shareUrlCached.value) {
// 使用已缓存的分享链接,避免二次 API 调用
uni.setClipboardData({
data: shareUrlCached.value,
success: () => { uni.showToast({ title: '链接已复制' }); loadData() },
fail: () => { uni.showToast({ title: '复制失败,请长按选择复制', icon: 'none' }) },
})
return
}
// 缓存未命中时兜底:调 API 生成
try { try {
const res = await uni.request({ const res = await uni.request({
url: api('/share/create'), url: api('/share/create'), method: 'POST',
method: 'POST',
data: { type: 'app', title: '我在AI磁场·职引练习面试', description: 'AI模拟面试+简历优化,快来一起提升吧' }, data: { type: 'app', title: '我在AI磁场·职引练习面试', description: 'AI模拟面试+简历优化,快来一起提升吧' },
header: { Authorization: `Bearer ${token}` }, header: { Authorization: `Bearer ${token}` },
}) })
if (res.statusCode !== 200) return if (res.statusCode < 200 || res.statusCode >= 300) {
const shareUrl = `https://zhiyinwx.yzrcloud.cn/share/${res.data.shareCode}` uni.showToast({ title: `创建失败(${res.statusCode}),请重试`, icon: 'none' })
uni.setClipboardData({ data: shareUrl, success: () => { uni.showToast({ title: '链接已复制' }); loadData() } }) return
} catch (e) { console.error(e) } }
const data = res.data?.data || res.data
if (!data.shareCode) {
uni.showToast({ title: '返回数据异常,请重试', icon: 'none' })
return
}
const shareUrl = `https://zhiyinwx.yzrcloud.cn/api/share/${data.shareCode}`
shareUrlCached.value = shareUrl
uni.setClipboardData({
data: shareUrl,
success: () => { uni.showToast({ title: '链接已复制' }); loadData() },
fail: () => { uni.showToast({ title: '复制失败,请长按选择复制', icon: 'none' }) },
})
} catch (e) {
console.error('[share] copyLink error:', e)
uni.showToast({ title: '网络错误,请重试', icon: 'none' })
}
}
function showTimelineHint() {
uni.showToast({ title: '请点击右上角 ··· 选择分享到朋友圈', icon: 'none', duration: 3000 })
} }
function typeLabel(type) { function typeLabel(type) {
@@ -236,11 +322,18 @@ function formatTime(t) {
.today-hint { font-size: 18rpx; color: var(--color-text-tertiary); text-align: center; } .today-hint { font-size: 18rpx; color: var(--color-text-tertiary); text-align: center; }
/* Share buttons */ /* Share buttons */
.share-actions { padding: 0 32rpx; display: flex; gap: 20rpx; } .share-actions { padding: 0 32rpx; display: flex; flex-direction: column; gap: 16rpx; }
.share-btn { flex: 1; display: flex; align-items: center; justify-content: center; height: 88rpx; border-radius: var(--radius-md); font-size: 26rpx; font-weight: 500; border: none; } .share-btn-row { display: flex; gap: 20rpx; }
.share-btn {
flex: 1; display: flex; align-items: center; justify-content: center; height: 88rpx; border-radius: var(--radius-md);
font-size: 26rpx; font-weight: 500; border: none; padding: 0; white-space: nowrap;
}
.share-btn:active { transform: scale(0.96); } .share-btn:active { transform: scale(0.96); }
.btn-icon { margin-right: 8rpx; font-size: 28rpx; } .btn-icon { margin-right: 8rpx; font-size: 28rpx; }
.wx-share { background: #07C160; color: #FFFFFF; } .wx-share { background: #07C160; color: #FFFFFF; }
.wx-timeline { background: #FF6600; color: #FFFFFF; }
.wx-timeline-hint { flex: 1; display: flex; align-items: center; justify-content: center; height: 88rpx; border-radius: var(--radius-md); font-size: 26rpx; font-weight: 500; background: #FF6600; color: #FFFFFF; white-space: nowrap; }
.wx-timeline-hint:active { transform: scale(0.96); }
.link-share { background: #FFFFFF; color: var(--color-text); border: 2rpx solid var(--color-border); } .link-share { background: #FFFFFF; color: var(--color-text); border: 2rpx solid var(--color-border); }
/* Tabs */ /* Tabs */
+202 -4
View File
@@ -6,6 +6,7 @@
<image class="avatar" :src="userInfo.avatar || '/static/avatar-default.png'" mode="aspectFill" /> <image class="avatar" :src="userInfo.avatar || '/static/avatar-default.png'" mode="aspectFill" />
<view class="profile-info"> <view class="profile-info">
<text class="nickname">{{ userInfo.nickname || '未设置昵称' }}</text> <text class="nickname">{{ userInfo.nickname || '未设置昵称' }}</text>
<text class="user-id">ID: {{ maskedId }}</text>
<view class="plan-badge">{{ userInfo.plan || '免费版' }}</view> <view class="plan-badge">{{ userInfo.plan || '免费版' }}</view>
</view> </view>
<text class="header-arrow"></text> <text class="header-arrow"></text>
@@ -38,31 +39,73 @@
</view> </view>
</view> </view>
<!-- 引力值卡片代替原配额卡片 -->
<view class="gravity-card" v-if="isLoggedIn">
<view class="gravity-card-inner">
<view class="gravity-top-row">
<view class="gravity-header">
<text class="gravity-icon"></text>
<text class="gravity-label">我的引力值</text>
</view>
<text class="gravity-num">{{ memberInfo.gravity ?? 0 }}</text>
</view>
<text class="gravity-hint">每次面试消耗 5 引力值 · 分享可获得更多</text>
<view class="gravity-actions">
<text class="gravity-btn share" @click="goSharePage">分享得引力值</text>
<text class="gravity-btn contribute" @click="goContributePage">贡献面经</text>
<text class="gravity-btn h5buy" @click="goH5Buy">官网购买</text>
</view>
</view>
</view>
<!-- 菜单列表 --> <!-- 菜单列表 -->
<view class="menu-area"> <view class="menu-area">
<view class="menu-group"> <view class="menu-group">
<view class="menu-item" @click="requireLogin(goCareer, '择业顾问')">
<view class="menu-icon-wrap wrap-teal"><text class="menu-icon">🧭</text></view>
<text class="menu-text">择业顾问</text>
<text class="menu-tag">NEW</text>
<text class="menu-arrow"></text>
</view>
<view class="menu-item" @click="requireLogin(goHistory, '面试记录')"> <view class="menu-item" @click="requireLogin(goHistory, '面试记录')">
<view class="menu-icon-wrap wrap-blue"><text class="menu-icon">📋</text></view> <view class="menu-icon-wrap wrap-blue"><text class="menu-icon">📋</text></view>
<text class="menu-text">面试记录</text> <text class="menu-text">面试记录</text>
<text class="menu-arrow"></text> <text class="menu-arrow"></text>
</view> </view>
<view class="menu-item" @click="requireLogin(goReviewReview, '面试复盘')">
<view class="menu-icon-wrap wrap-orange"><text class="menu-icon">🎙</text></view>
<text class="menu-text">面试复盘</text>
<text class="menu-arrow"></text>
</view>
<!-- 会员中心菜单注释隐藏微信审核 -->
<!--
<view class="menu-item" @click="goVip"> <view class="menu-item" @click="goVip">
<view class="menu-icon-wrap wrap-purple"><text class="menu-icon">💎</text></view> <view class="menu-icon-wrap wrap-purple"><text class="menu-icon">💎</text></view>
<text class="menu-text">会员中心</text> <text class="menu-text">会员中心</text>
<text class="menu-arrow"></text> <text class="menu-arrow"></text>
</view> </view>
-->
<view class="menu-item" @click="requireLogin(goResume, '我的简历')"> <view class="menu-item" @click="requireLogin(goResume, '我的简历')">
<view class="menu-icon-wrap wrap-green"><text class="menu-icon">📄</text></view> <view class="menu-icon-wrap wrap-green"><text class="menu-icon">📄</text></view>
<text class="menu-text">我的简历</text> <text class="menu-text">我的简历</text>
<text class="menu-arrow"></text> <text class="menu-arrow"></text>
</view> </view>
<view class="menu-item" @click="requireLogin(goShare, '我的分享')"> <view class="menu-item" @click="requireLogin(goSharePage, '我的分享')">
<view class="menu-icon-wrap wrap-orange"><text class="menu-icon">📤</text></view> <view class="menu-icon-wrap wrap-orange"><text class="menu-icon">📤</text></view>
<text class="menu-text">我的分享</text> <text class="menu-text">我的分享</text>
<text class="menu-arrow"></text> <text class="menu-arrow"></text>
</view> </view>
</view> </view>
<view class="menu-group"> <view class="menu-group">
<!-- #ifdef MP-WEIXIN -->
<view class="menu-item">
<button class="contact-btn-inner" open-type="contact">
<view class="menu-icon-wrap wrap-gray"><text class="menu-icon">💬</text></view>
<text class="menu-text">联系客服</text>
<text class="menu-arrow"></text>
</button>
</view>
<!-- #endif -->
<view class="menu-item" @click="goAbout"> <view class="menu-item" @click="goAbout">
<view class="menu-icon-wrap wrap-gray"><text class="menu-icon"></text></view> <view class="menu-icon-wrap wrap-gray"><text class="menu-icon"></text></view>
<text class="menu-text">关于</text> <text class="menu-text">关于</text>
@@ -78,32 +121,109 @@
<button class="logout-btn" @click="doLogout">退出登录</button> <button class="logout-btn" @click="doLogout">退出登录</button>
</view> </view>
</view> </view>
<!-- 引力值获取方式 -->
<view class="modal-overlay" v-if="showGetGravityModal" @click="showGetGravityModal = false">
<view class="modal-content" @click.stop>
<text class="modal-title">获取引力值</text>
<text class="modal-hint">引力值可用于面试简历优化下载等</text>
<!-- 分享得引力值 -->
<view class="gp-get-method" @click="goSharePage">
<view class="gp-method-icon-wrap"><text class="gp-method-icon">🎁</text></view>
<view class="gp-method-info">
<text class="gp-method-name">分享给好友</text>
<text class="gp-method-desc">每成功邀请一位好友注册可得 5 引力值上不封顶</text>
</view>
<text class="gp-method-arrow"></text>
</view>
<!-- 贡献面经 -->
<view class="gp-get-method" @click="goContributePage">
<view class="gp-method-icon-wrap"><text class="gp-method-icon">📝</text></view>
<view class="gp-method-info">
<text class="gp-method-name">贡献面经</text>
<text class="gp-method-desc">每发布一篇面经可获得 3 引力值助力他人也提升自己</text>
</view>
<text class="gp-method-arrow"></text>
</view>
<!-- 官网购买 -->
<view class="gp-get-method" @click="goH5Buy">
<view class="gp-method-icon-wrap"><text class="gp-method-icon">🛒</text></view>
<view class="gp-method-info">
<text class="gp-method-name">官网购买</text>
<text class="gp-method-desc">通过官网网页端可购买引力值套餐支持更多支付方式</text>
</view>
<text class="gp-method-arrow"></text>
</view>
<text class="modal-close" @click="showGetGravityModal = false">关闭</text>
</view>
</view>
</view> </view>
</template> </template>
<script setup> <script setup>
import { ref, computed, onMounted } from 'vue' import { ref, computed, onMounted } from 'vue'
// #ifdef MP-WEIXIN
import { onShow, onShareAppMessage, onShareTimeline } from '@dcloudio/uni-app'
// #endif
// #ifndef MP-WEIXIN
import { onShow } from '@dcloudio/uni-app' import { onShow } from '@dcloudio/uni-app'
// #endif
import { api } from '../../config' import { api } from '../../config'
// #ifdef MP-WEIXIN
onShareAppMessage(() => ({ title: '职引 - 个人中心 | AI模拟面试', path: '/pages/user/user' }))
onShareTimeline(() => ({ title: '职引 - 个人中心 | AI模拟面试' }))
// #endif
const userInfo = ref({}) const userInfo = ref({})
const isAdmin = ref(false) const isAdmin = ref(false)
const stats = ref({ interviewCount: 0, avgScore: '--', completedCount: 0 }) const stats = ref({ interviewCount: 0, avgScore: '--', completedCount: 0 })
const token = ref('') const token = ref('')
const memberInfo = ref({ plan: 'free', planName: '免费版', remaining: 0, gravity: 0 })
const isLoggedIn = computed(() => !!token.value) const isLoggedIn = computed(() => !!token.value)
const maskedId = computed(() => {
const id = userInfo.value.id || ''
return id.length > 6 ? `****${id.slice(-6)}` : id || '--'
})
const refreshState = () => { const refreshState = () => {
token.value = uni.getStorageSync('token') || '' token.value = uni.getStorageSync('token') || ''
if (!token.value) return if (!token.value) return
try { const s = uni.getStorageSync('userInfo'); if (s) userInfo.value = JSON.parse(s) } catch(e) {} try { const s = uni.getStorageSync('userInfo'); if (s) userInfo.value = JSON.parse(s) } catch(e) {}
loadStats() loadStats()
loadMemberStatus()
checkAdmin() checkAdmin()
fetchUserInfo()
}
const fetchUserInfo = async () => {
try {
const res = await uni.request({ url: api('/user/info'), method: 'GET', header: { 'Authorization': `Bearer ${token.value}` } })
if (res.statusCode === 200 && res.data) {
userInfo.value = res.data
uni.setStorageSync('userInfo', JSON.stringify(res.data))
}
} catch(e) { /* silent */ }
} }
onMounted(refreshState) onMounted(refreshState)
onShow(refreshState) onShow(refreshState)
const loadMemberStatus = async () => {
if (!token.value) return
try {
const res = await uni.request({ url: api('/member/status'), method: 'GET', header: { 'Authorization': `Bearer ${token.value}` } })
if (res.statusCode >= 200 && res.statusCode < 300 && res.data) {
memberInfo.value = { plan: res.data.plan || 'free', planName: res.data.planName || '免费版', remaining: res.data.remaining ?? 0, gravity: res.data.gravity ?? 0 }
}
} catch(e) { /* silent */ }
}
const loadStats = async () => { const loadStats = async () => {
try { try {
const res = await uni.request({ url: api('/interview/stats/mine'), method: 'GET', header: { 'Authorization': `Bearer ${token.value}` } }) const res = await uni.request({ url: api('/interview/stats/mine'), method: 'GET', header: { 'Authorization': `Bearer ${token.value}` } })
@@ -126,10 +246,36 @@ const checkAdmin = () => {
} }
const goLogin = () => uni.navigateTo({ url: '/pages/login/login' }) const goLogin = () => uni.navigateTo({ url: '/pages/login/login' })
//
const showGetGravityModal = ref(false)
const openBuyModal = () => { showGetGravityModal.value = true }
const goH5Buy = () => {
const token = uni.getStorageSync('token') || ''
const url = `https://zhiyin.yzrcloud.cn/?buy=gravity${token ? '&token=' + token : ''}`
// #ifdef H5
uni.navigateTo({ url: '/pages/member/member' })
// #endif
// #ifdef MP-WEIXIN
uni.setClipboardData({
data: url,
success: () => {
uni.showToast({ title: '链接已复制,请在手机浏览器中打开', icon: 'none', duration: 3000 })
},
fail: () => {
uni.showToast({ title: '复制失败,请手动访问 zhiyin.yzrcloud.cn', icon: 'none', duration: 3000 })
},
})
// #endif
}
const goCareer = () => uni.navigateTo({ url: '/pages/career/career' })
const goHistory = () => uni.switchTab({ url: '/pages/history/history' }) const goHistory = () => uni.switchTab({ url: '/pages/history/history' })
const goReviewReview = () => uni.navigateTo({ url: '/pages/review/review' })
const goVip = () => uni.navigateTo({ url: '/pages/member/member' }) const goVip = () => uni.navigateTo({ url: '/pages/member/member' })
const goResume = () => uni.navigateTo({ url: '/pages/resume/resume' }) const goResume = () => uni.navigateTo({ url: '/pages/resume/resume' })
const goShare = () => uni.navigateTo({ url: '/pages/share/share' }) const goSharePage = () => uni.navigateTo({ url: '/pages/share/share' })
const goContributePage = () => uni.navigateTo({ url: '/pages/contribute/contribute' })
const goAdmin = () => uni.navigateTo({ url: '/pages/admin/admin' }) const goAdmin = () => uni.navigateTo({ url: '/pages/admin/admin' })
const goAbout = () => uni.navigateTo({ url: '/pages/about/about' }) const goAbout = () => uni.navigateTo({ url: '/pages/about/about' })
@@ -153,6 +299,7 @@ const doLogout = () => {
.profile-info { flex: 1; display: flex; flex-direction: column; } .profile-info { flex: 1; display: flex; flex-direction: column; }
.nickname { font-size: 34rpx; font-weight: 700; color: #FFFFFF; } .nickname { font-size: 34rpx; font-weight: 700; color: #FFFFFF; }
.plan-badge { font-size: 20rpx; color: rgba(255,255,255,0.9); background: rgba(255,255,255,0.2); padding: 4rpx 14rpx; border-radius: 8rpx; align-self: flex-start; margin-top: 8rpx; } .plan-badge { font-size: 20rpx; color: rgba(255,255,255,0.9); background: rgba(255,255,255,0.2); padding: 4rpx 14rpx; border-radius: 8rpx; align-self: flex-start; margin-top: 8rpx; }
.user-id { font-size: 20rpx; color: rgba(255,255,255,0.5); margin-top: 4rpx; }
.header-arrow { font-size: 36rpx; color: rgba(255,255,255,0.5); } .header-arrow { font-size: 36rpx; color: rgba(255,255,255,0.5); }
.stats-bar { display: flex; align-items: center; background: rgba(255,255,255,0.15); border-radius: var(--radius-lg); padding: 24rpx 0; } .stats-bar { display: flex; align-items: center; background: rgba(255,255,255,0.15); border-radius: var(--radius-lg); padding: 24rpx 0; }
.stat { flex: 1; display: flex; flex-direction: column; align-items: center; } .stat { flex: 1; display: flex; flex-direction: column; align-items: center; }
@@ -166,23 +313,74 @@ const doLogout = () => {
.guest-icon { font-size: 40rpx; } .guest-icon { font-size: 40rpx; }
.guest-info { flex: 1; } .guest-info { flex: 1; }
.guest-name { font-size: 30rpx; font-weight: 600; color: #FFFFFF; } .guest-name { font-size: 30rpx; font-weight: 600; color: #FFFFFF; }
.guest-hint { font-size: 22rpx; color: rgba(255,255,255,0.7); margin-top: 4rpx; }
.menu-area { padding: 0 32rpx 32rpx; margin-top: -40rpx; } /* 引力值卡片(代替原配额卡片) */
.gravity-card { margin: -48rpx 32rpx 16rpx; position: relative; z-index: 1; }
.gravity-card-inner {
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
border-radius: var(--radius-xl);
padding: 28rpx 24rpx 24rpx;
box-shadow: 0 8rpx 32rpx rgba(102,126,234,0.25);
}
.gravity-top-row { display: flex; align-items: baseline; justify-content: space-between; margin-bottom: 8rpx; }
.gravity-header { display: flex; align-items: center; gap: 8rpx; }
.gravity-icon { font-size: 32rpx; }
.gravity-label { font-size: 24rpx; color: rgba(255,255,255,0.8); font-weight: 500; }
.gravity-num { font-size: 48rpx; font-weight: 800; color: #FFFFFF; line-height: 1; }
.gravity-hint { font-size: 20rpx; color: rgba(255,255,255,0.6); margin-bottom: 20rpx; }
.gravity-actions { display: flex; gap: 16rpx; }
.gravity-btn {
flex: 1; text-align: center; padding: 18rpx 0; border-radius: var(--radius-md);
font-size: 26rpx; font-weight: 600; white-space: nowrap;
}
.gravity-btn.share { background: rgba(255,255,255,0.2); color: #FFFFFF; border: 2rpx solid rgba(255,255,255,0.3); }
.gravity-btn.h5buy { background: #FFFFFF; color: #667eea; }
.gravity-btn.contribute { background: rgba(255,255,255,0.15); color: #FFFFFF; border: 2rpx solid rgba(255,255,255,0.2); }
.gravity-btn:active { transform: scale(0.96); }
.menu-area { padding: 0 32rpx 32rpx; margin-top: 8rpx; }
.menu-group { background: #FFFFFF; border-radius: var(--radius-lg); overflow: hidden; margin-bottom: 24rpx; box-shadow: var(--shadow-sm); } .menu-group { background: #FFFFFF; border-radius: var(--radius-lg); overflow: hidden; margin-bottom: 24rpx; box-shadow: var(--shadow-sm); }
.menu-item { display: flex; align-items: center; padding: 28rpx 32rpx; border-bottom: 1rpx solid var(--color-border); } .menu-item { display: flex; align-items: center; padding: 28rpx 32rpx; border-bottom: 1rpx solid var(--color-border); }
.contact-btn-inner { width: 100%; background: transparent; border: none; border-radius: 0; padding: 0; margin: 0; line-height: inherit; font-size: inherit; text-align: left; display: flex; align-items: center; min-height: auto; }
.contact-btn-inner::after { border: none; }
.menu-item:last-child { border-bottom: none; } .menu-item:last-child { border-bottom: none; }
.menu-item:active { background: #F9FAFB; } .menu-item:active { background: #F9FAFB; }
.menu-icon-wrap { width: 60rpx; height: 60rpx; border-radius: var(--radius-md); display: flex; align-items: center; justify-content: center; margin-right: 20rpx; flex-shrink: 0; } .menu-icon-wrap { width: 60rpx; height: 60rpx; border-radius: var(--radius-md); display: flex; align-items: center; justify-content: center; margin-right: 20rpx; flex-shrink: 0; }
.menu-icon { font-size: 28rpx; } .menu-icon { font-size: 28rpx; }
.menu-text { flex: 1; font-size: 28rpx; color: var(--color-text); font-weight: 500; } .menu-text { flex: 1; font-size: 28rpx; color: var(--color-text); font-weight: 500; }
.menu-tag { font-size: 18rpx; color: #fff; background: var(--color-primary); padding: 2rpx 12rpx; border-radius: 20rpx; margin-right: 12rpx; }
.menu-arrow { font-size: 32rpx; color: #D1D5DB; } .menu-arrow { font-size: 32rpx; color: #D1D5DB; }
.wrap-blue { background: #EEF2FF; } .wrap-blue { background: #EEF2FF; }
.wrap-purple { background: #F5F3FF; } .wrap-purple { background: #F5F3FF; }
.wrap-green { background: #ECFDF5; } .wrap-green { background: #ECFDF5; }
.wrap-orange { background: #FFF7ED; } .wrap-orange { background: #FFF7ED; }
.wrap-gray { background: #F3F4F6; } .wrap-gray { background: #F3F4F6; }
.wrap-teal { background: #E6FFFA; }
.logout-wrap { margin-top: 8rpx; } .logout-wrap { margin-top: 8rpx; }
.logout-btn { background: #FFFFFF; color: var(--color-error); font-size: 28rpx; font-weight: 500; border-radius: var(--radius-md); height: 88rpx; line-height: 88rpx; border: 2rpx solid #FECACA; } .logout-btn { background: #FFFFFF; color: var(--color-error); font-size: 28rpx; font-weight: 500; border-radius: var(--radius-md); height: 88rpx; line-height: 88rpx; border: 2rpx solid #FECACA; }
.logout-btn:active { background: #FEF2F2; transform: scale(0.96); } .logout-btn:active { background: #FEF2F2; transform: scale(0.96); }
/* 弹窗通用 */
.modal-overlay { position: fixed; top: 0; left: 0; right: 0; bottom: 0; background: rgba(0,0,0,0.5); display: flex; align-items: center; justify-content: center; z-index: 100; }
.modal-content { background: #FFF; border-radius: var(--radius-xl); padding: 40rpx 32rpx; width: 600rpx; max-height: 80vh; overflow-y: auto; display: flex; flex-direction: column; align-items: center; gap: 16rpx; }
.modal-title { font-size: 30rpx; font-weight: 700; color: var(--color-text); }
.modal-hint { font-size: 22rpx; color: #6B7280; text-align: center; }
.modal-close { font-size: 24rpx; color: #9CA3AF; padding: 12rpx 24rpx; }
/* 获取引力值弹窗 - 分享/贡献方法 */
.gp-get-method {
width: 100%; display: flex; align-items: center; gap: 16rpx;
background: #F9FAFB; border-radius: var(--radius-md); padding: 20rpx;
border: 2rpx solid #E5E7EB; margin-top: 8rpx;
}
.gp-get-method:active { transform: scale(0.97); background: #F3F4F6; }
.gp-method-icon-wrap {
width: 64rpx; height: 64rpx; border-radius: var(--radius-md);
background: #EEF2FF; display: flex; align-items: center; justify-content: center; flex-shrink: 0;
}
.gp-method-icon { font-size: 32rpx; }
.gp-method-info { flex: 1; display: flex; flex-direction: column; gap: 4rpx; }
.gp-method-name { font-size: 26rpx; font-weight: 600; color: var(--color-text); }
.gp-method-desc { font-size: 20rpx; color: #6B7280; line-height: 1.4; }
.gp-method-arrow { font-size: 32rpx; color: #D1D5DB; }
</style> </style>
+16 -1
View File
@@ -22,7 +22,7 @@ async function request<T = any>(url: string, method: string = 'POST', data?: any
} }
} }
export const apiService = { const apiService = {
user: { user: {
sendCode: (phone: string) => request(API_ENDPOINTS.USER.SEND_CODE, 'POST', { phone }), sendCode: (phone: string) => request(API_ENDPOINTS.USER.SEND_CODE, 'POST', { phone }),
login: (phone: string, code: string) => request(API_ENDPOINTS.USER.LOGIN, 'POST', { phone, code }), login: (phone: string, code: string) => request(API_ENDPOINTS.USER.LOGIN, 'POST', { phone, code }),
@@ -90,6 +90,21 @@ export const apiService = {
records: () => request(API_ENDPOINTS.SHARE.RECORDS, 'GET', undefined, true), records: () => request(API_ENDPOINTS.SHARE.RECORDS, 'GET', undefined, true),
visitors: () => request(API_ENDPOINTS.SHARE.VISITORS, 'GET', undefined, true), visitors: () => request(API_ENDPOINTS.SHARE.VISITORS, 'GET', undefined, true),
}, },
review: {
list: (page = 1, limit = 20) =>
request(`${API_ENDPOINTS.REVIEW.LIST}?page=${page}&limit=${limit}`, 'GET', undefined, true),
detail: (id: string) => request(API_ENDPOINTS.REVIEW.DETAIL(id), 'GET', undefined, true),
delete: (id: string) => request(API_ENDPOINTS.REVIEW.DELETE(id), 'DELETE', undefined, true),
submitText: (position: string, text: string, company?: string) =>
request(API_ENDPOINTS.REVIEW.TEXT, 'POST', { position, text, company: company || '' }, true),
},
career: {
analyze: (profile: { major: string; grade?: string; interests?: string; gpa?: string; goal?: string }) =>
request(API_ENDPOINTS.CAREER.ANALYZE, 'POST', profile, true),
chat: (message: string, history: { role: string; content: string }[]) =>
request(API_ENDPOINTS.CAREER.CHAT, 'POST', { message, history }, true),
positions: () => request(API_ENDPOINTS.CAREER.POSITIONS, 'GET', undefined, true),
},
} }
export default apiService export default apiService
+3
View File
@@ -0,0 +1,3 @@
User-agent: *
Allow: /
Sitemap: https://zhiyin.yzrcloud.cn/sitemap.xml
+8
View File
@@ -0,0 +1,8 @@
<?xml version="1.0" encoding="UTF-8"?>
<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9">
<url>
<loc>https://zhiyin.yzrcloud.cn</loc>
<changefreq>daily</changefreq>
<priority>1.0</priority>
</url>
</urlset>
+9
View File
@@ -1,8 +1,17 @@
import { defineConfig } from 'vite'; import { defineConfig } from 'vite';
import { execSync } from 'child_process';
import uni from '@dcloudio/vite-plugin-uni'; import uni from '@dcloudio/vite-plugin-uni';
let appVersion = '1.0.0';
try {
appVersion = execSync('git describe --tags --abbrev=0 2>/dev/null || echo "1.0.0"', { encoding: 'utf8' }).trim().replace(/^v/, '');
} catch {}
export default defineConfig({ export default defineConfig({
plugins: [uni()], plugins: [uni()],
define: {
__APP_VERSION__: JSON.stringify(appVersion),
},
server: { server: {
port: 8888, port: 8888,
strictPort: true, strictPort: true,