Compare commits
11 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 8ee27fdd32 | |||
| a1e1f0b3c3 | |||
| 2fbab1072f | |||
| c2ba810a02 | |||
| 3f1239c35e | |||
| 1be5b34906 | |||
| c58bb27575 | |||
| e0de29fdd0 | |||
| 6a3cc8544e | |||
| c161ffbc3c | |||
| 7e1bf669ab |
@@ -1,10 +1,10 @@
|
|||||||
{
|
{
|
||||||
"sessionID": "ses_15492f54bffepl01q9FkaRyN28",
|
"sessionID": "ses_15492f54bffepl01q9FkaRyN28",
|
||||||
"updatedAt": "2026-06-16T01:20:56.855Z",
|
"updatedAt": "2026-06-19T07:14:26.627Z",
|
||||||
"sources": {
|
"sources": {
|
||||||
"background-task": {
|
"background-task": {
|
||||||
"state": "idle",
|
"state": "idle",
|
||||||
"updatedAt": "2026-06-16T01:20:56.855Z"
|
"updatedAt": "2026-06-19T07:14:26.627Z"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -21,9 +21,10 @@ zhiyin/
|
|||||||
├── zhiyin-app/ # uni-app 3.x 前端 (H5 + 微信小程序)
|
├── zhiyin-app/ # uni-app 3.x 前端 (H5 + 微信小程序)
|
||||||
│ └── src/
|
│ └── src/
|
||||||
│ ├── pages/ # 20 个页面 (pages.json 路由)
|
│ ├── pages/ # 20 个页面 (pages.json 路由)
|
||||||
|
│ ├── composables/ # 可复用组合式函数(如 useGravityPurchase)
|
||||||
│ ├── services/api.ts # API 调用封装 (uni.request)
|
│ ├── services/api.ts # API 调用封装 (uni.request)
|
||||||
│ ├── config.ts # 端点定义 + api() 辅助函数
|
│ ├── config.ts # 端点定义 + api() 辅助函数
|
||||||
│ └── App.vue # 设计 Token + 全局样式
|
│ └── App.vue # 设计 Token + 全局样式 + H5 URL 参数处理
|
||||||
└── docs/ # 产品/架构/部署/路线图文档
|
└── docs/ # 产品/架构/部署/路线图文档
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -40,7 +41,7 @@ zhiyin/
|
|||||||
| `payment` | 微信支付 v3(Native + JSAPI + 回调) |
|
| `payment` | 微信支付 v3(Native + JSAPI + 回调) |
|
||||||
| `progress` | 进步轨迹雷达图 / 打卡日历 / 行业基准 / 岗位匹配 |
|
| `progress` | 进步轨迹雷达图 / 打卡日历 / 行业基准 / 岗位匹配 |
|
||||||
| `contribution` | 面经贡献 + 公司题库(数据飞轮核心) |
|
| `contribution` | 面经贡献 + 公司题库(数据飞轮核心) |
|
||||||
| `schedule` | 定时任务:VIP 过期降级、每日一题推送、微信 token 刷新 |
|
| `schedule` | 定时任务:VIP 过期降级、每日一题推送、微信 token 刷新、月度引力值补给 |
|
||||||
| `share` | 分享链接生成 / 访问追踪 / 积分奖励 |
|
| `share` | 分享链接生成 / 访问追踪 / 积分奖励 |
|
||||||
| `tts` | 语音合成(TTS) |
|
| `tts` | 语音合成(TTS) |
|
||||||
| `admin` | 管理后台 API |
|
| `admin` | 管理后台 API |
|
||||||
@@ -96,15 +97,22 @@ zhiyin/
|
|||||||
- 环境变量: `AI_PRIMARY_KEY`, `AI_BACKUP_KEY`
|
- 环境变量: `AI_PRIMARY_KEY`, `AI_BACKUP_KEY`
|
||||||
|
|
||||||
### 支付(微信支付 v3)
|
### 支付(微信支付 v3)
|
||||||
- Native 支付(H5 扫码): `POST /payment/create`
|
- Native 支付(H5 扫码): `POST /payment/create-product`(按量购买引力值)
|
||||||
- JSAPI 支付(小程序内): `POST /payment/jsapi`
|
- JSAPI 支付(小程序内): `POST /payment/jsapi-product`(按量购买引力值)
|
||||||
- 支付回调: `POST /payment/notify`(@Public,验签 + 解密 + 自动开会员)
|
- 支付回调: `POST /payment/notify`(@Public,验签 + 解密 + 自动到账)
|
||||||
|
- 支付结果轮询: `GET /payment/check/:outTradeNo`
|
||||||
|
- 产品定价: `GET /member/plans`(含 products 字段,定义引力值单价和赠送量)
|
||||||
- 需要微信商户证书文件(通过 postbuild 复制到 dist)
|
- 需要微信商户证书文件(通过 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/
|
# 路径: backend/
|
||||||
@@ -130,12 +138,12 @@ npm test # 前端单元测试(vitest,7 个)
|
|||||||
### 构建检查
|
### 构建检查
|
||||||
```bash
|
```bash
|
||||||
# 后端构建(注意 OOM:需 NODE_OPTIONS="--max-old-space-size=2048")
|
# 后端构建(注意 OOM:需 NODE_OPTIONS="--max-old-space-size=2048")
|
||||||
cd backend && NODE_OPTIONS="--max-old-space-size=2048" npx nest build
|
cd backend && npm run build
|
||||||
```
|
```
|
||||||
|
|
||||||
### 部署后端
|
### 部署后端
|
||||||
```bash
|
```bash
|
||||||
cd backend && npx nest build
|
cd backend && npm run build
|
||||||
cp -rf dist/* /www/wwwroot/server/zhiyin/backend/dist/
|
cp -rf dist/* /www/wwwroot/server/zhiyin/backend/dist/
|
||||||
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
|
||||||
@@ -144,7 +152,7 @@ sleep 3 && curl -s http://localhost:3006/api/user/wx-login -X POST -H "Content-T
|
|||||||
|
|
||||||
### 部署前端 H5
|
### 部署前端 H5
|
||||||
```bash
|
```bash
|
||||||
cd zhiyin-app && npx uni build
|
cd zhiyin-app && npm run build:h5
|
||||||
rm -rf /www/wwwroot/zhiyin.yzrcloud.cn/assets
|
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/index.html /www/wwwroot/zhiyin.yzrcloud.cn/
|
||||||
cp -r dist/build/h5/assets /www/wwwroot/zhiyin.yzrcloud.cn/
|
cp -r dist/build/h5/assets /www/wwwroot/zhiyin.yzrcloud.cn/
|
||||||
@@ -153,9 +161,11 @@ chown -R www:www /www/wwwroot/zhiyin.yzrcloud.cn/index.html /www/wwwroot/zhiyin.
|
|||||||
grep -oP '["'"'"']([a-zA-Z0-9_-]+\.[a-z]+(\.js|\.css|\.png|\.svg))["'"'"']' /www/wwwroot/zhiyin.yzrcloud.cn/assets/index-*.js | sort -u
|
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
|
```bash
|
||||||
cd zhiyin-app && npm run build:mp-weixin && node scripts/upload-mp.js
|
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 构建
|
||||||
```
|
```
|
||||||
|
|
||||||
---
|
---
|
||||||
@@ -171,26 +181,27 @@ cd zhiyin-app && npm run build:mp-weixin && node scripts/upload-mp.js
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## 五、定时任务(3 个 cron,在 schedule 模块)
|
## 五、定时任务(4 个 cron,在 schedule 模块)
|
||||||
|
|
||||||
| 服务 | 周期 | 职责 |
|
| 服务 | 周期 | 职责 |
|
||||||
|------|------|------|
|
|------|------|------|
|
||||||
| `VipExpiryService` | 每日 00:00 | 扫描过期 VIP 并降级为 free 计划 |
|
| `VipExpiryService` | 每日 00:00 | 扫描过期 VIP 并降级为 free 计划 |
|
||||||
| `DailyQuestionPushService` | 每日 09:00 | 通过微信订阅消息推送每日一题(需配置模板 ID) |
|
| `DailyQuestionPushService` | 每日 09:00 | 通过微信订阅消息推送每日一题(需配置模板 ID) |
|
||||||
| `WechatTokenService` | 每 2 小时 | 刷新微信 access_token(缓存到 Redis) |
|
| `WechatTokenService` | 每 2 小时 | 刷新微信 access_token(缓存到 Redis) |
|
||||||
|
| `GravityTopUpService` | 每日 02:00 | 给所有未过期的成长版/冲刺版用户补给月度引力值 |
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
## 六、项目状态与开发阶段
|
## 六、项目状态与开发阶段
|
||||||
|
|
||||||
**当前**: Phase 0.5 完成,Phase 1(MVP 上线)进行中
|
**当前**: Phase 1(MVP 上线)进行中 — v1.0.16
|
||||||
|
|
||||||
| 阶段 | 状态 | 关键交付 |
|
| 阶段 | 状态 | 关键交付 |
|
||||||
|------|------|---------|
|
|------|------|---------|
|
||||||
| Phase 0: 战略升级 | ✅ 完成 | 定价重构(免费 + ¥19.9/月),三层壁垒设计 |
|
| Phase 0: 战略升级 | ✅ 完成 | 定价重构(免费 + ¥19.9/月),三层壁垒设计 |
|
||||||
| Phase 0.5: 壁垒构建 | ✅ 完成 | 数据飞轮(面经贡献+题库),留存入围(进步轨迹+打卡日历+每日一题) |
|
| Phase 0.5: 壁垒构建 | ✅ 完成 | 数据飞轮(面经贡献+题库),留存入围(进步轨迹+打卡日历+每日一题) |
|
||||||
| Phase 1: MVP 上线 | 🚧 当前 | 小程序 v1.0.11 已上传、H5 已部署、生产模式已启用、SMTP 邮箱验证码已配置 |
|
| Phase 1: MVP 上线 | 🚧 当前 | 小程序 v1.0.16 已上传、引力值体系统一(订阅制改为按量购买)、管理后台完善、H5 已部署、生产模式已启用 |
|
||||||
| Phase 1.5: 商业化 | 📋 规划 | 冲刺版 ¥49.9/月、每日一题定时推送、PMF 验证 |
|
| Phase 1.5: 商业化 | 📋 规划 | 引力值运营策略、每日一题定时推送、PMF 验证 |
|
||||||
| Phase 2: 增强 + 题库 | 📋 规划 | 50+ 校招岗位、技能缺口分析、公司真题库建设 |
|
| Phase 2: 增强 + 题库 | 📋 规划 | 50+ 校招岗位、技能缺口分析、公司真题库建设 |
|
||||||
| Phase 3: 秋招冲刺 | 📋 规划 | 高校合作、B 端服务、KOC 推广 |
|
| Phase 3: 秋招冲刺 | 📋 规划 | 高校合作、B 端服务、KOC 推广 |
|
||||||
|
|
||||||
@@ -251,7 +262,7 @@ VITE_APP_NAME=AI磁场
|
|||||||
|
|
||||||
- 远程仓库: `http://127.0.0.1:2999/txai-dev/zhiyin.git`(本机 Gitea,带 token 认证)
|
- 远程仓库: `http://127.0.0.1:2999/txai-dev/zhiyin.git`(本机 Gitea,带 token 认证)
|
||||||
- 默认分支: `master`
|
- 默认分支: `master`
|
||||||
- 最新 tag: `v1.0.11`(小程序上传版本号源自 git tag)
|
- 最新 tag: `v1.0.16`(小程序上传版本号源自 git tag)
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
@@ -265,9 +276,12 @@ VITE_APP_NAME=AI磁场
|
|||||||
6. **API 限流**: 100 次/60 秒(在 `app.module.ts` 中配置),注意避免在定时任务和批量操作中被限
|
6. **API 限流**: 100 次/60 秒(在 `app.module.ts` 中配置),注意避免在定时任务和批量操作中被限
|
||||||
7. **验证码**: 生产模式(`NODE_ENV=production`)使用真实 SMTP 发邮件验证码;非生产模式手机验证码固定为 `123456`、邮件验证码在响应中返回 `devCode`
|
7. **验证码**: 生产模式(`NODE_ENV=production`)使用真实 SMTP 发邮件验证码;非生产模式手机验证码固定为 `123456`、邮件验证码在响应中返回 `devCode`
|
||||||
8. **MongoDB**: 8 个核心集合 + 2 个分享集合
|
8. **MongoDB**: 8 个核心集合 + 2 个分享集合
|
||||||
9. **api.ts 陷阱**: 对象字面量必须在 `export const apiService = {` 或 `const apiService = { ... export default apiService` 中包裹,否则 uni-app 构建报错 `Expected ";" but found ":"`。git pull 后经常丢失这行声明,需手动补回
|
9. **引力值体系**: 所有计划统一走引力值消耗(面试 5、优化 3、下载 2)。VIP 不再免额度,成长版每月 250 引力值,冲刺版每月 600 引力值,每日凌晨 2 点定时补给。免费用户注册送 5 引力值。小程序内通过分享得引力值/贡献面经/复制官网链接到浏览器打开购买三种方式获取引力值;H5 直接扫码支付按量购买(¥5/份)。
|
||||||
10. **H5 构建 assets 清理**: `assets/` 中的旧 hash 文件不能随意删除——`index-*.js`(主 bundle)动态 import 了所有 page chunk,删除仍在引用的文件会导致浏览器 `NS_ERROR_CORRUPTED_CONTENT`
|
10. **api.ts 陷阱**: 对象字面量必须在 `export const apiService = {` 或 `const apiService = { ... export default apiService` 中包裹,否则 uni-app 构建报错 `Expected ";" but found ":"`。git pull 后经常丢失这行声明,需手动补回
|
||||||
11. **管理后台自动验证**: `admin.vue` 中 `onMounted` 自动调用 `doVerify()`,进入后台即检测 JWT 中 `role` 是否为 `admin`,不再需要手动点击"验证管理员身份"按钮
|
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` 参数自动登录跳转)
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
@@ -295,7 +309,7 @@ VITE_APP_NAME=AI磁场
|
|||||||
|
|
||||||
### Step 4: 测试验证
|
### Step 4: 测试验证
|
||||||
```bash
|
```bash
|
||||||
cd backend && NODE_OPTIONS="--max-old-space-size=2048" npx nest build
|
cd backend && npm run build
|
||||||
npm test -- --forceExit --detectOpenHandles
|
npm test -- --forceExit --detectOpenHandles
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -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)
|
||||||
@@ -113,7 +113,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 +122,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)
|
||||||
}
|
}
|
||||||
@@ -279,29 +279,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 +409,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 +422,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 次/月'],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,44 +22,47 @@ export class AiService {
|
|||||||
|
|
||||||
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 (deepseek-v4-flash on sensenova)
|
// 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 primary fallback...`)
|
this.logger.warn(`Primary AI failed: ${(e as Error).message}, trying primary fallback...`)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try primary fallback model (sensenova-6.7-flash-lite, same provider)
|
// Try primary fallback model (sensenova-6.7-flash-lite, same provider)
|
||||||
try {
|
try {
|
||||||
const result = await this.callApi(this.primaryUrl, this.primaryKey, this.primaryFallbackModel, systemPrompt, userMessage, temperature, maxTokens)
|
const result = await this.callApi(this.primaryUrl, this.primaryKey, this.primaryFallbackModel, systemPrompt, userMessage, temperature, maxTokens, 60000)
|
||||||
if (result) return result
|
if (result) return result
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
this.logger.warn(`Primary fallback AI also failed: ${(e as Error).message}, trying backup...`)
|
this.logger.warn(`Primary fallback AI also failed: ${(e as Error).message}, trying backup...`)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try backup AI (NVIDIA)
|
// Try backup AI (NVIDIA - meta/llama-3.1-8b-instruct)
|
||||||
try {
|
try {
|
||||||
const result = await this.callApi(this.backupUrl, this.backupKey, this.backupModel, systemPrompt, userMessage, temperature, maxTokens)
|
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 \u670d\u52a1\u6682\u65f6\u4e0d\u53ef\u7528\uff0c\u8bf7\u7a0d\u540e\u91cd\u8bd5")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
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`,
|
||||||
{
|
{
|
||||||
@@ -76,11 +79,17 @@ export class AiService {
|
|||||||
"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
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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 {}
|
||||||
|
|||||||
@@ -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: ['免费版全部权益', 'AI 数字人面试无限次', '详细面试报告(四维评分)', '进步轨迹雷达图 + 打卡', '每日一题推送', '参考回答思路', '公司真题库', '简历优化 20 次/月', '简历下载 10 次/月'] },
|
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: ['成长版全部权益', 'AI 语音分析(语气词/语速检测)', '技能缺口分析报告', '学习路径推荐', '真人导师 1v1 点评(每月 1 次)', '简历精修(每月 1 次)', '内推优先', '简历优化 50 次/月', '简历下载 30 次/月'] },
|
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 } },
|
||||||
|
|||||||
@@ -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}`)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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,119 +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
|
|
||||||
|
|
||||||
// Try paid credits first
|
// 迁移旧字段
|
||||||
const paid = await this.userModel.findOneAndUpdate(
|
if ((user.gravity ?? 0) <= 0 && this.hasOldCredits(user)) {
|
||||||
{ _id: userId, resumeOptimizeCredits: { $gt: 0 } },
|
await this.migrateOldCredits(userId)
|
||||||
{ $inc: { resumeOptimizeCredits: -1 } },
|
}
|
||||||
).exec()
|
|
||||||
if (paid) return
|
|
||||||
|
|
||||||
// Try old remaining credits (backward compat)
|
// 免费优化次数
|
||||||
const oldRemaining = await this.userModel.findOneAndUpdate(
|
|
||||||
{ _id: userId, remaining: { $gt: 0 } },
|
|
||||||
{ $inc: { remaining: -1 } },
|
|
||||||
).exec()
|
|
||||||
if (oldRemaining) 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()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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)
|
||||||
|
|||||||
@@ -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'
|
||||||
|
|||||||
@@ -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,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+12
-10
@@ -1,8 +1,8 @@
|
|||||||
# 职引项目 · 状态报告 v4.5
|
# 职引项目 · 状态报告 v4.6
|
||||||
|
|
||||||
> **项目版本**: v4.5
|
> **项目版本**: v4.6
|
||||||
> **更新时间**: 2026-06-17
|
> **更新时间**: 2026-06-19
|
||||||
> **项目状态**: ✅ 面试复盘上线 + AI 择业顾问 MVP
|
> **项目状态**: ✅ 引力值体系统一 + 管理后台完善
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
@@ -76,8 +76,10 @@
|
|||||||
| 功能 | 后端 | 前端 | 状态 |
|
| 功能 | 后端 | 前端 | 状态 |
|
||||||
|------|------|------|------|
|
|------|------|------|------|
|
||||||
| 免费版额度(日2次/5轮) | ✅ | ✅ | **完成** |
|
| 免费版额度(日2次/5轮) | ✅ | ✅ | **完成** |
|
||||||
| 成长版 ¥19.9/月 | ✅ | ✅ | **完成** |
|
| 成长版 ¥19.9/月(250 引力值) | ✅ | ✅ | **完成** |
|
||||||
| 冲刺版 ¥49.9/月(含权益扣减) | ✅ | ✅ | **完成** |
|
| 冲刺版 ¥49.9/月(600 引力值) | ✅ | ✅ | **完成** |
|
||||||
|
| 引力值统一体系(取消 VIP 无限面试) | ✅ | ✅ | **完成** |
|
||||||
|
| 会员月度引力值自动配发(cron) | ✅ | N/A | **完成** |
|
||||||
| 微信支付 Native QR / JSAPI | ✅ | ✅ H5+MP | **完成** |
|
| 微信支付 Native QR / JSAPI | ✅ | ✅ H5+MP | **完成** |
|
||||||
| 支付回调/自动开会员 | ✅ | N/A | **完成** |
|
| 支付回调/自动开会员 | ✅ | N/A | **完成** |
|
||||||
| 每日一题定时推送 | ✅ | N/A | **完成**(需配置模板ID) |
|
| 每日一题定时推送 | ✅ | N/A | **完成**(需配置模板ID) |
|
||||||
@@ -174,7 +176,7 @@
|
|||||||
| `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 评析 + 口语分析 |
|
| `interview-review` | controller + service + schema + asr service | ✅ | 面试复盘:音频 ASR + AI 评析 + 口语分析 |
|
||||||
| `career-advice` | controller + service + module | ✅ | AI 择业顾问:专业分析 + 岗位匹配 + 多轮对话 |
|
| `career-advice` | controller + service + module | ✅ | AI 择业顾问:专业分析 + 岗位匹配 + 多轮对话 |
|
||||||
| `admin` | controller + module | ✅ | 管理后台 |
|
| `admin` | controller + module | ✅ | 管理后台 |
|
||||||
@@ -200,7 +202,7 @@
|
|||||||
| 面试复盘 | review/review | ✅ 三种模式(列表/上传/报告) |
|
| 面试复盘 | review/review | ✅ 三种模式(列表/上传/报告) |
|
||||||
| 择业顾问 | career/career | ✅ AI 专业分析 + 岗位匹配 + 多轮对话 |
|
| 择业顾问 | career/career | ✅ AI 专业分析 + 岗位匹配 + 多轮对话 |
|
||||||
| 实习搜索 | internship/internship | ✅ 热门岗位 |
|
| 实习搜索 | internship/internship | ✅ 热门岗位 |
|
||||||
| 管理后台 | admin/admin | ✅ 仪表盘 |
|
| 管理后台 | admin/admin | ✅ 仪表盘/用户/面试/简历/订单/定价/分享/岗位/诊断/管理员 · 全 CRUD |
|
||||||
| 关于/协议/隐私 | about/agreement/privacy | ✅ |
|
| 关于/协议/隐私 | about/agreement/privacy | ✅ |
|
||||||
|
|
||||||
---
|
---
|
||||||
@@ -217,11 +219,11 @@
|
|||||||
---
|
---
|
||||||
|
|
||||||
## 十、变更记录
|
## 十、变更记录
|
||||||
| 2026-06-17 | v4.5 | AI 择业顾问 MVP:后端模块 + 前端职业分析页面 + 热门岗位联动 | AI |
|
|
||||||
|
|
||||||
| 日期 | 版本 | 变更内容 | 操作者 |
|
| 日期 | 版本 | 变更内容 | 操作者 |
|
||||||
|------|------|----------|--------|
|
|------|------|----------|--------|
|
||||||
| 2026-06-02 | v1.0 | 项目状态初版 | 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 |
|
||||||
|
|||||||
@@ -6,8 +6,28 @@ onLaunch(() => {
|
|||||||
// #ifdef MP-WEIXIN
|
// #ifdef MP-WEIXIN
|
||||||
initPrivacy()
|
initPrivacy()
|
||||||
// #endif
|
// #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
|
// #ifdef MP-WEIXIN
|
||||||
function initPrivacy() {
|
function initPrivacy() {
|
||||||
if (wx.onNeedPrivacyAuthorization) {
|
if (wx.onNeedPrivacyAuthorization) {
|
||||||
|
|||||||
@@ -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,
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
{
|
{
|
||||||
"name": "宇之然AI磁场",
|
"name": "宇之然AI磁场",
|
||||||
"appid": "__UNI__DEV__",
|
"appid": "__UNI__DEV__",
|
||||||
"versionName": "1.0.12",
|
"versionName": "1.0.15",
|
||||||
"versionCode": "112",
|
"versionCode": "115",
|
||||||
"description": "职引 - 宇之然AI磁场旗下AI模拟面试平台,提供AI面试官模拟练习、简历智能优化、大厂面经题库,助你轻松应对校招面试。",
|
"description": "职引 - 宇之然AI磁场旗下AI模拟面试平台,提供AI面试官模拟练习、简历智能优化、大厂面经题库,助你轻松应对校招面试。",
|
||||||
"h5": {
|
"h5": {
|
||||||
"title": "职引 - AI模拟面试 | 宇之然AI磁场",
|
"title": "职引 - AI模拟面试 | 宇之然AI磁场",
|
||||||
|
|||||||
@@ -22,7 +22,8 @@
|
|||||||
<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 === 'positions' }" @click="switchTab('positions')">岗位</text>
|
<text class="tab" :class="{ active: tab === 'positions' }" @click="switchTab('positions')">岗位</text>
|
||||||
<text class="tab" :class="{ active: tab === 'admins' }" @click="switchTab('admins')">管理</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>
|
||||||
|
|
||||||
<!-- 概览 -->
|
<!-- 概览 -->
|
||||||
@@ -66,10 +67,8 @@
|
|||||||
</view>
|
</view>
|
||||||
<view class="user-badges">
|
<view class="user-badges">
|
||||||
<text class="user-plan" :class="{ vip: u.plan === 'growth' || u.plan === 'sprint' }">{{ u.plan === 'growth' || u.plan === 'sprint' ? '会员' : '免费' }}</text>
|
<text class="user-plan" :class="{ vip: u.plan === 'growth' || u.plan === 'sprint' }">{{ u.plan === 'growth' || u.plan === 'sprint' ? '会员' : '免费' }}</text>
|
||||||
<text class="user-credit">面试:{{ u.interviewCredits ?? 0 }}</text>
|
<text class="user-credit">引力值:{{ u.gravity ?? 0 }}</text>
|
||||||
<text class="user-credit">优化:{{ u.resumeOptimizeCredits ?? 0 }}</text>
|
<text class="user-credit share" v-if="u.shareCredits > 0">分享:{{ u.shareCredits }}</text>
|
||||||
<text class="user-credit">下载:{{ u.resumeDownloadCredits ?? 0 }}</text>
|
|
||||||
<text class="user-credit share">分享:{{ u.shareCredits ?? 0 }}</text>
|
|
||||||
</view>
|
</view>
|
||||||
<view class="user-actions">
|
<view class="user-actions">
|
||||||
<text class="user-action-btn" v-if="u.plan === 'free'" @click="setVip(u._id)">设为会员</text>
|
<text class="user-action-btn" v-if="u.plan === 'free'" @click="setVip(u._id)">设为会员</text>
|
||||||
@@ -83,6 +82,12 @@
|
|||||||
|
|
||||||
<!-- 面试 -->
|
<!-- 面试 -->
|
||||||
<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">
|
||||||
@@ -95,13 +100,20 @@
|
|||||||
<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>
|
||||||
|
<text class="iv-time">{{ iv.createdAt?.slice(0,16).replace('T',' ') }}</text>
|
||||||
</view>
|
</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">
|
||||||
@@ -114,6 +126,9 @@
|
|||||||
<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>
|
<text class="resume-time">{{ r.createdAt?.slice(0,10) }}</text>
|
||||||
|
<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>
|
||||||
@@ -142,8 +157,10 @@
|
|||||||
{{ 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>
|
<text class="order-time">{{ o.createdAt ? o.createdAt.slice(0,16).replace('T',' ') : '--' }}</text>
|
||||||
<view class="order-actions" v-if="o.status === 'pending'">
|
<view class="order-actions">
|
||||||
<text class="sync-btn" @click="syncOrder(o.outTradeNo)">同步</text>
|
<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>
|
||||||
</view>
|
</view>
|
||||||
@@ -174,12 +191,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" />
|
||||||
@@ -205,7 +242,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">
|
||||||
@@ -298,13 +337,27 @@
|
|||||||
<text class="loading-text" v-if="posLoading">加载中...</text>
|
<text class="loading-text" v-if="posLoading">加载中...</text>
|
||||||
</view>
|
</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>
|
||||||
@@ -331,6 +384,10 @@
|
|||||||
<text class="cfg-val">{{ posForm.active ? '启用' : '停用' }}</text>
|
<text class="cfg-val">{{ posForm.active ? '启用' : '停用' }}</text>
|
||||||
</picker>
|
</picker>
|
||||||
</view>
|
</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">
|
<view class="modal-actions">
|
||||||
<button class="modal-btn cancel" @click="closePositionModal">取消</button>
|
<button class="modal-btn cancel" @click="closePositionModal">取消</button>
|
||||||
<button class="modal-btn confirm" @click="savePosition">保存</button>
|
<button class="modal-btn confirm" @click="savePosition">保存</button>
|
||||||
@@ -338,6 +395,29 @@
|
|||||||
</view>
|
</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">
|
||||||
@@ -364,11 +444,10 @@
|
|||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, computed, onMounted } 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)
|
||||||
@@ -397,9 +476,10 @@ 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)
|
||||||
@@ -407,6 +487,14 @@ 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
|
// Position management
|
||||||
const positions = ref([])
|
const positions = ref([])
|
||||||
@@ -420,6 +508,8 @@ const posForm = reactive({
|
|||||||
sort: 0,
|
sort: 0,
|
||||||
active: true,
|
active: true,
|
||||||
category: 'ai',
|
category: 'ai',
|
||||||
|
description: '',
|
||||||
|
requirements: '',
|
||||||
})
|
})
|
||||||
|
|
||||||
const growthPriceDisplay = computed(() => growthPriceTemp.value.toFixed(1))
|
const growthPriceDisplay = computed(() => growthPriceTemp.value.toFixed(1))
|
||||||
@@ -441,12 +531,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') || ''
|
||||||
|
|
||||||
@@ -492,6 +637,7 @@ const switchTab = (t) => {
|
|||||||
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 () => {
|
||||||
@@ -514,22 +660,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 {
|
||||||
@@ -545,6 +724,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 {
|
||||||
@@ -635,6 +823,8 @@ const openPositionModal = (position) => {
|
|||||||
posForm.sort = position.sort ?? 0
|
posForm.sort = position.sort ?? 0
|
||||||
posForm.active = position.active ?? true
|
posForm.active = position.active ?? true
|
||||||
posForm.category = position.category || 'traditional'
|
posForm.category = position.category || 'traditional'
|
||||||
|
posForm.description = position.description || ''
|
||||||
|
posForm.requirements = position.requirements || ''
|
||||||
posModal.value = { show: true, isNew: false }
|
posModal.value = { show: true, isNew: false }
|
||||||
} else {
|
} else {
|
||||||
posForm.name = ''
|
posForm.name = ''
|
||||||
@@ -644,6 +834,8 @@ const openPositionModal = (position) => {
|
|||||||
posForm.sort = positions.value.length + 1
|
posForm.sort = positions.value.length + 1
|
||||||
posForm.active = true
|
posForm.active = true
|
||||||
posForm.category = 'ai'
|
posForm.category = 'ai'
|
||||||
|
posForm.description = ''
|
||||||
|
posForm.requirements = ''
|
||||||
posModal.value = { show: true, isNew: true }
|
posModal.value = { show: true, isNew: true }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -749,12 +941,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 }
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -766,12 +953,10 @@ 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()
|
||||||
@@ -790,8 +975,8 @@ onMounted(() => { doVerify() })
|
|||||||
.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); }
|
||||||
@@ -860,6 +1045,7 @@ onMounted(() => { doVerify() })
|
|||||||
.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; }
|
||||||
@@ -901,4 +1087,8 @@ onMounted(() => { doVerify() })
|
|||||||
.pos-mgr-btn { font-size: 20rpx; padding: 4rpx 16rpx; border-radius: var(--radius-round); }
|
.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.edit { color: var(--color-primary); border: 2rpx solid var(--color-primary); }
|
||||||
.pos-mgr-btn.del { color: #EF4444; border: 2rpx solid #EF4444; }
|
.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; }
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -150,7 +150,7 @@ const doAnalyze = async () => {
|
|||||||
data: { ...profile },
|
data: { ...profile },
|
||||||
header: { 'Authorization': `Bearer ${token.value}`, 'Content-Type': 'application/json' },
|
header: { 'Authorization': `Bearer ${token.value}`, 'Content-Type': 'application/json' },
|
||||||
})
|
})
|
||||||
if (res.statusCode === 200) {
|
if (res.statusCode >= 200 && res.statusCode < 300) {
|
||||||
if (res.data.error) {
|
if (res.data.error) {
|
||||||
error.value = res.data.error
|
error.value = res.data.error
|
||||||
return
|
return
|
||||||
@@ -184,7 +184,7 @@ const doChat = async () => {
|
|||||||
data: { message: msg, history: chatHistory.value.slice(0, -1) },
|
data: { message: msg, history: chatHistory.value.slice(0, -1) },
|
||||||
header: { 'Authorization': `Bearer ${token.value}`, 'Content-Type': 'application/json' },
|
header: { 'Authorization': `Bearer ${token.value}`, 'Content-Type': 'application/json' },
|
||||||
})
|
})
|
||||||
if (res.statusCode === 200) {
|
if (res.statusCode >= 200 && res.statusCode < 300) {
|
||||||
chatHistory.value.push({ role: 'assistant', content: res.data.reply || (res.data.error || '') })
|
chatHistory.value.push({ role: 'assistant', content: res.data.reply || (res.data.error || '') })
|
||||||
} else {
|
} else {
|
||||||
chatHistory.value.push({ role: 'assistant', content: '回复失败,请稍后重试' })
|
chatHistory.value.push({ role: 'assistant', content: '回复失败,请稍后重试' })
|
||||||
@@ -211,12 +211,13 @@ const goInterview = (position) => {
|
|||||||
.hero-title { font-size: 40rpx; font-weight: 700; color: var(--color-text); margin-top: 16rpx; }
|
.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; }
|
.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); }
|
.form-card { background: #fff; border-radius: var(--radius-lg); padding: 32rpx; box-shadow: var(--shadow-sm); overflow: hidden; }
|
||||||
.form-group { margin-bottom: 28rpx; }
|
.form-group { margin-bottom: 28rpx; width: 100%; }
|
||||||
.form-group:last-child { margin-bottom: 0; }
|
.form-group:last-child { margin-bottom: 0; }
|
||||||
.form-label { font-size: 26rpx; font-weight: 600; color: var(--color-text); display: block; margin-bottom: 12rpx; }
|
.form-label { font-size: 26rpx; font-weight: 600; color: var(--color-text); display: block; margin-bottom: 12rpx; }
|
||||||
.required { color: var(--color-error); }
|
.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; }
|
.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; }
|
.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; }
|
.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; }
|
||||||
|
|
||||||
|
|||||||
@@ -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.interviewCredits > 0 ? '剩余 ' + userInfo.interviewCredits + ' 次' : '已用完' }}</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>
|
||||||
|
|||||||
@@ -66,10 +66,47 @@
|
|||||||
</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>
|
||||||
@@ -79,16 +116,20 @@ import { ref, computed, onMounted, onBeforeUnmount, nextTick } from 'vue'
|
|||||||
import { onLoad } from '@dcloudio/uni-app'
|
import { onLoad } from '@dcloudio/uni-app'
|
||||||
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'
|
||||||
|
|
||||||
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([])
|
||||||
@@ -116,9 +157,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()) startInterview()
|
if (!position.value) {
|
||||||
|
// 未传入岗位,展示选择弹窗(无论是否登录)
|
||||||
|
loadPositions()
|
||||||
|
showPositionPicker.value = true
|
||||||
|
} else if (token()) {
|
||||||
|
startInterview()
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
onBeforeUnmount(() => {
|
onBeforeUnmount(() => {
|
||||||
@@ -155,6 +224,11 @@ const startInterview = async () => {
|
|||||||
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 {
|
} else {
|
||||||
const msg = res.data?.message || '创建面试失败'
|
const msg = res.data?.message || '创建面试失败'
|
||||||
messages.value.push({ role: 'ai', content: msg })
|
messages.value.push({ role: 'ai', content: msg })
|
||||||
@@ -198,8 +272,10 @@ const sendAnswer = async () => {
|
|||||||
answeredCount.value = res.data.questionCount || answeredCount.value + 1
|
answeredCount.value = res.data.questionCount || answeredCount.value + 1
|
||||||
if (res.data.totalQuestions) MAX_QUESTIONS = res.data.totalQuestions
|
if (res.data.totalQuestions) MAX_QUESTIONS = res.data.totalQuestions
|
||||||
} else if (res.statusCode === 403) {
|
} else if (res.statusCode === 403) {
|
||||||
messages.value.push({ role: 'ai', content: res.data?.message || '面试次数已用完' })
|
const errMsg = res.data?.message || '面试次数已用完'
|
||||||
|
messages.value.push({ role: 'ai', content: errMsg + ' 👉 购买后可继续面试' })
|
||||||
isComplete.value = true
|
isComplete.value = true
|
||||||
|
completedReason.value = 'noCredits'
|
||||||
} else {
|
} else {
|
||||||
messages.value.push({ role: 'ai', content: res.data?.message || '回答提交失败' })
|
messages.value.push({ role: 'ai', content: res.data?.message || '回答提交失败' })
|
||||||
}
|
}
|
||||||
@@ -240,6 +316,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) })
|
||||||
}
|
}
|
||||||
@@ -389,7 +486,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>
|
||||||
|
|||||||
@@ -33,7 +33,7 @@
|
|||||||
<text class="field-label">密码</text>
|
<text class="field-label">密码</text>
|
||||||
<input class="input" type="password" v-model="password" placeholder="请输入密码" @confirm="doPasswordLogin" />
|
<input class="input" type="password" v-model="password" placeholder="请输入密码" @confirm="doPasswordLogin" />
|
||||||
</view>
|
</view>
|
||||||
<button class="login-btn" :disabled="!canPasswordLogin || pwdLoading || !agreed" @click="doPasswordLogin">
|
<button class="login-btn" :disabled="!canPasswordLogin || pwdLoading" @click="doPasswordLogin">
|
||||||
{{ pwdLoading ? '登录中...' : '登录' }}
|
{{ pwdLoading ? '登录中...' : '登录' }}
|
||||||
</button>
|
</button>
|
||||||
<view class="switch-hint" @click="loginMode='code'">忘记密码?使用验证码登录</view>
|
<view class="switch-hint" @click="loginMode='code'">忘记密码?使用验证码登录</view>
|
||||||
@@ -54,7 +54,7 @@
|
|||||||
<text class="field-label">验证码</text>
|
<text class="field-label">验证码</text>
|
||||||
<input class="input" type="number" maxlength="6" v-model="emailCode" placeholder="请输入6位验证码" />
|
<input class="input" type="number" maxlength="6" v-model="emailCode" placeholder="请输入6位验证码" />
|
||||||
</view>
|
</view>
|
||||||
<button class="login-btn" :disabled="!emailSent || !emailCode || emailLoading || !agreed" @click="doEmailLogin">
|
<button class="login-btn" :disabled="!emailSent || !emailCode || emailLoading" @click="doEmailLogin">
|
||||||
{{ emailLoading ? '登录中...' : '登录' }}
|
{{ emailLoading ? '登录中...' : '登录' }}
|
||||||
</button>
|
</button>
|
||||||
<view class="switch-hint" @click="loginMode='password'">已有密码?使用密码登录</view>
|
<view class="switch-hint" @click="loginMode='password'">已有密码?使用密码登录</view>
|
||||||
@@ -77,7 +77,7 @@
|
|||||||
<text class="field-label">确认密码</text>
|
<text class="field-label">确认密码</text>
|
||||||
<input class="input" type="password" v-model="confirmPassword" placeholder="再次输入密码" @confirm="doRegister" />
|
<input class="input" type="password" v-model="confirmPassword" placeholder="再次输入密码" @confirm="doRegister" />
|
||||||
</view>
|
</view>
|
||||||
<button class="login-btn" :disabled="!canRegister || regLoading || !agreed" @click="doRegister">
|
<button class="login-btn" :disabled="!canRegister || regLoading" @click="doRegister">
|
||||||
{{ regLoading ? '注册中...' : '注册' }}
|
{{ regLoading ? '注册中...' : '注册' }}
|
||||||
</button>
|
</button>
|
||||||
<view class="switch-hint" @click="mainTab='login'">已有账号?去登录</view>
|
<view class="switch-hint" @click="mainTab='login'">已有账号?去登录</view>
|
||||||
@@ -87,7 +87,7 @@
|
|||||||
<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" :disabled="wxLoading || !agreed" @click="doWxLogin">{{ wxLoading ? '登录中...' : '微信一键登录' }}</button>
|
<button class="login-btn wx-btn" :disabled="wxLoading" @click="doWxLogin">{{ wxLoading ? '登录中...' : '微信一键登录' }}</button>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
<!-- 法律声明 - 用户自主勾选同意 -->
|
<!-- 法律声明 - 用户自主勾选同意 -->
|
||||||
@@ -156,6 +156,10 @@ 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))
|
||||||
@@ -165,7 +169,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({
|
||||||
@@ -227,7 +231,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({
|
||||||
@@ -250,7 +254,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({
|
||||||
@@ -292,6 +296,7 @@ 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()
|
||||||
|
|||||||
@@ -1,342 +1,221 @@
|
|||||||
<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">{{ sprintPriceText }}</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')">{{ sprintPriceText }}/月 立即开通</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'
|
||||||
import { api } from '../../config'
|
import { api } from '../../config'
|
||||||
import UQRCode from 'uqrcodejs'
|
|
||||||
|
|
||||||
const isLoggedIn = ref(false)
|
const goBack = () => uni.switchTab({ url: '/pages/user/user' })
|
||||||
const isMp = ref(false)
|
|
||||||
const plan = ref('free')
|
// #ifdef H5
|
||||||
const currentPlanName = ref('免费版')
|
const buyQty = ref(1)
|
||||||
const paySuccess = ref(false)
|
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 sprintPriceText = ref('¥49.9')
|
|
||||||
const currentOutTradeNo = ref('')
|
|
||||||
const freeFeatures = ref(['AI 模拟面试 1 次(体验)', '基础面试报告', '通用题库随机出题', '简历优化(限 3 次免费)'])
|
|
||||||
const growthFeatures = ref(['免费版全部权益', 'AI 数字人面试无限次', '详细面试报告(四维评分)', '进步轨迹雷达图 + 打卡', '每日一题推送', '参考回答思路', '公司真题库', '每场最多 10 轮 AI 对话'])
|
|
||||||
const sprintFeatures = ref(['成长版全部权益', 'AI 语音分析(语气词/语速检测)', '技能缺口分析报告', '学习路径推荐', '真人导师 1v1 点评(每月 1 次)', '简历精修(每月 1 次)', '内推优先', 'AI 实时提示功能'])
|
|
||||||
|
|
||||||
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) {
|
|
||||||
const plans = Array.isArray(lres.data.plans) ? lres.data.plans : (Array.isArray(lres.data) ? lres.data : [])
|
|
||||||
const growth = plans.find((p) => p.id === 'growth')
|
|
||||||
const sprint = plans.find((p) => p.id === 'sprint')
|
|
||||||
if (growth) {
|
|
||||||
growthPriceText.value = `¥${(growth.price / 100).toFixed(1)}`
|
|
||||||
if (growth.features?.length) growthFeatures.value = growth.features
|
|
||||||
}
|
|
||||||
if (sprint?.features?.length) sprintFeatures.value = sprint.features
|
|
||||||
if (sprint) sprintPriceText.value = `¥${(sprint.price / 100).toFixed(1)}`
|
|
||||||
if (lres.data.price?.monthly) {
|
|
||||||
growthPriceText.value = `¥${(lres.data.price.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>
|
||||||
@@ -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([])
|
||||||
|
|
||||||
@@ -127,6 +154,18 @@ const todayStats = computed(() => ({
|
|||||||
|
|
||||||
const isWechat = ref(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(() => {
|
||||||
// #ifdef H5
|
// #ifdef H5
|
||||||
isWechat.value = navigator?.userAgent?.toLowerCase().includes('micromessenger') || false
|
isWechat.value = navigator?.userAgent?.toLowerCase().includes('micromessenger') || false
|
||||||
@@ -139,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 }),
|
||||||
@@ -162,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}`
|
||||||
@@ -181,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) {
|
||||||
@@ -238,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 */
|
||||||
|
|||||||
@@ -39,6 +39,25 @@
|
|||||||
</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">
|
||||||
@@ -58,17 +77,20 @@
|
|||||||
<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="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>
|
||||||
@@ -90,6 +112,46 @@
|
|||||||
<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>
|
||||||
|
|
||||||
@@ -102,6 +164,7 @@ 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 maskedId = computed(() => {
|
||||||
@@ -114,6 +177,7 @@ const refreshState = () => {
|
|||||||
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()
|
fetchUserInfo()
|
||||||
}
|
}
|
||||||
@@ -131,6 +195,16 @@ const fetchUserInfo = async () => {
|
|||||||
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}` } })
|
||||||
@@ -153,12 +227,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 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 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' })
|
||||||
|
|
||||||
@@ -197,7 +295,31 @@ const doLogout = () => {
|
|||||||
.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; }
|
||||||
|
|
||||||
.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); }
|
||||||
.menu-item:last-child { border-bottom: none; }
|
.menu-item:last-child { border-bottom: none; }
|
||||||
@@ -216,4 +338,28 @@ const doLogout = () => {
|
|||||||
.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>
|
||||||
|
|||||||
Reference in New Issue
Block a user