Files
zhiyin/zhiyin-app/src/pages/report/report.vue
T
2026-06-16 13:18:36 +08:00

372 lines
14 KiB
Vue

<template>
<view class="page">
<view class="header" v-if="!loading && report">
<text class="report-title">面试报告</text>
<text class="report-position">{{ report.position }}</text>
</view>
<view v-if="loading" class="loading-box"><text>加载中...</text></view>
<view v-else-if="report" class="body">
<view class="score-card">
<view class="score-circle" :class="scoreLevel(report.totalScore)">
<text class="score-num">{{ report.totalScore }}</text>
<text class="score-label">总分</text>
</view>
</view>
<view class="info-row">
<text class="info-label">面试岗位</text>
<text class="info-value">{{ report.position }}</text>
</view>
<view class="info-row">
<text class="info-label">面试题数</text>
<text class="info-value">{{ report.questionCount }} </text>
</view>
<view class="section" v-if="report.dimensions">
<view class="section-title">📊 四维能力评估</view>
<view class="dim-grid">
<view class="dim-item" v-for="dim in dimList" :key="dim.key">
<view class="dim-header">
<text class="dim-name">{{ dim.label }}</text>
<text class="dim-score">{{ dim.value }}</text>
</view>
<view class="dim-bar-bg">
<view class="dim-bar-fill" :style="{ width: dim.value + '%', background: dim.color }"></view>
</view>
</view>
</view>
</view>
<view class="section" v-if="report.summary">
<view class="section-title">📝 评估总结</view>
<text class="summary-text">{{ report.summary }}</text>
</view>
<view class="section">
<view class="section-title">💬 完整对话</view>
<view class="msg-list">
<view v-for="(msg, idx) in report.messages" :key="idx" class="msg-item" :class="msg.role">
<view class="msg-label">{{ msg.role === 'ai' ? '面试官' : '你' }}</view>
<text class="msg-content">{{ msg.content }}</text>
</view>
</view>
</view>
<view class="actions">
<button class="btn-primary" @click="generateCard">📸 生成分享卡片</button>
<button class="btn-outline" @click="retryInterview">再面一次</button>
<button class="btn-ghost" @click="goHistory">返回记录</button>
</view>
</view>
<view v-else class="empty-box"><text>暂无报告数据</text></view>
<!-- Share card canvas (hidden) -->
<canvas canvas-id="shareCard" class="hidden-canvas" :style="{ width: cardWidth + 'px', height: cardHeight + 'px' }"></canvas>
</view>
</template>
<script setup>
import { ref, nextTick } from 'vue'
import { onLoad } from '@dcloudio/uni-app'
import { api } from '../../config'
const loading = ref(true)
const report = ref(null)
const dimList = ref([])
const cardWidth = 600
const cardHeight = 900
const dimDefs = [
{ key: 'logic', label: '逻辑思维', color: '#6366F1' },
{ key: 'expression', label: '表达能力', color: '#10B981' },
{ key: 'professionalism', label: '专业度', color: '#F59E0B' },
{ key: 'stability', label: '稳定性', color: '#EF4444' },
]
onLoad(async (options) => {
const interviewId = options?.interviewId || ''
if (!interviewId) { loading.value = false; return }
try {
const token = uni.getStorageSync('token') || ''
if (!token) { loading.value = false; return }
const res = await uni.request({
url: api(`/interview/${interviewId}`),
method: 'GET',
header: { 'Authorization': `Bearer ${token}` },
})
if (res.statusCode === 200) {
const data = res.data
const rawSummary = data.summary || ''
// Try to parse summary JSON for structured display
let summaryText = rawSummary
let dimensions = null
try {
const parsed = JSON.parse(rawSummary)
dimensions = {
logic: parsed['逻辑思维'] || 0,
expression: parsed['表达能力'] || 0,
professionalism: parsed['专业度'] || 0,
stability: parsed['稳定性'] || 0,
}
const parts = []
if (parsed['优点']) parts.push('✅ 优点:' + parsed['优点'].join('、'))
if (parsed['不足']) parts.push('⚠️ 不足:' + parsed['不足'].join('、'))
if (parsed['建议']) parts.push('💡 建议:' + parsed['建议'].join('、'))
summaryText = parts.join('\n')
} catch {}
// Use backend dimensions if available, else parsed
const dims = data.dimensions || dimensions
if (dims) {
dimList.value = dimDefs.map(d => ({
...d,
value: Math.min(100, Math.max(0, Math.round(dims[d.key] || 0))),
}))
}
report.value = {
position: data.position || '通用岗位',
totalScore: data.totalScore || 0,
questionCount: data.questionCount || 0,
summary: summaryText,
messages: data.messages || [],
dimensions: dims,
interviewId,
}
if (data.status === 'in_progress') {
uni.request({
url: api(`/interview/${interviewId}/complete`),
method: 'POST',
header: { 'Authorization': `Bearer ${token}` },
}).then(c => {
if (c.statusCode === 200 && c.data) {
report.value.totalScore = c.data.totalScore || report.value.totalScore
report.value.summary = c.data.summary || report.value.summary
if (c.data.dimensions) {
dimList.value = dimDefs.map(d => ({
...d,
value: Math.min(100, Math.max(0, Math.round(c.data.dimensions[d.key] || 0))),
}))
}
}
}).catch(e => {
console.error('[report] auto-complete failed:', e)
})
}
}
} catch(e) { console.error(e) }
finally { loading.value = false }
})
const scoreLevel = (s) => { if (s >= 80) return 'good'; if (s >= 60) return 'medium'; return 'poor' }
const retryInterview = () => uni.switchTab({ url: '/pages/index/index' })
const goHistory = () => uni.switchTab({ url: '/pages/history/history' })
async function generateCard() {
if (!report.value) return
uni.showLoading({ title: '生成中...' })
await nextTick()
const ctx = uni.createCanvasContext('shareCard')
const w = cardWidth
const h = cardHeight
const r = dimList.value
const total = report.value.totalScore
const pos = report.value.position
const date = new Date().toLocaleDateString('zh-CN')
// Background gradient
const gradient = ctx.createLinearGradient(0, 0, 0, h)
gradient.addColorStop(0, '#1E1B4B')
gradient.addColorStop(0.5, '#312E81')
gradient.addColorStop(1, '#1E1B4B')
ctx.setFillStyle(gradient)
ctx.fillRect(0, 0, w, h)
// Border decoration
ctx.setStrokeStyle('#4F46E5')
ctx.setLineWidth(4)
ctx.strokeRect(16, 16, w - 32, h - 32)
// Title
ctx.setFillStyle('#FFFFFF')
ctx.setFontSize(36)
ctx.setTextAlign('center')
ctx.fillText('🎯 模拟面试报告', w / 2, 80)
// Position
ctx.setFontSize(24)
ctx.setFillStyle('#A5B4FC')
ctx.fillText(pos + ' | ' + date, w / 2, 120)
// Total score circle
const cx = w / 2
const cy = 220
const radius = 72
// Glow effect
ctx.setStrokeStyle('rgba(99,102,241,0.3)')
ctx.setLineWidth(16)
ctx.beginPath()
ctx.arc(cx, cy, radius + 10, 0, Math.PI * 2)
ctx.stroke()
// Circle background
ctx.setFillStyle('#2D2A6E')
ctx.beginPath()
ctx.arc(cx, cy, radius, 0, Math.PI * 2)
ctx.fill()
// Total score text
ctx.setFillStyle(total >= 80 ? '#34D399' : total >= 60 ? '#FBBF24' : '#F87171')
ctx.setFontSize(52)
ctx.setTextAlign('center')
ctx.fillText(String(total), cx, cy - 8)
ctx.setFontSize(16)
ctx.setFillStyle('#A5B4FC')
ctx.fillText('总分', cx, cy + 34)
// Dimension bars
const barColors = ['#6366F1', '#10B981', '#F59E0B', '#EF4444']
const barStartY = 330
const barH = 36
const barGap = 16
const barMaxW = 380
const barX = 130
for (let i = 0; i < r.length; i++) {
const d = r[i]
const y = barStartY + i * (barH + barGap)
const v = d.value
// Label
ctx.setFontSize(20)
ctx.setFillStyle('#A5B4FC')
ctx.setTextAlign('left')
ctx.fillText(d.label, 40, y + 26)
// Bar background
ctx.setFillStyle('rgba(255,255,255,0.08)')
ctx.beginPath()
ctx.roundRect ? ctx.roundRect(barX, y, barMaxW, barH, barH / 2) : ctx.rect(barX, y, barMaxW, barH)
ctx.fill()
// Bar fill
ctx.setFillStyle(barColors[i])
ctx.beginPath()
const fillW = Math.max(0, (v / 100) * barMaxW)
ctx.roundRect ? ctx.roundRect(barX, y, fillW, barH, barH / 2) : ctx.rect(barX, y, fillW, barH)
ctx.fill()
// Score text
ctx.setFontSize(20)
ctx.setFillStyle('#FFFFFF')
ctx.setTextAlign('right')
ctx.fillText(v + '分', w - 40, y + 26)
}
// Divider line
ctx.setStrokeStyle('rgba(165,180,252,0.3)')
ctx.setLineWidth(1)
ctx.beginPath()
ctx.moveTo(40, 600)
ctx.lineTo(w - 40, 600)
ctx.stroke()
// Bottom info
ctx.setFontSize(20)
ctx.setFillStyle('#A5B4FC')
ctx.setTextAlign('center')
ctx.fillText('「职引」- AI 模拟面试助手', w / 2, 650)
ctx.setFontSize(16)
ctx.setFillStyle('rgba(165,180,252,0.5)')
ctx.fillText('扫码开始你的模拟面试 → 在微信搜索"职引"小程序', w / 2, 690)
// QR text hint
ctx.setFillStyle('#FFFFFF')
ctx.setFontSize(16)
ctx.setTextAlign('center')
ctx.fillText('在微信搜索「职引」小程序', w / 2, 760)
ctx.fillText('查看完整面试报告', w / 2, 790)
ctx.draw(false, async () => {
try {
const tempRes = await uni.canvasToTempFilePath({ canvasId: 'shareCard' })
uni.hideLoading()
uni.showActionSheet({
itemList: ['保存到相册', '分享给好友'],
success: (res) => {
if (res.tapIndex === 0) {
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' })
}
},
})
} catch (e) {
uni.hideLoading()
uni.showToast({ title: '卡片生成失败', icon: 'none' })
}
})
}
</script>
<style scoped>
.page { background: #F3F4F6; min-height: 100vh; }
.header {
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; min-height: 90rpx;
}
.report-title { font-size: 36rpx; font-weight: 700; color: #FFFFFF; display: block; }
.report-position { font-size: 24rpx; color: rgba(255,255,255,0.8); margin-top: 8rpx; display: block; }
.loading-box { padding: 200rpx 0; text-align: center; color: #9CA3AF; font-size: 28rpx; }
.body { padding: 0 32rpx 48rpx; }
.score-card { display: flex; justify-content: center; margin: -40rpx 0 30rpx; }
.score-circle {
width: 180rpx; height: 180rpx; border-radius: 50%;
background: #FFFFFF; display: flex; flex-direction: column;
align-items: center; justify-content: center;
box-shadow: 0 8rpx 30rpx rgba(79,70,229,0.2);
}
.score-circle.good { box-shadow: 0 8rpx 30rpx rgba(5,150,105,0.2); }
.score-circle.medium { box-shadow: 0 8rpx 30rpx rgba(217,119,6,0.2); }
.score-circle.poor { box-shadow: 0 8rpx 30rpx rgba(220,38,38,0.2); }
.score-num { font-size: 56rpx; font-weight: 700; color: #4F46E5; line-height: 1.2; }
.good .score-num { color: #059669; }
.medium .score-num { color: #D97706; }
.poor .score-num { color: #DC2626; }
.score-label { font-size: 20rpx; color: #9CA3AF; margin-top: 4rpx; }
.info-row { display: flex; justify-content: space-between; padding: 20rpx 0; border-bottom: 1rpx solid #E5E7EB; }
.info-label { font-size: 24rpx; color: #6B7280; }
.info-value { font-size: 24rpx; color: #111827; font-weight: 500; }
.section { background: #FFFFFF; border-radius: 20rpx; padding: 28rpx; margin-top: 24rpx; box-shadow: 0 2rpx 12rpx rgba(0,0,0,0.04); }
.section-title { font-size: 28rpx; font-weight: 600; color: #111827; margin-bottom: 16rpx; }
.summary-text { font-size: 24rpx; color: #374151; line-height: 1.8; white-space: pre-wrap; }
.msg-list { display: flex; flex-direction: column; gap: 16rpx; }
.msg-item { padding: 16rpx; border-radius: 12rpx; }
.msg-item.ai { background: #F9FAFB; border-left: 4rpx solid #4F46E5; }
.msg-item.user { background: #EEF2FF; border-left: 4rpx solid #818CF8; }
.msg-label { font-size: 20rpx; font-weight: 600; color: #4F46E5; margin-bottom: 8rpx; }
.msg-content { font-size: 24rpx; color: #111827; line-height: 1.7; white-space: pre-wrap; }
.actions { display: flex; flex-direction: column; gap: 16rpx; margin-top: 32rpx; }
.btn-primary { background: linear-gradient(135deg,#4F46E5,#7C3AED); color: #FFFFFF; border-radius: 16rpx; height: 88rpx; line-height: 88rpx; font-size: 28rpx; font-weight: 600; }
.btn-outline { background: #FFFFFF; color: #4F46E5; border-radius: 16rpx; height: 88rpx; line-height: 88rpx; font-size: 28rpx; border: 2rpx solid #4F46E5; }
.btn-ghost { background: transparent; color: #6B7280; border-radius: 16rpx; height: 72rpx; line-height: 72rpx; font-size: 24rpx; }
.empty-box { padding: 200rpx 0; text-align: center; color: #9CA3AF; font-size: 28rpx; }
.hidden-canvas { position: fixed; left: -9999px; top: -9999px; pointer-events: none; }
.dim-grid { display: flex; flex-direction: column; gap: 20rpx; }
.dim-header { display: flex; justify-content: space-between; margin-bottom: 8rpx; }
.dim-name { font-size: 24rpx; color: #374151; }
.dim-score { font-size: 24rpx; color: #4F46E5; font-weight: 600; }
.dim-bar-bg { height: 20rpx; background: #F3F4F6; border-radius: 10rpx; overflow: hidden; }
.dim-bar-fill { height: 100%; border-radius: 10rpx; transition: width 0.6s ease; }
</style>