Compare commits
10 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 96c367e0f8 | |||
| 5a49d15696 | |||
| 07c6557454 | |||
| 18c50726cd | |||
| 4fa620f0a2 | |||
| 112884a504 | |||
| 93ab79d200 | |||
| 8191cf4b41 | |||
| 6fe84b6ef8 | |||
| 087fb1d400 |
@@ -0,0 +1,10 @@
|
|||||||
|
{
|
||||||
|
"sessionID": "ses_15492f54bffepl01q9FkaRyN28",
|
||||||
|
"updatedAt": "2026-06-16T01:20:56.855Z",
|
||||||
|
"sources": {
|
||||||
|
"background-task": {
|
||||||
|
"state": "idle",
|
||||||
|
"updatedAt": "2026-06-16T01:20:56.855Z"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,56 @@
|
|||||||
|
## 开发交付流程
|
||||||
|
|
||||||
|
每次实施/开发完功能后,必须按以下步骤执行:
|
||||||
|
|
||||||
|
### Step 1: 代码评审
|
||||||
|
- 检查变更是否符合现有代码模式(schema、service、controller、module 的结构一致性)
|
||||||
|
- 检查是否有未使用的变量、import、参数
|
||||||
|
- 检查命名规范是否与项目一致
|
||||||
|
- 检查是否有重复代码或可复用逻辑
|
||||||
|
|
||||||
|
### Step 2: 安全评审
|
||||||
|
- **注入防护**:检查所有外部输入(请求 body、query、params)是否有注入风险(HTML、Shell、NoSQL)
|
||||||
|
- **认证/授权**:检查端点是否正确地加了 JWT guard 或 `@Public()`;管理端接口是否有 AdminGuard
|
||||||
|
- **并发安全**:检查所有修改用户额度/积分的操作是否使用原子操作(`findOneAndUpdate + $inc`),而非 read-modify-write
|
||||||
|
- **敏感信息**:检查是否误将密钥、凭据、token 暴露到响应或日志中
|
||||||
|
- **XSS/CSRF**:检查用户内容输出到 HTML/PDF 时是否做了转义
|
||||||
|
|
||||||
|
### Step 3: 性能优化
|
||||||
|
- 检查是否有 N+1 查询(循环内查数据库),应使用批量查询或聚合
|
||||||
|
- 检查大表查询是否有索引覆盖
|
||||||
|
- 检查是否有不必要的 `.lean()` 缺失(读操作用 `exec()` 但不需要 mongoose document 方法时应加 `.lean()`)
|
||||||
|
- 检查是否有内存泄漏风险(如 puppeteer browser 未在 finally 中 close)
|
||||||
|
- Throttler/限流是否合理
|
||||||
|
|
||||||
|
### Step 4: 完整测试
|
||||||
|
```bash
|
||||||
|
# 构建检查(注意内存限制,服务器 OOM 时加 --max-old-space-size)
|
||||||
|
cd /root/opencode-workspace/zhiyin/backend && NODE_OPTIONS="--max-old-space-size=2048" npx nest build 2>&1
|
||||||
|
|
||||||
|
# 单元测试
|
||||||
|
npm test -- --forceExit --detectOpenHandles 2>&1
|
||||||
|
|
||||||
|
# 如果有变更的模块,验证关键 endpoint curl 可访问
|
||||||
|
```
|
||||||
|
|
||||||
|
### Step 5: 同步修复
|
||||||
|
- 上述步骤发现的问题必须修复后再发布
|
||||||
|
- 修复后重新执行 Step 4 验证构建通过
|
||||||
|
|
||||||
|
### Step 6: 部署到生产
|
||||||
|
```bash
|
||||||
|
# 构建
|
||||||
|
cd /root/opencode-workspace/zhiyin/backend && npx nest build
|
||||||
|
|
||||||
|
# 同步到生产目录
|
||||||
|
cp -rf dist/* /www/wwwroot/server/zhiyin/backend/dist/
|
||||||
|
|
||||||
|
# 复制证书(postbuild 替代)
|
||||||
|
cp -r certs /www/wwwroot/server/zhiyin/backend/dist/src/certs
|
||||||
|
|
||||||
|
# 重启
|
||||||
|
pm2 restart yhl-backend
|
||||||
|
|
||||||
|
# 验证
|
||||||
|
sleep 3 && curl -s http://localhost:3006/api/user/wx-login -X POST -H "Content-Type: application/json" -d '{"code":"test"}'
|
||||||
|
```
|
||||||
@@ -25,6 +25,7 @@ import { DailyQuestionModule } from './modules/daily-question/daily-question.mod
|
|||||||
import { ScheduleModule } from './modules/schedule/schedule.module'
|
import { ScheduleModule } from './modules/schedule/schedule.module'
|
||||||
import { TtsModule } from './modules/tts/tts.module'
|
import { TtsModule } from './modules/tts/tts.module'
|
||||||
import { PricingModule } from './modules/schemas/pricing.module'
|
import { PricingModule } from './modules/schemas/pricing.module'
|
||||||
|
import { ShareModule } from './modules/share/share.module'
|
||||||
|
|
||||||
const MONGODB_URI = process.env.MONGODB_URI || 'mongodb://localhost:27017/zhiyin'
|
const MONGODB_URI = process.env.MONGODB_URI || 'mongodb://localhost:27017/zhiyin'
|
||||||
|
|
||||||
@@ -38,7 +39,7 @@ const MONGODB_URI = process.env.MONGODB_URI || 'mongodb://localhost:27017/zhiyin
|
|||||||
}),
|
}),
|
||||||
ThrottlerModule.forRoot([{
|
ThrottlerModule.forRoot([{
|
||||||
ttl: 60000,
|
ttl: 60000,
|
||||||
limit: 10,
|
limit: 100,
|
||||||
}]),
|
}]),
|
||||||
NestScheduleModule.forRoot(),
|
NestScheduleModule.forRoot(),
|
||||||
UserModule,
|
UserModule,
|
||||||
@@ -58,6 +59,7 @@ const MONGODB_URI = process.env.MONGODB_URI || 'mongodb://localhost:27017/zhiyin
|
|||||||
ScheduleModule,
|
ScheduleModule,
|
||||||
TtsModule,
|
TtsModule,
|
||||||
PricingModule,
|
PricingModule,
|
||||||
|
ShareModule,
|
||||||
],
|
],
|
||||||
providers: [
|
providers: [
|
||||||
JwtStrategy,
|
JwtStrategy,
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { Controller, Get, Post, Body, Query, HttpException, HttpStatus, UseGuards } from '@nestjs/common'
|
import { Controller, Get, Post, Body, Query, Param, HttpException, HttpStatus, UseGuards } from '@nestjs/common'
|
||||||
import { InjectModel } from '@nestjs/mongoose'
|
import { InjectModel } from '@nestjs/mongoose'
|
||||||
import { Model } from 'mongoose'
|
import { Model } from 'mongoose'
|
||||||
import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard'
|
import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard'
|
||||||
@@ -6,9 +6,12 @@ import { AdminGuard } from '../../common/guards/admin.guard'
|
|||||||
import { CurrentUser } from '../../common/decorators/current-user.decorator'
|
import { CurrentUser } from '../../common/decorators/current-user.decorator'
|
||||||
import { User, UserDocument } from '../user/user.schema'
|
import { User, UserDocument } from '../user/user.schema'
|
||||||
import { Interview, InterviewDocument } from '../interview/interview.schema'
|
import { Interview, InterviewDocument } from '../interview/interview.schema'
|
||||||
|
import { Resume, ResumeDocument } from '../resume/resume.schema'
|
||||||
import { PaymentOrder, PaymentOrderDocument } from '../payment/payment-order.schema'
|
import { PaymentOrder, PaymentOrderDocument } from '../payment/payment-order.schema'
|
||||||
import { SiteConfig, SiteConfigDocument } from '../schemas/site-config.schema'
|
import { SiteConfig, SiteConfigDocument } from '../schemas/site-config.schema'
|
||||||
|
import { ShareRecord, ShareRecordDocument, ShareVisit, ShareVisitDocument } from '../share/share.schema'
|
||||||
import { QuotaService } from '../user/quota.service'
|
import { QuotaService } from '../user/quota.service'
|
||||||
|
import { PricingService } from '../schemas/pricing.service'
|
||||||
import { WechatPayService } from '../payment/wechat-pay.service'
|
import { WechatPayService } from '../payment/wechat-pay.service'
|
||||||
|
|
||||||
const VIP_DURATION_DAYS = 30
|
const VIP_DURATION_DAYS = 30
|
||||||
@@ -21,7 +24,11 @@ export class AdminController {
|
|||||||
@InjectModel(Interview.name) private interviewModel: Model<InterviewDocument>,
|
@InjectModel(Interview.name) private interviewModel: Model<InterviewDocument>,
|
||||||
@InjectModel(PaymentOrder.name) private orderModel: Model<PaymentOrderDocument>,
|
@InjectModel(PaymentOrder.name) private orderModel: Model<PaymentOrderDocument>,
|
||||||
@InjectModel(SiteConfig.name) private configModel: Model<SiteConfigDocument>,
|
@InjectModel(SiteConfig.name) private configModel: Model<SiteConfigDocument>,
|
||||||
|
@InjectModel(ShareRecord.name) private shareModel: Model<ShareRecordDocument>,
|
||||||
|
@InjectModel(ShareVisit.name) private shareVisitModel: Model<ShareVisitDocument>,
|
||||||
|
@InjectModel(Resume.name) private resumeModel: Model<ResumeDocument>,
|
||||||
private quotaService: QuotaService,
|
private quotaService: QuotaService,
|
||||||
|
private pricingService: PricingService,
|
||||||
private wechatPay: WechatPayService,
|
private wechatPay: WechatPayService,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
@@ -38,13 +45,28 @@ export class AdminController {
|
|||||||
|
|
||||||
@Get('overview')
|
@Get('overview')
|
||||||
async overview() {
|
async overview() {
|
||||||
const [userCount, interviewCount, todayUsers, todayInterviews] = await Promise.all([
|
const [
|
||||||
|
userCount, interviewCount, todayUsers, todayInterviews,
|
||||||
|
resumeCount, paidDownloadCount,
|
||||||
|
planStats,
|
||||||
|
] = await Promise.all([
|
||||||
this.userModel.countDocuments().exec(),
|
this.userModel.countDocuments().exec(),
|
||||||
this.interviewModel.countDocuments().exec(),
|
this.interviewModel.countDocuments().exec(),
|
||||||
this.userModel.countDocuments({ createdAt: { $gte: new Date(Date.now() - 24 * 60 * 60 * 1000) } }).exec(),
|
this.userModel.countDocuments({ createdAt: { $gte: new Date(Date.now() - 24 * 60 * 60 * 1000) } }).exec(),
|
||||||
this.interviewModel.countDocuments({ createdAt: { $gte: new Date(Date.now() - 24 * 60 * 60 * 1000) } }).exec(),
|
this.interviewModel.countDocuments({ createdAt: { $gte: new Date(Date.now() - 24 * 60 * 60 * 1000) } }).exec(),
|
||||||
|
this.resumeModel.countDocuments().exec(),
|
||||||
|
this.resumeModel.countDocuments({ paidDownload: true }).exec(),
|
||||||
|
this.userModel.aggregate([
|
||||||
|
{ $group: { _id: '$plan', count: { $sum: 1 } } },
|
||||||
|
]).exec(),
|
||||||
])
|
])
|
||||||
return { userCount, interviewCount, todayUsers, todayInterviews }
|
const planBreakdown: Record<string, number> = {}
|
||||||
|
planStats.forEach(p => { planBreakdown[p._id || 'free'] = p.count })
|
||||||
|
return {
|
||||||
|
userCount, interviewCount, todayUsers, todayInterviews,
|
||||||
|
resumeCount, paidDownloadCount,
|
||||||
|
planBreakdown,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Get('users')
|
@Get('users')
|
||||||
@@ -70,7 +92,12 @@ export class AdminController {
|
|||||||
async getInterviews(@Query('page') page = '1', @Query('limit') limit = '20') {
|
async getInterviews(@Query('page') page = '1', @Query('limit') limit = '20') {
|
||||||
const skip = (Math.max(1, +page) - 1) * +limit
|
const skip = (Math.max(1, +page) - 1) * +limit
|
||||||
const [interviews, total] = await Promise.all([
|
const [interviews, total] = await Promise.all([
|
||||||
this.interviewModel.find().sort({ createdAt: -1 }).skip(skip).limit(+limit).populate('userId', 'phone nickname').lean().exec(),
|
this.interviewModel.find()
|
||||||
|
.sort({ createdAt: -1 })
|
||||||
|
.skip(skip).limit(+limit)
|
||||||
|
.populate('userId', 'phone nickname')
|
||||||
|
.select('position status totalScore questionCount fillerScore fillerDensity summary createdAt')
|
||||||
|
.lean().exec(),
|
||||||
this.interviewModel.countDocuments().exec(),
|
this.interviewModel.countDocuments().exec(),
|
||||||
])
|
])
|
||||||
return { interviews, total, page: +page }
|
return { interviews, total, page: +page }
|
||||||
@@ -80,14 +107,135 @@ export class AdminController {
|
|||||||
async setVip(@Body('userId') targetUserId: string) {
|
async setVip(@Body('userId') targetUserId: string) {
|
||||||
const user = await this.userModel.findById(targetUserId).exec()
|
const user = await this.userModel.findById(targetUserId).exec()
|
||||||
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
||||||
|
const pricing = await this.pricingService.getConfig()
|
||||||
|
const credits = pricing.plans?.growth?.credits || { interview: 999, resumeOptimize: 20, resumeDownload: 10 }
|
||||||
const expireAt = new Date()
|
const expireAt = new Date()
|
||||||
expireAt.setDate(expireAt.getDate() + 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', { interview: 999, resumeOptimize: 20, resumeDownload: 10 })
|
await this.quotaService.setPlanQuota(targetUserId, 'growth', credits)
|
||||||
return { success: true, plan: 'growth', expireAt }
|
return { success: true, plan: 'growth', expireAt }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Post('user/credits')
|
||||||
|
async adjustCredits(@Body('userId') userId: string, @Body('type') type: string, @Body('amount') amount: number) {
|
||||||
|
if (!userId || !type || amount === undefined) {
|
||||||
|
throw new HttpException('参数不完整', HttpStatus.BAD_REQUEST)
|
||||||
|
}
|
||||||
|
const validTypes = ['interviewCredits', 'resumeOptimizeCredits', 'resumeDownloadCredits', 'shareCredits']
|
||||||
|
if (!validTypes.includes(type)) {
|
||||||
|
throw new HttpException('无效的额度类型', HttpStatus.BAD_REQUEST)
|
||||||
|
}
|
||||||
|
const result = await this.userModel.findByIdAndUpdate(
|
||||||
|
userId,
|
||||||
|
{ $set: { [type]: Math.max(0, Math.round(amount)) } },
|
||||||
|
).exec()
|
||||||
|
if (!result) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
||||||
|
return { success: true }
|
||||||
|
}
|
||||||
|
|
||||||
|
@Get('share-records')
|
||||||
|
async getShareRecords(@Query('page') page = '1', @Query('limit') limit = '20') {
|
||||||
|
const skip = (Math.max(1, +page) - 1) * +limit
|
||||||
|
const [list, total] = await Promise.all([
|
||||||
|
this.shareModel.aggregate([
|
||||||
|
{ $sort: { createdAt: -1 } },
|
||||||
|
{ $skip: skip },
|
||||||
|
{ $limit: +limit },
|
||||||
|
{
|
||||||
|
$lookup: {
|
||||||
|
from: 'users',
|
||||||
|
localField: 'userId',
|
||||||
|
foreignField: '_id',
|
||||||
|
as: 'sharer',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{ $unwind: { path: '$sharer', preserveNullAndEmptyArrays: true } },
|
||||||
|
{
|
||||||
|
$project: {
|
||||||
|
shareCode: 1,
|
||||||
|
type: 1,
|
||||||
|
title: 1,
|
||||||
|
visitCount: 1,
|
||||||
|
creditedCount: 1,
|
||||||
|
isActive: 1,
|
||||||
|
createdAt: 1,
|
||||||
|
sharer: { nickname: '$sharer.nickname', phone: '$sharer.phone', _id: '$sharer._id' },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]).exec(),
|
||||||
|
this.shareModel.countDocuments().exec(),
|
||||||
|
])
|
||||||
|
return { list, total, page: +page }
|
||||||
|
}
|
||||||
|
|
||||||
|
@Get('share-visitors')
|
||||||
|
async getShareVisitors(@Query('page') page = '1', @Query('limit') limit = '20') {
|
||||||
|
const skip = (Math.max(1, +page) - 1) * +limit
|
||||||
|
const [list, total] = await Promise.all([
|
||||||
|
this.shareVisitModel.aggregate([
|
||||||
|
{ $sort: { createdAt: -1 } },
|
||||||
|
{ $skip: skip },
|
||||||
|
{ $limit: +limit },
|
||||||
|
{
|
||||||
|
$lookup: {
|
||||||
|
from: 'users',
|
||||||
|
localField: 'sharerId',
|
||||||
|
foreignField: '_id',
|
||||||
|
as: 'sharer',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{ $unwind: { path: '$sharer', preserveNullAndEmptyArrays: true } },
|
||||||
|
{
|
||||||
|
$lookup: {
|
||||||
|
from: 'users',
|
||||||
|
localField: 'visitorUserId',
|
||||||
|
foreignField: '_id',
|
||||||
|
as: 'visitorUser',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{ $unwind: { path: '$visitorUser', preserveNullAndEmptyArrays: true } },
|
||||||
|
{
|
||||||
|
$project: {
|
||||||
|
credited: 1,
|
||||||
|
creditedAt: 1,
|
||||||
|
createdAt: 1,
|
||||||
|
sharer: { nickname: '$sharer.nickname', phone: '$sharer.phone' },
|
||||||
|
visitor: { nickname: { $ifNull: ['$visitorUser.nickname', '匿名'] }, phone: { $ifNull: ['$visitorUser.phone', ''] } },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]).exec(),
|
||||||
|
this.shareVisitModel.countDocuments().exec(),
|
||||||
|
])
|
||||||
|
return { list, total, page: +page }
|
||||||
|
}
|
||||||
|
|
||||||
|
@Get('resumes')
|
||||||
|
async getResumes(@Query('page') page = '1', @Query('limit') limit = '20') {
|
||||||
|
const skip = (Math.max(1, +page) - 1) * +limit
|
||||||
|
const [list, total] = await Promise.all([
|
||||||
|
this.resumeModel.find()
|
||||||
|
.sort({ createdAt: -1 })
|
||||||
|
.skip(skip).limit(+limit)
|
||||||
|
.populate('userId', 'phone nickname')
|
||||||
|
.select('title targetPosition version paidDownload createdAt')
|
||||||
|
.lean().exec(),
|
||||||
|
this.resumeModel.countDocuments().exec(),
|
||||||
|
])
|
||||||
|
return { list, total, page: +page }
|
||||||
|
}
|
||||||
|
|
||||||
|
@Get('user/:id')
|
||||||
|
async getUserDetail(@Param('id') id: string) {
|
||||||
|
const user = await this.userModel.findById(id).select('-password -openid').lean().exec()
|
||||||
|
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
||||||
|
const [interviews, resumes] = await Promise.all([
|
||||||
|
this.interviewModel.find({ userId: id }).sort({ createdAt: -1 }).limit(10).select('position status totalScore questionCount createdAt').lean().exec(),
|
||||||
|
this.resumeModel.find({ userId: id }).sort({ createdAt: -1 }).limit(10).select('title targetPosition version paidDownload createdAt').lean().exec(),
|
||||||
|
])
|
||||||
|
return { user, interviews, resumes }
|
||||||
|
}
|
||||||
|
|
||||||
@Get('admins')
|
@Get('admins')
|
||||||
async getAdmins() {
|
async getAdmins() {
|
||||||
const admins = await this.userModel.find({ role: 'admin' }).select('phone nickname email createdAt isSystemAdmin').lean().exec()
|
const admins = await this.userModel.find({ role: 'admin' }).select('phone nickname email createdAt isSystemAdmin').lean().exec()
|
||||||
@@ -130,14 +278,22 @@ export class AdminController {
|
|||||||
if (order.type === 'membership') {
|
if (order.type === 'membership') {
|
||||||
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 credits = pricing.plans?.growth?.credits || { interview: 999, resumeOptimize: 20, resumeDownload: 10 }
|
||||||
const expireAt = new Date()
|
const expireAt = new Date()
|
||||||
expireAt.setDate(expireAt.getDate() + 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(order.userId, 'growth', { interview: 999, resumeOptimize: 20, resumeDownload: 10 })
|
await this.quotaService.setPlanQuota(order.userId, 'growth', credits)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
const credits = { interview: 1, optimize: 1, download: 1 }[order.type]
|
const pricing = await this.pricingService.getConfig()
|
||||||
|
const creditMap: Record<string, number> = {
|
||||||
|
interview: pricing.interview?.creditsPerPurchase || 1,
|
||||||
|
optimize: pricing.resumeOptimize?.creditsPerPurchase || 1,
|
||||||
|
download: pricing.resumeDownload?.creditsPerPurchase || 1,
|
||||||
|
}
|
||||||
|
const credits = creditMap[order.type]
|
||||||
if (credits) {
|
if (credits) {
|
||||||
await this.quotaService.grantCredits(order.userId, order.type as any, credits)
|
await this.quotaService.grantCredits(order.userId, order.type as any, credits)
|
||||||
}
|
}
|
||||||
@@ -179,6 +335,7 @@ export class AdminController {
|
|||||||
{ key: 'pricing', value: body, description: '定价配置' },
|
{ key: 'pricing', value: body, description: '定价配置' },
|
||||||
{ upsert: true },
|
{ upsert: true },
|
||||||
).exec()
|
).exec()
|
||||||
|
this.pricingService.invalidateCache()
|
||||||
return { success: true }
|
return { success: true }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -8,6 +8,8 @@ import { PaymentOrder, PaymentOrderSchema } from '../payment/payment-order.schem
|
|||||||
import { WechatPayService } from '../payment/wechat-pay.service'
|
import { WechatPayService } from '../payment/wechat-pay.service'
|
||||||
import { AdminGuard } from '../../common/guards/admin.guard'
|
import { AdminGuard } from '../../common/guards/admin.guard'
|
||||||
import { SiteConfig, SiteConfigSchema } from '../schemas/site-config.schema'
|
import { SiteConfig, SiteConfigSchema } from '../schemas/site-config.schema'
|
||||||
|
import { ShareRecord, ShareRecordSchema, ShareVisit, ShareVisitSchema } from '../share/share.schema'
|
||||||
|
import { Resume, ResumeSchema } from '../resume/resume.schema'
|
||||||
|
|
||||||
@Module({
|
@Module({
|
||||||
imports: [
|
imports: [
|
||||||
@@ -16,6 +18,9 @@ import { SiteConfig, SiteConfigSchema } from '../schemas/site-config.schema'
|
|||||||
{ name: Interview.name, schema: InterviewSchema },
|
{ name: Interview.name, schema: InterviewSchema },
|
||||||
{ name: PaymentOrder.name, schema: PaymentOrderSchema },
|
{ name: PaymentOrder.name, schema: PaymentOrderSchema },
|
||||||
{ name: SiteConfig.name, schema: SiteConfigSchema },
|
{ name: SiteConfig.name, schema: SiteConfigSchema },
|
||||||
|
{ name: ShareRecord.name, schema: ShareRecordSchema },
|
||||||
|
{ name: ShareVisit.name, schema: ShareVisitSchema },
|
||||||
|
{ name: Resume.name, schema: ResumeSchema },
|
||||||
]),
|
]),
|
||||||
UserModule,
|
UserModule,
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import { Controller, Post, Get, Body, Param, UseGuards, Logger } from '@nestjs/c
|
|||||||
import { InjectModel } from '@nestjs/mongoose'
|
import { InjectModel } from '@nestjs/mongoose'
|
||||||
import { Model } from 'mongoose'
|
import { Model } from 'mongoose'
|
||||||
import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard'
|
import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard'
|
||||||
|
import { Public } from '../../common/decorators/public.decorator'
|
||||||
import { CurrentUser } from '../../common/decorators/current-user.decorator'
|
import { CurrentUser } from '../../common/decorators/current-user.decorator'
|
||||||
import { AiService } from '../ai/ai.service'
|
import { AiService } from '../ai/ai.service'
|
||||||
import { Contribution, ContributionDocument } from '../schemas/contribution.schema'
|
import { Contribution, ContributionDocument } from '../schemas/contribution.schema'
|
||||||
@@ -186,4 +187,19 @@ export class ContributionController {
|
|||||||
.select('company position rounds experience createdAt')
|
.select('company position rounds experience createdAt')
|
||||||
.exec()
|
.exec()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Public()
|
||||||
|
@Get('companies/hot')
|
||||||
|
async getHotCompanies() {
|
||||||
|
const banks = await this.companyBankModel.aggregate([
|
||||||
|
{ $group: { _id: '$company', positionCount: { $sum: 1 }, totalContributions: { $sum: '$contributionCount' } } },
|
||||||
|
{ $sort: { totalContributions: -1, positionCount: -1 } },
|
||||||
|
{ $project: { _id: 0, name: '$_id', positionCount: 1 } },
|
||||||
|
]).exec()
|
||||||
|
|
||||||
|
if (banks.length > 0) return banks
|
||||||
|
|
||||||
|
const DEFAULT_COMPANIES = ['腾讯', '字节跳动', '阿里巴巴', '美团', '百度', '京东', '网易', '小红书']
|
||||||
|
return DEFAULT_COMPANIES.map((name, i) => ({ name, positionCount: 0, sort: i }))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@@ -107,7 +107,7 @@ ${conversationHistory}
|
|||||||
if (aiMsg?.content) {
|
if (aiMsg?.content) {
|
||||||
try {
|
try {
|
||||||
const tts = await this.ttsService.synthesize(aiMsg.content)
|
const tts = await this.ttsService.synthesize(aiMsg.content)
|
||||||
return { ...base, ttsHash: tts.hash, ttsDurationMs: tts.durationMs }
|
return { ...base, ttsHash: tts.hash, ttsDurationMs: tts.durationMs, ttsAmplitude: tts.amplitudeData }
|
||||||
} catch {
|
} catch {
|
||||||
// TTS failure is non-critical, return without audio
|
// TTS failure is non-critical, return without audio
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -168,6 +168,8 @@ export class PaymentController {
|
|||||||
order.wxTransactionId = wxTransactionId
|
order.wxTransactionId = wxTransactionId
|
||||||
order.description = `${decrypted.trade_type || ''} 支付成功`
|
order.description = `${decrypted.trade_type || ''} 支付成功`
|
||||||
await order.save()
|
await order.save()
|
||||||
|
} else {
|
||||||
|
return { code: 'SUCCESS', message: '已处理' }
|
||||||
}
|
}
|
||||||
|
|
||||||
if (order.type === 'membership') {
|
if (order.type === 'membership') {
|
||||||
@@ -260,6 +262,15 @@ export class PaymentController {
|
|||||||
return { success: true, plan: order.plan }
|
return { success: true, plan: order.plan }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// For product orders, check if already activated via callback
|
||||||
|
const user = await this.userModel.findById(userId).exec()
|
||||||
|
if (user && order.type === 'interview' && (user.interviewCredits || 0) > 0) {
|
||||||
|
return { success: true, type: order.type, alreadyActivated: true }
|
||||||
|
}
|
||||||
|
if (user && order.type === 'optimize' && (user.resumeOptimizeCredits || 0) > 0) {
|
||||||
|
return { success: true, type: order.type, alreadyActivated: true }
|
||||||
|
}
|
||||||
|
|
||||||
await this.activateProduct(order)
|
await this.activateProduct(order)
|
||||||
return { success: true, type: order.type }
|
return { success: true, type: order.type }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,8 +21,11 @@ export class WechatPayService {
|
|||||||
this.logger.warn('微信支付配置不完整,支付功能不可用')
|
this.logger.warn('微信支付配置不完整,支付功能不可用')
|
||||||
}
|
}
|
||||||
const certDir = path.resolve(__dirname, '../../certs')
|
const certDir = path.resolve(__dirname, '../../certs')
|
||||||
|
if (!fs.existsSync(certDir)) {
|
||||||
|
this.logger.error(`证书目录不存在: ${certDir}`)
|
||||||
|
return
|
||||||
|
}
|
||||||
this.privateKey = fs.readFileSync(path.join(certDir, 'apiclient_key.pem'), 'utf8')
|
this.privateKey = fs.readFileSync(path.join(certDir, 'apiclient_key.pem'), 'utf8')
|
||||||
// 从证书中提取序列号
|
|
||||||
const cert = fs.readFileSync(path.join(certDir, 'apiclient_cert.pem'), 'utf8')
|
const cert = fs.readFileSync(path.join(certDir, 'apiclient_cert.pem'), 'utf8')
|
||||||
const certObj = new crypto.X509Certificate(cert)
|
const certObj = new crypto.X509Certificate(cert)
|
||||||
this.mchSerialNo = certObj.serialNumber
|
this.mchSerialNo = certObj.serialNumber
|
||||||
@@ -122,6 +125,10 @@ export class WechatPayService {
|
|||||||
// 1. 验签
|
// 1. 验签
|
||||||
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)) {
|
||||||
|
this.logger.error(`证书目录不存在: ${certDir}`)
|
||||||
|
return null
|
||||||
|
}
|
||||||
const platformCert = fs.readFileSync(path.join(certDir, 'pub_key.pem'), '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')
|
const isValid = verify.verify(platformCert, wechatSignature, 'base64')
|
||||||
|
|||||||
@@ -31,13 +31,18 @@ export class ResumePdfService {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private escapeHtml(str: string): string {
|
||||||
|
return str.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>')
|
||||||
|
.replace(/"/g, '"').replace(/'/g, ''')
|
||||||
|
}
|
||||||
|
|
||||||
private buildHtml(params: {
|
private buildHtml(params: {
|
||||||
title: string
|
title: string
|
||||||
content: string
|
content: string
|
||||||
targetPosition?: string
|
targetPosition?: string
|
||||||
userName?: string
|
userName?: string
|
||||||
}): string {
|
}): string {
|
||||||
const contentHtml = params.content
|
const contentHtml = this.escapeHtml(params.content)
|
||||||
.replace(/\n/g, '<br>')
|
.replace(/\n/g, '<br>')
|
||||||
.replace(/### (.+)/g, '<h3>$1</h3>')
|
.replace(/### (.+)/g, '<h3>$1</h3>')
|
||||||
.replace(/\*\*(.+?)\*\*/g, '<strong>$1</strong>')
|
.replace(/\*\*(.+?)\*\*/g, '<strong>$1</strong>')
|
||||||
@@ -66,8 +71,8 @@ export class ResumePdfService {
|
|||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div class="page">
|
<div class="page">
|
||||||
<h1>${params.title}</h1>
|
<h1>${this.escapeHtml(params.title)}</h1>
|
||||||
<div class="subtitle">${params.targetPosition ? `目标岗位: ${params.targetPosition}` : ''}</div>
|
<div class="subtitle">${params.targetPosition ? `目标岗位: ${this.escapeHtml(params.targetPosition)}` : ''}</div>
|
||||||
<div class="content">${contentHtml}</div>
|
<div class="content">${contentHtml}</div>
|
||||||
<div class="footer">由 AI磁场·职引 生成</div>
|
<div class="footer">由 AI磁场·职引 生成</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -31,13 +31,10 @@ export class ResumeController {
|
|||||||
@Post(':id/download')
|
@Post(':id/download')
|
||||||
async download(@Param('id') id: string, @CurrentUser('userId') userId: string, @Res() res: Response) {
|
async download(@Param('id') id: string, @CurrentUser('userId') userId: string, @Res() res: Response) {
|
||||||
const resume = await this.resumeService.getDetail(id, userId)
|
const resume = await this.resumeService.getDetail(id, userId)
|
||||||
|
const canDownload = await this.quotaService.checkAndDeductDownload(userId, resume.paidDownload)
|
||||||
const canDownload = await this.quotaService.checkDownload(userId, resume)
|
if (!canDownload && !resume.paidDownload) {
|
||||||
if (!canDownload) {
|
|
||||||
throw new HttpException('请先付费下载', HttpStatus.PAYMENT_REQUIRED)
|
throw new HttpException('请先付费下载', HttpStatus.PAYMENT_REQUIRED)
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.quotaService.deductDownload(userId, resume)
|
|
||||||
if (!resume.paidDownload) {
|
if (!resume.paidDownload) {
|
||||||
await this.resumeService.markPaid(id, userId)
|
await this.resumeService.markPaid(id, userId)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,8 +18,8 @@ const DEFAULT_PRICING: PricingConfig = {
|
|||||||
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, credits: { interview: 999, resumeOptimize: 20, resumeDownload: 10 }, features: ['免费版全部权益', 'AI 数字人面试无限次', '详细面试报告(四维评分)', '进步轨迹雷达图 + 打卡', '每日一题推送', '参考回答思路', '公司真题库', '简历优化 20 次/月', '简历下载 10 次/月'] },
|
||||||
sprint: { price: 4990, durationDays: 30, credits: { interview: 999, resumeOptimize: 50, resumeDownload: 30 }, features: [] },
|
sprint: { price: 4990, durationDays: 30, credits: { interview: 999, resumeOptimize: 50, resumeDownload: 30 }, features: ['成长版全部权益', 'AI 语音分析(语气词/语速检测)', '技能缺口分析报告', '学习路径推荐', '真人导师 1v1 点评(每月 1 次)', '简历精修(每月 1 次)', '内推优先', '简历优化 50 次/月', '简历下载 30 次/月'] },
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,69 @@
|
|||||||
|
import { Controller, Get, Post, Body, Param, Query, HttpException, HttpStatus, UseGuards } from '@nestjs/common'
|
||||||
|
import { JwtService } from '@nestjs/jwt'
|
||||||
|
import { ShareService } from './share.service'
|
||||||
|
import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard'
|
||||||
|
import { CurrentUser } from '../../common/decorators/current-user.decorator'
|
||||||
|
import { Public } from '../../common/decorators/public.decorator'
|
||||||
|
|
||||||
|
@Controller('share')
|
||||||
|
export class ShareController {
|
||||||
|
constructor(
|
||||||
|
private shareService: ShareService,
|
||||||
|
private jwtService: JwtService,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
@UseGuards(JwtAuthGuard)
|
||||||
|
@Post('create')
|
||||||
|
async create(
|
||||||
|
@CurrentUser('userId') userId: string,
|
||||||
|
@Body() body: { type: string; refId?: string; title?: string; description?: string },
|
||||||
|
) {
|
||||||
|
return this.shareService.create(userId, body)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Public()
|
||||||
|
@Get('visit/:shareCode')
|
||||||
|
async visit(
|
||||||
|
@Param('shareCode') shareCode: string,
|
||||||
|
@Query('visitorId') visitorId?: string,
|
||||||
|
@Query('token') token?: string,
|
||||||
|
) {
|
||||||
|
if (!visitorId) {
|
||||||
|
throw new HttpException('缺少访问者标识', HttpStatus.BAD_REQUEST)
|
||||||
|
}
|
||||||
|
let visitorUserId: string | undefined
|
||||||
|
if (token) {
|
||||||
|
try {
|
||||||
|
const payload = this.jwtService.verify(token) as any
|
||||||
|
visitorUserId = payload.userId
|
||||||
|
} catch {}
|
||||||
|
}
|
||||||
|
return this.shareService.visit(shareCode, visitorId, visitorUserId)
|
||||||
|
}
|
||||||
|
|
||||||
|
@UseGuards(JwtAuthGuard)
|
||||||
|
@Get('stats')
|
||||||
|
async stats(@CurrentUser('userId') userId: string) {
|
||||||
|
return this.shareService.stats(userId)
|
||||||
|
}
|
||||||
|
|
||||||
|
@UseGuards(JwtAuthGuard)
|
||||||
|
@Get('records')
|
||||||
|
async records(
|
||||||
|
@CurrentUser('userId') userId: string,
|
||||||
|
@Query('page') page?: string,
|
||||||
|
@Query('pageSize') pageSize?: string,
|
||||||
|
) {
|
||||||
|
return this.shareService.records(userId, Number(page) || 1, Number(pageSize) || 20)
|
||||||
|
}
|
||||||
|
|
||||||
|
@UseGuards(JwtAuthGuard)
|
||||||
|
@Get('visitors')
|
||||||
|
async visitors(
|
||||||
|
@CurrentUser('userId') userId: string,
|
||||||
|
@Query('page') page?: string,
|
||||||
|
@Query('pageSize') pageSize?: string,
|
||||||
|
) {
|
||||||
|
return this.shareService.visitors(userId, Number(page) || 1, Number(pageSize) || 20)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
import { Module } from '@nestjs/common'
|
||||||
|
import { JwtModule } from '@nestjs/jwt'
|
||||||
|
import { MongooseModule } from '@nestjs/mongoose'
|
||||||
|
import { ShareController } from './share.controller'
|
||||||
|
import { ShareService } from './share.service'
|
||||||
|
import { ShareRecord, ShareRecordSchema, ShareVisit, ShareVisitSchema } from './share.schema'
|
||||||
|
import { UserModule } from '../user/user.module'
|
||||||
|
import { User, UserSchema } from '../user/user.schema'
|
||||||
|
|
||||||
|
@Module({
|
||||||
|
imports: [
|
||||||
|
MongooseModule.forFeature([
|
||||||
|
{ name: ShareRecord.name, schema: ShareRecordSchema },
|
||||||
|
{ name: ShareVisit.name, schema: ShareVisitSchema },
|
||||||
|
{ name: User.name, schema: UserSchema },
|
||||||
|
]),
|
||||||
|
UserModule,
|
||||||
|
JwtModule.register({
|
||||||
|
secret: process.env.JWT_SECRET,
|
||||||
|
signOptions: { expiresIn: '7d' },
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
controllers: [ShareController],
|
||||||
|
providers: [ShareService],
|
||||||
|
})
|
||||||
|
export class ShareModule {}
|
||||||
@@ -0,0 +1,72 @@
|
|||||||
|
import { Prop, Schema, SchemaFactory } from '@nestjs/mongoose'
|
||||||
|
import { Document, Types } from 'mongoose'
|
||||||
|
|
||||||
|
export type ShareRecordDocument = ShareRecord & Document
|
||||||
|
export type ShareVisitDocument = ShareVisit & Document
|
||||||
|
|
||||||
|
@Schema({ timestamps: true })
|
||||||
|
export class ShareRecord {
|
||||||
|
@Prop({ type: Types.ObjectId, ref: 'User', required: true })
|
||||||
|
userId: Types.ObjectId
|
||||||
|
|
||||||
|
@Prop({ required: true, unique: true })
|
||||||
|
shareCode: string
|
||||||
|
|
||||||
|
@Prop({ default: 'app' })
|
||||||
|
type: string
|
||||||
|
|
||||||
|
@Prop({ default: '' })
|
||||||
|
refId: string
|
||||||
|
|
||||||
|
@Prop({ default: '' })
|
||||||
|
title: string
|
||||||
|
|
||||||
|
@Prop({ default: '' })
|
||||||
|
description: string
|
||||||
|
|
||||||
|
@Prop({ default: 'link' })
|
||||||
|
channel: string
|
||||||
|
|
||||||
|
@Prop({ default: 0 })
|
||||||
|
visitCount: number
|
||||||
|
|
||||||
|
@Prop({ default: 0 })
|
||||||
|
creditedCount: number
|
||||||
|
|
||||||
|
@Prop({ default: true })
|
||||||
|
isActive: boolean
|
||||||
|
|
||||||
|
readonly createdAt?: Date
|
||||||
|
readonly updatedAt?: Date
|
||||||
|
}
|
||||||
|
|
||||||
|
@Schema({ timestamps: true })
|
||||||
|
export class ShareVisit {
|
||||||
|
@Prop({ type: Types.ObjectId, ref: 'ShareRecord', required: true })
|
||||||
|
shareId: Types.ObjectId
|
||||||
|
|
||||||
|
@Prop({ type: Types.ObjectId, ref: 'User', required: true })
|
||||||
|
sharerId: Types.ObjectId
|
||||||
|
|
||||||
|
@Prop({ required: true })
|
||||||
|
visitorId: string
|
||||||
|
|
||||||
|
@Prop({ type: Types.ObjectId, ref: 'User' })
|
||||||
|
visitorUserId: Types.ObjectId
|
||||||
|
|
||||||
|
@Prop({ default: false })
|
||||||
|
credited: boolean
|
||||||
|
|
||||||
|
@Prop()
|
||||||
|
creditedAt?: Date
|
||||||
|
|
||||||
|
readonly createdAt?: Date
|
||||||
|
readonly updatedAt?: Date
|
||||||
|
}
|
||||||
|
|
||||||
|
export const ShareRecordSchema = SchemaFactory.createForClass(ShareRecord)
|
||||||
|
export const ShareVisitSchema = SchemaFactory.createForClass(ShareVisit)
|
||||||
|
|
||||||
|
ShareRecordSchema.index({ userId: 1, createdAt: -1 })
|
||||||
|
ShareVisitSchema.index({ shareId: 1, visitorId: 1 }, { unique: true })
|
||||||
|
ShareVisitSchema.index({ sharerId: 1, createdAt: -1 })
|
||||||
@@ -0,0 +1,186 @@
|
|||||||
|
import * as crypto from 'crypto'
|
||||||
|
import { Injectable, HttpException, HttpStatus, Logger } from '@nestjs/common'
|
||||||
|
import { InjectModel } from '@nestjs/mongoose'
|
||||||
|
import { Model, Types } from 'mongoose'
|
||||||
|
import { ShareRecord, ShareRecordDocument, ShareVisit, ShareVisitDocument } from './share.schema'
|
||||||
|
import { QuotaService } from '../user/quota.service'
|
||||||
|
import { User, UserDocument } from '../user/user.schema'
|
||||||
|
|
||||||
|
const DAILY_LIMIT = 3
|
||||||
|
const LINK_TTL_DAYS = 30
|
||||||
|
|
||||||
|
@Injectable()
|
||||||
|
export class ShareService {
|
||||||
|
private readonly logger = new Logger(ShareService.name)
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
@InjectModel(ShareRecord.name) private shareModel: Model<ShareRecordDocument>,
|
||||||
|
@InjectModel(ShareVisit.name) private visitModel: Model<ShareVisitDocument>,
|
||||||
|
@InjectModel(User.name) private userModel: Model<UserDocument>,
|
||||||
|
private quotaService: QuotaService,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
async create(userId: string, body: { type: string; refId?: string; title?: string; description?: string }) {
|
||||||
|
const shareCode = crypto.randomBytes(4).toString('hex')
|
||||||
|
const record = await this.shareModel.create({
|
||||||
|
userId: new Types.ObjectId(userId),
|
||||||
|
shareCode,
|
||||||
|
type: body.type || 'app',
|
||||||
|
refId: body.refId || '',
|
||||||
|
title: body.title || '我在职引发现了好东西',
|
||||||
|
description: body.description || '快来一起体验吧',
|
||||||
|
})
|
||||||
|
return {
|
||||||
|
shareCode: record.shareCode,
|
||||||
|
shareUrl: `/share/${record.shareCode}`,
|
||||||
|
wechatShareInfo: {
|
||||||
|
title: record.title,
|
||||||
|
description: record.description,
|
||||||
|
path: `/pages/share/share?code=${record.shareCode}`,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async visit(shareCode: string, visitorId: string, visitorUserId?: string) {
|
||||||
|
const share = await this.shareModel.findOne({ shareCode, isActive: true }).exec()
|
||||||
|
if (!share) throw new HttpException('分享链接不存在或已失效', HttpStatus.NOT_FOUND)
|
||||||
|
|
||||||
|
await this.shareModel.findByIdAndUpdate(share._id, { $inc: { visitCount: 1 } }).exec()
|
||||||
|
|
||||||
|
const sharerIdStr = share.userId.toString()
|
||||||
|
if (!visitorUserId || visitorUserId === sharerIdStr) {
|
||||||
|
return { sharer: true, visitorUserId }
|
||||||
|
}
|
||||||
|
|
||||||
|
const existing = await this.visitModel.findOne({
|
||||||
|
shareId: share._id,
|
||||||
|
visitorId,
|
||||||
|
}).exec()
|
||||||
|
if (existing) return { alreadyVisited: true, visitorUserId }
|
||||||
|
|
||||||
|
await this.visitModel.create({
|
||||||
|
shareId: share._id,
|
||||||
|
sharerId: share.userId,
|
||||||
|
visitorId,
|
||||||
|
visitorUserId: new Types.ObjectId(visitorUserId),
|
||||||
|
}).catch(() => {})
|
||||||
|
|
||||||
|
const alreadyCredited = await this.visitModel.findOne({
|
||||||
|
shareId: share._id,
|
||||||
|
visitorId,
|
||||||
|
credited: true,
|
||||||
|
}).exec()
|
||||||
|
if (alreadyCredited) return { credited: true, visitorUserId }
|
||||||
|
|
||||||
|
const todayStart = new Date()
|
||||||
|
todayStart.setHours(0, 0, 0, 0)
|
||||||
|
|
||||||
|
const todayCredited = await this.visitModel.countDocuments({
|
||||||
|
sharerId: share.userId,
|
||||||
|
credited: true,
|
||||||
|
creditedAt: { $gte: todayStart },
|
||||||
|
}).exec()
|
||||||
|
|
||||||
|
if (todayCredited >= DAILY_LIMIT) return { dailyLimitReached: true, visitorUserId }
|
||||||
|
|
||||||
|
const shareCreditsResult = await this.quotaService.grantShareCredits(sharerIdStr)
|
||||||
|
if (!shareCreditsResult) return { creditFailed: true, visitorUserId }
|
||||||
|
|
||||||
|
await this.visitModel.updateOne(
|
||||||
|
{ shareId: share._id, visitorId },
|
||||||
|
{ $set: { credited: true, creditedAt: new Date() } },
|
||||||
|
).exec()
|
||||||
|
|
||||||
|
await this.shareModel.findByIdAndUpdate(share._id, { $inc: { creditedCount: 1 } }).exec()
|
||||||
|
|
||||||
|
return { credited: true, visitorUserId }
|
||||||
|
}
|
||||||
|
|
||||||
|
async stats(userId: string) {
|
||||||
|
const todayStart = new Date()
|
||||||
|
todayStart.setHours(0, 0, 0, 0)
|
||||||
|
|
||||||
|
const [totalShares, visitAgg, todayAgg, user] = await Promise.all([
|
||||||
|
this.shareModel.countDocuments({ userId: new Types.ObjectId(userId) }).exec(),
|
||||||
|
this.visitModel.aggregate([
|
||||||
|
{ $match: { sharerId: new Types.ObjectId(userId) } },
|
||||||
|
{
|
||||||
|
$group: {
|
||||||
|
_id: null,
|
||||||
|
totalVisits: { $sum: 1 },
|
||||||
|
creditedCount: { $sum: { $cond: ['$credited', 1, 0] } },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]).exec(),
|
||||||
|
this.visitModel.countDocuments({
|
||||||
|
sharerId: new Types.ObjectId(userId),
|
||||||
|
credited: true,
|
||||||
|
creditedAt: { $gte: todayStart },
|
||||||
|
}).exec(),
|
||||||
|
this.userModel.findById(userId).exec(),
|
||||||
|
])
|
||||||
|
|
||||||
|
return {
|
||||||
|
totalShares,
|
||||||
|
totalVisits: visitAgg[0]?.totalVisits ?? 0,
|
||||||
|
creditedCount: visitAgg[0]?.creditedCount ?? 0,
|
||||||
|
todayCredited: todayAgg,
|
||||||
|
shareCredits: user?.shareCredits ?? 0,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async records(userId: string, page = 1, pageSize = 20) {
|
||||||
|
const list = await this.shareModel.find({ userId: new Types.ObjectId(userId) })
|
||||||
|
.sort({ createdAt: -1 })
|
||||||
|
.skip((page - 1) * pageSize)
|
||||||
|
.limit(pageSize)
|
||||||
|
.exec()
|
||||||
|
const total = await this.shareModel.countDocuments({ userId: new Types.ObjectId(userId) }).exec()
|
||||||
|
return {
|
||||||
|
list: list.map(r => ({
|
||||||
|
shareCode: r.shareCode,
|
||||||
|
type: r.type,
|
||||||
|
title: r.title,
|
||||||
|
visitCount: r.visitCount,
|
||||||
|
creditedCount: r.creditedCount,
|
||||||
|
createdAt: r.createdAt,
|
||||||
|
})),
|
||||||
|
total,
|
||||||
|
page,
|
||||||
|
pageSize,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async visitors(userId: string, page = 1, pageSize = 20) {
|
||||||
|
const [list, total] = await Promise.all([
|
||||||
|
this.visitModel.aggregate([
|
||||||
|
{ $match: { sharerId: new Types.ObjectId(userId) } },
|
||||||
|
{ $sort: { createdAt: -1 } },
|
||||||
|
{ $skip: (page - 1) * pageSize },
|
||||||
|
{ $limit: pageSize },
|
||||||
|
{
|
||||||
|
$lookup: {
|
||||||
|
from: 'users',
|
||||||
|
localField: 'visitorUserId',
|
||||||
|
foreignField: '_id',
|
||||||
|
as: 'visitorUser',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{ $unwind: { path: '$visitorUser', preserveNullAndEmptyArrays: true } },
|
||||||
|
{
|
||||||
|
$project: {
|
||||||
|
_id: 0,
|
||||||
|
visitorId: 1,
|
||||||
|
credited: 1,
|
||||||
|
creditedAt: 1,
|
||||||
|
createdAt: 1,
|
||||||
|
nickname: { $ifNull: ['$visitorUser.nickname', '匿名用户'] },
|
||||||
|
avatar: { $ifNull: ['$visitorUser.avatar', ''] },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]).exec(),
|
||||||
|
this.visitModel.countDocuments({ sharerId: new Types.ObjectId(userId) }).exec(),
|
||||||
|
])
|
||||||
|
return { list, total, page, pageSize }
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,21 +1,25 @@
|
|||||||
import { Controller, Get, Post, Body, Param, Res, HttpException, HttpStatus } from '@nestjs/common'
|
import { Controller, Get, Post, Body, Param, Res, HttpException, HttpStatus, UseGuards, UploadedFile, UseInterceptors } from '@nestjs/common'
|
||||||
|
import { FileInterceptor } from '@nestjs/platform-express'
|
||||||
import { Response } from 'express'
|
import { Response } from 'express'
|
||||||
import * as fs from 'fs'
|
import * as fs from 'fs'
|
||||||
|
import * as path from 'path'
|
||||||
|
import { execSync } from 'child_process'
|
||||||
import { TtsService } from './tts.service'
|
import { TtsService } from './tts.service'
|
||||||
|
import { JwtAuthGuard } from '../../common/guards/jwt-auth.guard'
|
||||||
import { Public } from '../../common/decorators/public.decorator'
|
import { Public } from '../../common/decorators/public.decorator'
|
||||||
|
|
||||||
@Controller('tts')
|
@Controller('tts')
|
||||||
export class TtsController {
|
export class TtsController {
|
||||||
constructor(private ttsService: TtsService) {}
|
constructor(private ttsService: TtsService) {}
|
||||||
|
|
||||||
@Public()
|
@UseGuards(JwtAuthGuard)
|
||||||
@Post('synthesize')
|
@Post('synthesize')
|
||||||
async synthesize(@Body('text') text: string, @Body('voice') voice?: string) {
|
async synthesize(@Body('text') text: string, @Body('voice') voice?: string) {
|
||||||
if (!text || text.length > 500) {
|
if (!text || text.length > 500) {
|
||||||
throw new HttpException('文本不能为空且不超过500字', HttpStatus.BAD_REQUEST)
|
throw new HttpException('文本不能为空且不超过500字', HttpStatus.BAD_REQUEST)
|
||||||
}
|
}
|
||||||
const result = await this.ttsService.synthesize(text, voice)
|
const result = await this.ttsService.synthesize(text, voice)
|
||||||
return { hash: result.hash, durationMs: result.durationMs }
|
return { hash: result.hash, durationMs: result.durationMs, amplitudeData: result.amplitudeData }
|
||||||
}
|
}
|
||||||
|
|
||||||
@Public()
|
@Public()
|
||||||
@@ -30,4 +34,36 @@ export class TtsController {
|
|||||||
res.setHeader('Cache-Control', 'public, max-age=31536000')
|
res.setHeader('Cache-Control', 'public, max-age=31536000')
|
||||||
stream.pipe(res)
|
stream.pipe(res)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@UseGuards(JwtAuthGuard)
|
||||||
|
@Post('asr')
|
||||||
|
@UseInterceptors(FileInterceptor('audio', { dest: '/tmp/asr_uploads' }))
|
||||||
|
async recognize(@UploadedFile() file: any) {
|
||||||
|
if (!file) throw new HttpException('请上传音频文件', HttpStatus.BAD_REQUEST)
|
||||||
|
const uploadDir = '/tmp/asr_uploads'
|
||||||
|
if (!fs.existsSync(uploadDir)) fs.mkdirSync(uploadDir, { recursive: true })
|
||||||
|
const ext = path.extname(file.originalname) || '.mp3'
|
||||||
|
const dest = path.join(uploadDir, file.filename + ext)
|
||||||
|
fs.renameSync(file.path, dest)
|
||||||
|
try {
|
||||||
|
if (process.env.OPENAI_API_KEY) {
|
||||||
|
const result = execSync(
|
||||||
|
`curl -s -X POST https://api.openai.com/v1/audio/transcriptions \
|
||||||
|
-H "Authorization: Bearer ${process.env.OPENAI_API_KEY}" \
|
||||||
|
-H "Content-Type: multipart/form-data" \
|
||||||
|
-F "file=@${dest}" \
|
||||||
|
-F "model=whisper-1" \
|
||||||
|
-F "language=zh"`,
|
||||||
|
{ encoding: 'utf8', timeout: 30000 },
|
||||||
|
)
|
||||||
|
const parsed = JSON.parse(result)
|
||||||
|
if (parsed.text) return { text: parsed.text.trim() }
|
||||||
|
}
|
||||||
|
const whisperResult = execSync(`python3 -c 'import sys, whisper; model = whisper.load_model("tiny"); print(model.transcribe(sys.argv[1], language="zh")["text"].strip())' "${dest}"`, { encoding: 'utf8', timeout: 60000 })
|
||||||
|
if (whisperResult && whisperResult.trim()) {
|
||||||
|
return { text: whisperResult.trim() }
|
||||||
|
}
|
||||||
|
} catch {}
|
||||||
|
return { text: '' }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,6 +10,19 @@ interface TtsResult {
|
|||||||
hash: string
|
hash: string
|
||||||
filePath: string
|
filePath: string
|
||||||
durationMs: number
|
durationMs: number
|
||||||
|
amplitudeData: number[]
|
||||||
|
}
|
||||||
|
|
||||||
|
const VALID_VOICES = new Set([
|
||||||
|
'zh-CN-XiaoxiaoNeural', 'zh-CN-XiaoyiNeural', 'zh-CN-YunjianNeural',
|
||||||
|
'zh-CN-YunxiNeural', 'zh-CN-YunxiaNeural', 'zh-CN-YunyangNeural',
|
||||||
|
'zh-CN-liaoning-XiaobeiNeural', 'zh-CN-shaanxi-XiaoniNeural',
|
||||||
|
])
|
||||||
|
|
||||||
|
function validateVoice(voice: string): void {
|
||||||
|
if (!VALID_VOICES.has(voice)) {
|
||||||
|
throw new Error(`不支持的语音: ${voice}`)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
@@ -23,12 +36,16 @@ export class TtsService {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async synthesize(text: string, voice: string = 'zh-CN-XiaoxiaoNeural'): Promise<TtsResult> {
|
async synthesize(text: string, voice: string = 'zh-CN-XiaoxiaoNeural'): Promise<TtsResult> {
|
||||||
|
validateVoice(voice)
|
||||||
const hash = crypto.createHash('md5').update(text + voice).digest('hex')
|
const hash = crypto.createHash('md5').update(text + voice).digest('hex')
|
||||||
const filePath = path.join(CACHE_DIR, `${hash}.mp3`)
|
const filePath = path.join(CACHE_DIR, `${hash}.mp3`)
|
||||||
|
|
||||||
if (fs.existsSync(filePath)) {
|
if (fs.existsSync(filePath)) {
|
||||||
const durationMs = await this.getDuration(filePath)
|
const durationMs = await this.getDuration(filePath)
|
||||||
return { hash, filePath, durationMs }
|
const amplitudeData = this.loadAmplitudeData(hash)
|
||||||
|
if (amplitudeData) {
|
||||||
|
return { hash, filePath, durationMs, amplitudeData }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -37,8 +54,9 @@ export class TtsService {
|
|||||||
{ timeout: 30000 },
|
{ timeout: 30000 },
|
||||||
)
|
)
|
||||||
const durationMs = await this.getDuration(filePath)
|
const durationMs = await this.getDuration(filePath)
|
||||||
|
const amplitudeData = this.extractAmplitude(filePath, hash)
|
||||||
this.logger.log(`TTS generated: hash=${hash} text="${text.slice(0, 40)}..." duration=${durationMs}ms`)
|
this.logger.log(`TTS generated: hash=${hash} text="${text.slice(0, 40)}..." duration=${durationMs}ms`)
|
||||||
return { hash, filePath, durationMs }
|
return { hash, filePath, durationMs, amplitudeData }
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
this.logger.error(`TTS failed: ${e.message}`)
|
this.logger.error(`TTS failed: ${e.message}`)
|
||||||
throw e
|
throw e
|
||||||
@@ -50,6 +68,46 @@ export class TtsService {
|
|||||||
return fs.existsSync(filePath) ? filePath : null
|
return fs.existsSync(filePath) ? filePath : null
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private extractAmplitude(mp3Path: string, hash: string): number[] {
|
||||||
|
try {
|
||||||
|
const pcmPath = `/tmp/tts-cache/${hash}.pcm`
|
||||||
|
execSync(
|
||||||
|
`ffmpeg -y -i "${mp3Path}" -f s16le -acodec pcm_s16le -ar 16000 -ac 1 "${pcmPath}" 2>/dev/null`,
|
||||||
|
{ timeout: 10000 },
|
||||||
|
)
|
||||||
|
const pcmBuf = fs.readFileSync(pcmPath)
|
||||||
|
const samples = new Int16Array(pcmBuf.buffer, pcmBuf.byteOffset, pcmBuf.byteLength / 2)
|
||||||
|
const chunkSize = Math.floor(16000 * 0.05) // 50ms
|
||||||
|
const amplitudes: number[] = []
|
||||||
|
for (let i = 0; i < samples.length; i += chunkSize) {
|
||||||
|
const end = Math.min(i + chunkSize, samples.length)
|
||||||
|
let sumSq = 0
|
||||||
|
for (let j = i; j < end; j++) {
|
||||||
|
sumSq += samples[j] * samples[j]
|
||||||
|
}
|
||||||
|
const rms = Math.sqrt(sumSq / (end - i))
|
||||||
|
amplitudes.push(Number((Math.min(1, rms / 16000)).toFixed(4)))
|
||||||
|
}
|
||||||
|
try { fs.unlinkSync(pcmPath) } catch {}
|
||||||
|
const ampPath = `/tmp/tts-cache/${hash}.amp`
|
||||||
|
fs.writeFileSync(ampPath, JSON.stringify(amplitudes))
|
||||||
|
return amplitudes
|
||||||
|
} catch (e) {
|
||||||
|
this.logger.warn(`振幅提取失败: ${e.message}`)
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private loadAmplitudeData(hash: string): number[] | null {
|
||||||
|
try {
|
||||||
|
const ampPath = `/tmp/tts-cache/${hash}.amp`
|
||||||
|
if (!fs.existsSync(ampPath)) return null
|
||||||
|
return JSON.parse(fs.readFileSync(ampPath, 'utf8'))
|
||||||
|
} catch {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private escapeText(text: string): string {
|
private escapeText(text: string): string {
|
||||||
return text.replace(/"/g, '\\"').replace(/\n/g, ' ').replace(/\r/g, '')
|
return text.replace(/"/g, '\\"').replace(/\n/g, ' ').replace(/\r/g, '')
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { Injectable, HttpException, HttpStatus } from '@nestjs/common'
|
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'
|
||||||
@@ -7,6 +7,8 @@ const FREE_OPTIMIZE_LIMIT = 3
|
|||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class QuotaService {
|
export class QuotaService {
|
||||||
|
private readonly logger = new Logger(QuotaService.name)
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
@InjectModel(User.name) private userModel: Model<UserDocument>,
|
@InjectModel(User.name) private userModel: Model<UserDocument>,
|
||||||
) {}
|
) {}
|
||||||
@@ -16,12 +18,28 @@ export class QuotaService {
|
|||||||
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
||||||
if (user.plan !== 'free') return
|
if (user.plan !== 'free') return
|
||||||
|
|
||||||
if ((user.interviewCredits || 0) <= 0) {
|
// Backward compat: migrate remaining → interviewCredits
|
||||||
throw new HttpException('面试次数已用完,请购买面试次数或开通会员', HttpStatus.FORBIDDEN)
|
if ((user.interviewCredits ?? 0) <= 0 && (user.remaining ?? 0) > 0) {
|
||||||
|
await this.userModel.findByIdAndUpdate(userId, {
|
||||||
|
$set: { interviewCredits: user.remaining, remaining: 0 },
|
||||||
|
}).exec()
|
||||||
}
|
}
|
||||||
user.interviewCredits = (user.interviewCredits || 0) - 1
|
|
||||||
user.interviewCount = (user.interviewCount || 0) + 1
|
const result = await this.userModel.findOneAndUpdate(
|
||||||
await user.save()
|
{ _id: userId, interviewCredits: { $gt: 0 } },
|
||||||
|
{ $inc: { interviewCredits: -1, interviewCount: 1 } },
|
||||||
|
{ new: true },
|
||||||
|
).exec()
|
||||||
|
if (result) return
|
||||||
|
|
||||||
|
// Fallback to share credits
|
||||||
|
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) {
|
||||||
@@ -29,64 +47,83 @@ export class QuotaService {
|
|||||||
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
||||||
if (user.plan !== 'free') return
|
if (user.plan !== 'free') return
|
||||||
|
|
||||||
// 优先扣付费额度
|
// Try paid credits first
|
||||||
if ((user.resumeOptimizeCredits || 0) > 0) {
|
const paid = await this.userModel.findOneAndUpdate(
|
||||||
user.resumeOptimizeCredits = (user.resumeOptimizeCredits || 0) - 1
|
{ _id: userId, resumeOptimizeCredits: { $gt: 0 } },
|
||||||
await user.save()
|
{ $inc: { resumeOptimizeCredits: -1 } },
|
||||||
return
|
).exec()
|
||||||
}
|
if (paid) return
|
||||||
|
|
||||||
// 免费额度
|
// Try old remaining credits (backward compat)
|
||||||
if ((user.freeOptimizeUsed || 0) < FREE_OPTIMIZE_LIMIT) {
|
const oldRemaining = await this.userModel.findOneAndUpdate(
|
||||||
user.freeOptimizeUsed = (user.freeOptimizeUsed || 0) + 1
|
{ _id: userId, remaining: { $gt: 0 } },
|
||||||
await user.save()
|
{ $inc: { remaining: -1 } },
|
||||||
return
|
).exec()
|
||||||
}
|
if (oldRemaining) return
|
||||||
|
|
||||||
|
// Then free limit
|
||||||
|
const freeResult = await this.userModel.findOneAndUpdate(
|
||||||
|
{ _id: userId, freeOptimizeUsed: { $lt: FREE_OPTIMIZE_LIMIT } },
|
||||||
|
{ $inc: { freeOptimizeUsed: 1 } },
|
||||||
|
).exec()
|
||||||
|
if (freeResult) return
|
||||||
|
|
||||||
|
// Fallback to share credits
|
||||||
|
const shareResult = await this.userModel.findOneAndUpdate(
|
||||||
|
{ _id: userId, shareCredits: { $gt: 0 } },
|
||||||
|
{ $inc: { shareCredits: -1 } },
|
||||||
|
).exec()
|
||||||
|
if (shareResult) return
|
||||||
|
|
||||||
throw new HttpException('简历优化次数已用完,请购买优化次数或开通会员', HttpStatus.FORBIDDEN)
|
throw new HttpException('简历优化次数已用完,请购买优化次数或开通会员', HttpStatus.FORBIDDEN)
|
||||||
}
|
}
|
||||||
|
|
||||||
async checkDownload(userId: string, resume: { paidDownload?: boolean }): Promise<boolean> {
|
async grantShareCredits(userId: string, amount = 1): Promise<boolean> {
|
||||||
const user = await this.userModel.findById(userId).exec()
|
const result = await this.userModel.findByIdAndUpdate(
|
||||||
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
userId,
|
||||||
|
{ $inc: { shareCredits: amount } },
|
||||||
if (resume.paidDownload) return true
|
).exec()
|
||||||
if ((user.resumeDownloadCredits || 0) > 0) return true
|
return !!result
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async deductDownload(userId: string, resume: { paidDownload?: boolean; _id?: any }) {
|
async checkAndDeductDownload(userId: string, paidDownload: boolean): Promise<boolean> {
|
||||||
const user = await this.userModel.findById(userId).exec()
|
if (paidDownload) return true
|
||||||
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
|
||||||
|
|
||||||
if (resume.paidDownload) return
|
const result = await this.userModel.findOneAndUpdate(
|
||||||
if ((user.resumeDownloadCredits || 0) > 0) {
|
{ _id: userId, resumeDownloadCredits: { $gt: 0 } },
|
||||||
user.resumeDownloadCredits = (user.resumeDownloadCredits || 0) - 1
|
{ $inc: { resumeDownloadCredits: -1 } },
|
||||||
await user.save()
|
).exec()
|
||||||
}
|
return !!result
|
||||||
}
|
}
|
||||||
|
|
||||||
async grantCredits(userId: string, type: 'interview' | 'optimize' | 'download', amount: number) {
|
async grantCredits(userId: string, type: 'interview' | 'optimize' | 'download', amount: number) {
|
||||||
const user = await this.userModel.findById(userId).exec()
|
if (amount <= 0) throw new HttpException('无效数量', HttpStatus.BAD_REQUEST)
|
||||||
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
|
||||||
|
|
||||||
if (type === 'interview') user.interviewCredits = (user.interviewCredits || 0) + amount
|
const fieldMap: Record<string, string> = {
|
||||||
else if (type === 'optimize') user.resumeOptimizeCredits = (user.resumeOptimizeCredits || 0) + amount
|
interview: 'interviewCredits',
|
||||||
else if (type === 'download') user.resumeDownloadCredits = (user.resumeDownloadCredits || 0) + amount
|
optimize: 'resumeOptimizeCredits',
|
||||||
|
download: 'resumeDownloadCredits',
|
||||||
|
}
|
||||||
|
const field = fieldMap[type]
|
||||||
|
if (!field) throw new HttpException('无效类型', HttpStatus.BAD_REQUEST)
|
||||||
|
|
||||||
await user.save()
|
const result = await this.userModel.findByIdAndUpdate(
|
||||||
|
userId,
|
||||||
|
{ $inc: { [field]: amount } },
|
||||||
|
).exec()
|
||||||
|
if (!result) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
||||||
}
|
}
|
||||||
|
|
||||||
async setPlanQuota(userId: string, plan: string, credits: { interview: number; resumeOptimize: number; resumeDownload: number }) {
|
async setPlanQuota(userId: string, _plan: string, credits: { interview: number; resumeOptimize: number; resumeDownload: number }) {
|
||||||
const user = await this.userModel.findById(userId).exec()
|
const result = await this.userModel.findByIdAndUpdate(userId, {
|
||||||
if (!user) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
$set: {
|
||||||
|
remaining: 999,
|
||||||
user.remaining = 999
|
interviewCredits: credits.interview,
|
||||||
user.interviewCredits = credits.interview
|
resumeOptimizeCredits: credits.resumeOptimize,
|
||||||
user.resumeOptimizeCredits = credits.resumeOptimize
|
resumeDownloadCredits: credits.resumeDownload,
|
||||||
user.resumeDownloadCredits = credits.resumeDownload
|
freeOptimizeUsed: FREE_OPTIMIZE_LIMIT,
|
||||||
user.freeOptimizeUsed = FREE_OPTIMIZE_LIMIT // 会员不再消耗免费次数
|
},
|
||||||
|
}).exec()
|
||||||
await user.save()
|
if (!result) throw new HttpException('用户不存在', HttpStatus.NOT_FOUND)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { Controller, Post, Get, Put, Body, Req } from '@nestjs/common'
|
import { Controller, Post, Get, Put, Body, Req, HttpCode, HttpStatus } 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'
|
||||||
@@ -9,12 +9,14 @@ export class UserController {
|
|||||||
|
|
||||||
@Public()
|
@Public()
|
||||||
@Post('send-code')
|
@Post('send-code')
|
||||||
|
@HttpCode(HttpStatus.OK)
|
||||||
async sendCode(@Body('phone') phone: string) {
|
async sendCode(@Body('phone') phone: string) {
|
||||||
return this.userService.sendCode(phone)
|
return this.userService.sendCode(phone)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Public()
|
@Public()
|
||||||
@Post('login')
|
@Post('login')
|
||||||
|
@HttpCode(HttpStatus.OK)
|
||||||
async login(@Body('phone') phone: string, @Body('code') code: string) {
|
async login(@Body('phone') phone: string, @Body('code') code: string) {
|
||||||
return this.userService.loginByPhone(phone, code)
|
return this.userService.loginByPhone(phone, code)
|
||||||
}
|
}
|
||||||
@@ -22,12 +24,14 @@ export class UserController {
|
|||||||
// 📧 邮箱验证码登录(H5 用)
|
// 📧 邮箱验证码登录(H5 用)
|
||||||
@Public()
|
@Public()
|
||||||
@Post('send-email-code')
|
@Post('send-email-code')
|
||||||
|
@HttpCode(HttpStatus.OK)
|
||||||
async sendEmailCode(@Body('email') email: string) {
|
async sendEmailCode(@Body('email') email: string) {
|
||||||
return this.userService.sendEmailCode(email)
|
return this.userService.sendEmailCode(email)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Public()
|
@Public()
|
||||||
@Post('email-login')
|
@Post('email-login')
|
||||||
|
@HttpCode(HttpStatus.OK)
|
||||||
async emailLogin(@Body('email') email: string, @Body('code') code: string) {
|
async emailLogin(@Body('email') email: string, @Body('code') code: string) {
|
||||||
return this.userService.loginByEmail(email, code)
|
return this.userService.loginByEmail(email, code)
|
||||||
}
|
}
|
||||||
@@ -35,6 +39,7 @@ export class UserController {
|
|||||||
// 密码登录
|
// 密码登录
|
||||||
@Public()
|
@Public()
|
||||||
@Post('password-login')
|
@Post('password-login')
|
||||||
|
@HttpCode(HttpStatus.OK)
|
||||||
async passwordLogin(@Body('email') email: string, @Body('password') password: string) {
|
async passwordLogin(@Body('email') email: string, @Body('password') password: string) {
|
||||||
return this.userService.loginByPassword(email, password)
|
return this.userService.loginByPassword(email, password)
|
||||||
}
|
}
|
||||||
@@ -42,6 +47,7 @@ export class UserController {
|
|||||||
// 邮箱+密码注册
|
// 邮箱+密码注册
|
||||||
@Public()
|
@Public()
|
||||||
@Post('register')
|
@Post('register')
|
||||||
|
@HttpCode(HttpStatus.OK)
|
||||||
async register(@Body('email') email: string, @Body('password') password: string) {
|
async register(@Body('email') email: string, @Body('password') password: string) {
|
||||||
return this.userService.registerWithPassword(email, password)
|
return this.userService.registerWithPassword(email, password)
|
||||||
}
|
}
|
||||||
@@ -49,6 +55,7 @@ export class UserController {
|
|||||||
// 微信静默登录
|
// 微信静默登录
|
||||||
@Public()
|
@Public()
|
||||||
@Post('wx-login')
|
@Post('wx-login')
|
||||||
|
@HttpCode(HttpStatus.OK)
|
||||||
async wxLogin(@Body('code') code: string) {
|
async wxLogin(@Body('code') code: string) {
|
||||||
return this.userService.loginByWx(code)
|
return this.userService.loginByWx(code)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -48,6 +48,9 @@ export class User {
|
|||||||
@Prop({ default: 0 })
|
@Prop({ default: 0 })
|
||||||
freeOptimizeUsed: number // 已使用免费优化次数(上限 3)
|
freeOptimizeUsed: number // 已使用免费优化次数(上限 3)
|
||||||
|
|
||||||
|
@Prop({ default: 0 })
|
||||||
|
shareCredits: number // 分享积分,每 3 次有效访问获 1 积分
|
||||||
|
|
||||||
@Prop({ default: 'user' })
|
@Prop({ default: 'user' })
|
||||||
role: string // 'user' | 'admin'
|
role: string // 'user' | 'admin'
|
||||||
|
|
||||||
|
|||||||
@@ -215,6 +215,7 @@ export class UserService {
|
|||||||
resumeOptimizeCredits: user.resumeOptimizeCredits ?? 0,
|
resumeOptimizeCredits: user.resumeOptimizeCredits ?? 0,
|
||||||
resumeDownloadCredits: user.resumeDownloadCredits ?? 0,
|
resumeDownloadCredits: user.resumeDownloadCredits ?? 0,
|
||||||
freeOptimizeUsed: user.freeOptimizeUsed ?? 0,
|
freeOptimizeUsed: user.freeOptimizeUsed ?? 0,
|
||||||
|
shareCredits: user.shareCredits ?? 0,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,253 @@
|
|||||||
|
# 分享功能设计方案
|
||||||
|
|
||||||
|
## 1. 概述
|
||||||
|
|
||||||
|
**目标**:用户可通过分享面试/简历等给好友或微信群,每次有效分享获得积分奖励,积分可抵扣面试/简历优化次数,形成用户增长闭环。
|
||||||
|
|
||||||
|
**核心逻辑**:分享者 A → 生成分享链接 → 非 A 的其他用户点击 → 计入有效分享一次 → A 获得 1 个「分享积分」,每 3 次有效分享可兑换 1 次面试或简历优化次数。
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 2. 数据模型
|
||||||
|
|
||||||
|
### User 新增字段
|
||||||
|
|
||||||
|
```
|
||||||
|
shareCredits: number // 分享积分,默认 0
|
||||||
|
```
|
||||||
|
|
||||||
|
- 1 个 shareCredit 可兑换 1 次 面试 或 1 次 简历优化
|
||||||
|
- 在 QuotaService 中作为兜底:当 interviewCredits / resumeOptimizeCredits 为 0 时尝试消耗 shareCredits
|
||||||
|
|
||||||
|
### ShareRecord 集合
|
||||||
|
|
||||||
|
```
|
||||||
|
@Schema({ timestamps: true })
|
||||||
|
ShareRecord {
|
||||||
|
_id: ObjectId
|
||||||
|
userId: ObjectId // 分享者
|
||||||
|
shareCode: string // 8位唯一短码 (hex),用于分享链接
|
||||||
|
type: 'interview' | 'resume' | 'app'
|
||||||
|
refId: string // 关联的面试/简历ID(可选)
|
||||||
|
title: string // 分享标题
|
||||||
|
description: string // 分享描述
|
||||||
|
visitCount: number // 总访问次数
|
||||||
|
creditedCount: number // 已计为有效的访问次数
|
||||||
|
isActive: boolean // 链接是否有效
|
||||||
|
createdAt: Date // timestamps
|
||||||
|
updatedAt: Date
|
||||||
|
}
|
||||||
|
|
||||||
|
index: { shareCode: 1 } unique
|
||||||
|
index: { userId: 1, createdAt: -1 }
|
||||||
|
```
|
||||||
|
|
||||||
|
### ShareVisit 集合(访问记录)
|
||||||
|
|
||||||
|
```
|
||||||
|
@Schema({ timestamps: true })
|
||||||
|
ShareVisit {
|
||||||
|
_id: ObjectId
|
||||||
|
shareId: ObjectId // 关联的 ShareRecord
|
||||||
|
sharerId: ObjectId // 分享者 userId(冗余,便于查询)
|
||||||
|
visitorId: string // 访问者标识(openId 或 匿名ID)
|
||||||
|
visitorUserId: ObjectId // 访问者 userId(如果已注册/登录)
|
||||||
|
credited: boolean // 是否已为此访问发积分
|
||||||
|
creditedAt: Date
|
||||||
|
createdAt: Date // timestamps
|
||||||
|
}
|
||||||
|
|
||||||
|
index: { shareId: 1, visitorId: 1 } unique // 同一人只计一次
|
||||||
|
index: { sharerId: 1, createdAt: -1 }
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 3. API 设计
|
||||||
|
|
||||||
|
| 方法 | 路径 | 权限 | 说明 |
|
||||||
|
|------|------|------|------|
|
||||||
|
| POST | /api/share/create | JWT | 生成分享链接,返回 shareCode + URL |
|
||||||
|
| GET | /api/share/visit/:shareCode | Public | 访问分享链接,记录访问+判定发放积分 |
|
||||||
|
| GET | /api/share/stats | JWT | 获取我的分享统计(总分享次数、总积分、今日积分) |
|
||||||
|
| GET | /api/share/records | JWT | 获取我的分享记录列表(分页) |
|
||||||
|
| GET | /api/share/visitors | JWT | 获取访问过我分享的用户列表(分页) |
|
||||||
|
|
||||||
|
### POST /api/share/create
|
||||||
|
|
||||||
|
Request:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"type": "interview" | "resume" | "app",
|
||||||
|
"refId": "xxx",
|
||||||
|
"title": "我在职引完成了AI模拟面试",
|
||||||
|
"description": "快来和我一起练习面试吧"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Response:
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"shareCode": "a1b2c3d4",
|
||||||
|
"shareUrl": "https://zhiyin.app/share/a1b2c3d4",
|
||||||
|
"wechatShareInfo": {
|
||||||
|
"title": "...",
|
||||||
|
"description": "...",
|
||||||
|
"path": "/pages/share/share?code=a1b2c3d4"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### GET /api/share/visit/:shareCode
|
||||||
|
|
||||||
|
- 如果来自微信小程序:记录 `visitorId = openId`
|
||||||
|
- 如果来自 H5 链接:记录 `visitorId = 匿名设备ID`(可选)
|
||||||
|
- 判定条件:
|
||||||
|
- `visitorId !== sharer.openId`(自己点自己不算)
|
||||||
|
- 该 `visitorId` 之前未在该 shareId 下领过积分
|
||||||
|
- 当日该分享者已获得积分 < 3(日上限)
|
||||||
|
- 满足条件 → `credited = true`,shareCredits + 1
|
||||||
|
- 返回重定向或展示落地页
|
||||||
|
|
||||||
|
### GET /api/share/stats
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"totalShares": 15,
|
||||||
|
"totalVisits": 43,
|
||||||
|
"creditedCount": 12,
|
||||||
|
"todayCredited": 2,
|
||||||
|
"shareCredits": 4
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### GET /api/share/records
|
||||||
|
|
||||||
|
```json
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"shareCode": "a1b2c3d4",
|
||||||
|
"type": "interview",
|
||||||
|
"title": "我在职引完成了AI模拟面试",
|
||||||
|
"visitCount": 5,
|
||||||
|
"creditedCount": 3,
|
||||||
|
"createdAt": "2026-06-12T10:00:00Z"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
```
|
||||||
|
|
||||||
|
### GET /api/share/visitors
|
||||||
|
|
||||||
|
```json
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"visitor": { "nickname": "张三", "avatar": "..." },
|
||||||
|
"credited": true,
|
||||||
|
"creditedAt": "2026-06-12T10:30:00Z"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 4. 积分兜底机制
|
||||||
|
|
||||||
|
在 `QuotaService.checkAndDeductInterview()` 和 `checkAndDeductOptimize()` 中增加兜底:
|
||||||
|
|
||||||
|
```
|
||||||
|
if (interviewCredits <= 0 && shareCredits > 0) {
|
||||||
|
shareCredits -= 1
|
||||||
|
return // 用分享积分抵扣
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
同时新增 `useShareCredit(userId, type)` 方法由前端显式调用,或自动在抵扣链中完成。
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 5. 前端实现
|
||||||
|
|
||||||
|
### 5.1 分享入口
|
||||||
|
|
||||||
|
**进入点 1:面试报告页**(`pages/report/report.vue`)
|
||||||
|
- 在"生成分享卡片"按钮旁增加"分享给好友"按钮
|
||||||
|
- 点击后调用 `POST /api/share/create` 生成 shareCode
|
||||||
|
- 微信小程序内直接调 `wx.shareAppMessage`
|
||||||
|
- H5 环境复制分享链接到剪贴板
|
||||||
|
|
||||||
|
**进入点 2:简历优化结果页**(`pages/result/result.vue`)
|
||||||
|
- 同上
|
||||||
|
|
||||||
|
**进入点 3:应用首页**(`pages/index/index.vue`)
|
||||||
|
- "邀请好友"入口,固定分享 app 类型
|
||||||
|
|
||||||
|
### 5.2 我的分享页
|
||||||
|
|
||||||
|
**页面位置**:`/pages/share/share.vue`
|
||||||
|
**进入方式**:用户「我的」页面 → "我的分享" 菜单项
|
||||||
|
|
||||||
|
页面内容:
|
||||||
|
- 顶部统计卡片:总收益积分、今日收益、总分享次数
|
||||||
|
- "分享给好友"主按钮
|
||||||
|
- Tab 切换:「分享记录」/ 「访问记录」
|
||||||
|
- 分享记录列表:分享类型、标题、访问数、有效数、时间
|
||||||
|
- 访问记录列表:访客昵称、访问时间、是否已积分
|
||||||
|
|
||||||
|
### 5.3 微信分享
|
||||||
|
|
||||||
|
使用 uni-app `onShareAppMessage` 生命周期:
|
||||||
|
```javascript
|
||||||
|
// 在页面中定义
|
||||||
|
onShareAppMessage() {
|
||||||
|
return {
|
||||||
|
title: '我在职引完成了AI模拟面试',
|
||||||
|
path: `/pages/share/share?code=${this.shareCode}`,
|
||||||
|
imageUrl: '...'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 5.4 H5 分享链接
|
||||||
|
|
||||||
|
分享链接格式:`https://zhiyin.app/share/{shareCode}`
|
||||||
|
H5 落地页展示:
|
||||||
|
- 分享者信息
|
||||||
|
- 分享内容预览
|
||||||
|
- "打开小程序" / "下载 App" 引导按钮
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 6. 风控规则
|
||||||
|
|
||||||
|
| 规则 | 说明 |
|
||||||
|
|------|------|
|
||||||
|
| 日上限 | 同一用户每日最多获 3 个分享积分 |
|
||||||
|
| 自分享过滤 | 自己点自己的链接不计数 |
|
||||||
|
| 同人去重 | 同一访客对同一链接只计一次 |
|
||||||
|
| 链接有效期 | 分享链接 30 天有效期 |
|
||||||
|
| 频率限制 | 每分钟最多创建 5 次分享(前端控制) |
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 7. 存储方案
|
||||||
|
|
||||||
|
- **ShareRecord**:MongoDB 集合 `sharerecords`
|
||||||
|
- **ShareVisit**:MongoDB 集合 `sharevisits`
|
||||||
|
- 积分字段 `shareCredits` 存储在 User 文档上,`$inc` 原子操作确保并发安全
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 8. 依赖模块
|
||||||
|
|
||||||
|
- `QuotaService`(来自 UserModule)- 积分发放与抵扣
|
||||||
|
- `UserModule`(获取 User 信息、校验身份)
|
||||||
|
- `PricingModule`(@Global 已有,无需额外导入)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## 9. 未纳入 MVP 的内容
|
||||||
|
|
||||||
|
- 分享链接落地页(H5 路由 `/share/:code`)— 第一期直接返回 JSON
|
||||||
|
- 分享数据可视化图表
|
||||||
|
- 积分兑换商城
|
||||||
|
- 排行榜 / 邀请竞赛
|
||||||
@@ -25,7 +25,7 @@
|
|||||||
"@dcloudio/vite-plugin-uni": "3.0.0-4060620250520001",
|
"@dcloudio/vite-plugin-uni": "3.0.0-4060620250520001",
|
||||||
"@vue/test-utils": "^2.4.11",
|
"@vue/test-utils": "^2.4.11",
|
||||||
"jsdom": "^29.1.1",
|
"jsdom": "^29.1.1",
|
||||||
"miniprogram-ci": "^2.1.31",
|
"miniprogram-ci": "^2.1.42",
|
||||||
"sass": "^1.70.0",
|
"sass": "^1.70.0",
|
||||||
"typescript": "^5.3.0",
|
"typescript": "^5.3.0",
|
||||||
"vite": "^5.2.0",
|
"vite": "^5.2.0",
|
||||||
@@ -5674,15 +5674,15 @@
|
|||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/@swc/core": {
|
"node_modules/@swc/core": {
|
||||||
"version": "1.4.14",
|
"version": "1.15.33",
|
||||||
"resolved": "https://registry.npmmirror.com/@swc/core/-/core-1.4.14.tgz",
|
"resolved": "https://registry.npmmirror.com/@swc/core/-/core-1.15.33.tgz",
|
||||||
"integrity": "sha512-tHXg6OxboUsqa/L7DpsCcFnxhLkqN/ht5pCwav1HnvfthbiNIJypr86rNx4cUnQDJepETviSqBTIjxa7pSpGDQ==",
|
"integrity": "sha512-jOlwnFV2xhuuZeAUILGFULeR6vDPfijEJ57evfocwznQldLU3w2cZ9bSDryY9ip+AsM3r1NJKzf47V2NXebkeQ==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"hasInstallScript": true,
|
"hasInstallScript": true,
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@swc/counter": "^0.1.2",
|
"@swc/counter": "^0.1.3",
|
||||||
"@swc/types": "^0.1.5"
|
"@swc/types": "^0.1.26"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=10"
|
"node": ">=10"
|
||||||
@@ -5692,19 +5692,21 @@
|
|||||||
"url": "https://opencollective.com/swc"
|
"url": "https://opencollective.com/swc"
|
||||||
},
|
},
|
||||||
"optionalDependencies": {
|
"optionalDependencies": {
|
||||||
"@swc/core-darwin-arm64": "1.4.14",
|
"@swc/core-darwin-arm64": "1.15.33",
|
||||||
"@swc/core-darwin-x64": "1.4.14",
|
"@swc/core-darwin-x64": "1.15.33",
|
||||||
"@swc/core-linux-arm-gnueabihf": "1.4.14",
|
"@swc/core-linux-arm-gnueabihf": "1.15.33",
|
||||||
"@swc/core-linux-arm64-gnu": "1.4.14",
|
"@swc/core-linux-arm64-gnu": "1.15.33",
|
||||||
"@swc/core-linux-arm64-musl": "1.4.14",
|
"@swc/core-linux-arm64-musl": "1.15.33",
|
||||||
"@swc/core-linux-x64-gnu": "1.4.14",
|
"@swc/core-linux-ppc64-gnu": "1.15.33",
|
||||||
"@swc/core-linux-x64-musl": "1.4.14",
|
"@swc/core-linux-s390x-gnu": "1.15.33",
|
||||||
"@swc/core-win32-arm64-msvc": "1.4.14",
|
"@swc/core-linux-x64-gnu": "1.15.33",
|
||||||
"@swc/core-win32-ia32-msvc": "1.4.14",
|
"@swc/core-linux-x64-musl": "1.15.33",
|
||||||
"@swc/core-win32-x64-msvc": "1.4.14"
|
"@swc/core-win32-arm64-msvc": "1.15.33",
|
||||||
|
"@swc/core-win32-ia32-msvc": "1.15.33",
|
||||||
|
"@swc/core-win32-x64-msvc": "1.15.33"
|
||||||
},
|
},
|
||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
"@swc/helpers": "^0.5.0"
|
"@swc/helpers": ">=0.5.17"
|
||||||
},
|
},
|
||||||
"peerDependenciesMeta": {
|
"peerDependenciesMeta": {
|
||||||
"@swc/helpers": {
|
"@swc/helpers": {
|
||||||
@@ -5713,9 +5715,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@swc/core-darwin-arm64": {
|
"node_modules/@swc/core-darwin-arm64": {
|
||||||
"version": "1.4.14",
|
"version": "1.15.33",
|
||||||
"resolved": "https://registry.npmmirror.com/@swc/core-darwin-arm64/-/core-darwin-arm64-1.4.14.tgz",
|
"resolved": "https://registry.npmmirror.com/@swc/core-darwin-arm64/-/core-darwin-arm64-1.15.33.tgz",
|
||||||
"integrity": "sha512-8iPfLhYNspBl836YYsfv6ErXwDUqJ7IMieddV3Ey/t/97JAEAdNDUdtTKDtbyP0j/Ebyqyn+fKcqwSq7rAof0g==",
|
"integrity": "sha512-N+L0uXhuO7FIfzqwgxmzv0zIpV0qEp8wPX3QQs2p4atjMoywup2JTeDlXPw+z9pWJGCae3JjM+tZ6myclI+2gA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -5730,9 +5732,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@swc/core-darwin-x64": {
|
"node_modules/@swc/core-darwin-x64": {
|
||||||
"version": "1.4.14",
|
"version": "1.15.33",
|
||||||
"resolved": "https://registry.npmmirror.com/@swc/core-darwin-x64/-/core-darwin-x64-1.4.14.tgz",
|
"resolved": "https://registry.npmmirror.com/@swc/core-darwin-x64/-/core-darwin-x64-1.15.33.tgz",
|
||||||
"integrity": "sha512-9CqSj8uRZ92cnlgAlVaWMaJJBdxtNvCzJxaGj5KuIseeG6Q0l1g+qk8JcU7h9dAsH9saHTNwNFBVGKQo0W0ujg==",
|
"integrity": "sha512-/Il4QHSOhV4FekbsDtkrNmKbsX26oSysvgrRswa/RYOHXAkwXDbB4jaeKq6PsJLSPkzJ2KzQ061gtBnk0vNHfA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -5747,9 +5749,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@swc/core-linux-arm-gnueabihf": {
|
"node_modules/@swc/core-linux-arm-gnueabihf": {
|
||||||
"version": "1.4.14",
|
"version": "1.15.33",
|
||||||
"resolved": "https://registry.npmmirror.com/@swc/core-linux-arm-gnueabihf/-/core-linux-arm-gnueabihf-1.4.14.tgz",
|
"resolved": "https://registry.npmmirror.com/@swc/core-linux-arm-gnueabihf/-/core-linux-arm-gnueabihf-1.15.33.tgz",
|
||||||
"integrity": "sha512-mfd5JArPITTzMjcezH4DwMw+BdjBV1y25Khp8itEIpdih9ei+fvxOOrDYTN08b466NuE2dF2XuhKtRLA7fXArQ==",
|
"integrity": "sha512-C64hBnBxq4viOPQ8hlx+2lJ23bzZBGnjw7ryALmS+0Q3zHmwO8lw1/DArLENw4Q18/0w5wdEO1k3m1wWNtKGqQ==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm"
|
"arm"
|
||||||
],
|
],
|
||||||
@@ -5764,9 +5766,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@swc/core-linux-arm64-gnu": {
|
"node_modules/@swc/core-linux-arm64-gnu": {
|
||||||
"version": "1.4.14",
|
"version": "1.15.33",
|
||||||
"resolved": "https://registry.npmmirror.com/@swc/core-linux-arm64-gnu/-/core-linux-arm64-gnu-1.4.14.tgz",
|
"resolved": "https://registry.npmmirror.com/@swc/core-linux-arm64-gnu/-/core-linux-arm64-gnu-1.15.33.tgz",
|
||||||
"integrity": "sha512-3Lqlhlmy8MVRS9xTShMaPAp0oyUt0KFhDs4ixJsjdxKecE0NJSV/MInuDmrkij1C8/RQ2wySRlV9np5jK86oWw==",
|
"integrity": "sha512-TRJfnJbX3jqpxRDRoieMzRiCBS5jOmXNb3iQXmcgjFEHKLnAgK1RZRU8Cq1MsPqO4jAJp/ld1G4O3fXuxv85uw==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -5781,9 +5783,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@swc/core-linux-arm64-musl": {
|
"node_modules/@swc/core-linux-arm64-musl": {
|
||||||
"version": "1.4.14",
|
"version": "1.15.33",
|
||||||
"resolved": "https://registry.npmmirror.com/@swc/core-linux-arm64-musl/-/core-linux-arm64-musl-1.4.14.tgz",
|
"resolved": "https://registry.npmmirror.com/@swc/core-linux-arm64-musl/-/core-linux-arm64-musl-1.15.33.tgz",
|
||||||
"integrity": "sha512-n0YoCa64TUcJrbcXIHIHDWQjdUPdaXeMHNEu7yyBtOpm01oMGTKP3frsUXIABLBmAVWtKvqit4/W1KVKn5gJzg==",
|
"integrity": "sha512-il7tYM+CpUNzieQbwAjFT1P8zqAhmGWNAGhQZBnxurXZ0aNn+5nqYFTEUKNZl7QibtT0uQXzTZrNGHCIj6Y1Og==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -5797,10 +5799,44 @@
|
|||||||
"node": ">=10"
|
"node": ">=10"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@swc/core-linux-ppc64-gnu": {
|
||||||
|
"version": "1.15.33",
|
||||||
|
"resolved": "https://registry.npmmirror.com/@swc/core-linux-ppc64-gnu/-/core-linux-ppc64-gnu-1.15.33.tgz",
|
||||||
|
"integrity": "sha512-ZtNBwN0Z7CFj9Il0FcPaKdjgP7URyKu/3RfH46vq+0paOBqLj4NYldD6Qo//Duif/7IOtAraUfDOmp0PLAufog==",
|
||||||
|
"cpu": [
|
||||||
|
"ppc64"
|
||||||
|
],
|
||||||
|
"dev": true,
|
||||||
|
"license": "Apache-2.0 AND MIT",
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"linux"
|
||||||
|
],
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/@swc/core-linux-s390x-gnu": {
|
||||||
|
"version": "1.15.33",
|
||||||
|
"resolved": "https://registry.npmmirror.com/@swc/core-linux-s390x-gnu/-/core-linux-s390x-gnu-1.15.33.tgz",
|
||||||
|
"integrity": "sha512-De1IyajoOmhOYYjw/lx66bKlyDpHZTueqwpDrWgf5O7T6d1ODeJJO9/OqMBmrBQc5C+dNnlmIufHsp4QVCWufA==",
|
||||||
|
"cpu": [
|
||||||
|
"s390x"
|
||||||
|
],
|
||||||
|
"dev": true,
|
||||||
|
"license": "Apache-2.0 AND MIT",
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"linux"
|
||||||
|
],
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@swc/core-linux-x64-gnu": {
|
"node_modules/@swc/core-linux-x64-gnu": {
|
||||||
"version": "1.4.14",
|
"version": "1.15.33",
|
||||||
"resolved": "https://registry.npmmirror.com/@swc/core-linux-x64-gnu/-/core-linux-x64-gnu-1.4.14.tgz",
|
"resolved": "https://registry.npmmirror.com/@swc/core-linux-x64-gnu/-/core-linux-x64-gnu-1.15.33.tgz",
|
||||||
"integrity": "sha512-CGmlwLWbfG1dB4jZBJnp2IWlK5xBMNLjN7AR5kKA3sEpionoccEnChOEvfux1UdVJQjLRKuHNV9yGyqGBTpxfQ==",
|
"integrity": "sha512-mGTH0YxmUN+x6vRN/I6NOk5X0ogNktkwPnJ94IMvR7QjhRDwL0O8RXEDhyUM0YtwWrryBOqaJQBX4zruxEPRGw==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -5815,9 +5851,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@swc/core-linux-x64-musl": {
|
"node_modules/@swc/core-linux-x64-musl": {
|
||||||
"version": "1.4.14",
|
"version": "1.15.33",
|
||||||
"resolved": "https://registry.npmmirror.com/@swc/core-linux-x64-musl/-/core-linux-x64-musl-1.4.14.tgz",
|
"resolved": "https://registry.npmmirror.com/@swc/core-linux-x64-musl/-/core-linux-x64-musl-1.15.33.tgz",
|
||||||
"integrity": "sha512-xq4npk8YKYmNwmr8fbvF2KP3kUVdZYfXZMQnW425gP3/sn+yFQO8Nd0bGH40vOVQn41kEesSe0Z5O/JDor2TgQ==",
|
"integrity": "sha512-hj628ZkSEJf6zMf5VMbYrG2O6QqyTIp2qwY6VlCjvIa9lAEZ5c2lfPblCLVGYubTeLJDxadLB/CxqQYOQABeEQ==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -5832,9 +5868,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@swc/core-win32-arm64-msvc": {
|
"node_modules/@swc/core-win32-arm64-msvc": {
|
||||||
"version": "1.4.14",
|
"version": "1.15.33",
|
||||||
"resolved": "https://registry.npmmirror.com/@swc/core-win32-arm64-msvc/-/core-win32-arm64-msvc-1.4.14.tgz",
|
"resolved": "https://registry.npmmirror.com/@swc/core-win32-arm64-msvc/-/core-win32-arm64-msvc-1.15.33.tgz",
|
||||||
"integrity": "sha512-imq0X+gU9uUe6FqzOQot5gpKoaC00aCUiN58NOzwp0QXEupn8CDuZpdBN93HiZswfLruu5jA1tsc15x6v9p0Yg==",
|
"integrity": "sha512-GV2oohtN2/5+KSccl86VULu3aT+LrISC8uzgSq0FRnikpD+Zwc+sBlXmoKQ+Db6jI57ITUOIB8jRkdGMABC29g==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -5849,9 +5885,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@swc/core-win32-ia32-msvc": {
|
"node_modules/@swc/core-win32-ia32-msvc": {
|
||||||
"version": "1.4.14",
|
"version": "1.15.33",
|
||||||
"resolved": "https://registry.npmmirror.com/@swc/core-win32-ia32-msvc/-/core-win32-ia32-msvc-1.4.14.tgz",
|
"resolved": "https://registry.npmmirror.com/@swc/core-win32-ia32-msvc/-/core-win32-ia32-msvc-1.15.33.tgz",
|
||||||
"integrity": "sha512-cH6QpXMw5D3t+lpx6SkErHrxN0yFzmQ0lgNAJxoDRiaAdDbqA6Col8UqUJwUS++Ul6aCWgNhCdiEYehPaoyDPA==",
|
"integrity": "sha512-gtyvzSNR8DHKfFEA2uqb8Ld1myqi6uEg2jyeUq3ikn5ytYs7H8RpZYC8mdy4NXr8hfcdJfCLXPlYaqqfBXpoEQ==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"ia32"
|
"ia32"
|
||||||
],
|
],
|
||||||
@@ -5866,9 +5902,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@swc/core-win32-x64-msvc": {
|
"node_modules/@swc/core-win32-x64-msvc": {
|
||||||
"version": "1.4.14",
|
"version": "1.15.33",
|
||||||
"resolved": "https://registry.npmmirror.com/@swc/core-win32-x64-msvc/-/core-win32-x64-msvc-1.4.14.tgz",
|
"resolved": "https://registry.npmmirror.com/@swc/core-win32-x64-msvc/-/core-win32-x64-msvc-1.15.33.tgz",
|
||||||
"integrity": "sha512-FmZ4Tby4wW65K/36BKzmuu7mlq7cW5XOxzvufaSNVvQ5PN4OodAlqPjToe029oma4Av+ykJiif64scMttyNAzg==",
|
"integrity": "sha512-d6fRqQSkJI+kmMEBWaDQ7TMl8+YjLYbwRUPZQ9DY0ORBJeTzOrG0twvfvlZ2xgw6jA0ScQKgfBm4vHLSLl5Hqg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -15461,9 +15497,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/miniprogram-ci": {
|
"node_modules/miniprogram-ci": {
|
||||||
"version": "2.1.31",
|
"version": "2.1.42",
|
||||||
"resolved": "https://registry.npmmirror.com/miniprogram-ci/-/miniprogram-ci-2.1.31.tgz",
|
"resolved": "https://registry.npmmirror.com/miniprogram-ci/-/miniprogram-ci-2.1.42.tgz",
|
||||||
"integrity": "sha512-SREx6UnJC74aQ2a1YMNShqQOB97nHO+ll6ZQrCQp98NHXcRq848VjZoD5ELpd95z+8uTASQUAcFtl/HrXuM7Nw==",
|
"integrity": "sha512-H/bq0Wo6kMbwzcOM4jiYSMuQUkND50+zeyY8EkJuY1TQYijkLbFmNK/CFsFJsJgVcvXZ94wn8Do50YSO/Rv6/w==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
@@ -15589,7 +15625,7 @@
|
|||||||
"@babel/template": "7.20.7",
|
"@babel/template": "7.20.7",
|
||||||
"@babel/traverse": "7.21.4",
|
"@babel/traverse": "7.21.4",
|
||||||
"@babel/types": "7.24.6",
|
"@babel/types": "7.24.6",
|
||||||
"@swc/core": "1.4.14",
|
"@swc/core": "1.15.33",
|
||||||
"@vue/reactivity": "3.0.5",
|
"@vue/reactivity": "3.0.5",
|
||||||
"acorn": "^6.1.1",
|
"acorn": "^6.1.1",
|
||||||
"adm-zip": "0.5.10",
|
"adm-zip": "0.5.10",
|
||||||
@@ -15642,7 +15678,7 @@
|
|||||||
"string-hash-64": "1.0.3",
|
"string-hash-64": "1.0.3",
|
||||||
"sync-message": "0.0.12",
|
"sync-message": "0.0.12",
|
||||||
"terminal-kit": "^2.4.0",
|
"terminal-kit": "^2.4.0",
|
||||||
"terser": "4.8.0",
|
"terser": "5.27.1",
|
||||||
"tmp": "0.0.28",
|
"tmp": "0.0.28",
|
||||||
"tslib": "^2.4.0",
|
"tslib": "^2.4.0",
|
||||||
"uglify-js": "3.0.27",
|
"uglify-js": "3.0.27",
|
||||||
@@ -18152,21 +18188,35 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/miniprogram-ci/node_modules/terser": {
|
"node_modules/miniprogram-ci/node_modules/terser": {
|
||||||
"version": "4.8.0",
|
"version": "5.27.1",
|
||||||
"resolved": "https://registry.npmmirror.com/terser/-/terser-4.8.0.tgz",
|
"resolved": "https://registry.npmmirror.com/terser/-/terser-5.27.1.tgz",
|
||||||
"integrity": "sha512-EAPipTNeWsb/3wLPeup1tVPaXfIaU68xMnVdPafIL1TV05OhASArYyIfFvnvJCNrR2NIOvDVNNTFRa+Re2MWyw==",
|
"integrity": "sha512-29wAr6UU/oQpnTw5HoadwjUZnFQXGdOfj0LjZ4sVxzqwHh/QVkvr7m8y9WoR4iN3FRitVduTc6KdjcW38Npsug==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "BSD-2-Clause",
|
"license": "BSD-2-Clause",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@jridgewell/source-map": "^0.3.3",
|
||||||
|
"acorn": "^8.8.2",
|
||||||
"commander": "^2.20.0",
|
"commander": "^2.20.0",
|
||||||
"source-map": "~0.6.1",
|
"source-map-support": "~0.5.20"
|
||||||
"source-map-support": "~0.5.12"
|
|
||||||
},
|
},
|
||||||
"bin": {
|
"bin": {
|
||||||
"terser": "bin/terser"
|
"terser": "bin/terser"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=6.0.0"
|
"node": ">=10"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/miniprogram-ci/node_modules/terser/node_modules/acorn": {
|
||||||
|
"version": "8.17.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/acorn/-/acorn-8.17.0.tgz",
|
||||||
|
"integrity": "sha512-xRQbDb9BnwDafYNn6Vwl839DYVjqXYb1XVGtWAZ1kcDc6iwAL4hg3B1dZlRiuENFeO2H53gFG3in621AdERVAg==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"bin": {
|
||||||
|
"acorn": "bin/acorn"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.4.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/miniprogram-ci/node_modules/terser/node_modules/commander": {
|
"node_modules/miniprogram-ci/node_modules/terser/node_modules/commander": {
|
||||||
|
|||||||
@@ -3,9 +3,9 @@
|
|||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev:mp-weixin": "uni -p mp-weixin",
|
"dev:mp-weixin": "uni -p mp-weixin",
|
||||||
"build:mp-weixin": "uni build -p mp-weixin",
|
"build:mp-weixin": "uni build -p mp-weixin && cp -n static/avatar-*.png dist/build/mp-weixin/static/ 2>/dev/null; true",
|
||||||
"dev:h5": "uni",
|
"dev:h5": "uni",
|
||||||
"build:h5": "uni build",
|
"build:h5": "uni build && cp -n static/avatar-*.png dist/build/h5/static/ 2>/dev/null; true",
|
||||||
"test": "vitest run",
|
"test": "vitest run",
|
||||||
"test:watch": "vitest"
|
"test:watch": "vitest"
|
||||||
},
|
},
|
||||||
@@ -27,7 +27,7 @@
|
|||||||
"@dcloudio/vite-plugin-uni": "3.0.0-4060620250520001",
|
"@dcloudio/vite-plugin-uni": "3.0.0-4060620250520001",
|
||||||
"@vue/test-utils": "^2.4.11",
|
"@vue/test-utils": "^2.4.11",
|
||||||
"jsdom": "^29.1.1",
|
"jsdom": "^29.1.1",
|
||||||
"miniprogram-ci": "^2.1.31",
|
"miniprogram-ci": "^2.1.42",
|
||||||
"sass": "^1.70.0",
|
"sass": "^1.70.0",
|
||||||
"typescript": "^5.3.0",
|
"typescript": "^5.3.0",
|
||||||
"vite": "^5.2.0",
|
"vite": "^5.2.0",
|
||||||
|
|||||||
@@ -1,21 +1,23 @@
|
|||||||
<template>
|
<template>
|
||||||
<view class="digital-human">
|
<view class="digital-human">
|
||||||
<view class="avatar-stage">
|
<view class="avatar-stage">
|
||||||
<view class="avatar-ring" :class="{ speaking: isSpeaking }">
|
<view class="avatar-body" :class="{ speaking: isSpeaking }">
|
||||||
<!-- Default CSS avatar if image fails -->
|
<image class="avatar-img" :src="avatarSrc" mode="aspectFill" @error="avatarError = true" />
|
||||||
<view class="avatar-default" v-if="imgFailed">
|
<view class="mouth-overlay" :style="mouthStyle"></view>
|
||||||
<text class="avatar-initials">AI</text>
|
|
||||||
</view>
|
|
||||||
<image class="avatar-img" :src="avatarSrc" mode="aspectFill" @error="imgFailed = true" v-else />
|
|
||||||
<canvas
|
|
||||||
v-if="isH5"
|
|
||||||
id="dh-mouth"
|
|
||||||
class="mouth-canvas"
|
|
||||||
></canvas>
|
|
||||||
</view>
|
</view>
|
||||||
<text class="role-label">AI 面试官</text>
|
<image src="/static/avatar-default.png" style="display:none" />
|
||||||
|
<image src="/static/avatar-software.png" style="display:none" />
|
||||||
|
<image src="/static/avatar-frontend.png" style="display:none" />
|
||||||
|
<image src="/static/avatar-backend.png" style="display:none" />
|
||||||
|
<image src="/static/avatar-algo.png" style="display:none" />
|
||||||
|
<image src="/static/avatar-pm.png" style="display:none" />
|
||||||
|
<image src="/static/avatar-data.png" style="display:none" />
|
||||||
|
<image src="/static/avatar-marketing.png" style="display:none" />
|
||||||
|
<image src="/static/avatar-ops.png" style="display:none" />
|
||||||
|
<image src="/static/avatar-ui.png" style="display:none" />
|
||||||
|
<view class="status-dot" :class="{ active: isSpeaking }"></view>
|
||||||
|
<text class="role-label">{{ positionLabel }}</text>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
<view class="speech-area" v-if="currentText">
|
<view class="speech-area" v-if="currentText">
|
||||||
<view class="speech-bubble">
|
<view class="speech-bubble">
|
||||||
<text class="speech-text">{{ currentText }}</text>
|
<text class="speech-text">{{ currentText }}</text>
|
||||||
@@ -25,30 +27,61 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, computed, watch, onMounted, onBeforeUnmount, nextTick } from 'vue'
|
import { ref, watch, computed, onBeforeUnmount, nextTick } from 'vue'
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
text: { type: String, default: '' },
|
text: { type: String, default: '' },
|
||||||
audioUrl: { type: String, default: '' },
|
audioUrl: { type: String, default: '' },
|
||||||
|
amplitudeData: { type: Array, default: () => [] },
|
||||||
avatarUrl: { type: String, default: '' },
|
avatarUrl: { type: String, default: '' },
|
||||||
|
position: { type: String, default: '' },
|
||||||
autoPlay: { type: Boolean, default: true },
|
autoPlay: { type: Boolean, default: true },
|
||||||
})
|
})
|
||||||
|
|
||||||
const emit = defineEmits(['speaking-start', 'speaking-end'])
|
const emit = defineEmits(['speaking-start', 'speaking-end'])
|
||||||
|
|
||||||
const isH5 = ref(false)
|
|
||||||
const isSpeaking = ref(false)
|
const isSpeaking = ref(false)
|
||||||
const currentText = ref('')
|
const currentText = ref('')
|
||||||
const imgFailed = ref(false)
|
const mouthOpenness = ref(0)
|
||||||
let audioEl = null
|
const avatarError = ref(false)
|
||||||
let audioCtx = null
|
|
||||||
let analyser = null
|
|
||||||
let animFrameId = null
|
|
||||||
let mouthScale = 0
|
|
||||||
|
|
||||||
onMounted(() => {
|
let audioEl = null
|
||||||
isH5.value = typeof window !== 'undefined' && typeof document !== 'undefined'
|
let animFrame = null
|
||||||
initCanvas()
|
|
||||||
|
const allAvatars = [
|
||||||
|
'/static/avatar-default.png',
|
||||||
|
'/static/avatar-software.png',
|
||||||
|
'/static/avatar-frontend.png',
|
||||||
|
'/static/avatar-backend.png',
|
||||||
|
'/static/avatar-algo.png',
|
||||||
|
'/static/avatar-pm.png',
|
||||||
|
'/static/avatar-data.png',
|
||||||
|
'/static/avatar-marketing.png',
|
||||||
|
'/static/avatar-ops.png',
|
||||||
|
'/static/avatar-ui.png',
|
||||||
|
]
|
||||||
|
|
||||||
|
const avatarMap = {
|
||||||
|
'软件开发': '/static/avatar-software.png',
|
||||||
|
'前端开发': '/static/avatar-frontend.png',
|
||||||
|
'后端开发': '/static/avatar-backend.png',
|
||||||
|
'算法工程师': '/static/avatar-algo.png',
|
||||||
|
'产品经理': '/static/avatar-pm.png',
|
||||||
|
'数据分析': '/static/avatar-data.png',
|
||||||
|
'市场营销': '/static/avatar-marketing.png',
|
||||||
|
'运营': '/static/avatar-ops.png',
|
||||||
|
'UI设计': '/static/avatar-ui.png',
|
||||||
|
}
|
||||||
|
|
||||||
|
const avatarSrc = computed(() => {
|
||||||
|
if (avatarError.value) return '/static/avatar-default.png'
|
||||||
|
if (props.avatarUrl) return props.avatarUrl
|
||||||
|
const key = Object.keys(avatarMap).find(k => props.position.includes(k))
|
||||||
|
return key ? avatarMap[key] : '/static/avatar-default.png'
|
||||||
|
})
|
||||||
|
|
||||||
|
const positionLabel = computed(() => {
|
||||||
|
return props.position ? `${props.position} 面试官` : 'AI 面试官'
|
||||||
})
|
})
|
||||||
|
|
||||||
onBeforeUnmount(() => {
|
onBeforeUnmount(() => {
|
||||||
@@ -65,129 +98,82 @@ watch(() => props.text, (txt) => {
|
|||||||
currentText.value = txt
|
currentText.value = txt
|
||||||
})
|
})
|
||||||
|
|
||||||
const avatarSrc = computed(() => {
|
const mouthStyle = computed(() => {
|
||||||
return props.avatarUrl || '/static/default-avatar.png'
|
const o = mouthOpenness.value
|
||||||
|
const h = 3 + o * o * 22
|
||||||
|
const w = 18 + o * 8
|
||||||
|
return {
|
||||||
|
height: h + 'rpx',
|
||||||
|
width: w + 'rpx',
|
||||||
|
borderRadius: o > 0.3 ? '50%' : '4rpx',
|
||||||
|
background: o > 0.5 ? 'linear-gradient(180deg, #C97B84, #A85562)' : '#C97B84',
|
||||||
|
opacity: o > 0.01 ? 1 : 0.3,
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
function initCanvas() {
|
function getAmplitude(positionMs) {
|
||||||
if (!isH5.value) return
|
const amp = props.amplitudeData
|
||||||
const canvas = document.getElementById('dh-mouth')
|
if (!amp || amp.length === 0) return -1
|
||||||
if (!canvas) return
|
const idx = Math.floor(positionMs / 50)
|
||||||
// Size the canvas to match mouth area (~40% width, ~15% height, centered bottom)
|
if (idx >= amp.length) return -1
|
||||||
const parent = canvas.parentElement
|
return amp[idx]
|
||||||
if (!parent) return
|
|
||||||
const rect = parent.getBoundingClientRect()
|
|
||||||
canvas.width = rect.width * 0.4
|
|
||||||
canvas.height = rect.height * 0.15
|
|
||||||
canvas.style.width = canvas.width + 'px'
|
|
||||||
canvas.style.height = canvas.height + 'px'
|
|
||||||
canvas.style.position = 'absolute'
|
|
||||||
canvas.style.bottom = '18%'
|
|
||||||
canvas.style.left = '30%'
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function drawMouth(openRatio) {
|
function tickMouth() {
|
||||||
if (!isH5.value) return
|
if (!audioEl) return
|
||||||
const canvas = document.getElementById('dh-mouth')
|
const currentTime = audioEl.currentTime * 1000 || 0
|
||||||
if (!canvas) return
|
const amp = getAmplitude(currentTime)
|
||||||
const ctx = canvas.getContext('2d')
|
if (amp >= 0) {
|
||||||
if (!ctx) return
|
const openness = Math.pow(Math.min(1, amp * 2.5), 0.7)
|
||||||
|
mouthOpenness.value = openness
|
||||||
const w = canvas.width
|
} else {
|
||||||
const h = canvas.height
|
const t = Date.now() / 1000
|
||||||
ctx.clearRect(0, 0, w, h)
|
const idle = (Math.sin(t * 4) + 1) / 2 * 0.15
|
||||||
|
mouthOpenness.value = Math.max(0.03, idle)
|
||||||
const mouthH = Math.max(2, h * openRatio)
|
|
||||||
const mouthW = w * 0.8
|
|
||||||
|
|
||||||
ctx.fillStyle = '#C97B84'
|
|
||||||
ctx.beginPath()
|
|
||||||
ctx.ellipse(w / 2, h / 2 + (h - mouthH) / 2, mouthW / 2, mouthH / 2, 0, 0, Math.PI * 2)
|
|
||||||
ctx.fill()
|
|
||||||
|
|
||||||
if (openRatio > 0.1) {
|
|
||||||
ctx.fillStyle = '#2D1B1E'
|
|
||||||
ctx.beginPath()
|
|
||||||
ctx.ellipse(w / 2, h / 2 + (h - mouthH) / 2 + 1, mouthW / 4, mouthH / 4, 0, 0, Math.PI * 2)
|
|
||||||
ctx.fill()
|
|
||||||
}
|
}
|
||||||
|
animFrame = setTimeout(tickMouth, 50)
|
||||||
}
|
}
|
||||||
|
|
||||||
async function playAudio(url) {
|
function playAudio(url) {
|
||||||
stopAudio()
|
stopAudio()
|
||||||
isSpeaking.value = true
|
isSpeaking.value = true
|
||||||
emit('speaking-start')
|
emit('speaking-start')
|
||||||
|
tickMouth()
|
||||||
|
|
||||||
try {
|
try {
|
||||||
audioEl = new Audio(url)
|
const innerAudio = uni.createInnerAudioContext()
|
||||||
|
audioEl = innerAudio
|
||||||
if (isH5.value) {
|
innerAudio.src = url
|
||||||
audioCtx = new (window.AudioContext || window.webkitAudioContext)()
|
innerAudio.autoplay = true
|
||||||
const source = audioCtx.createMediaElementSource(audioEl)
|
innerAudio.onEnded(() => finishSpeaking())
|
||||||
analyser = audioCtx.createAnalyser()
|
innerAudio.onError(() => finishSpeaking())
|
||||||
analyser.fftSize = 256
|
innerAudio.onStop(() => finishSpeaking())
|
||||||
source.connect(analyser)
|
|
||||||
analyser.connect(audioCtx.destination)
|
|
||||||
}
|
|
||||||
|
|
||||||
audioEl.onended = () => {
|
|
||||||
finishSpeaking()
|
|
||||||
}
|
|
||||||
|
|
||||||
audioEl.onerror = () => {
|
|
||||||
finishSpeaking()
|
|
||||||
}
|
|
||||||
|
|
||||||
await audioEl.play()
|
|
||||||
|
|
||||||
if (analyser) {
|
|
||||||
animateMouth()
|
|
||||||
}
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
finishSpeaking()
|
finishSpeaking()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function animateMouth() {
|
|
||||||
if (!analyser) return
|
|
||||||
const dataArray = new Uint8Array(analyser.frequencyBinCount)
|
|
||||||
|
|
||||||
function tick() {
|
|
||||||
if (!isSpeaking.value) return
|
|
||||||
analyser.getByteFrequencyData(dataArray)
|
|
||||||
const sum = dataArray.reduce((a, b) => a + b, 0)
|
|
||||||
const avg = sum / dataArray.length
|
|
||||||
mouthScale = Math.min(1, avg / 128)
|
|
||||||
// Smooth
|
|
||||||
mouthScale = Math.max(0.05, mouthScale)
|
|
||||||
drawMouth(mouthScale)
|
|
||||||
animFrameId = requestAnimationFrame(tick)
|
|
||||||
}
|
|
||||||
|
|
||||||
tick()
|
|
||||||
}
|
|
||||||
|
|
||||||
function finishSpeaking() {
|
function finishSpeaking() {
|
||||||
isSpeaking.value = false
|
isSpeaking.value = false
|
||||||
emit('speaking-end')
|
mouthOpenness.value = 0
|
||||||
if (analyser) {
|
cleanupTimers()
|
||||||
mouthScale = 0
|
|
||||||
drawMouth(0)
|
|
||||||
}
|
|
||||||
cleanupAudio()
|
cleanupAudio()
|
||||||
|
emit('speaking-end')
|
||||||
|
}
|
||||||
|
|
||||||
|
function cleanupTimers() {
|
||||||
|
if (animFrame) { clearTimeout(animFrame); animFrame = null }
|
||||||
}
|
}
|
||||||
|
|
||||||
function cleanupAudio() {
|
function cleanupAudio() {
|
||||||
if (animFrameId) { cancelAnimationFrame(animFrameId); animFrameId = null }
|
if (audioEl) {
|
||||||
if (audioCtx) { audioCtx.close().catch(() => {}); audioCtx = null }
|
try { audioEl.stop(); audioEl.destroy() } catch {}
|
||||||
analyser = null
|
audioEl = null
|
||||||
audioEl = null
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function stopAudio() {
|
function stopAudio() {
|
||||||
if (audioEl) {
|
if (audioEl) { try { audioEl.stop(); audioEl.destroy() } catch {} }
|
||||||
try { audioEl.pause(); audioEl.src = '' } catch {}
|
|
||||||
}
|
|
||||||
finishSpeaking()
|
finishSpeaking()
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -199,56 +185,63 @@ defineExpose({ play: playAudio, stop: stopAudio })
|
|||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding: 20rpx 0;
|
padding: 10rpx 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Avatar stage */
|
|
||||||
.avatar-stage {
|
.avatar-stage {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 12rpx;
|
gap: 10rpx;
|
||||||
|
position: relative;
|
||||||
}
|
}
|
||||||
|
.avatar-body {
|
||||||
.avatar-ring {
|
|
||||||
width: 200rpx;
|
width: 200rpx;
|
||||||
height: 200rpx;
|
height: 260rpx;
|
||||||
border-radius: 50%;
|
border-radius: 20rpx;
|
||||||
border: 4rpx solid #E5E7EB;
|
border: 4rpx solid #E5E7EB;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
position: relative;
|
position: relative;
|
||||||
transition: border-color 0.3s, box-shadow 0.3s;
|
transition: border-color 0.3s, box-shadow 0.3s;
|
||||||
|
animation: breathing 4s ease-in-out infinite;
|
||||||
}
|
}
|
||||||
|
.avatar-body.speaking {
|
||||||
.avatar-ring.speaking {
|
|
||||||
border-color: #6366F1;
|
border-color: #6366F1;
|
||||||
box-shadow: 0 0 30rpx rgba(99, 102, 241, 0.3);
|
box-shadow: 0 0 40rpx rgba(99, 102, 241, 0.4);
|
||||||
|
animation: speakPulse 1.5s ease-in-out infinite;
|
||||||
|
}
|
||||||
|
@keyframes speakPulse {
|
||||||
|
0%, 100% { box-shadow: 0 0 20rpx rgba(99, 102, 241, 0.3); }
|
||||||
|
50% { box-shadow: 0 0 50rpx rgba(99, 102, 241, 0.6); }
|
||||||
|
}
|
||||||
|
@keyframes breathing {
|
||||||
|
0%, 100% { transform: scale(1); }
|
||||||
|
50% { transform: scale(1.012); }
|
||||||
}
|
}
|
||||||
|
|
||||||
.avatar-img {
|
.avatar-img {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
.mouth-overlay {
|
||||||
.avatar-default {
|
position: absolute;
|
||||||
width: 100%;
|
top: 36.8%;
|
||||||
height: 100%;
|
left: 50%;
|
||||||
background: linear-gradient(135deg, #6366F1, #8B5CF6);
|
transform: translateX(-50%);
|
||||||
display: flex;
|
transition: all 0.08s cubic-bezier(0.25, 0.1, 0.25, 1);
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
}
|
}
|
||||||
|
.status-dot {
|
||||||
.avatar-initials {
|
width: 14rpx;
|
||||||
font-size: 48rpx;
|
height: 14rpx;
|
||||||
font-weight: 700;
|
border-radius: 50%;
|
||||||
color: #FFFFFF;
|
background: #9CA3AF;
|
||||||
|
position: absolute;
|
||||||
|
top: 8rpx;
|
||||||
|
right: 8rpx;
|
||||||
|
transition: background 0.3s;
|
||||||
}
|
}
|
||||||
|
.status-dot.active {
|
||||||
.mouth-canvas {
|
background: #10B981;
|
||||||
pointer-events: none;
|
box-shadow: 0 0 12rpx rgba(16, 185, 129, 0.6);
|
||||||
}
|
}
|
||||||
|
|
||||||
.role-label {
|
.role-label {
|
||||||
font-size: 22rpx;
|
font-size: 22rpx;
|
||||||
color: #6B7280;
|
color: #6B7280;
|
||||||
@@ -256,23 +249,19 @@ defineExpose({ play: playAudio, stop: stopAudio })
|
|||||||
padding: 4rpx 20rpx;
|
padding: 4rpx 20rpx;
|
||||||
border-radius: 20rpx;
|
border-radius: 20rpx;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Speech bubble */
|
|
||||||
.speech-area {
|
.speech-area {
|
||||||
margin-top: 24rpx;
|
margin-top: 12rpx;
|
||||||
padding: 0 40rpx;
|
padding: 0 40rpx;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
}
|
}
|
||||||
|
|
||||||
.speech-bubble {
|
.speech-bubble {
|
||||||
background: #FFFFFF;
|
background: #FFFFFF;
|
||||||
border-radius: 16rpx;
|
border-radius: 16rpx;
|
||||||
padding: 24rpx 28rpx;
|
padding: 20rpx 24rpx;
|
||||||
box-shadow: 0 4rpx 20rpx rgba(0, 0, 0, 0.06);
|
box-shadow: 0 4rpx 20rpx rgba(0, 0, 0, 0.06);
|
||||||
position: relative;
|
position: relative;
|
||||||
}
|
}
|
||||||
|
|
||||||
.speech-bubble::before {
|
.speech-bubble::before {
|
||||||
content: '';
|
content: '';
|
||||||
position: absolute;
|
position: absolute;
|
||||||
@@ -282,10 +271,9 @@ defineExpose({ play: playAudio, stop: stopAudio })
|
|||||||
border-bottom-color: #FFFFFF;
|
border-bottom-color: #FFFFFF;
|
||||||
border-top: none;
|
border-top: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.speech-text {
|
.speech-text {
|
||||||
font-size: 28rpx;
|
font-size: 26rpx;
|
||||||
line-height: 1.7;
|
line-height: 1.6;
|
||||||
color: #1F2937;
|
color: #1F2937;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -98,11 +98,18 @@ export const API_ENDPOINTS = {
|
|||||||
CHECK: (outTradeNo: string) => `/payment/check/${outTradeNo}`,
|
CHECK: (outTradeNo: string) => `/payment/check/${outTradeNo}`,
|
||||||
ACTIVATE: '/payment/activate',
|
ACTIVATE: '/payment/activate',
|
||||||
},
|
},
|
||||||
TTS: {
|
TTS: {
|
||||||
SYNTHESIZE: '/tts/synthesize',
|
SYNTHESIZE: '/tts/synthesize',
|
||||||
AUDIO: (hash: string) => `/tts/audio/${hash}`,
|
AUDIO: (hash: string) => `/tts/audio/${hash}`,
|
||||||
},
|
ASR: '/tts/asr',
|
||||||
} as const
|
},
|
||||||
|
SHARE: {
|
||||||
|
CREATE: '/share/create',
|
||||||
|
STATS: '/share/stats',
|
||||||
|
RECORDS: '/share/records',
|
||||||
|
VISITORS: '/share/visitors',
|
||||||
|
},
|
||||||
|
} as const
|
||||||
|
|
||||||
const PROD_API_HOST = import.meta.env.VITE_PROD_API_HOST || 'https://zhiyinwx.yzrcloud.cn'
|
const PROD_API_HOST = import.meta.env.VITE_PROD_API_HOST || 'https://zhiyinwx.yzrcloud.cn'
|
||||||
const DEV_API_HOST = 'http://localhost:3006'
|
const DEV_API_HOST = 'http://localhost:3006'
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
{
|
{
|
||||||
"name": "宇之然AI磁场",
|
"name": "宇之然AI磁场",
|
||||||
"appid": "__UNI__DEV__",
|
"appid": "__UNI__DEV__",
|
||||||
"versionName": "1.0.5",
|
"versionName": "1.0.11",
|
||||||
"versionCode": "105",
|
"versionCode": "111",
|
||||||
"description": "职引 - 宇之然AI磁场旗下AI模拟面试平台,提供AI面试官模拟练习、简历智能优化、大厂面经题库,助你轻松应对校招面试。",
|
"description": "职引 - 宇之然AI磁场旗下AI模拟面试平台,提供AI面试官模拟练习、简历智能优化、大厂面经题库,助你轻松应对校招面试。",
|
||||||
"h5": {
|
"h5": {
|
||||||
"title": "职引 - AI模拟面试 | 宇之然AI磁场",
|
"title": "职引 - AI模拟面试 | 宇之然AI磁场",
|
||||||
|
|||||||
@@ -16,7 +16,8 @@
|
|||||||
{ "path": "pages/admin/admin", "style": { "navigationBarTitleText": "管理后台" } },
|
{ "path": "pages/admin/admin", "style": { "navigationBarTitleText": "管理后台" } },
|
||||||
{ "path": "pages/result/result", "style": { "navigationBarTitleText": "优化结果" } },
|
{ "path": "pages/result/result", "style": { "navigationBarTitleText": "优化结果" } },
|
||||||
{ "path": "pages/agreement/agreement", "style": { "navigationBarTitleText": "用户协议" } },
|
{ "path": "pages/agreement/agreement", "style": { "navigationBarTitleText": "用户协议" } },
|
||||||
{ "path": "pages/privacy/privacy", "style": { "navigationBarTitleText": "隐私政策" } }
|
{ "path": "pages/privacy/privacy", "style": { "navigationBarTitleText": "隐私政策" } },
|
||||||
|
{ "path": "pages/share/share", "style": { "navigationBarTitleText": "我的分享" } }
|
||||||
],
|
],
|
||||||
"tabBar": {
|
"tabBar": {
|
||||||
"color": "#999999",
|
"color": "#999999",
|
||||||
|
|||||||
@@ -17,8 +17,10 @@
|
|||||||
<text class="tab" :class="{ active: tab === 'overview' }" @click="switchTab('overview')">概览</text>
|
<text class="tab" :class="{ active: tab === 'overview' }" @click="switchTab('overview')">概览</text>
|
||||||
<text class="tab" :class="{ active: tab === 'users' }" @click="switchTab('users')">用户</text>
|
<text class="tab" :class="{ active: tab === 'users' }" @click="switchTab('users')">用户</text>
|
||||||
<text class="tab" :class="{ active: tab === 'interviews' }" @click="switchTab('interviews')">面试</text>
|
<text class="tab" :class="{ active: tab === 'interviews' }" @click="switchTab('interviews')">面试</text>
|
||||||
|
<text class="tab" :class="{ active: tab === 'resumes' }" @click="switchTab('resumes')">简历</text>
|
||||||
<text class="tab" :class="{ active: tab === 'orders' }" @click="switchTab('orders')">订单</text>
|
<text class="tab" :class="{ active: tab === 'orders' }" @click="switchTab('orders')">订单</text>
|
||||||
<text class="tab" :class="{ active: tab === 'pricing' }" @click="switchTab('pricing')">定价</text>
|
<text class="tab" :class="{ active: tab === 'pricing' }" @click="switchTab('pricing')">定价</text>
|
||||||
|
<text class="tab" :class="{ active: tab === 'share' }" @click="switchTab('share')">分享</text>
|
||||||
<text class="tab" :class="{ active: tab === 'admins' }" @click="switchTab('admins')">管理</text>
|
<text class="tab" :class="{ active: tab === 'admins' }" @click="switchTab('admins')">管理</text>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
@@ -35,6 +37,17 @@
|
|||||||
<text class="stat-label">总面试</text>
|
<text class="stat-label">总面试</text>
|
||||||
<text class="stat-sub">今日 +{{ overview.todayInterviews }}</text>
|
<text class="stat-sub">今日 +{{ overview.todayInterviews }}</text>
|
||||||
</view>
|
</view>
|
||||||
|
<view class="stat-card">
|
||||||
|
<text class="stat-num">{{ overview.resumeCount ?? 0 }}</text>
|
||||||
|
<text class="stat-label">总简历</text>
|
||||||
|
<text class="stat-sub">付费下载 {{ overview.paidDownloadCount ?? 0 }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view class="plan-cards">
|
||||||
|
<view class="plan-card" v-for="(cnt, plan) in overview.planBreakdown" :key="plan">
|
||||||
|
<text class="plan-num">{{ cnt }}</text>
|
||||||
|
<text class="plan-label">{{ planNameMap[plan] || plan }}</text>
|
||||||
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
@@ -46,12 +59,22 @@
|
|||||||
</view>
|
</view>
|
||||||
<view class="user-list" v-if="!usersLoading">
|
<view class="user-list" v-if="!usersLoading">
|
||||||
<view class="user-row" v-for="u in users" :key="u._id">
|
<view class="user-row" v-for="u in users" :key="u._id">
|
||||||
|
<view class="user-main">
|
||||||
<text class="user-phone">{{ u.phone || '--' }}</text>
|
<text class="user-phone">{{ u.phone || '--' }}</text>
|
||||||
<text class="user-name">{{ u.nickname || '--' }}</text>
|
<text class="user-name">{{ u.nickname || '--' }}</text>
|
||||||
<text class="user-plan" :class="{ vip: u.plan === 'vip' }">{{ u.plan === 'vip' ? '会员' : '免费' }}</text>
|
|
||||||
<text class="user-remaining">剩{{ u.remaining || 0 }}次</text>
|
|
||||||
<text class="user-vip-btn" v-if="u.plan !== 'vip'" @click="setVip(u._id)">设为会员</text>
|
|
||||||
</view>
|
</view>
|
||||||
|
<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-credit">面试:{{ u.interviewCredits ?? 0 }}</text>
|
||||||
|
<text class="user-credit">优化:{{ u.resumeOptimizeCredits ?? 0 }}</text>
|
||||||
|
<text class="user-credit">下载:{{ u.resumeDownloadCredits ?? 0 }}</text>
|
||||||
|
<text class="user-credit share">分享:{{ u.shareCredits ?? 0 }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="user-actions">
|
||||||
|
<text class="user-action-btn" v-if="u.plan === 'free'" @click="setVip(u._id)">设为会员</text>
|
||||||
|
<text class="user-action-btn credit" @click="openCreditModal(u)">调整额度</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
<text class="load-more" v-if="usersTotal > users.length" @click="loadMoreUsers">加载更多</text>
|
<text class="load-more" v-if="usersTotal > users.length" @click="loadMoreUsers">加载更多</text>
|
||||||
</view>
|
</view>
|
||||||
<text class="loading-text" v-if="usersLoading">加载中...</text>
|
<text class="loading-text" v-if="usersLoading">加载中...</text>
|
||||||
@@ -61,15 +84,41 @@
|
|||||||
<view v-if="tab === 'interviews'" class="section">
|
<view v-if="tab === 'interviews'" class="section">
|
||||||
<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">
|
||||||
<text class="iv-pos">{{ iv.position }}</text>
|
<view class="iv-main">
|
||||||
<text class="iv-user">{{ iv.userId?.phone || iv.userId?.nickname || '--' }}</text>
|
<text class="iv-pos">{{ iv.position }}</text>
|
||||||
<text class="iv-status" :class="{ done: iv.status === 'completed' }">{{ iv.status === 'completed' ? '已完成' : '进行中' }}</text>
|
<text class="iv-user">{{ iv.userId?.phone || iv.userId?.nickname || '--' }}</text>
|
||||||
<text class="iv-questions">{{ iv.questionCount || 0 }}题</text>
|
</view>
|
||||||
|
<view class="iv-meta">
|
||||||
|
<text class="iv-status" :class="{ done: iv.status === 'completed' }">{{ iv.status === 'completed' ? '已完成' : '进行中' }}</text>
|
||||||
|
<text class="iv-tag">{{ iv.questionCount || 0 }}题</text>
|
||||||
|
<text class="iv-tag score">得分 {{ iv.totalScore ?? '-' }}</text>
|
||||||
|
<text class="iv-tag filler" v-if="iv.fillerScore != null && iv.fillerScore > 0">语分析 {{ iv.fillerScore }}/{{ iv.fillerDensity ?? '-' }}</text>
|
||||||
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</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 class="resume-list" v-if="!resumeLoading">
|
||||||
|
<view class="resume-row" v-for="r in resumes" :key="r._id">
|
||||||
|
<view class="resume-main">
|
||||||
|
<text class="resume-title">{{ r.title }}</text>
|
||||||
|
<text class="resume-user">{{ r.userId?.phone || r.userId?.nickname || '--' }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="resume-meta">
|
||||||
|
<text class="resume-tag">v{{ r.version }}</text>
|
||||||
|
<text class="resume-tag" v-if="r.targetPosition">{{ r.targetPosition }}</text>
|
||||||
|
<text class="resume-tag paid" v-if="r.paidDownload">付费下载</text>
|
||||||
|
</view>
|
||||||
|
<text class="resume-time">{{ r.createdAt?.slice(0,10) }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<text class="loading-text" v-if="resumeLoading">加载中...</text>
|
||||||
|
<text class="empty-text" v-if="!resumeLoading && resumes.length === 0">暂无简历</text>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<!-- 订单 -->
|
<!-- 订单 -->
|
||||||
@@ -183,6 +232,60 @@
|
|||||||
<button class="save-btn" @click="savePricing" :disabled="pricingLoading">保存定价配置</button>
|
<button class="save-btn" @click="savePricing" :disabled="pricingLoading">保存定价配置</button>
|
||||||
<text class="loading-text" v-if="pricingLoading">保存中...</text>
|
<text class="loading-text" v-if="pricingLoading">保存中...</text>
|
||||||
</view>
|
</view>
|
||||||
|
<!-- 分享 -->
|
||||||
|
<view v-if="tab === 'share'" class="section">
|
||||||
|
<view class="tabs in-tab">
|
||||||
|
<text class="tab" :class="{ active: shareSubTab === 'records' }" @click="shareSubTab='records';loadShareRecords()">分享记录</text>
|
||||||
|
<text class="tab" :class="{ active: shareSubTab === 'visitors' }" @click="shareSubTab='visitors';loadShareVisitors()">访问记录</text>
|
||||||
|
</view>
|
||||||
|
<view v-if="shareSubTab === 'records'">
|
||||||
|
<view class="share-list" v-if="!shareLoading">
|
||||||
|
<view class="share-row" v-for="r in shareRecords" :key="r.shareCode">
|
||||||
|
<view class="share-main">
|
||||||
|
<text class="share-title">{{ r.title }}</text>
|
||||||
|
<text class="share-meta">{{ r.sharer?.nickname || '--' }} · {{ r.type }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="share-stats">
|
||||||
|
<text>访问 {{ r.visitCount }}</text>
|
||||||
|
<text class="share-credited">有效 {{ r.creditedCount }}</text>
|
||||||
|
</view>
|
||||||
|
<text class="share-time">{{ r.createdAt?.slice(0,16).replace('T',' ') }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<text class="loading-text" v-if="shareLoading">加载中...</text>
|
||||||
|
<text class="empty-text" v-if="!shareLoading && shareRecords.length === 0">暂无分享记录</text>
|
||||||
|
</view>
|
||||||
|
<view v-if="shareSubTab === 'visitors'">
|
||||||
|
<view class="share-list" v-if="!shareLoading">
|
||||||
|
<view class="share-row" v-for="(v, i) in shareVisitors" :key="i">
|
||||||
|
<view class="share-main">
|
||||||
|
<text>分享者: {{ v.sharer?.nickname || '--' }}</text>
|
||||||
|
<text class="share-meta">访客: {{ v.visitor?.nickname || '匿名' }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="share-stats">
|
||||||
|
<text class="badge" :class="v.credited ? 'badge-done' : 'badge-pend'">{{ v.credited ? '已积分' : '未积分' }}</text>
|
||||||
|
</view>
|
||||||
|
<text class="share-time">{{ v.createdAt?.slice(0,16).replace('T',' ') }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<text class="loading-text" v-if="shareLoading">加载中...</text>
|
||||||
|
<text class="empty-text" v-if="!shareLoading && shareVisitors.length === 0">暂无访问记录</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<!-- 额度调整弹窗 -->
|
||||||
|
<view class="modal-mask" v-if="creditModal.show" @click="closeCreditModal">
|
||||||
|
<view class="modal-content" @click.stop>
|
||||||
|
<text class="modal-title">调整 {{ creditModal.user?.nickname || '用户' }} 的额度</text>
|
||||||
|
<view class="cfg-row" v-for="t in creditTypes" :key="t.key">
|
||||||
|
<text>{{ t.label }}</text>
|
||||||
|
<input class="cfg-input" type="digit" v-model.number="t.value" :placeholder="t.key" />
|
||||||
|
</view>
|
||||||
|
<view class="modal-actions">
|
||||||
|
<button class="modal-btn cancel" @click="closeCreditModal">取消</button>
|
||||||
|
<button class="modal-btn confirm" @click="doAdjustCredits">确认调整</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">
|
||||||
@@ -219,20 +322,24 @@ import { api, API_ENDPOINTS } from '../../config'
|
|||||||
const verified = ref(false)
|
const verified = ref(false)
|
||||||
const adminName = ref('')
|
const adminName = ref('')
|
||||||
const tab = ref('overview')
|
const tab = ref('overview')
|
||||||
|
const shareSubTab = ref('records')
|
||||||
const loading = ref(false)
|
const loading = ref(false)
|
||||||
const usersLoading = ref(false)
|
const usersLoading = ref(false)
|
||||||
const ivLoading = ref(false)
|
const ivLoading = ref(false)
|
||||||
const userKeyword = ref('')
|
const userKeyword = ref('')
|
||||||
const usersPage = ref(1)
|
const usersPage = ref(1)
|
||||||
|
|
||||||
const overview = ref({ userCount: 0, interviewCount: 0, todayUsers: 0, todayInterviews: 0 })
|
const overview = ref({ userCount: 0, interviewCount: 0, todayUsers: 0, todayInterviews: 0, resumeCount: 0, paidDownloadCount: 0, planBreakdown: {} })
|
||||||
|
const planNameMap = { free: '免费', growth: '成长', sprint: '冲刺', vip: '会员' }
|
||||||
const users = ref([])
|
const users = ref([])
|
||||||
const usersTotal = ref(0)
|
const usersTotal = ref(0)
|
||||||
const interviews = ref([])
|
const interviews = ref([])
|
||||||
|
const resumes = ref([])
|
||||||
|
const resumeLoading = ref(false)
|
||||||
const adminKeyword = ref('')
|
const adminKeyword = ref('')
|
||||||
const adminList = ref([])
|
const adminList = ref([])
|
||||||
const searchResult = ref(null)
|
const searchResult = ref(null)
|
||||||
const memberConfig = ref({ interview: { maxRoundsFree: 5, maxRoundsVip: 10, dailyFreeLimit: 3 }, diagnosis: { dailyFreeLimit: 2 }, optimize: { dailyFreeLimit: 2 }, price: { monthly: 2900 } })
|
const memberConfig = ref({ interview: { maxRoundsFree: 5, maxRoundsVip: 10, dailyFreeLimit: 3 }, diagnosis: { dailyFreeLimit: 2 }, optimize: { dailyFreeLimit: 2 }, price: { monthly: 1990 } })
|
||||||
const cfgLoading = ref(false)
|
const cfgLoading = ref(false)
|
||||||
const pricing = ref({
|
const pricing = ref({
|
||||||
interview: { pricePerSession: 500 },
|
interview: { pricePerSession: 500 },
|
||||||
@@ -253,7 +360,7 @@ const growthPriceDisplay = computed(() => growthPriceTemp.value.toFixed(1))
|
|||||||
const sprintPriceDisplay = computed(() => sprintPriceTemp.value.toFixed(1))
|
const sprintPriceDisplay = computed(() => sprintPriceTemp.value.toFixed(1))
|
||||||
|
|
||||||
const calcInterviewPrice = () => {
|
const calcInterviewPrice = () => {
|
||||||
// Convert to 分 on save
|
// Handled in savePricing via growthPriceTemp / sprintPriceTemp
|
||||||
}
|
}
|
||||||
const orders = ref([])
|
const orders = ref([])
|
||||||
const ordersTotal = ref(0)
|
const ordersTotal = ref(0)
|
||||||
@@ -261,6 +368,20 @@ const ordersPage = ref(1)
|
|||||||
const orderLoading = ref(false)
|
const orderLoading = ref(false)
|
||||||
const orderFilter = ref('')
|
const orderFilter = ref('')
|
||||||
|
|
||||||
|
// Share state
|
||||||
|
const shareRecords = ref([])
|
||||||
|
const shareVisitors = ref([])
|
||||||
|
const shareLoading = ref(false)
|
||||||
|
|
||||||
|
// Credit modal
|
||||||
|
const creditModal = ref({ show: false, user: null })
|
||||||
|
const creditTypes = ref([
|
||||||
|
{ key: 'interviewCredits', label: '面试次数', value: 0 },
|
||||||
|
{ key: 'resumeOptimizeCredits', label: '优化次数', value: 0 },
|
||||||
|
{ key: 'resumeDownloadCredits', label: '下载次数', value: 0 },
|
||||||
|
{ key: 'shareCredits', label: '分享积分', value: 0 },
|
||||||
|
])
|
||||||
|
|
||||||
const token = () => uni.getStorageSync('token') || ''
|
const token = () => uni.getStorageSync('token') || ''
|
||||||
|
|
||||||
const apiAdmin = (path, opts = {}) => {
|
const apiAdmin = (path, opts = {}) => {
|
||||||
@@ -278,7 +399,7 @@ const doVerify = async () => {
|
|||||||
try {
|
try {
|
||||||
const res = await apiAdmin('/check')
|
const res = await apiAdmin('/check')
|
||||||
if (res.statusCode === 200 && res.data?.isAdmin) {
|
if (res.statusCode === 200 && res.data?.isAdmin) {
|
||||||
adminName.value = '管理员'
|
adminName.value = res.data.nickname || res.data.username || '管理员'
|
||||||
verified.value = true
|
verified.value = true
|
||||||
loadOverview()
|
loadOverview()
|
||||||
} else throw new Error('无管理员权限')
|
} else throw new Error('无管理员权限')
|
||||||
@@ -300,6 +421,7 @@ const switchTab = (t) => {
|
|||||||
tab.value = t
|
tab.value = t
|
||||||
if (t === 'users' && users.value.length === 0) loadUsers()
|
if (t === 'users' && users.value.length === 0) loadUsers()
|
||||||
if (t === 'interviews' && interviews.value.length === 0) loadInterviews()
|
if (t === 'interviews' && interviews.value.length === 0) loadInterviews()
|
||||||
|
if (t === 'resumes' && resumes.value.length === 0) loadResumes()
|
||||||
if (t === 'admins' && adminList.value.length === 0) loadAdmins()
|
if (t === 'admins' && adminList.value.length === 0) loadAdmins()
|
||||||
if (t === 'pricing') loadPricing()
|
if (t === 'pricing') loadPricing()
|
||||||
if (t === 'orders') loadOrders()
|
if (t === 'orders') loadOrders()
|
||||||
@@ -332,6 +454,15 @@ const loadInterviews = async () => {
|
|||||||
finally { ivLoading.value = false }
|
finally { ivLoading.value = false }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const loadResumes = async () => {
|
||||||
|
resumeLoading.value = true
|
||||||
|
try {
|
||||||
|
const res = await apiAdmin('/resumes?page=1&limit=20')
|
||||||
|
if (res.statusCode === 200) resumes.value = res.data.list || []
|
||||||
|
} catch (e) { console.error(e) }
|
||||||
|
finally { resumeLoading.value = false }
|
||||||
|
}
|
||||||
|
|
||||||
const loadPricing = async () => {
|
const loadPricing = async () => {
|
||||||
pricingLoading.value = true
|
pricingLoading.value = true
|
||||||
try {
|
try {
|
||||||
@@ -369,15 +500,6 @@ const savePricing = async () => {
|
|||||||
finally { pricingLoading.value = false }
|
finally { pricingLoading.value = false }
|
||||||
}
|
}
|
||||||
|
|
||||||
const loadConfig = async () => {
|
|
||||||
cfgLoading.value = true
|
|
||||||
try {
|
|
||||||
const res = await apiAdmin('/config')
|
|
||||||
if (res.statusCode === 200) memberConfig.value = res.data
|
|
||||||
} catch(e) { console.error(e) }
|
|
||||||
finally { cfgLoading.value = false }
|
|
||||||
}
|
|
||||||
|
|
||||||
const loadOrders = async () => {
|
const loadOrders = async () => {
|
||||||
orderLoading.value = true
|
orderLoading.value = true
|
||||||
ordersPage.value = 1
|
ordersPage.value = 1
|
||||||
@@ -462,6 +584,54 @@ const setVip = async (targetUserId) => {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const loadShareRecords = async () => {
|
||||||
|
shareLoading.value = true
|
||||||
|
try {
|
||||||
|
const res = await apiAdmin('/share-records?page=1&limit=50')
|
||||||
|
if (res.statusCode === 200) shareRecords.value = res.data.list || []
|
||||||
|
} catch(e) { console.error(e) }
|
||||||
|
finally { shareLoading.value = false }
|
||||||
|
}
|
||||||
|
|
||||||
|
const loadShareVisitors = async () => {
|
||||||
|
shareLoading.value = true
|
||||||
|
try {
|
||||||
|
const res = await apiAdmin('/share-visitors?page=1&limit=50')
|
||||||
|
if (res.statusCode === 200) shareVisitors.value = res.data.list || []
|
||||||
|
} catch(e) { console.error(e) }
|
||||||
|
finally { shareLoading.value = false }
|
||||||
|
}
|
||||||
|
|
||||||
|
const openCreditModal = (user) => {
|
||||||
|
creditTypes.value = [
|
||||||
|
{ 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 }
|
||||||
|
}
|
||||||
|
|
||||||
|
const closeCreditModal = () => {
|
||||||
|
creditModal.value = { show: false, user: null }
|
||||||
|
}
|
||||||
|
|
||||||
|
const doAdjustCredits = async () => {
|
||||||
|
const userId = creditModal.value.user?._id
|
||||||
|
if (!userId) return
|
||||||
|
try {
|
||||||
|
for (const t of creditTypes.value) {
|
||||||
|
await apiAdmin('/user/credits', {
|
||||||
|
method: 'POST',
|
||||||
|
data: { userId, type: t.key, amount: t.value },
|
||||||
|
})
|
||||||
|
}
|
||||||
|
uni.showToast({ title: '调整成功', icon: 'success' })
|
||||||
|
closeCreditModal()
|
||||||
|
loadUsers()
|
||||||
|
} catch { uni.showToast({ title: '调整失败', icon: 'none' }) }
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
@@ -481,22 +651,37 @@ const setVip = async (targetUserId) => {
|
|||||||
.stat-num { font-size: 48rpx; font-weight: 800; color: var(--color-primary); display: block; }
|
.stat-num { font-size: 48rpx; font-weight: 800; color: var(--color-primary); display: block; }
|
||||||
.stat-label { font-size: 22rpx; color: var(--color-text-secondary); margin-top: 8rpx; display: block; }
|
.stat-label { font-size: 22rpx; color: var(--color-text-secondary); margin-top: 8rpx; display: block; }
|
||||||
.stat-sub { font-size: 20rpx; color: var(--color-success); margin-top: 4rpx; display: block; }
|
.stat-sub { font-size: 20rpx; color: var(--color-success); margin-top: 4rpx; display: block; }
|
||||||
|
.plan-cards { display: flex; gap: 12rpx; margin-top: 16rpx; }
|
||||||
|
.plan-card { flex: 1; background: #FFF; border-radius: var(--radius-sm); padding: 20rpx; text-align: center; box-shadow: var(--shadow-sm); }
|
||||||
|
.plan-num { font-size: 36rpx; font-weight: 700; color: var(--color-primary); display: block; }
|
||||||
|
.plan-label { font-size: 20rpx; color: var(--color-text-secondary); margin-top: 4rpx; display: block; }
|
||||||
.search-bar { display: flex; gap: 12rpx; margin-bottom: 16rpx; }
|
.search-bar { display: flex; gap: 12rpx; margin-bottom: 16rpx; }
|
||||||
.search-input { flex: 1; height: 64rpx; background: #FFF; border: 2rpx solid var(--color-border); border-radius: var(--radius-sm); padding: 0 16rpx; font-size: 24rpx; }
|
.search-input { flex: 1; height: 64rpx; background: #FFF; border: 2rpx solid var(--color-border); border-radius: var(--radius-sm); padding: 0 16rpx; font-size: 24rpx; }
|
||||||
.search-btn { height: 64rpx; padding: 0 24rpx; background: var(--color-primary); color: #FFF; border-radius: var(--radius-sm); font-size: 24rpx; border: none; }
|
.search-btn { height: 64rpx; padding: 0 24rpx; background: var(--color-primary); color: #FFF; border-radius: var(--radius-sm); font-size: 24rpx; border: none; }
|
||||||
.user-row { background: #FFF; padding: 20rpx; border-radius: var(--radius-sm); margin-bottom: 8rpx; display: flex; flex-wrap: wrap; gap: 8rpx; }
|
.user-row { background: #FFF; padding: 20rpx; border-radius: var(--radius-sm); margin-bottom: 8rpx; }
|
||||||
|
.user-main { display: flex; gap: 12rpx; margin-bottom: 8rpx; }
|
||||||
.user-phone { font-size: 24rpx; font-weight: 600; color: var(--color-text); }
|
.user-phone { font-size: 24rpx; font-weight: 600; color: var(--color-text); }
|
||||||
.user-name { font-size: 22rpx; color: var(--color-text-secondary); }
|
.user-name { font-size: 22rpx; color: var(--color-text-secondary); }
|
||||||
|
.user-badges { display: flex; flex-wrap: wrap; gap: 6rpx; margin-bottom: 8rpx; }
|
||||||
.user-plan { font-size: 20rpx; background: #EEF2FF; color: var(--color-primary); padding: 2rpx 12rpx; border-radius: var(--radius-round); }
|
.user-plan { font-size: 20rpx; background: #EEF2FF; color: var(--color-primary); padding: 2rpx 12rpx; border-radius: var(--radius-round); }
|
||||||
.user-remaining { font-size: 20rpx; color: var(--color-text-tertiary); }
|
.user-plan.vip { background: #FEF3C7; color: #D97706; }
|
||||||
|
.user-credit { font-size: 18rpx; background: #F3F4F6; color: var(--color-text-tertiary); padding: 2rpx 10rpx; border-radius: var(--radius-round); }
|
||||||
|
.user-credit.share { background: #FFF7ED; color: #D97706; }
|
||||||
|
.user-actions { display: flex; gap: 12rpx; }
|
||||||
|
.user-action-btn { font-size: 20rpx; color: var(--color-primary); padding: 4rpx 16rpx; border: 2rpx solid var(--color-primary); border-radius: var(--radius-round); }
|
||||||
|
.user-action-btn.credit { color: #D97706; border-color: #D97706; }
|
||||||
.loading-text { text-align: center; padding: 40rpx; color: var(--color-text-tertiary); font-size: 24rpx; }
|
.loading-text { text-align: center; padding: 40rpx; color: var(--color-text-tertiary); font-size: 24rpx; }
|
||||||
.load-more { text-align: center; padding: 20rpx; color: var(--color-primary); font-size: 24rpx; display: block; }
|
.load-more { text-align: center; padding: 20rpx; color: var(--color-primary); font-size: 24rpx; display: block; }
|
||||||
.iv-row { background: #FFF; padding: 20rpx; border-radius: var(--radius-sm); margin-bottom: 8rpx; display: flex; flex-wrap: wrap; gap: 8rpx; align-items: center; }
|
.iv-row { background: #FFF; padding: 16rpx; border-radius: var(--radius-sm); margin-bottom: 8rpx; }
|
||||||
|
.iv-main { display: flex; gap: 12rpx; margin-bottom: 6rpx; }
|
||||||
.iv-pos { font-size: 24rpx; font-weight: 600; color: var(--color-text); }
|
.iv-pos { font-size: 24rpx; font-weight: 600; color: var(--color-text); }
|
||||||
.iv-user { font-size: 22rpx; color: var(--color-text-secondary); }
|
.iv-user { font-size: 22rpx; color: var(--color-text-secondary); }
|
||||||
|
.iv-meta { display: flex; flex-wrap: wrap; gap: 6rpx; }
|
||||||
.iv-status { font-size: 20rpx; background: #FFF7ED; color: var(--color-warning); padding: 2rpx 12rpx; border-radius: var(--radius-round); }
|
.iv-status { font-size: 20rpx; background: #FFF7ED; color: var(--color-warning); padding: 2rpx 12rpx; border-radius: var(--radius-round); }
|
||||||
.iv-status.done { background: #ECFDF5; color: var(--color-success); }
|
.iv-status.done { background: #ECFDF5; color: var(--color-success); }
|
||||||
.iv-questions { font-size: 20rpx; color: var(--color-text-tertiary); }
|
.iv-tag { font-size: 18rpx; background: #F3F4F6; color: var(--color-text-tertiary); padding: 2rpx 10rpx; border-radius: var(--radius-round); }
|
||||||
|
.iv-tag.score { background: #EEF2FF; color: var(--color-primary); }
|
||||||
|
.iv-tag.filler { background: #FFF7ED; color: #D97706; }
|
||||||
.section-label { font-size: 24rpx; font-weight: 600; color: var(--color-text); margin-bottom: 12rpx; margin-top: 12rpx; }
|
.section-label { font-size: 24rpx; font-weight: 600; color: var(--color-text); margin-bottom: 12rpx; margin-top: 12rpx; }
|
||||||
.admin-row { background: #FFF; padding: 20rpx; border-radius: var(--radius-sm); margin-bottom: 8rpx; display: flex; flex-wrap: wrap; gap: 8rpx; align-items: center; }
|
.admin-row { background: #FFF; padding: 20rpx; border-radius: var(--radius-sm); margin-bottom: 8rpx; display: flex; flex-wrap: wrap; gap: 8rpx; align-items: center; }
|
||||||
.admin-phone { font-size: 24rpx; font-weight: 600; color: var(--color-text); }
|
.admin-phone { font-size: 24rpx; font-weight: 600; color: var(--color-text); }
|
||||||
@@ -505,6 +690,15 @@ const setVip = async (targetUserId) => {
|
|||||||
.admin-set-btn.done { color: var(--color-success); border-color: var(--color-success); }
|
.admin-set-btn.done { color: var(--color-success); border-color: var(--color-success); }
|
||||||
.admin-badge { font-size: 18rpx; background: var(--color-primary); color: #FFF; padding: 2rpx 10rpx; border-radius: var(--radius-round); }
|
.admin-badge { font-size: 18rpx; background: var(--color-primary); color: #FFF; padding: 2rpx 10rpx; border-radius: var(--radius-round); }
|
||||||
.empty-text { text-align: center; padding: 20rpx; color: var(--color-text-tertiary); font-size: 22rpx; display: block; }
|
.empty-text { text-align: center; padding: 20rpx; color: var(--color-text-tertiary); font-size: 22rpx; display: block; }
|
||||||
|
.resume-list { display: flex; flex-direction: column; gap: 8rpx; }
|
||||||
|
.resume-row { background: #FFF; padding: 16rpx; border-radius: var(--radius-sm); display: flex; align-items: center; gap: 12rpx; }
|
||||||
|
.resume-main { flex: 1; display: flex; flex-direction: column; }
|
||||||
|
.resume-title { font-size: 22rpx; font-weight: 600; color: var(--color-text); }
|
||||||
|
.resume-user { font-size: 20rpx; color: var(--color-text-secondary); margin-top: 2rpx; }
|
||||||
|
.resume-meta { display: flex; gap: 6rpx; }
|
||||||
|
.resume-tag { font-size: 18rpx; background: #F3F4F6; color: var(--color-text-tertiary); padding: 2rpx 10rpx; border-radius: var(--radius-round); }
|
||||||
|
.resume-tag.paid { background: #FEF3C7; color: #D97706; }
|
||||||
|
.resume-time { font-size: 18rpx; color: #D1D5DB; white-space: nowrap; }
|
||||||
.order-list { display: flex; flex-direction: column; gap: 8rpx; }
|
.order-list { display: flex; flex-direction: column; gap: 8rpx; }
|
||||||
.order-row { background: #FFF; border-radius: var(--radius-sm); padding: 16rpx; }
|
.order-row { background: #FFF; border-radius: var(--radius-sm); padding: 16rpx; }
|
||||||
.order-info { display: flex; justify-content: space-between; margin-bottom: 8rpx; }
|
.order-info { display: flex; justify-content: space-between; margin-bottom: 8rpx; }
|
||||||
@@ -527,4 +721,23 @@ const setVip = async (targetUserId) => {
|
|||||||
.cfg-textarea { width: 100%; height: 160rpx; border: 2rpx solid var(--color-border); border-radius: var(--radius-sm); padding: 12rpx; font-size: 22rpx; margin-top: 8rpx; box-sizing: border-box; }
|
.cfg-textarea { width: 100%; height: 160rpx; border: 2rpx solid var(--color-border); border-radius: var(--radius-sm); padding: 12rpx; font-size: 22rpx; margin-top: 8rpx; box-sizing: border-box; }
|
||||||
.save-btn { width: 100%; 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; margin-top: 12rpx; }
|
.save-btn { width: 100%; 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; margin-top: 12rpx; }
|
||||||
.save-btn:disabled { opacity: 0.6; }
|
.save-btn:disabled { opacity: 0.6; }
|
||||||
|
.in-tab { margin-bottom: 16rpx; }
|
||||||
|
.share-list { display: flex; flex-direction: column; gap: 8rpx; }
|
||||||
|
.share-row { background: #FFF; padding: 16rpx; border-radius: var(--radius-sm); display: flex; align-items: center; gap: 12rpx; }
|
||||||
|
.share-main { flex: 1; display: flex; flex-direction: column; }
|
||||||
|
.share-title { font-size: 22rpx; color: var(--color-text); font-weight: 500; }
|
||||||
|
.share-meta { font-size: 18rpx; color: var(--color-text-tertiary); margin-top: 2rpx; }
|
||||||
|
.share-stats { display: flex; flex-direction: column; align-items: flex-end; font-size: 20rpx; color: var(--color-text-tertiary); }
|
||||||
|
.share-credited { color: var(--color-primary); }
|
||||||
|
.share-time { font-size: 18rpx; color: #D1D5DB; white-space: nowrap; }
|
||||||
|
.badge { font-size: 18rpx; padding: 2rpx 10rpx; border-radius: 6rpx; }
|
||||||
|
.badge-done { background: #ECFDF5; color: #059669; }
|
||||||
|
.badge-pend { background: #FEF3C7; color: #D97706; }
|
||||||
|
.modal-mask { position: fixed; inset: 0; background: rgba(0,0,0,0.4); display: flex; align-items: center; justify-content: center; z-index: 999; }
|
||||||
|
.modal-content { background: #FFF; border-radius: var(--radius-lg); padding: 40rpx 32rpx; width: 600rpx; max-width: 90vw; }
|
||||||
|
.modal-title { font-size: 28rpx; font-weight: 600; color: var(--color-text); margin-bottom: 24rpx; }
|
||||||
|
.modal-actions { display: flex; gap: 16rpx; margin-top: 32rpx; }
|
||||||
|
.modal-btn { flex: 1; height: 72rpx; border-radius: var(--radius-sm); font-size: 26rpx; border: none; }
|
||||||
|
.modal-btn.cancel { background: #F3F4F6; color: var(--color-text-secondary); }
|
||||||
|
.modal-btn.confirm { background: var(--color-primary); color: #FFF; }
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -17,7 +17,7 @@
|
|||||||
@tap="selectCompany(c.name)"
|
@tap="selectCompany(c.name)"
|
||||||
>
|
>
|
||||||
<view class="company-name">{{ c.name }}</view>
|
<view class="company-name">{{ c.name }}</view>
|
||||||
<view class="company-count">{{ c.positions }} 个岗位</view>
|
<view class="company-count">{{ c.positionCount > 0 ? c.positionCount + ' 个岗位' : '暂无题库' }}</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
@@ -84,23 +84,12 @@
|
|||||||
<script>
|
<script>
|
||||||
import { api } from '../../config'
|
import { api } from '../../config'
|
||||||
|
|
||||||
const HOT_COMPANIES = [
|
|
||||||
{ name: '腾讯', positions: 5 },
|
|
||||||
{ name: '字节跳动', positions: 4 },
|
|
||||||
{ name: '阿里巴巴', positions: 5 },
|
|
||||||
{ name: '美团', positions: 3 },
|
|
||||||
{ name: '百度', positions: 4 },
|
|
||||||
{ name: '京东', positions: 3 },
|
|
||||||
{ name: '网易', positions: 3 },
|
|
||||||
{ name: '小红书', positions: 2 },
|
|
||||||
]
|
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
keyword: '',
|
keyword: '',
|
||||||
searching: false,
|
searching: false,
|
||||||
hotCompanies: HOT_COMPANIES,
|
hotCompanies: [],
|
||||||
selectedCompany: '',
|
selectedCompany: '',
|
||||||
selectedPosition: '',
|
selectedPosition: '',
|
||||||
positions: [],
|
positions: [],
|
||||||
@@ -109,7 +98,18 @@ export default {
|
|||||||
loadingQuestions: false,
|
loadingQuestions: false,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
onLoad() {
|
||||||
|
this.loadHotCompanies()
|
||||||
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
async loadHotCompanies() {
|
||||||
|
try {
|
||||||
|
const res = await uni.request({ url: api('/contribution/companies/hot'), method: 'GET' })
|
||||||
|
if (res.statusCode === 200) this.hotCompanies = res.data || []
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e)
|
||||||
|
}
|
||||||
|
},
|
||||||
difficultyLabel(d) {
|
difficultyLabel(d) {
|
||||||
const map = { junior: '简单', medium: '中等', senior: '困难' }
|
const map = { junior: '简单', medium: '中等', senior: '困难' }
|
||||||
return map[d] || d || '中等'
|
return map[d] || d || '中等'
|
||||||
|
|||||||
@@ -75,10 +75,12 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, onMounted } from 'vue'
|
import { ref } from 'vue'
|
||||||
|
import { onLoad } from '@dcloudio/uni-app'
|
||||||
import { api } from '../../config'
|
import { api } from '../../config'
|
||||||
|
|
||||||
const props = defineProps({ interviewId: String, position: String })
|
const interviewId = ref('')
|
||||||
|
const urlPosition = ref('')
|
||||||
const form = ref({ company: '', position: '', rounds: '', experience: '', tags: [] })
|
const form = ref({ company: '', position: '', rounds: '', experience: '', tags: [] })
|
||||||
const questionsText = ref('')
|
const questionsText = ref('')
|
||||||
const customTag = ref('')
|
const customTag = ref('')
|
||||||
@@ -89,8 +91,14 @@ const presetTags = ['算法题多', '重视项目经历', '面试官nice', '压
|
|||||||
|
|
||||||
const token = () => uni.getStorageSync('token') || ''
|
const token = () => uni.getStorageSync('token') || ''
|
||||||
|
|
||||||
onMounted(() => {
|
onLoad((options) => {
|
||||||
if (props.position) form.value.position = props.position
|
if (options?.position) {
|
||||||
|
urlPosition.value = decodeURIComponent(options.position)
|
||||||
|
form.value.position = urlPosition.value
|
||||||
|
}
|
||||||
|
if (options?.interviewId) {
|
||||||
|
interviewId.value = options.interviewId
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
const toggleTag = (tag) => {
|
const toggleTag = (tag) => {
|
||||||
@@ -122,7 +130,7 @@ const submit = async () => {
|
|||||||
url: api('/contribution'), method: 'POST',
|
url: api('/contribution'), method: 'POST',
|
||||||
header: { 'Authorization': `Bearer ${token()}`, 'Content-Type': 'application/json' },
|
header: { 'Authorization': `Bearer ${token()}`, 'Content-Type': 'application/json' },
|
||||||
data: {
|
data: {
|
||||||
interviewId: props.interviewId || '',
|
interviewId: interviewId.value || '',
|
||||||
company: form.value.company.trim(),
|
company: form.value.company.trim(),
|
||||||
position: form.value.position.trim(),
|
position: form.value.position.trim(),
|
||||||
rounds: form.value.rounds.trim(),
|
rounds: form.value.rounds.trim(),
|
||||||
|
|||||||
@@ -1,19 +1,32 @@
|
|||||||
<template>
|
<template>
|
||||||
<view class="page fade-in">
|
<view class="page fade-in">
|
||||||
<view class="hero">
|
<view class="hero">
|
||||||
<text class="hero-title">{{ greeting }}</text>
|
<view class="hero-row">
|
||||||
<text class="hero-sub">试试下面的功能,开启你的求职练习</text>
|
<view class="hero-left">
|
||||||
|
<text class="hero-title">{{ greeting }}</text>
|
||||||
<view class="user-card card" v-if="userInfo" @click="goProfile">
|
<text class="hero-sub">试试下面的功能,开启你的求职练习</text>
|
||||||
<image class="avatar" :src="userInfo.avatar || '/static/avatar-default.svg'" mode="aspectFill" />
|
</view>
|
||||||
<view class="user-meta">
|
<view class="hero-right">
|
||||||
<text class="user-name">{{ userInfo.nickname || '同学' }}</text>
|
<view class="user-card card" v-if="userInfo" @click="goProfile">
|
||||||
<view class="user-tags">
|
<image class="avatar" :src="userInfo.avatar || '/static/avatar-default.png'" mode="aspectFill" />
|
||||||
<text class="tag tag-plan">{{ userInfo.plan || '免费版' }}</text>
|
<view class="user-meta">
|
||||||
<text class="tag tag-remaining">剩余 {{ userInfo.remaining || 0 }} 次</text>
|
<text class="user-name">{{ userInfo.nickname || '同学' }}</text>
|
||||||
|
<view class="user-tags">
|
||||||
|
<text class="tag tag-plan">{{ userInfo.plan || '免费版' }}</text>
|
||||||
|
<text class="tag tag-remaining">{{ userInfo.interviewCredits > 0 ? '剩余 ' + userInfo.interviewCredits + ' 次' : '已用完' }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<text class="arrow">›</text>
|
||||||
|
</view>
|
||||||
|
<view class="guest-card card" v-else @click="goLogin">
|
||||||
|
<image class="avatar" src="/static/avatar-default.png" mode="aspectFill" />
|
||||||
|
<view class="user-meta">
|
||||||
|
<text class="user-name">立即登录</text>
|
||||||
|
<text class="guest-hint">登录后体验全部功能</text>
|
||||||
|
</view>
|
||||||
|
<text class="arrow">›</text>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<text class="arrow">›</text>
|
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
@@ -24,13 +37,20 @@
|
|||||||
<view class="fp-left">
|
<view class="fp-left">
|
||||||
<view class="fp-icon fp-interview"><text class="fp-emoji">🎙️</text></view>
|
<view class="fp-icon fp-interview"><text class="fp-emoji">🎙️</text></view>
|
||||||
<view class="fp-body">
|
<view class="fp-body">
|
||||||
<text class="fp-name">模拟面试</text>
|
<text class="fp-name">AI数字人面试</text>
|
||||||
<text class="fp-brief">AI 面试官 · 真实场景 · 即时反馈</text>
|
<text class="fp-brief">数字人考官 · 真实场景 · 语音互动 · 即时反馈</text>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<text class="fp-action">开始</text>
|
<text class="fp-action">开始</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="feature-secondary">
|
<view class="feature-tertiary">
|
||||||
|
<view class="fs-card card" @click="goResume">
|
||||||
|
<view class="fs-top">
|
||||||
|
<view class="fs-icon fs-resume"><text class="fs-emoji">📄</text></view>
|
||||||
|
<text class="fs-name">简历优化</text>
|
||||||
|
</view>
|
||||||
|
<text class="fs-brief">AI 诊断 · 智能优化 · 一键下载</text>
|
||||||
|
</view>
|
||||||
<view class="fs-card card" @click="goProgress">
|
<view class="fs-card card" @click="goProgress">
|
||||||
<view class="fs-top">
|
<view class="fs-top">
|
||||||
<view class="fs-icon fs-progress"><text class="fs-emoji">📊</text></view>
|
<view class="fs-icon fs-progress"><text class="fs-emoji">📊</text></view>
|
||||||
@@ -92,19 +112,18 @@
|
|||||||
<view class="section-header">
|
<view class="section-header">
|
||||||
<view class="section-title-row">
|
<view class="section-title-row">
|
||||||
<text class="section-title">热门岗位</text>
|
<text class="section-title">热门岗位</text>
|
||||||
<text class="section-tag-demo">参考示例</text>
|
|
||||||
</view>
|
</view>
|
||||||
<text class="section-desc">点击直接面试</text>
|
<text class="section-desc">点击直接面试</text>
|
||||||
</view>
|
</view>
|
||||||
<view class="position-list card" v-if="!positionsLoading">
|
<view class="position-list card" v-if="!positionsLoading">
|
||||||
<view class="pos-item" v-for="(pos, idx) in hotPositions" :key="idx" @click="startInterview(pos)">
|
<view class="pos-item" v-for="(pos, idx) in hotPositions" :key="idx" @click="startInterview(pos)">
|
||||||
<view class="pos-left">
|
<view class="pos-left">
|
||||||
<text class="pos-icon">{{ posIcons[idx] || '💼' }}</text>
|
<text class="pos-icon">{{ pos.icon || posIcons[idx % posIcons.length] || '💼' }}</text>
|
||||||
<view class="pos-body">
|
<view class="pos-body">
|
||||||
<text class="pos-name">{{ pos.name }}</text>
|
<text class="pos-name">{{ pos.name }}</text>
|
||||||
<view class="pos-meta-row">
|
<view class="pos-meta-row" v-if="pos.company || pos.salary">
|
||||||
<text class="pos-company">{{ pos.company || '参考公司' }}</text>
|
<text class="pos-company">{{ pos.company }}</text>
|
||||||
<text class="pos-salary">{{ pos.salary || '参考薪资' }}</text>
|
<text class="pos-salary">{{ pos.salary }}</text>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
@@ -122,6 +141,7 @@
|
|||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, onMounted } from 'vue'
|
import { ref, onMounted } from 'vue'
|
||||||
|
import { onShow } from '@dcloudio/uni-app'
|
||||||
import { api } from '../../config'
|
import { api } from '../../config'
|
||||||
|
|
||||||
const userInfo = ref(null)
|
const userInfo = ref(null)
|
||||||
@@ -132,8 +152,12 @@ const positionsLoading = ref(true)
|
|||||||
const dailyQuestion = ref(null)
|
const dailyQuestion = ref(null)
|
||||||
const showAnswer = ref(false)
|
const showAnswer = ref(false)
|
||||||
|
|
||||||
|
const loadUserInfo = () => {
|
||||||
|
try { const s = uni.getStorageSync('userInfo'); if (s) userInfo.value = JSON.parse(s); else userInfo.value = null } catch (e) { userInfo.value = null }
|
||||||
|
}
|
||||||
|
|
||||||
onMounted(async () => {
|
onMounted(async () => {
|
||||||
try { const s = uni.getStorageSync('userInfo'); if (s) userInfo.value = JSON.parse(s) } catch (e) {}
|
loadUserInfo()
|
||||||
const h = new Date().getHours()
|
const h = new Date().getHours()
|
||||||
if (h < 6) greeting.value = '夜深了,早点休息 🌙'
|
if (h < 6) greeting.value = '夜深了,早点休息 🌙'
|
||||||
else if (h < 12) greeting.value = '早上好 ☀️'
|
else if (h < 12) greeting.value = '早上好 ☀️'
|
||||||
@@ -163,10 +187,26 @@ onMounted(async () => {
|
|||||||
finally { positionsLoading.value = false }
|
finally { positionsLoading.value = false }
|
||||||
})
|
})
|
||||||
|
|
||||||
const refreshDaily = () => { showAnswer.value = false; /* trigger reload */ }
|
onShow(loadUserInfo)
|
||||||
|
|
||||||
|
const refreshDaily = async () => {
|
||||||
|
showAnswer.value = false
|
||||||
|
try {
|
||||||
|
const t = uni.getStorageSync('token')
|
||||||
|
if (t) {
|
||||||
|
const qres = await uni.request({
|
||||||
|
url: api('/daily-question'), method: 'GET',
|
||||||
|
header: { 'Authorization': `Bearer ${t}` }
|
||||||
|
})
|
||||||
|
if (qres.statusCode === 200 && qres.data) dailyQuestion.value = qres.data
|
||||||
|
}
|
||||||
|
} catch (e) { /* silent */ }
|
||||||
|
}
|
||||||
|
|
||||||
const goProfile = () => uni.switchTab({ url: '/pages/user/user' })
|
const goProfile = () => uni.switchTab({ url: '/pages/user/user' })
|
||||||
|
const goLogin = () => uni.navigateTo({ url: '/pages/login/login' })
|
||||||
const goInterview = () => uni.navigateTo({ url: '/pages/interview/interview' })
|
const goInterview = () => uni.navigateTo({ url: '/pages/interview/interview' })
|
||||||
|
const goResume = () => uni.navigateTo({ url: '/pages/resume/resume' })
|
||||||
const goProgress = () => uni.navigateTo({ url: '/pages/progress/progress' })
|
const goProgress = () => uni.navigateTo({ url: '/pages/progress/progress' })
|
||||||
const goContribute = () => uni.navigateTo({ url: '/pages/contribute/contribute' })
|
const goContribute = () => uni.navigateTo({ url: '/pages/contribute/contribute' })
|
||||||
const goBank = () => uni.navigateTo({ url: '/pages/company-bank/bank' })
|
const goBank = () => uni.navigateTo({ url: '/pages/company-bank/bank' })
|
||||||
@@ -182,16 +222,24 @@ const startInterview = (pos) => uni.navigateTo({ url: `/pages/interview/intervie
|
|||||||
background: linear-gradient(135deg, var(--color-gradient-start) 0%, var(--color-gradient-mid) 50%, var(--color-gradient-end) 100%);
|
background: linear-gradient(135deg, var(--color-gradient-start) 0%, var(--color-gradient-mid) 50%, var(--color-gradient-end) 100%);
|
||||||
padding: 48rpx 32rpx 72rpx; border-radius: 0 0 48rpx 48rpx;
|
padding: 48rpx 32rpx 72rpx; border-radius: 0 0 48rpx 48rpx;
|
||||||
}
|
}
|
||||||
|
.hero-row { display: flex; align-items: flex-start; gap: 24rpx; }
|
||||||
|
.hero-left { flex: 1; min-width: 0; padding-top: 8rpx; }
|
||||||
|
.hero-right { flex-shrink: 0; width: 320rpx; }
|
||||||
.hero-title { font-size: 40rpx; font-weight: 700; color: #FFF; display: block; line-height: 1.3; }
|
.hero-title { font-size: 40rpx; font-weight: 700; color: #FFF; display: block; line-height: 1.3; }
|
||||||
.hero-sub { font-size: 22rpx; color: rgba(255,255,255,0.7); margin-top: 8rpx; display: block; }
|
.hero-sub { font-size: 22rpx; color: rgba(255,255,255,0.7); margin-top: 8rpx; display: block; }
|
||||||
|
|
||||||
.user-card {
|
.user-card, .guest-card {
|
||||||
background: rgba(255,255,255,0.95); backdrop-filter: blur(20rpx);
|
background: rgba(255,255,255,0.95); backdrop-filter: blur(20rpx);
|
||||||
border-radius: var(--radius-xl); padding: 24rpx 28rpx;
|
border-radius: var(--radius-xl); padding: 20rpx 24rpx;
|
||||||
display: flex; align-items: center; margin-top: 24rpx;
|
display: flex; align-items: center;
|
||||||
box-shadow: 0 8rpx 32rpx rgba(0,0,0,0.1);
|
box-shadow: 0 8rpx 32rpx rgba(0,0,0,0.1);
|
||||||
}
|
}
|
||||||
.avatar { width: 88rpx; height: 88rpx; border-radius: 50%; margin-right: 20rpx; border: 3rpx solid var(--color-primary-light); flex-shrink: 0; }
|
.guest-card { background: rgba(255,255,255,0.15); backdrop-filter: blur(10rpx); }
|
||||||
|
.guest-card .avatar { border-color: rgba(255,255,255,0.3); }
|
||||||
|
.guest-card .user-name { font-size: 26rpx; color: #FFF; }
|
||||||
|
.guest-hint { font-size: 20rpx; color: rgba(255,255,255,0.6); margin-top: 4rpx; display: block; }
|
||||||
|
.guest-card .arrow { color: rgba(255,255,255,0.4); }
|
||||||
|
.avatar { width: 72rpx; height: 72rpx; border-radius: 50%; margin-right: 16rpx; border: 3rpx solid var(--color-primary-light); flex-shrink: 0; }
|
||||||
.user-meta { flex: 1; min-width: 0; }
|
.user-meta { flex: 1; min-width: 0; }
|
||||||
.user-name { font-size: 30rpx; font-weight: 600; color: var(--color-text); }
|
.user-name { font-size: 30rpx; font-weight: 600; color: var(--color-text); }
|
||||||
.user-tags { display: flex; gap: 10rpx; margin-top: 10rpx; }
|
.user-tags { display: flex; gap: 10rpx; margin-top: 10rpx; }
|
||||||
@@ -205,7 +253,6 @@ const startInterview = (pos) => uni.navigateTo({ url: `/pages/interview/intervie
|
|||||||
.section-header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 20rpx; }
|
.section-header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 20rpx; }
|
||||||
.section-title { font-size: 30rpx; font-weight: 700; color: var(--color-text); }
|
.section-title { font-size: 30rpx; font-weight: 700; color: var(--color-text); }
|
||||||
.section-title-row { display: flex; align-items: center; gap: 12rpx; }
|
.section-title-row { display: flex; align-items: center; gap: 12rpx; }
|
||||||
.section-tag-demo { font-size: 18rpx; color: #9CA3AF; background: #F3F4F6; padding: 2rpx 10rpx; border-radius: 6rpx; }
|
|
||||||
.section-desc { font-size: 22rpx; color: var(--color-primary); }
|
.section-desc { font-size: 22rpx; color: var(--color-primary); }
|
||||||
|
|
||||||
.feature-list { display: flex; flex-direction: column; gap: 16rpx; }
|
.feature-list { display: flex; flex-direction: column; gap: 16rpx; }
|
||||||
@@ -223,6 +270,7 @@ const startInterview = (pos) => uni.navigateTo({ url: `/pages/interview/intervie
|
|||||||
.fp-brief { font-size: 20rpx; color: var(--color-text-secondary); margin-top: 4rpx; display: block; }
|
.fp-brief { font-size: 20rpx; color: var(--color-text-secondary); margin-top: 4rpx; display: block; }
|
||||||
.fp-action { font-size: 28rpx; color: var(--color-primary); font-weight: 600; flex-shrink: 0; }
|
.fp-action { font-size: 28rpx; color: var(--color-primary); font-weight: 600; flex-shrink: 0; }
|
||||||
.feature-secondary { display: grid; grid-template-columns: 1fr 1fr; gap: 16rpx; }
|
.feature-secondary { display: grid; grid-template-columns: 1fr 1fr; gap: 16rpx; }
|
||||||
|
.feature-tertiary { display: grid; grid-template-columns: 1fr 1fr 1fr; gap: 16rpx; }
|
||||||
.fs-card { padding: 20rpx; border-radius: var(--radius-lg); }
|
.fs-card { padding: 20rpx; border-radius: var(--radius-lg); }
|
||||||
.fs-top { display: flex; align-items: center; gap: 10rpx; }
|
.fs-top { display: flex; align-items: center; gap: 10rpx; }
|
||||||
.fs-icon { width: 44rpx; height: 44rpx; border-radius: 12rpx; display: flex; align-items: center; justify-content: center; flex-shrink: 0; }
|
.fs-icon { width: 44rpx; height: 44rpx; border-radius: 12rpx; display: flex; align-items: center; justify-content: center; flex-shrink: 0; }
|
||||||
@@ -230,7 +278,8 @@ const startInterview = (pos) => uni.navigateTo({ url: `/pages/interview/intervie
|
|||||||
.fs-name { font-size: 26rpx; font-weight: 600; color: var(--color-text); }
|
.fs-name { font-size: 26rpx; font-weight: 600; color: var(--color-text); }
|
||||||
.fs-brief { font-size: 18rpx; color: var(--color-text-secondary); margin-top: 10rpx; display: block; }
|
.fs-brief { font-size: 18rpx; color: var(--color-text-secondary); margin-top: 10rpx; display: block; }
|
||||||
.fs-progress { background: linear-gradient(135deg, #EEF2FF, #C7D2FE); }
|
.fs-progress { background: linear-gradient(135deg, #EEF2FF, #C7D2FE); }
|
||||||
.fs-contribute { background: linear-gradient(135deg, #ECFDF5, #A7F3D0); }
|
.fs-resume { background: linear-gradient(135deg, #ECFDF5, #A7F3D0); }
|
||||||
|
.fs-contribute { background: linear-gradient(135deg, #FFF7ED, #FDBA74); }
|
||||||
|
|
||||||
/* 每日一题 */
|
/* 每日一题 */
|
||||||
.daily-card { padding: 24rpx; border-radius: var(--radius-lg); }
|
.daily-card { padding: 24rpx; border-radius: var(--radius-lg); }
|
||||||
|
|||||||
@@ -27,6 +27,8 @@
|
|||||||
ref="dhRef"
|
ref="dhRef"
|
||||||
:text="aiSpeechText"
|
:text="aiSpeechText"
|
||||||
:audio-url="aiAudioUrl"
|
:audio-url="aiAudioUrl"
|
||||||
|
:amplitude-data="aiAmplitudeData"
|
||||||
|
:position="position"
|
||||||
:auto-play="true"
|
:auto-play="true"
|
||||||
@speaking-start="onAvatarSpeaking"
|
@speaking-start="onAvatarSpeaking"
|
||||||
@speaking-end="onAvatarSilent"
|
@speaking-end="onAvatarSilent"
|
||||||
@@ -53,11 +55,14 @@
|
|||||||
</scroll-view>
|
</scroll-view>
|
||||||
|
|
||||||
<view class="input-bar" v-if="!isComplete">
|
<view class="input-bar" v-if="!isComplete">
|
||||||
|
<view class="mic-btn" :class="{ recording: isRecording }" @touchstart="startRecord" @touchend="stopRecord" @touchcancel="stopRecord" @mousedown="startRecord" @mouseup="stopRecord" @mouseleave="stopRecord">
|
||||||
|
<text class="mic-icon">🎤</text>
|
||||||
|
</view>
|
||||||
<view class="input-box">
|
<view class="input-box">
|
||||||
<textarea class="input-area" v-model="inputText" placeholder="输入你的回答..." :auto-height="true" :maxlength="2000" :disabled="aiLoading" @confirm="sendAnswer" />
|
<textarea class="input-area" v-model="inputText" placeholder="输入你的回答..." :auto-height="true" :maxlength="2000" :disabled="aiLoading" @confirm="sendAnswer" />
|
||||||
</view>
|
</view>
|
||||||
<view class="send-btn" :class="{ disabled: !inputText.trim() || aiLoading }" @click="sendAnswer">
|
<view class="send-btn" :class="{ disabled: (!inputText.trim() && !isRecording) || aiLoading }" @click="sendAnswer">
|
||||||
<text class="send-icon">➤</text>
|
<text class="send-icon">{{ isRecording ? '◉' : '➤' }}</text>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
@@ -83,21 +88,25 @@ const answeredCount = ref(0)
|
|||||||
const isComplete = ref(false)
|
const isComplete = ref(false)
|
||||||
const scrollToId = ref('')
|
const scrollToId = ref('')
|
||||||
const position = ref('')
|
const position = ref('')
|
||||||
const avatarMode = ref(false)
|
const avatarMode = ref(true)
|
||||||
const aiSpeechText = ref('')
|
const aiSpeechText = ref('')
|
||||||
const aiAudioUrl = ref('')
|
const aiAudioUrl = ref('')
|
||||||
|
const aiAmplitudeData = ref([])
|
||||||
const isSpeaking = ref(false)
|
const isSpeaking = ref(false)
|
||||||
const dhRef = ref(null)
|
const dhRef = ref(null)
|
||||||
|
const isRecording = ref(false)
|
||||||
|
let recorder = null
|
||||||
|
|
||||||
let timerSeconds = 0
|
let timerSeconds = 0
|
||||||
let timerInterval = null
|
let timerInterval = null
|
||||||
|
|
||||||
const progressPercent = computed(() => Math.min((answeredCount.value / 5) * 100, 100))
|
let MAX_QUESTIONS = 10
|
||||||
|
const progressPercent = computed(() => Math.min((answeredCount.value / MAX_QUESTIONS) * 100, 100))
|
||||||
const formatTime = computed(() => {
|
const formatTime = computed(() => {
|
||||||
const m = Math.floor(timerSeconds / 60); const s = timerSeconds % 60
|
const m = Math.floor(timerSeconds / 60); const s = timerSeconds % 60
|
||||||
return `${String(m).padStart(2, '0')}:${String(s).padStart(2, '0')}`
|
return `${String(m).padStart(2, '0')}:${String(s).padStart(2, '0')}`
|
||||||
})
|
})
|
||||||
const token = computed(() => uni.getStorageSync('token') || '')
|
const token = () => uni.getStorageSync('token') || ''
|
||||||
|
|
||||||
onLoad((options) => {
|
onLoad((options) => {
|
||||||
if (options?.position) {
|
if (options?.position) {
|
||||||
@@ -109,7 +118,7 @@ onLoad((options) => {
|
|||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
timerInterval = setInterval(() => timerSeconds++, 1000)
|
timerInterval = setInterval(() => timerSeconds++, 1000)
|
||||||
if (token.value) startInterview()
|
if (token()) startInterview()
|
||||||
})
|
})
|
||||||
|
|
||||||
onBeforeUnmount(() => {
|
onBeforeUnmount(() => {
|
||||||
@@ -117,7 +126,7 @@ onBeforeUnmount(() => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
const checkLogin = () => {
|
const checkLogin = () => {
|
||||||
if (!token.value) {
|
if (!token()) {
|
||||||
uni.showModal({
|
uni.showModal({
|
||||||
title: '请先登录', content: '登录后即可开始 AI 模拟面试', confirmText: '去登录',
|
title: '请先登录', content: '登录后即可开始 AI 模拟面试', confirmText: '去登录',
|
||||||
success: (r) => { if (r.confirm) uni.navigateTo({ url: '/pages/login/login' }) },
|
success: (r) => { if (r.confirm) uni.navigateTo({ url: '/pages/login/login' }) },
|
||||||
@@ -133,18 +142,22 @@ const startInterview = async () => {
|
|||||||
try {
|
try {
|
||||||
const res = await uni.request({
|
const res = await uni.request({
|
||||||
url: api('/interview/create'), method: 'POST',
|
url: api('/interview/create'), method: 'POST',
|
||||||
header: { 'Authorization': `Bearer ${token.value}`, 'Content-Type': 'application/json' },
|
header: { 'Authorization': `Bearer ${token()}`, 'Content-Type': 'application/json' },
|
||||||
data: { position: position.value },
|
data: { position: position.value },
|
||||||
})
|
})
|
||||||
if (res.statusCode === 200 && res.data) {
|
if (res.statusCode === 200 && res.data) {
|
||||||
interviewId.value = res.data.id
|
interviewId.value = res.data.id
|
||||||
messages.value = res.data.messages || messages.value
|
messages.value = res.data.messages || messages.value
|
||||||
answeredCount.value = res.data.questionCount || 0
|
answeredCount.value = res.data.questionCount || 0
|
||||||
|
if (res.data.totalQuestions) MAX_QUESTIONS = res.data.totalQuestions
|
||||||
// Speak first question in avatar mode
|
// Speak first question in avatar mode
|
||||||
if (avatarMode.value && res.data.messages?.length) {
|
if (avatarMode.value && res.data.messages?.length) {
|
||||||
const last = res.data.messages[res.data.messages.length - 1]
|
const last = res.data.messages[res.data.messages.length - 1]
|
||||||
if (last?.role === 'ai') await speakAiText(last.content)
|
if (last?.role === 'ai') await speakAiText(last.content)
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
const msg = res.data?.message || '创建面试失败'
|
||||||
|
messages.value.push({ role: 'ai', content: msg })
|
||||||
}
|
}
|
||||||
} catch {
|
} catch {
|
||||||
messages.value.push({ role: 'ai', content: '创建面试失败,请重试' })
|
messages.value.push({ role: 'ai', content: '创建面试失败,请重试' })
|
||||||
@@ -156,8 +169,11 @@ const startInterview = async () => {
|
|||||||
|
|
||||||
const sendAnswer = async () => {
|
const sendAnswer = async () => {
|
||||||
if (!inputText.value.trim() || aiLoading.value || isComplete.value) return
|
if (!inputText.value.trim() || aiLoading.value || isComplete.value) return
|
||||||
if (!token.value) { checkLogin(); return }
|
if (!token()) { checkLogin(); return }
|
||||||
if (!interviewId.value) { await startInterview(); return }
|
if (!interviewId.value) {
|
||||||
|
await startInterview()
|
||||||
|
if (!interviewId.value) return // creation failed, don't discard answer
|
||||||
|
}
|
||||||
|
|
||||||
const answer = inputText.value.trim()
|
const answer = inputText.value.trim()
|
||||||
messages.value.push({ role: 'user', content: answer })
|
messages.value.push({ role: 'user', content: answer })
|
||||||
@@ -168,21 +184,24 @@ const sendAnswer = async () => {
|
|||||||
try {
|
try {
|
||||||
const res = await uni.request({
|
const res = await uni.request({
|
||||||
url: api(`/interview/${interviewId.value}/answer`), method: 'POST',
|
url: api(`/interview/${interviewId.value}/answer`), method: 'POST',
|
||||||
header: { 'Authorization': `Bearer ${token.value}`, 'Content-Type': 'application/json' },
|
header: { 'Authorization': `Bearer ${token()}`, 'Content-Type': 'application/json' },
|
||||||
data: avatarMode.value ? { answer, avatar: true } : { answer },
|
data: avatarMode.value ? { answer, avatar: true } : { answer },
|
||||||
})
|
})
|
||||||
if (res.statusCode === 200 && res.data?.messages) {
|
if (res.statusCode === 200 && res.data?.messages) {
|
||||||
const aiMsg = res.data.messages.find(m => m.role === 'ai')
|
const aiMsg = res.data.messages.find(m => m.role === 'ai')
|
||||||
|
// Only push AI messages from response to avoid duplicating the user message already added above
|
||||||
|
const newAiMessages = res.data.messages.filter(m => m.role === 'ai')
|
||||||
|
if (newAiMessages.length > 0) messages.value.push(...newAiMessages)
|
||||||
if (avatarMode.value && aiMsg) {
|
if (avatarMode.value && aiMsg) {
|
||||||
// In avatar mode, only show avatar speaking, don't add to chat
|
await speakAiText(aiMsg.content, res.data.ttsHash, res.data.ttsAmplitude)
|
||||||
await speakAiText(aiMsg.content, res.data.ttsHash)
|
|
||||||
} else {
|
|
||||||
messages.value.push(...res.data.messages)
|
|
||||||
}
|
}
|
||||||
answeredCount.value = res.data.questionCount || answeredCount.value + 1
|
answeredCount.value = res.data.questionCount || answeredCount.value + 1
|
||||||
if (res.data.ttsHash && !avatarMode.value) {
|
if (res.data.totalQuestions) MAX_QUESTIONS = res.data.totalQuestions
|
||||||
// Still got TTS but not in avatar mode, just show text
|
} else if (res.statusCode === 403) {
|
||||||
}
|
messages.value.push({ role: 'ai', content: res.data?.message || '面试次数已用完' })
|
||||||
|
isComplete.value = true
|
||||||
|
} else {
|
||||||
|
messages.value.push({ role: 'ai', content: res.data?.message || '回答提交失败' })
|
||||||
}
|
}
|
||||||
} catch {
|
} catch {
|
||||||
messages.value.push({ role: 'ai', content: '回答提交失败,请重试' })
|
messages.value.push({ role: 'ai', content: '回答提交失败,请重试' })
|
||||||
@@ -192,19 +211,21 @@ const sendAnswer = async () => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function speakAiText(text, ttsHash) {
|
async function speakAiText(text, ttsHash, ttsAmplitude) {
|
||||||
aiSpeechText.value = text
|
aiSpeechText.value = text
|
||||||
|
aiAmplitudeData.value = ttsAmplitude || []
|
||||||
if (ttsHash) {
|
if (ttsHash) {
|
||||||
aiAudioUrl.value = api(API_ENDPOINTS.TTS.AUDIO(ttsHash))
|
aiAudioUrl.value = api(API_ENDPOINTS.TTS.AUDIO(ttsHash))
|
||||||
} else {
|
} else {
|
||||||
try {
|
try {
|
||||||
const synthRes = await uni.request({
|
const synthRes = await uni.request({
|
||||||
url: api('/tts/synthesize'), method: 'POST',
|
url: api('/tts/synthesize'), method: 'POST',
|
||||||
header: { 'Content-Type': 'application/json' },
|
header: { 'Content-Type': 'application/json', 'Authorization': `Bearer ${token()}` },
|
||||||
data: { text },
|
data: { text },
|
||||||
})
|
})
|
||||||
if (synthRes.statusCode === 200 && synthRes.data?.hash) {
|
if (synthRes.statusCode === 200 && synthRes.data?.hash) {
|
||||||
aiAudioUrl.value = api(API_ENDPOINTS.TTS.AUDIO(synthRes.data.hash))
|
aiAudioUrl.value = api(API_ENDPOINTS.TTS.AUDIO(synthRes.data.hash))
|
||||||
|
aiAmplitudeData.value = synthRes.data?.amplitudeData || []
|
||||||
}
|
}
|
||||||
} catch {}
|
} catch {}
|
||||||
}
|
}
|
||||||
@@ -229,6 +250,51 @@ const confirmExit = () => {
|
|||||||
success: (r) => { if (r.confirm) uni.navigateBack() },
|
success: (r) => { if (r.confirm) uni.navigateBack() },
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function startRecord() {
|
||||||
|
if (aiLoading.value || isComplete.value) return
|
||||||
|
// #ifdef MP-WEIXIN
|
||||||
|
isRecording.value = true
|
||||||
|
recorder = uni.getRecorderManager()
|
||||||
|
recorder.onStart(() => {})
|
||||||
|
recorder.onError(() => { isRecording.value = false })
|
||||||
|
recorder.start({ format: 'mp3' })
|
||||||
|
uni.vibrateShort({ type: 'medium' })
|
||||||
|
// #endif
|
||||||
|
// #ifndef MP-WEIXIN
|
||||||
|
uni.showToast({ title: '语音输入仅支持小程序', icon: 'none' })
|
||||||
|
// #endif
|
||||||
|
}
|
||||||
|
|
||||||
|
function stopRecord() {
|
||||||
|
if (!recorder || !isRecording.value) return
|
||||||
|
isRecording.value = false
|
||||||
|
recorder.stop()
|
||||||
|
recorder.onStop(async (res) => {
|
||||||
|
if (!res.tempFilePath) return
|
||||||
|
const audioPath = res.tempFilePath
|
||||||
|
try {
|
||||||
|
const uploadRes = await uni.uploadFile({
|
||||||
|
url: api(API_ENDPOINTS.TTS.ASR),
|
||||||
|
filePath: audioPath,
|
||||||
|
name: 'audio',
|
||||||
|
header: { 'Authorization': `Bearer ${token()}` },
|
||||||
|
})
|
||||||
|
console.log('[ASR] upload response:', uploadRes.statusCode, typeof uploadRes.data === 'string' ? uploadRes.data.slice(0, 200) : JSON.stringify(uploadRes.data).slice(0, 200))
|
||||||
|
if (uploadRes.statusCode === 200 && uploadRes.data) {
|
||||||
|
const data = typeof uploadRes.data === 'string' ? JSON.parse(uploadRes.data) : uploadRes.data
|
||||||
|
if (data.text) {
|
||||||
|
inputText.value = data.text
|
||||||
|
uni.vibrateShort({ type: 'light' })
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.error('[ASR] upload error:', e?.message || e)
|
||||||
|
}
|
||||||
|
uni.showToast({ title: '语音识别失败,请手动输入', icon: 'none' })
|
||||||
|
})
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
@@ -302,6 +368,18 @@ const confirmExit = () => {
|
|||||||
}
|
}
|
||||||
.input-box { flex: 1; background: var(--color-bg); border-radius: var(--radius-md); padding: 12rpx 20rpx; }
|
.input-box { flex: 1; background: var(--color-bg); border-radius: var(--radius-md); padding: 12rpx 20rpx; }
|
||||||
.input-area { width: 100%; font-size: 26rpx; color: var(--color-text); max-height: 160rpx; line-height: 1.5; }
|
.input-area { width: 100%; font-size: 26rpx; color: var(--color-text); max-height: 160rpx; line-height: 1.5; }
|
||||||
|
.mic-btn {
|
||||||
|
width: 64rpx; height: 64rpx; border-radius: 50%; background: #F3F4F6;
|
||||||
|
display: flex; align-items: center; justify-content: center; flex-shrink: 0;
|
||||||
|
transition: all 0.2s;
|
||||||
|
}
|
||||||
|
.mic-btn:active { transform: scale(0.9); }
|
||||||
|
.mic-btn.recording { background: #FEE2E2; animation: mic-pulse 1s infinite; }
|
||||||
|
.mic-icon { font-size: 28rpx; }
|
||||||
|
@keyframes mic-pulse {
|
||||||
|
0%, 100% { box-shadow: 0 0 0 0 rgba(239, 68, 68, 0.4); }
|
||||||
|
50% { box-shadow: 0 0 0 16rpx rgba(239, 68, 68, 0); }
|
||||||
|
}
|
||||||
.send-btn {
|
.send-btn {
|
||||||
width: 80rpx; height: 80rpx; background: linear-gradient(135deg, var(--color-gradient-start), var(--color-gradient-mid));
|
width: 80rpx; height: 80rpx; background: linear-gradient(135deg, var(--color-gradient-start), var(--color-gradient-mid));
|
||||||
border-radius: 50%; display: flex; align-items: center; justify-content: center; flex-shrink: 0;
|
border-radius: 50%; display: flex; align-items: center; justify-content: center; flex-shrink: 0;
|
||||||
|
|||||||
@@ -147,7 +147,7 @@ onMounted(() => {
|
|||||||
// #endif
|
// #endif
|
||||||
})
|
})
|
||||||
|
|
||||||
onBeforeUnmount(() => { if (timer) clearInterval(timer) })
|
onBeforeUnmount(() => { if (timer) { clearTimeout(timer); timer = null } })
|
||||||
|
|
||||||
// 辅助
|
// 辅助
|
||||||
const showToast = (title, icon = 'none') => uni.showToast({ title, icon })
|
const showToast = (title, icon = 'none') => uni.showToast({ title, icon })
|
||||||
@@ -286,12 +286,25 @@ const doWxLogin = async () => {
|
|||||||
// #ifdef MP-WEIXIN
|
// #ifdef MP-WEIXIN
|
||||||
wxLoading.value = true
|
wxLoading.value = true
|
||||||
try {
|
try {
|
||||||
const { code } = await uni.login()
|
const wxResp = await uni.login()
|
||||||
const res = await uni.request({ url: api('/user/wx-login'), method: 'POST', data: { code } })
|
console.log('[wxLogin] uni.login success:', JSON.stringify(wxResp).slice(0, 300))
|
||||||
|
const { code, errMsg } = wxResp
|
||||||
|
if (!code) { console.error('[wxLogin] no code:', errMsg); showToast('获取微信凭证失败'); return }
|
||||||
|
const res = await uni.request({
|
||||||
|
url: api('/user/wx-login'), method: 'POST',
|
||||||
|
header: { 'Content-Type': 'application/json' },
|
||||||
|
data: { code },
|
||||||
|
})
|
||||||
|
console.log('[wxLogin] server response:', res.statusCode, JSON.stringify(res.data).slice(0, 500))
|
||||||
if (res.statusCode === 200 && res.data?.token) {
|
if (res.statusCode === 200 && res.data?.token) {
|
||||||
loginSuccess(res.data)
|
loginSuccess(res.data)
|
||||||
} else { showToast('微信登录失败') }
|
} else {
|
||||||
} catch { showToast('微信登录失败') }
|
showToast(res.data?.message || `登录失败(${res.statusCode})`)
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.error('[wxLogin] error:', JSON.stringify(e).slice(0, 500))
|
||||||
|
showToast('微信登录失败')
|
||||||
|
}
|
||||||
finally { wxLoading.value = false }
|
finally { wxLoading.value = false }
|
||||||
// #endif
|
// #endif
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -42,7 +42,7 @@
|
|||||||
<view class="plan-badge sprint-badge">🚀 冲刺</view>
|
<view class="plan-badge sprint-badge">🚀 冲刺</view>
|
||||||
<view class="plan-header">
|
<view class="plan-header">
|
||||||
<text class="plan-name">冲刺版</text>
|
<text class="plan-name">冲刺版</text>
|
||||||
<text class="plan-price"><text class="price-num price-sprint">¥49.9</text><text class="price-unit">/月</text></text>
|
<text class="plan-price"><text class="price-num price-sprint">{{ sprintPriceText }}</text><text class="price-unit">/月</text></text>
|
||||||
</view>
|
</view>
|
||||||
<view class="plan-features">
|
<view class="plan-features">
|
||||||
<text class="feat" v-for="f in sprintFeatures" :key="f">✓ {{ f }}</text>
|
<text class="feat" v-for="f in sprintFeatures" :key="f">✓ {{ f }}</text>
|
||||||
@@ -50,7 +50,7 @@
|
|||||||
<view class="plan-action" v-if="!isLoggedIn" @click="goLogin">登录后开通</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 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-if="plan === 'growth'" @click="startPay('sprint')">升级至冲刺版</view>
|
||||||
<view class="plan-action" v-else @click="startPay('sprint')">¥49.9/月 立即开通</view>
|
<view class="plan-action" v-else @click="startPay('sprint')">{{ sprintPriceText }}/月 立即开通</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
@@ -104,16 +104,11 @@ const payError = ref('')
|
|||||||
const payingPlanName = ref('')
|
const payingPlanName = ref('')
|
||||||
const payingPlan = ref('')
|
const payingPlan = ref('')
|
||||||
const growthPriceText = ref('¥19.9')
|
const growthPriceText = ref('¥19.9')
|
||||||
|
const sprintPriceText = ref('¥49.9')
|
||||||
const currentOutTradeNo = ref('')
|
const currentOutTradeNo = ref('')
|
||||||
const freeFeatures = ['每日 2 次 AI 模拟面试', '基础面试报告', '通用题库随机出题', '简历诊断(限 3 次)']
|
const freeFeatures = ref(['AI 模拟面试 1 次(体验)', '基础面试报告', '通用题库随机出题', '简历优化(限 3 次免费)'])
|
||||||
const growthFeatures = [
|
const growthFeatures = ref(['免费版全部权益', 'AI 数字人面试无限次', '详细面试报告(四维评分)', '进步轨迹雷达图 + 打卡', '每日一题推送', '参考回答思路', '公司真题库', '每场最多 10 轮 AI 对话'])
|
||||||
'免费版全部权益', '无限面试次数', '详细面试报告(四维评分)',
|
const sprintFeatures = ref(['成长版全部权益', 'AI 语音分析(语气词/语速检测)', '技能缺口分析报告', '学习路径推荐', '真人导师 1v1 点评(每月 1 次)', '简历精修(每月 1 次)', '内推优先', 'AI 实时提示功能'])
|
||||||
'进步轨迹雷达图 + 打卡', '每日一题推送', '参考回答思路', '公司真题库',
|
|
||||||
]
|
|
||||||
const sprintFeatures = [
|
|
||||||
'成长版全部权益', 'AI 语音分析(语气词/语速检测)', '技能缺口分析报告',
|
|
||||||
'学习路径推荐', '真人导师 1v1 点评(每月 1 次)', '简历精修(每月 1 次)', '内推优先',
|
|
||||||
]
|
|
||||||
|
|
||||||
const token = () => uni.getStorageSync('token') || ''
|
const token = () => uni.getStorageSync('token') || ''
|
||||||
|
|
||||||
@@ -136,9 +131,19 @@ onMounted(async () => {
|
|||||||
plan.value = d.plan || 'free'
|
plan.value = d.plan || 'free'
|
||||||
currentPlanName.value = d.planName || '免费版'
|
currentPlanName.value = d.planName || '免费版'
|
||||||
}
|
}
|
||||||
if (lres.statusCode === 200 && lres.data?.price) {
|
if (lres.statusCode === 200 && lres.data) {
|
||||||
const p = lres.data.price
|
const plans = Array.isArray(lres.data.plans) ? lres.data.plans : (Array.isArray(lres.data) ? lres.data : [])
|
||||||
growthPriceText.value = `¥${(p.monthly / 100).toFixed(1)}`
|
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) { /* ignore */ }
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -134,6 +134,7 @@
|
|||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, onMounted } from 'vue'
|
import { ref, onMounted } from 'vue'
|
||||||
|
import { onShow } from '@dcloudio/uni-app'
|
||||||
import { api } from '../../config'
|
import { api } from '../../config'
|
||||||
|
|
||||||
const stats = ref({ completedInterviews: 0, avgScore: 0, streak: 0 })
|
const stats = ref({ completedInterviews: 0, avgScore: 0, streak: 0 })
|
||||||
@@ -155,6 +156,16 @@ onMounted(async () => {
|
|||||||
const t = token()
|
const t = token()
|
||||||
if (!t) return
|
if (!t) return
|
||||||
|
|
||||||
|
await loadProgressData()
|
||||||
|
})
|
||||||
|
|
||||||
|
onShow(async () => {
|
||||||
|
if (token()) await loadProgressData()
|
||||||
|
})
|
||||||
|
|
||||||
|
async function loadProgressData() {
|
||||||
|
const t = token()
|
||||||
|
if (!t) return
|
||||||
try {
|
try {
|
||||||
// Load progress
|
// Load progress
|
||||||
const res = await uni.request({
|
const res = await uni.request({
|
||||||
@@ -166,7 +177,7 @@ onMounted(async () => {
|
|||||||
progress.value = d
|
progress.value = d
|
||||||
dimensions.value = dimensions.value.map(dim => ({
|
dimensions.value = dimensions.value.map(dim => ({
|
||||||
...dim,
|
...dim,
|
||||||
value: d.dimensions?.[dim.key] || Math.round(50 + Math.random() * 30),
|
value: d.dimensions?.[dim.key] || 0,
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
} catch (e) { console.error(e) }
|
} catch (e) { console.error(e) }
|
||||||
@@ -191,23 +202,29 @@ onMounted(async () => {
|
|||||||
}
|
}
|
||||||
} catch (e) { console.error(e) }
|
} catch (e) { console.error(e) }
|
||||||
|
|
||||||
// Build week days
|
buildWeekDays()
|
||||||
|
}
|
||||||
|
|
||||||
|
function buildWeekDays() {
|
||||||
const days = ['日', '一', '二', '三', '四', '五', '六']
|
const days = ['日', '一', '二', '三', '四', '五', '六']
|
||||||
const today = new Date()
|
const today = new Date()
|
||||||
const arr = []
|
const arr = []
|
||||||
|
const checkinDates = (progress.value.checkins || []).map((c) => {
|
||||||
|
const d = new Date(c.date || c.createdAt)
|
||||||
|
return `${d.getFullYear()}-${d.getMonth() + 1}-${d.getDate()}`
|
||||||
|
})
|
||||||
for (let i = 6; i >= 0; i--) {
|
for (let i = 6; i >= 0; i--) {
|
||||||
const d = new Date(today)
|
const d = new Date(today)
|
||||||
d.setDate(d.getDate() - i)
|
d.setDate(d.getDate() - i)
|
||||||
const isToday = i === 0
|
const key = `${d.getFullYear()}-${d.getMonth() + 1}-${d.getDate()}`
|
||||||
// Mark days with interviews (simulate based on streak)
|
|
||||||
arr.push({
|
arr.push({
|
||||||
label: days[d.getDay()],
|
label: days[d.getDay()],
|
||||||
isToday,
|
isToday: i === 0,
|
||||||
done: i < (stats.value.streak || 0),
|
done: checkinDates.includes(key),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
weekDays.value = arr
|
weekDays.value = arr
|
||||||
})
|
}
|
||||||
|
|
||||||
const formatDate = (d) => {
|
const formatDate = (d) => {
|
||||||
if (!d) return ''
|
if (!d) return ''
|
||||||
|
|||||||
@@ -156,7 +156,9 @@ onLoad(async (options) => {
|
|||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}).catch(() => {})
|
}).catch(e => {
|
||||||
|
console.error('[report] auto-complete failed:', e)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch(e) { console.error(e) }
|
} catch(e) { console.error(e) }
|
||||||
@@ -288,15 +290,12 @@ async function generateCard() {
|
|||||||
ctx.setFillStyle('rgba(165,180,252,0.5)')
|
ctx.setFillStyle('rgba(165,180,252,0.5)')
|
||||||
ctx.fillText('扫码开始你的模拟面试 → 在微信搜索"职引"小程序', w / 2, 690)
|
ctx.fillText('扫码开始你的模拟面试 → 在微信搜索"职引"小程序', w / 2, 690)
|
||||||
|
|
||||||
// QR code hint (simulated)
|
// QR text hint
|
||||||
ctx.setFillStyle('#FFFFFF')
|
ctx.setFillStyle('#FFFFFF')
|
||||||
ctx.setFontSize(12)
|
ctx.setFontSize(16)
|
||||||
ctx.setTextAlign('center')
|
ctx.setTextAlign('center')
|
||||||
ctx.fillText('⬛ ⬛ ⬛ ⬛ ⬛', w / 2, 760)
|
ctx.fillText('在微信搜索「职引」小程序', w / 2, 760)
|
||||||
ctx.fillText('⬛ ⬛ ⬛ ⬛ ⬛', w / 2, 780)
|
ctx.fillText('查看完整面试报告', w / 2, 790)
|
||||||
ctx.fillText('⬛ ⬛ ⬛ ⬛ ⬛', w / 2, 800)
|
|
||||||
ctx.fillText('⬛ ⬛ ⬛ ⬛ ⬛', w / 2, 820)
|
|
||||||
ctx.fillText('微信小程序', w / 2, 855)
|
|
||||||
|
|
||||||
ctx.draw(false, async () => {
|
ctx.draw(false, async () => {
|
||||||
try {
|
try {
|
||||||
@@ -306,7 +305,9 @@ async function generateCard() {
|
|||||||
itemList: ['保存到相册', '分享给好友'],
|
itemList: ['保存到相册', '分享给好友'],
|
||||||
success: (res) => {
|
success: (res) => {
|
||||||
if (res.tapIndex === 0) {
|
if (res.tapIndex === 0) {
|
||||||
uni.saveImageToPhotosAlbum({ filePath: tempRes.tempFilePath })
|
uni.saveImageToPhotosAlbum({ filePath: tempRes.tempFilePath, success: () => uni.showToast({ title: '已保存到相册', icon: 'success' }) })
|
||||||
|
} else if (res.tapIndex === 1) {
|
||||||
|
uni.shareAppMessage ? uni.shareAppMessage({ title: '我的面试报告', imageUrl: tempRes.tempFilePath }) : uni.showToast({ title: '请截图后分享', icon: 'none' })
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -16,7 +16,7 @@
|
|||||||
<text class="score-num">{{ diagnosisResult.score }}</text>
|
<text class="score-num">{{ diagnosisResult.score }}</text>
|
||||||
<text class="score-label">/100</text>
|
<text class="score-label">/100</text>
|
||||||
</view>
|
</view>
|
||||||
<text class="summary-text">{{ diagnosisResult.summary }}</text>
|
<text class="summary-text" v-if="diagnosisResult.summary">{{ diagnosisResult.summary }}</text>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
<!-- 岗位匹配度(诊断模式) -->
|
<!-- 岗位匹配度(诊断模式) -->
|
||||||
@@ -136,16 +136,19 @@ onLoad(async (options: any) => {
|
|||||||
function applyResult(data: any) {
|
function applyResult(data: any) {
|
||||||
loading.value = false;
|
loading.value = false;
|
||||||
if (isOptimize.value) {
|
if (isOptimize.value) {
|
||||||
optimizedContent.value = data.optimizedContent || '';
|
optimizedContent.value = data.optimized || '';
|
||||||
changes.value = data.changes || [];
|
changes.value = (data.changes || []).map((c: any) =>
|
||||||
highlights.value = data.highlights || [];
|
typeof c === 'string' ? { section: c, description: c } : c
|
||||||
|
);
|
||||||
|
highlights.value = [];
|
||||||
} else {
|
} else {
|
||||||
diagnosisResult.value = data;
|
diagnosisResult.value = data;
|
||||||
changes.value = (data.issues || []).map((i: any) => ({
|
changes.value = (data.issues || []).map((i: any) => ({
|
||||||
...i,
|
...i,
|
||||||
typeLabel: i.type === 'structure' ? '结构' : i.type === 'content' ? '内容' : i.type === 'keywords' ? '关键词' : i.type === 'achievement' ? '成就' : '格式',
|
typeLabel: i.level === 'high' ? '严重' : i.level === 'medium' ? '中等' : '轻微',
|
||||||
|
description: i.desc || i.description,
|
||||||
}));
|
}));
|
||||||
highlights.value = data.strengths || [];
|
highlights.value = data.suggestions || [];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,275 @@
|
|||||||
|
<template>
|
||||||
|
<view class="page fade-in">
|
||||||
|
<!-- 统计卡片 -->
|
||||||
|
<view class="stats-card">
|
||||||
|
<view class="stat-row">
|
||||||
|
<view class="stat-item">
|
||||||
|
<text class="stat-value">{{ stats.shareCredits || 0 }}</text>
|
||||||
|
<text class="stat-label">📦 我的积分</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view class="stat-sub-row">
|
||||||
|
<view class="stat-sub-item">
|
||||||
|
<text class="stat-sub-value">{{ stats.totalVisits || 0 }}</text>
|
||||||
|
<text class="stat-sub-label">总点击</text>
|
||||||
|
</view>
|
||||||
|
<view class="stat-arrow">→</view>
|
||||||
|
<view class="stat-sub-item">
|
||||||
|
<text class="stat-sub-value">{{ stats.creditedCount || 0 }}</text>
|
||||||
|
<text class="stat-sub-label">有效注册</text>
|
||||||
|
</view>
|
||||||
|
<view class="stat-arrow">→</view>
|
||||||
|
<view class="stat-sub-item">
|
||||||
|
<text class="stat-sub-value">{{ stats.shareCredits || 0 }}</text>
|
||||||
|
<text class="stat-sub-label">获得积分</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view class="hint">
|
||||||
|
💡 好友通过你的链接打开并 <text class="hint-em">登录/注册</text> 才算有效,每次有效得 1 积分
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 今日数据 -->
|
||||||
|
<view class="today-card">
|
||||||
|
<text class="today-title">今日数据</text>
|
||||||
|
<view class="today-row">
|
||||||
|
<view class="today-item">
|
||||||
|
<text class="today-value">{{ todayStats.visits }}</text>
|
||||||
|
<text class="today-label">点击</text>
|
||||||
|
</view>
|
||||||
|
<view class="today-item">
|
||||||
|
<text class="today-value">{{ todayStats.credited }}</text>
|
||||||
|
<text class="today-label">有效</text>
|
||||||
|
</view>
|
||||||
|
<view class="today-item">
|
||||||
|
<text class="today-value">{{ 3 - todayStats.credited > 0 ? 3 - todayStats.credited : 0 }}</text>
|
||||||
|
<text class="today-label">今日剩余</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view class="today-bar">
|
||||||
|
<view class="today-bar-fill" :style="{ width: Math.min(100, (todayStats.credited / 3) * 100) + '%' }"></view>
|
||||||
|
</view>
|
||||||
|
<text class="today-hint">每日最多 3 次有效积分</text>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 分享按钮 -->
|
||||||
|
<view class="share-actions">
|
||||||
|
<button class="share-btn wx-share" @click="shareToWechat" v-if="isWechat">
|
||||||
|
<text class="btn-icon">💬</text>
|
||||||
|
<text>分享给微信好友</text>
|
||||||
|
</button>
|
||||||
|
<button class="share-btn link-share" @click="copyLink">
|
||||||
|
<text class="btn-icon">🔗</text>
|
||||||
|
<text>复制分享链接</text>
|
||||||
|
</button>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- Tab 切换 -->
|
||||||
|
<view class="tab-bar">
|
||||||
|
<view class="tab-item" :class="{ active: tab === 'records' }" @click="tab = 'records'">分享记录</view>
|
||||||
|
<view class="tab-item" :class="{ active: tab === 'visitors' }" @click="tab = 'visitors'">访问明细</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 分享记录 -->
|
||||||
|
<view class="list" v-if="tab === 'records'">
|
||||||
|
<view class="list-item" v-for="item in records" :key="item.shareCode">
|
||||||
|
<view class="item-main">
|
||||||
|
<text class="item-type">{{ typeLabel(item.type) }}</text>
|
||||||
|
<text class="item-title">{{ item.title }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="item-stats">
|
||||||
|
<text class="item-stat">👀 {{ item.visitCount }}点击</text>
|
||||||
|
<text class="item-stat active">✅ {{ item.creditedCount }}有效</text>
|
||||||
|
</view>
|
||||||
|
<text class="item-time">{{ formatTime(item.createdAt) }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="empty" v-if="records.length === 0">
|
||||||
|
<text class="empty-icon">📤</text>
|
||||||
|
<text class="empty-text">还没有分享记录</text>
|
||||||
|
<text class="empty-hint">点击上方按钮分享给好友吧</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 访问明细 -->
|
||||||
|
<view class="list" v-if="tab === 'visitors'">
|
||||||
|
<view class="list-item" v-for="(item, idx) in visitors" :key="idx">
|
||||||
|
<image class="visitor-avatar" :src="item.avatar || '/static/avatar-default.svg'" />
|
||||||
|
<view class="item-main">
|
||||||
|
<text class="item-title">{{ item.nickname || '👤 未登录访客' }}</text>
|
||||||
|
<text class="item-time">{{ formatTime(item.createdAt) }}</text>
|
||||||
|
</view>
|
||||||
|
<view class="badge" :class="item.credited ? 'badge-done' : 'badge-pending'">
|
||||||
|
{{ item.credited ? '✅ 已积分' : '⏳ 未注册' }}
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view class="empty" v-if="visitors.length === 0">
|
||||||
|
<text class="empty-icon">👀</text>
|
||||||
|
<text class="empty-text">还没有访问记录</text>
|
||||||
|
<text class="empty-hint">分享后好友访问就会出现在这里</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup>
|
||||||
|
import { ref, computed, onMounted } from 'vue'
|
||||||
|
import { api } from '../../config'
|
||||||
|
|
||||||
|
const tab = ref('records')
|
||||||
|
const stats = ref({ totalShares: 0, totalVisits: 0, creditedCount: 0, todayCredited: 0, shareCredits: 0 })
|
||||||
|
const records = ref([])
|
||||||
|
const visitors = ref([])
|
||||||
|
|
||||||
|
const todayStats = computed(() => ({
|
||||||
|
visits: stats.value.todayCredited + Math.round(Math.random() * 0),
|
||||||
|
credited: stats.value.todayCredited || 0,
|
||||||
|
}))
|
||||||
|
|
||||||
|
const isWechat = ref(false)
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
// #ifdef H5
|
||||||
|
isWechat.value = navigator?.userAgent?.toLowerCase().includes('micromessenger') || false
|
||||||
|
// #endif
|
||||||
|
loadData()
|
||||||
|
})
|
||||||
|
|
||||||
|
async function loadData() {
|
||||||
|
const token = uni.getStorageSync('token')
|
||||||
|
if (!token) { uni.showToast({ title: '请先登录', icon: 'none' }); return }
|
||||||
|
const header = { Authorization: `Bearer ${token}` }
|
||||||
|
|
||||||
|
try {
|
||||||
|
const [statsRes, recordsRes, visitorsRes] = await Promise.all([
|
||||||
|
uni.request({ url: api('/share/stats'), method: 'GET', header }),
|
||||||
|
uni.request({ url: api('/share/records'), method: 'GET', header }),
|
||||||
|
uni.request({ url: api('/share/visitors'), method: 'GET', header }),
|
||||||
|
])
|
||||||
|
if (statsRes.statusCode === 200) stats.value = statsRes.data
|
||||||
|
if (recordsRes.statusCode === 200) records.value = recordsRes.data.list || []
|
||||||
|
if (visitorsRes.statusCode === 200) visitors.value = visitorsRes.data.list || []
|
||||||
|
} catch (e) { console.error(e) }
|
||||||
|
}
|
||||||
|
|
||||||
|
async function shareToWechat() {
|
||||||
|
const token = uni.getStorageSync('token')
|
||||||
|
if (!token) return
|
||||||
|
|
||||||
|
try {
|
||||||
|
const res = await uni.request({
|
||||||
|
url: api('/share/create'),
|
||||||
|
method: 'POST',
|
||||||
|
data: { type: 'app', title: '我在AI磁场·职引练习面试', description: 'AI模拟面试+简历优化,快来一起提升吧' },
|
||||||
|
header: { Authorization: `Bearer ${token}` },
|
||||||
|
})
|
||||||
|
if (res.statusCode !== 200) return
|
||||||
|
|
||||||
|
const data = res.data
|
||||||
|
const path = data.wechatShareInfo?.path || `/pages/share/share?code=${data.shareCode}`
|
||||||
|
|
||||||
|
uni.shareAppMessage({
|
||||||
|
provider: 'weixin',
|
||||||
|
title: data.wechatShareInfo?.title || 'AI磁场·职引',
|
||||||
|
description: data.wechatShareInfo?.description || '',
|
||||||
|
path,
|
||||||
|
imageUrl: 'https://zhiyinwx.yzrcloud.cn/static/share-card.png',
|
||||||
|
success: () => { uni.showToast({ title: '分享成功', icon: 'success' }); loadData() },
|
||||||
|
fail: () => { uni.showToast({ title: '分享取消', icon: 'none' }) },
|
||||||
|
})
|
||||||
|
} catch (e) { console.error(e) }
|
||||||
|
}
|
||||||
|
|
||||||
|
async function copyLink() {
|
||||||
|
const token = uni.getStorageSync('token')
|
||||||
|
if (!token) return
|
||||||
|
|
||||||
|
try {
|
||||||
|
const res = await uni.request({
|
||||||
|
url: api('/share/create'),
|
||||||
|
method: 'POST',
|
||||||
|
data: { type: 'app', title: '我在AI磁场·职引练习面试', description: 'AI模拟面试+简历优化,快来一起提升吧' },
|
||||||
|
header: { Authorization: `Bearer ${token}` },
|
||||||
|
})
|
||||||
|
if (res.statusCode !== 200) return
|
||||||
|
const shareUrl = `https://zhiyinwx.yzrcloud.cn/share/${res.data.shareCode}`
|
||||||
|
uni.setClipboardData({ data: shareUrl, success: () => { uni.showToast({ title: '链接已复制' }); loadData() } })
|
||||||
|
} catch (e) { console.error(e) }
|
||||||
|
}
|
||||||
|
|
||||||
|
function typeLabel(type) {
|
||||||
|
const map = { app: '应用', interview: '面试', resume: '简历' }
|
||||||
|
return map[type] || type
|
||||||
|
}
|
||||||
|
|
||||||
|
function formatTime(t) {
|
||||||
|
if (!t) return ''
|
||||||
|
const d = new Date(t)
|
||||||
|
return `${d.getMonth() + 1}/${d.getDate()} ${String(d.getHours()).padStart(2, '0')}:${String(d.getMinutes()).padStart(2, '0')}`
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.page { min-height: 100vh; background: var(--color-bg); padding-bottom: 48rpx; }
|
||||||
|
|
||||||
|
/* Stats card */
|
||||||
|
.stats-card { background: linear-gradient(135deg, #4F46E5 0%, #7C3AED 100%); margin: 24rpx 32rpx; border-radius: var(--radius-lg); padding: 32rpx; }
|
||||||
|
.stat-row { display: flex; justify-content: center; margin-bottom: 20rpx; }
|
||||||
|
.stat-item { display: flex; flex-direction: column; align-items: center; }
|
||||||
|
.stat-value { font-size: 56rpx; font-weight: 800; color: #FFFFFF; }
|
||||||
|
.stat-label { font-size: 22rpx; color: rgba(255,255,255,0.75); margin-top: 6rpx; }
|
||||||
|
|
||||||
|
.stat-sub-row { display: flex; align-items: center; justify-content: center; gap: 12rpx; padding: 16rpx 0; border-top: 1rpx solid rgba(255,255,255,0.15); border-bottom: 1rpx solid rgba(255,255,255,0.15); margin-bottom: 16rpx; }
|
||||||
|
.stat-sub-item { display: flex; flex-direction: column; align-items: center; }
|
||||||
|
.stat-sub-value { font-size: 32rpx; font-weight: 700; color: #FFFFFF; }
|
||||||
|
.stat-sub-label { font-size: 18rpx; color: rgba(255,255,255,0.6); margin-top: 4rpx; }
|
||||||
|
.stat-arrow { font-size: 24rpx; color: rgba(255,255,255,0.4); }
|
||||||
|
.hint { font-size: 20rpx; color: rgba(255,255,255,0.7); text-align: center; line-height: 1.6; }
|
||||||
|
.hint-em { color: #FCD34D; font-weight: 600; }
|
||||||
|
|
||||||
|
/* Today card */
|
||||||
|
.today-card { background: #FFFFFF; margin: 0 32rpx 24rpx; border-radius: var(--radius-md); padding: 24rpx; box-shadow: var(--shadow-sm); }
|
||||||
|
.today-title { font-size: 24rpx; font-weight: 600; color: var(--color-text); margin-bottom: 16rpx; }
|
||||||
|
.today-row { display: flex; gap: 24rpx; margin-bottom: 16rpx; }
|
||||||
|
.today-item { flex: 1; display: flex; flex-direction: column; align-items: center; }
|
||||||
|
.today-value { font-size: 36rpx; font-weight: 700; color: var(--color-primary); }
|
||||||
|
.today-label { font-size: 20rpx; color: var(--color-text-tertiary); margin-top: 4rpx; }
|
||||||
|
.today-bar { height: 8rpx; background: #F3F4F6; border-radius: 4rpx; overflow: hidden; margin-bottom: 8rpx; }
|
||||||
|
.today-bar-fill { height: 100%; background: linear-gradient(90deg, #4F46E5, #7C3AED); border-radius: 4rpx; transition: width 0.3s; }
|
||||||
|
.today-hint { font-size: 18rpx; color: var(--color-text-tertiary); text-align: center; }
|
||||||
|
|
||||||
|
/* Share buttons */
|
||||||
|
.share-actions { padding: 0 32rpx; 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; }
|
||||||
|
.share-btn:active { transform: scale(0.96); }
|
||||||
|
.btn-icon { margin-right: 8rpx; font-size: 28rpx; }
|
||||||
|
.wx-share { background: #07C160; color: #FFFFFF; }
|
||||||
|
.link-share { background: #FFFFFF; color: var(--color-text); border: 2rpx solid var(--color-border); }
|
||||||
|
|
||||||
|
/* Tabs */
|
||||||
|
.tab-bar { display: flex; margin: 32rpx 32rpx 0; border-bottom: 2rpx solid var(--color-border); }
|
||||||
|
.tab-item { flex: 1; text-align: center; padding: 20rpx 0; font-size: 28rpx; color: #9CA3AF; font-weight: 500; position: relative; }
|
||||||
|
.tab-item.active { color: var(--color-primary); }
|
||||||
|
.tab-item.active::after { content: ''; position: absolute; bottom: -2rpx; left: 30%; right: 30%; height: 4rpx; background: var(--color-primary); border-radius: 2rpx; }
|
||||||
|
|
||||||
|
/* List */
|
||||||
|
.list { padding: 0 32rpx; }
|
||||||
|
.list-item { background: #FFFFFF; border-radius: var(--radius-md); padding: 24rpx 28rpx; margin-top: 16rpx; display: flex; align-items: center; box-shadow: var(--shadow-sm); }
|
||||||
|
.item-main { flex: 1; display: flex; flex-direction: column; }
|
||||||
|
.item-type { font-size: 20rpx; color: var(--color-primary); background: #EEF2FF; padding: 2rpx 12rpx; border-radius: 6rpx; align-self: flex-start; margin-bottom: 4rpx; }
|
||||||
|
.item-title { font-size: 26rpx; color: var(--color-text); font-weight: 500; }
|
||||||
|
.item-stats { display: flex; flex-direction: column; align-items: flex-end; margin-left: 12rpx; }
|
||||||
|
.item-stat { font-size: 22rpx; color: #9CA3AF; white-space: nowrap; }
|
||||||
|
.item-stat.active { color: var(--color-primary); }
|
||||||
|
.item-time { font-size: 20rpx; color: #D1D5DB; white-space: nowrap; margin-left: 12rpx; }
|
||||||
|
|
||||||
|
.visitor-avatar { width: 56rpx; height: 56rpx; border-radius: 50%; margin-right: 16rpx; flex-shrink: 0; }
|
||||||
|
.badge { font-size: 20rpx; padding: 4rpx 16rpx; border-radius: 8rpx; white-space: nowrap; }
|
||||||
|
.badge-done { background: #ECFDF5; color: #059669; }
|
||||||
|
.badge-pending { background: #FEF3C7; color: #D97706; }
|
||||||
|
|
||||||
|
/* Empty state */
|
||||||
|
.empty { display: flex; flex-direction: column; align-items: center; padding: 80rpx 0; }
|
||||||
|
.empty-icon { font-size: 64rpx; margin-bottom: 16rpx; }
|
||||||
|
.empty-text { font-size: 28rpx; color: #9CA3AF; }
|
||||||
|
.empty-hint { font-size: 24rpx; color: #D1D5DB; margin-top: 8rpx; }
|
||||||
|
</style>
|
||||||
@@ -3,7 +3,7 @@
|
|||||||
<!-- 个人中心 -->
|
<!-- 个人中心 -->
|
||||||
<view class="header" v-if="isLoggedIn">
|
<view class="header" v-if="isLoggedIn">
|
||||||
<view class="profile-section">
|
<view class="profile-section">
|
||||||
<image class="avatar" :src="userInfo.avatar || '/static/avatar-default.svg'" mode="aspectFill" />
|
<image class="avatar" :src="userInfo.avatar || '/static/avatar-default.png'" mode="aspectFill" />
|
||||||
<view class="profile-info">
|
<view class="profile-info">
|
||||||
<text class="nickname">{{ userInfo.nickname || '未设置昵称' }}</text>
|
<text class="nickname">{{ userInfo.nickname || '未设置昵称' }}</text>
|
||||||
<view class="plan-badge">{{ userInfo.plan || '免费版' }}</view>
|
<view class="plan-badge">{{ userInfo.plan || '免费版' }}</view>
|
||||||
@@ -33,7 +33,6 @@
|
|||||||
<view class="guest-avatar"><text class="guest-icon">👤</text></view>
|
<view class="guest-avatar"><text class="guest-icon">👤</text></view>
|
||||||
<view class="guest-info">
|
<view class="guest-info">
|
||||||
<text class="guest-name">未登录 / 点击登录</text>
|
<text class="guest-name">未登录 / 点击登录</text>
|
||||||
<text class="guest-hint">登录后体验全部功能</text>
|
|
||||||
</view>
|
</view>
|
||||||
<text class="header-arrow">›</text>
|
<text class="header-arrow">›</text>
|
||||||
</view>
|
</view>
|
||||||
@@ -57,6 +56,11 @@
|
|||||||
<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-icon-wrap wrap-orange"><text class="menu-icon">📤</text></view>
|
||||||
|
<text class="menu-text">我的分享</text>
|
||||||
|
<text class="menu-arrow">›</text>
|
||||||
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="menu-group">
|
<view class="menu-group">
|
||||||
<view class="menu-item" @click="goAbout">
|
<view class="menu-item" @click="goAbout">
|
||||||
@@ -79,6 +83,7 @@
|
|||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import { ref, computed, onMounted } from 'vue'
|
import { ref, computed, onMounted } from 'vue'
|
||||||
|
import { onShow } from '@dcloudio/uni-app'
|
||||||
import { api } from '../../config'
|
import { api } from '../../config'
|
||||||
|
|
||||||
const userInfo = ref({})
|
const userInfo = ref({})
|
||||||
@@ -88,13 +93,28 @@ const token = ref('')
|
|||||||
|
|
||||||
const isLoggedIn = computed(() => !!token.value)
|
const isLoggedIn = computed(() => !!token.value)
|
||||||
|
|
||||||
onMounted(() => {
|
const refreshState = () => {
|
||||||
token.value = uni.getStorageSync('token') || ''
|
token.value = uni.getStorageSync('token') || ''
|
||||||
if (!token.value) return
|
if (!token.value) return
|
||||||
try { const s = uni.getStorageSync('userInfo'); if (s) userInfo.value = JSON.parse(s) } catch(e) {}
|
try { const s = uni.getStorageSync('userInfo'); if (s) userInfo.value = JSON.parse(s) } catch(e) {}
|
||||||
loadStats()
|
loadStats()
|
||||||
checkAdmin()
|
checkAdmin()
|
||||||
})
|
// Fetch fresh user info from API to update stale cache (e.g. credits changed after interview)
|
||||||
|
fetchUserInfo()
|
||||||
|
}
|
||||||
|
|
||||||
|
const fetchUserInfo = async () => {
|
||||||
|
try {
|
||||||
|
const res = await uni.request({ url: api('/user/info'), method: 'GET', header: { 'Authorization': `Bearer ${token.value}` } })
|
||||||
|
if (res.statusCode === 200 && res.data) {
|
||||||
|
userInfo.value = res.data
|
||||||
|
uni.setStorageSync('userInfo', JSON.stringify(res.data))
|
||||||
|
}
|
||||||
|
} catch(e) { /* silent - cached data is fallback */ }
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(refreshState)
|
||||||
|
onShow(refreshState)
|
||||||
|
|
||||||
const loadStats = async () => {
|
const loadStats = async () => {
|
||||||
try {
|
try {
|
||||||
@@ -121,6 +141,7 @@ const goLogin = () => uni.navigateTo({ url: '/pages/login/login' })
|
|||||||
const goHistory = () => uni.switchTab({ url: '/pages/history/history' })
|
const goHistory = () => uni.switchTab({ url: '/pages/history/history' })
|
||||||
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 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' })
|
||||||
|
|
||||||
@@ -171,6 +192,7 @@ const doLogout = () => {
|
|||||||
.wrap-blue { background: #EEF2FF; }
|
.wrap-blue { background: #EEF2FF; }
|
||||||
.wrap-purple { background: #F5F3FF; }
|
.wrap-purple { background: #F5F3FF; }
|
||||||
.wrap-green { background: #ECFDF5; }
|
.wrap-green { background: #ECFDF5; }
|
||||||
|
.wrap-orange { background: #FFF7ED; }
|
||||||
.wrap-gray { background: #F3F4F6; }
|
.wrap-gray { background: #F3F4F6; }
|
||||||
.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; }
|
||||||
|
|||||||
@@ -83,6 +83,13 @@ export const apiService = {
|
|||||||
byPosition: (position: string) =>
|
byPosition: (position: string) =>
|
||||||
request(API_ENDPOINTS.DAILY_QUESTION.BY_POSITION(position), 'GET', undefined, true),
|
request(API_ENDPOINTS.DAILY_QUESTION.BY_POSITION(position), 'GET', undefined, true),
|
||||||
},
|
},
|
||||||
|
share: {
|
||||||
|
create: (data: { type: string; refId?: string; title?: string; description?: string }) =>
|
||||||
|
request(API_ENDPOINTS.SHARE.CREATE, 'POST', data, true),
|
||||||
|
stats: () => request(API_ENDPOINTS.SHARE.STATS, 'GET', undefined, true),
|
||||||
|
records: () => request(API_ENDPOINTS.SHARE.RECORDS, 'GET', undefined, true),
|
||||||
|
visitors: () => request(API_ENDPOINTS.SHARE.VISITORS, 'GET', undefined, true),
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
export default apiService
|
export default apiService
|
||||||
|
|||||||
|
After Width: | Height: | Size: 3.9 KiB |
@@ -0,0 +1 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 180 180" fill="none" shape-rendering="auto"><metadata xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:dcterms="http://purl.org/dc/terms/"><rdf:RDF><rdf:Description><dc:title>Bottts</dc:title><dc:creator>Pablo Stanley</dc:creator><dc:source xsi:type="dcterms:URI">https://bottts.com/</dc:source><dcterms:license xsi:type="dcterms:URI">https://bottts.com/</dcterms:license><dc:rights>Remix of „Bottts” (https://bottts.com/) by „Pablo Stanley”, licensed under „Free for personal and commercial use” (https://bottts.com/)</dc:rights></rdf:Description></rdf:RDF></metadata><mask id="viewboxMask"><rect width="180" height="180" rx="0" ry="0" x="0" y="0" fill="#fff" /></mask><g mask="url(#viewboxMask)"><rect fill="#b6e3f4" width="180" height="180" x="0" y="0" /><g transform="translate(0 66)"><path d="M38 12c-2.95 11.7-19.9 6.67-23.37 18-3.46 11.35 8.03 20 17.53 20" stroke="#2A3544" stroke-width="6" opacity=".9"/><path d="M150 55c8.4 3.49 20.1-7.6 16-16.5-4.1-8.9-16-6.7-16-19.3" stroke="#2A3544" stroke-width="4" opacity=".9"/><mask id="sidesCables01-a" style="mask-type:luminance" maskUnits="userSpaceOnUse" x="21" y="6" width="138" height="58"><g fill="#fff"><rect x="21" y="35" width="16" height="22" rx="2"/><rect x="136" y="42" width="23" height="22" rx="2"/><rect x="136" y="6" width="23" height="18" rx="2"/></g></mask><g mask="url(#sidesCables01-a)"><path d="M0 0h180v76H0V0Z" fill="#546e7a"/><path d="M0 0h180v76H0V0Z" fill="#fff" fill-opacity=".3"/></g></g><g transform="translate(41)"><g filter="url(#topGlowingBulb01-a)"><path fill-rule="evenodd" clip-rule="evenodd" d="M32 24A16 16 0 0 1 48 8h4a16 16 0 0 1 16 16v8a8 8 0 0 1-8 8H40a8 8 0 0 1-8-8v-8Z" fill="#fff" fill-opacity=".3"/></g><path d="M49 11.5c4.93 0 9.37 2.13 12.44 5.52" stroke="#fff" stroke-width="2" stroke-linecap="round"/><path d="m49.83 29-9-9L38 22.83l10 10V40h4v-7.03l10.14-10.14L59.31 20l-9 9h-.48Z" fill="#fff" fill-opacity=".8"/><rect x="22" y="40" width="56" height="12" rx="1" fill="#48494B"/><defs><filter id="topGlowingBulb01-a" x="24" y="0" width="52" height="48" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB"><feFlood flood-opacity="0" result="BackgroundImageFix"/><feColorMatrix in="SourceAlpha" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/><feOffset/><feGaussianBlur stdDeviation="4"/><feColorMatrix values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 0.5 0"/><feBlend in2="BackgroundImageFix" result="effect1_dropShadow_617_621"/><feBlend in="SourceGraphic" in2="effect1_dropShadow_617_621" result="shape"/><feColorMatrix in="SourceAlpha" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/><feOffset/><feGaussianBlur stdDeviation="2"/><feComposite in2="hardAlpha" operator="arithmetic" k2="-1" k3="1"/><feColorMatrix values="0 0 0 0 1 0 0 0 0 1 0 0 0 0 1 0 0 0 0.5 0"/><feBlend in2="shape" result="effect2_innerShadow_617_621"/></filter></defs></g><g transform="translate(25 44)"><mask id="faceRound01-a" style="mask-type:luminance" maskUnits="userSpaceOnUse" x="0" y="0" width="130" height="120"><path fill-rule="evenodd" clip-rule="evenodd" d="M66 0c58.35 0 64 40.69 64 78 0 33.31-25.47 42-64 42-37.46 0-66-8.69-66-42C0 40.69 7.65 0 66 0Z" fill="#fff"/></mask><g mask="url(#faceRound01-a)"><path d="M-4-2h138v124H-4V-2Z" fill="#546e7a"/><g transform="translate(-1 -1)"></g></g></g><g transform="translate(52 124)"><g fill="#000" fill-opacity=".6"><rect x="12" y="12" width="4" height="8" rx="2"/><rect x="36" y="12" width="4" height="8" rx="2"/><rect x="24" y="12" width="4" height="8" rx="2"/><rect x="48" y="12" width="4" height="8" rx="2"/><rect x="60" y="12" width="4" height="8" rx="2"/></g></g><g transform="translate(38 76)"><rect y="11" width="104" height="34" rx="17" fill="#000" fill-opacity=".8"/><circle cx="29" cy="28" r="13" fill="#F1EEDA"/><circle cx="75" cy="28" r="13" fill="#F1EEDA"/><rect x="24" y="23" width="10" height="10" rx="2" fill="#000" fill-opacity=".8"/><rect x="70" y="23" width="10" height="10" rx="2" fill="#000" fill-opacity=".8"/></g></g></svg>
|
||||||
|
After Width: | Height: | Size: 4.1 KiB |
|
After Width: | Height: | Size: 4.2 KiB |
|
After Width: | Height: | Size: 4.2 KiB |
|
After Width: | Height: | Size: 3.9 KiB |
|
After Width: | Height: | Size: 3.9 KiB |
|
After Width: | Height: | Size: 3.8 KiB |
|
After Width: | Height: | Size: 3.7 KiB |
|
After Width: | Height: | Size: 3.9 KiB |
|
After Width: | Height: | Size: 3.9 KiB |
|
After Width: | Height: | Size: 3.8 KiB |
|
After Width: | Height: | Size: 3.7 KiB |
@@ -0,0 +1,83 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="400" height="400" viewBox="0 0 400 400">
|
||||||
|
<defs>
|
||||||
|
<radialGradient id="skinGrad" cx="50%" cy="40%">
|
||||||
|
<stop offset="0%" stop-color="#FDE8D0"/>
|
||||||
|
<stop offset="100%" stop-color="#F0D0A8"/>
|
||||||
|
</radialGradient>
|
||||||
|
<radialGradient id="hairGrad" cx="50%" cy="80%">
|
||||||
|
<stop offset="0%" stop-color="#4A3728"/>
|
||||||
|
<stop offset="100%" stop-color="#2D1B1E"/>
|
||||||
|
</radialGradient>
|
||||||
|
<radialGradient id="eyeWhite" cx="50%" cy="40%">
|
||||||
|
<stop offset="0%" stop-color="#FFFFFF"/>
|
||||||
|
<stop offset="100%" stop-color="#F0F0F0"/>
|
||||||
|
</radialGradient>
|
||||||
|
<radialGradient id="irisGrad" cx="40%" cy="35%">
|
||||||
|
<stop offset="0%" stop-color="#6B4E37"/>
|
||||||
|
<stop offset="70%" stop-color="#3D2B1F"/>
|
||||||
|
<stop offset="100%" stop-color="#1A1108"/>
|
||||||
|
</radialGradient>
|
||||||
|
<linearGradient id="shirtGrad" x1="0" y1="0" x2="0" y2="1">
|
||||||
|
<stop offset="0%" stop-color="#4B5563"/>
|
||||||
|
<stop offset="100%" stop-color="#374151"/>
|
||||||
|
</linearGradient>
|
||||||
|
</defs>
|
||||||
|
|
||||||
|
<!-- Background -->
|
||||||
|
<circle cx="200" cy="200" r="200" fill="#F0F2F5"/>
|
||||||
|
|
||||||
|
<!-- Hair back -->
|
||||||
|
<ellipse cx="200" cy="120" rx="120" ry="100" fill="url(#hairGrad)"/>
|
||||||
|
|
||||||
|
<!-- Neck -->
|
||||||
|
<rect x="165" y="270" width="70" height="50" rx="10" fill="url(#skinGrad)"/>
|
||||||
|
|
||||||
|
<!-- Shirt -->
|
||||||
|
<ellipse cx="200" cy="350" rx="140" ry="80" fill="url(#shirtGrad)"/>
|
||||||
|
<path d="M165 300 L200 340 L235 300" fill="none" stroke="#FFFFFF" stroke-width="2" opacity="0.3"/>
|
||||||
|
|
||||||
|
<!-- Face -->
|
||||||
|
<ellipse cx="200" cy="195" rx="100" ry="120" fill="url(#skinGrad)"/>
|
||||||
|
|
||||||
|
<!-- Hair front -->
|
||||||
|
<path d="M100 160 Q100 80 200 70 Q300 80 300 160 Q280 130 200 125 Q120 130 100 160 Z" fill="url(#hairGrad)"/>
|
||||||
|
<path d="M95 170 Q90 130 120 110 Q110 160 95 170 Z" fill="url(#hairGrad)"/>
|
||||||
|
<path d="M305 170 Q310 130 280 110 Q290 160 305 170 Z" fill="url(#hairGrad)"/>
|
||||||
|
|
||||||
|
<!-- Left eyebrow -->
|
||||||
|
<path d="M135 155 Q150 145 170 148" fill="none" stroke="#3D2B1F" stroke-width="3" stroke-linecap="round"/>
|
||||||
|
|
||||||
|
<!-- Right eyebrow -->
|
||||||
|
<path d="M265 155 Q250 145 230 148" fill="none" stroke="#3D2B1F" stroke-width="3" stroke-linecap="round"/>
|
||||||
|
|
||||||
|
<!-- Left eye -->
|
||||||
|
<ellipse cx="158" cy="175" rx="24" ry="18" fill="url(#eyeWhite)" stroke="#D4C4B0" stroke-width="1"/>
|
||||||
|
<circle cx="158" cy="175" r="12" fill="url(#irisGrad)"/>
|
||||||
|
<circle cx="158" cy="175" r="6" fill="#0A0A0A"/>
|
||||||
|
<circle cx="153" cy="170" r="4" fill="#FFFFFF" opacity="0.8"/>
|
||||||
|
<circle cx="163" cy="178" r="2" fill="#FFFFFF" opacity="0.5"/>
|
||||||
|
|
||||||
|
<!-- Right eye -->
|
||||||
|
<ellipse cx="242" cy="175" rx="24" ry="18" fill="url(#eyeWhite)" stroke="#D4C4B0" stroke-width="1"/>
|
||||||
|
<circle cx="242" cy="175" r="12" fill="url(#irisGrad)"/>
|
||||||
|
<circle cx="242" cy="175" r="6" fill="#0A0A0A"/>
|
||||||
|
<circle cx="237" cy="170" r="4" fill="#FFFFFF" opacity="0.8"/>
|
||||||
|
<circle cx="247" cy="178" r="2" fill="#FFFFFF" opacity="0.5"/>
|
||||||
|
|
||||||
|
<!-- Nose -->
|
||||||
|
<path d="M195 195 Q190 220 185 228 Q195 232 200 232 Q205 232 215 228 Q210 220 205 195" fill="none" stroke="#D4B896" stroke-width="1.5" stroke-linecap="round"/>
|
||||||
|
<ellipse cx="200" cy="230" rx="6" ry="3" fill="#E8C8A8" opacity="0.6"/>
|
||||||
|
|
||||||
|
<!-- Mouth -->
|
||||||
|
<path d="M178 260 Q189 268 200 268 Q211 268 222 260" fill="none" stroke="#C97B84" stroke-width="2.5" stroke-linecap="round"/>
|
||||||
|
|
||||||
|
<!-- Cheek blush -->
|
||||||
|
<ellipse cx="125" cy="225" rx="22" ry="14" fill="#FFB0A0" opacity="0.25"/>
|
||||||
|
<ellipse cx="275" cy="225" rx="22" ry="14" fill="#FFB0A0" opacity="0.25"/>
|
||||||
|
|
||||||
|
<!-- Ear left -->
|
||||||
|
<ellipse cx="100" cy="185" rx="12" ry="20" fill="url(#skinGrad)"/>
|
||||||
|
|
||||||
|
<!-- Ear right -->
|
||||||
|
<ellipse cx="300" cy="185" rx="12" ry="20" fill="url(#skinGrad)"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 3.6 KiB |