sass警告

This commit is contained in:
xuxin
2026-06-02 14:21:13 +08:00
parent dde72be9de
commit 557a6f30f2
19 changed files with 354 additions and 49 deletions
+2
View File
@@ -23,3 +23,5 @@ inclusion: always
## 注意事项
- 页面结构和ts的常量变量和方法都要加中文注释
- 新建 SCSS 文件如果使用了 variables.scss 中的变量(如 `$bg-white``$accent` 等),必须在文件顶部加 `@use '../variables' as *;`,否则通过 `@use` 方式引入 index.scss 时变量作用域不会穿透,会报 `Undefined variable` 错误
- 因为项目用了 rem(1rem=100px)适配方案,所有 Vue 页面和组件文件的最外层盒子都要加 `font-size: 0.14rem`,避免页面样式中受浏览器默认 rem 行高影响导致文字和布局异常
+1
View File
@@ -49,6 +49,7 @@ declare module 'vue' {
JobResumeCustomEditPanel: typeof import('./src/components/JobResumeCustomEditPanel.vue')['default']
JobResumeTemplate: typeof import('./src/components/JobResumeTemplate.vue')['default']
LoginDialog: typeof import('./src/components/LoginDialog.vue')['default']
MemberAccessDialog: typeof import('./src/components/MemberAccessDialog.vue')['default']
MemberDialog: typeof import('./src/components/MemberDialog.vue')['default']
ProfileEditDrawer: typeof import('./src/components/ProfileEditDrawer.vue')['default']
ProfilePageContent: typeof import('./src/components/ProfilePageContent.vue')['default']
+17 -3
View File
@@ -1,4 +1,5 @@
import request from '@/utils/request'
import store from '@/stores'
/** 通用响应结构 */
export interface ApiResult<T = any> {
@@ -31,13 +32,26 @@ export function sendSmsCode(mobileNumber: string) {
* POST /public/login/smsLogin
* Body: { mobileNumber, code, inviteCode? }
* 登录成功后后端会 Set-Cookie: Token=xxx
*
* inviteCode 优先使用传入参数,其次从全局 store 中读取(URL 邀请码)
* 登录成功后自动清空全局邀请码,避免重复发送
*/
export function smsLogin(mobileNumber: string, code: string, inviteCode?: string) {
return request.post<any, ApiResult<LoginData>>('/public/login/smsLogin', {
export async function smsLogin(mobileNumber: string, code: string, inviteCode?: string) {
// 优先使用显式传入的邀请码,否则从全局 store 取
const finalInviteCode = inviteCode || store.state.inviteCode || ''
const res = await request.post<any, ApiResult<LoginData>>('/public/login/smsLogin', {
mobileNumber,
code,
...(inviteCode ? { inviteCode } : {}),
...(finalInviteCode ? { inviteCode: finalInviteCode } : {}),
})
// 登录成功后清空全局邀请码
if (res.code === '0' && store.state.inviteCode) {
store.commit('SET_INVITE_CODE', '')
}
return res
}
/**
+65 -19
View File
@@ -1,15 +1,35 @@
import request from '@/utils/request'
import type { ApiResult } from '@/api/auth'
/**
* 获取当前用户有权限的路由列表
*
* 【mock 阶段】直接返回写死的数据,模拟后端接口
* 【对接真实接口时】把下面的 mock 数据替换成:
* return axios.get('/api/user/menus').then(res => res.data)
*
* 返回格式约定:
* path — 路由路径
* name — 路由名称(唯一标识,也用于 removeRoute
* component — 字符串,对应前端组件映射表的 key
* meta — 可选,传给路由的 meta 信息(图标、标题等)
* 路由菜单项 — 后端 /route/menu 接口返回的数据结构
*/
export interface RouteMenuVo {
/** 主键 */
id: string
/** 根节点ID */
rootId: string
/** 父级路由ID */
parentId: string
/** 路由名称(用于侧边栏显示) */
routeName: string
/** 前端路径 */
routePath: string
/** 前端组件路径(对应 componentMap 的 key */
component: string
/** 图标 key */
icon: string
/** 排序 */
sortOrder: number
/** 是否有使用权限:true=可用 false=无权限(需会员) */
accessible: boolean
/** 递归子菜单 */
children?: RouteMenuVo[]
}
/**
* 兼容旧版 MenuItemRaw 类型(供 SideNav 和 store 使用)
* 在新接口基础上补充原有字段映射
*/
export interface MenuItemRaw {
path: string
@@ -22,14 +42,40 @@ export interface MenuItemRaw {
/** 'footer' 表示该菜单项显示在底部区域而非主导航 */
position?: string
}
/** 排序字段 */
sortOrder: number
/** 是否有使用权限 */
accessible: boolean
}
export async function fetchUserRoutes(): Promise<MenuItemRaw[]> {
// TODO: 替换为真实接口 → return axios.get('/api/user/menus').then(res => res.data)
return [
{ path: '/resume', name: 'Resume', component: 'Resume', meta: { label: '简历', icon: 'nav-resume-icon' } },
{ path: '/profile', name: 'Profile', component: 'Profile', meta: { label: '个人资料', icon: 'nav-profile-icon' } },
{ path: '/agent', name: 'Agent', component: 'Agent', meta: { label: 'AI助手', icon: 'nav-agent-icon', badge: 'NEW' } },
{ path: '/settings', name: 'Settings', component: 'Settings', meta: { label: '设置', icon: 'nav-setting-icon', position: 'footer' } },
]
/**
* 将后端返回的 RouteMenuVo 转换为前端使用的 MenuItemRaw
*/
function mapRouteMenuToMenuItem(item: RouteMenuVo): MenuItemRaw {
return {
path: item.routePath,
name: item.component, // component 作为路由 name(与 componentMap 对应)
component: item.component,
meta: {
label: item.routeName,
icon: item.icon,
},
sortOrder: item.sortOrder,
accessible: item.accessible,
}
}
/**
* 获取当前用户有权限的路由菜单列表
* GET /route/menu
* Cookie 自动携带 Token
*/
export async function fetchUserRoutes(): Promise<MenuItemRaw[]> {
const res = await request.get<any, ApiResult<RouteMenuVo[]>>('/route/menu')
if (res.code === '0' && res.data) {
// 按 sortOrder 排序后转换格式
const sorted = [...res.data].sort((a, b) => a.sortOrder - b.sortOrder)
return sorted.map(mapRouteMenuToMenuItem)
}
return []
}
@@ -0,0 +1,77 @@
/* 会员权限拦截弹窗样式 */
@use '../variables' as *;
.member-access-dialog-overlay {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: $overlay-bg;
z-index: 9999;
display: flex;
align-items: center;
justify-content: center;
font-size: 0.14rem;
}
.member-access-dialog {
background: $bg-white;
border-radius: 0.12rem;
padding: 0.4rem 0.36rem 0.3rem;
width: 3.8rem;
text-align: center;
box-shadow: 0 0.04rem 0.2rem rgba(0, 0, 0, 0.15);
}
/* 锁图标 */
.member-access-dialog__icon {
margin-bottom: 0.2rem;
}
.member-access-dialog__lock-svg {
width: 0.48rem;
height: 0.48rem;
color: $accent;
}
/* 提示文字 */
.member-access-dialog__text {
font-size: 0.15rem;
color: $text-dark;
line-height: 1.6;
margin-bottom: 0.3rem;
}
/* 操作按钮区域 */
.member-access-dialog__actions {
display: flex;
gap: 0.16rem;
justify-content: center;
}
.member-access-dialog__btn {
flex: 1;
height: 0.4rem;
border-radius: 0.06rem;
font-size: 0.14rem;
cursor: pointer;
border: none;
transition: opacity 0.2s;
&:hover {
opacity: 0.85;
}
}
/* 灰色取消按钮 */
.member-access-dialog__btn--cancel {
background: $bg-main;
color: $text-middle;
}
/* 主题色确认按钮 */
.member-access-dialog__btn--confirm {
background: $btn-dark;
color: $bg-white;
}
@@ -1,3 +1,4 @@
@use 'sass:color';
@use '../variables' as *;
// ==================== 欢迎使用弹窗 ====================
@@ -56,12 +57,12 @@
margin-bottom: 0.14rem;
&:hover {
background: darken(#F3F4F5, 3%);
background: color.adjust(#F3F4F5, $lightness: -3%);
}
// 拖拽悬停时背景色变化
&.is-dragover {
background: darken(#F3F4F5, 6%);
background: color.adjust(#F3F4F5, $lightness: -6%);
}
}
@@ -1,3 +1,4 @@
@use 'sass:color';
@use '../variables' as *;
// ==================== 简历上传弹窗 ====================
@@ -83,7 +84,7 @@
// 拖拽悬停状态 — 背景色加深
&.is-dragover {
background: darken(#F6FCFC, 4%);
background: color.adjust(#F6FCFC, $lightness: -4%);
border-color: $accent;
}
@@ -1,3 +1,4 @@
@use 'sass:color';
@use '../variables' as *;
// ==================== 注销账号弹窗 ====================
@@ -247,7 +248,7 @@
color: $bg-white;
&:hover:not(:disabled) {
background: darken(#DC2626, 8%);
background: color.adjust(#DC2626, $lightness: -8%);
}
}
@@ -387,7 +388,7 @@
color: $bg-white;
&:hover {
background: darken(#DC2626, 8%);
background: color.adjust(#DC2626, $lightness: -8%);
}
}
}
+1
View File
@@ -37,6 +37,7 @@
@use './components/settings-invite-dialog.scss';
@use './components/resume-upload-dialog.scss';
@use './components/agreement-preview-dialog.scss';
@use './components/member-access-dialog.scss';
// 全局样式(优先级最高)
@use './auto.scss';
+51
View File
@@ -0,0 +1,51 @@
<template>
<!-- 会员权限拦截弹窗 无权限页面点击时弹出 -->
<Teleport to="body">
<div v-if="modelValue" class="member-access-dialog-overlay" @click="$emit('update:modelValue', false)">
<div class="member-access-dialog" @click.stop>
<!-- 提示图标 -->
<div class="member-access-dialog__icon">
<svg viewBox="0 0 48 48" fill="none" class="member-access-dialog__lock-svg">
<rect x="10" y="22" width="28" height="20" rx="3" stroke="currentColor" stroke-width="2.5"/>
<path d="M16 22V16a8 8 0 1116 0v6" stroke="currentColor" stroke-width="2.5" stroke-linecap="round"/>
<circle cx="24" cy="33" r="3" fill="currentColor"/>
</svg>
</div>
<!-- 提示文字 -->
<p class="member-access-dialog__text">该功能仅对会员开放升级会员享受更强大的AI求职功能</p>
<!-- 操作按钮 -->
<div class="member-access-dialog__actions">
<button class="member-access-dialog__btn member-access-dialog__btn--cancel" @click="$emit('update:modelValue', false)">
先不升级
</button>
<button class="member-access-dialog__btn member-access-dialog__btn--confirm" @click="handleViewDetail">
了解详情
</button>
</div>
</div>
</div>
</Teleport>
</template>
<script setup lang="ts">
/**
* 会员权限拦截弹窗
* 当用户点击无权限(accessible=false)的菜单时弹出
* "了解详情"按钮打开会员购买组件 MemberDialog
*/
/** 组件 Props */
defineProps<{ modelValue: boolean }>()
/** 组件 Emits */
const emit = defineEmits<{
(e: 'update:modelValue', value: boolean): void
(e: 'openMember'): void
}>()
/** 点击了解详情 — 关闭当前弹窗,通知父组件打开会员购买弹窗 */
function handleViewDetail() {
emit('update:modelValue', false)
emit('openMember')
}
</script>
+1 -1
View File
@@ -92,7 +92,7 @@ const inviteCode = computed(() => store.state.userInfo?.inviteCode || '')
/** 邀请链接文案 */
const inviteText = computed(() => {
const code = inviteCode.value
return `https://www.offerpai.com.cn/invite_code=${code}`
return `https://www.offerpai.com.cn?invite_code=${code}`
})
/** 复制链接到剪贴板 */
+36 -3
View File
@@ -123,6 +123,10 @@
<SettingsDialog v-model="showSettingsDialog" :initial-tab="store.state.settingsTab" />
<!-- 邀请注册送会员弹窗 -->
<SettingsInviteDialog v-model="showShareDialog" />
<!-- 会员权限拦截弹窗 -->
<MemberAccessDialog v-model="showMemberAccessDialog" @open-member="showMemberDialog = true" />
<!-- 会员购买弹窗 -->
<MemberDialog v-model="showMemberDialog" />
</div>
</template>
@@ -131,6 +135,8 @@ import { computed, ref, watch, onMounted } from 'vue'
import { useRoute, useRouter } from 'vue-router'
import { useStore } from 'vuex'
import SettingsDialog from '@/components/SettingsDialog.vue'
import MemberAccessDialog from '@/components/MemberAccessDialog.vue'
import MemberDialog from '@/components/MemberDialog.vue'
import { checkLogin } from '@/api/auth'
import { fetchMessageList, fetchUnreadCount, markMessageRead } from '@/api/message'
import { timestampToLocalDateTime } from '@/utils/time'
@@ -155,6 +161,10 @@ interface MenuItem {
iconImg: string
label: string
badge?: string
/** 排序字段 */
sortOrder: number
/** 是否有使用权限 */
accessible: boolean
}
/**
@@ -173,15 +183,17 @@ const iconMap: Record<string, string> = {
/**
* 静态菜单 — 不需要登录就能看到的导航项(比如"职位")
* 这些菜单始终显示,不依赖后端返回
* sortOrder 设为 0 确保在未获取到后端数据时也能正常显示
*/
const staticMenus: MenuItem[] = [
{ name: 'Jobs', path: '/jobs', iconImg: navJobsIcon, label: '职位' },
{ name: 'Jobs', path: '/jobs', iconImg: navJobsIcon, label: '职位', sortOrder: 0, accessible: true },
]
/**
* 动态菜单 — 从 store.state.dynamicMenus(后端返回的数据)转换而来
* 登录后才会有数据,登出后自动清空
* 过滤掉 position === 'footer' 的项(如"设置"),它们显示在底部区域
* 按 sortOrder 排序,routeNamemeta.label)作为显示名称
*/
const dynamicMenuItems = computed<MenuItem[]>(() => {
return store.state.dynamicMenus
@@ -192,19 +204,30 @@ const dynamicMenuItems = computed<MenuItem[]>(() => {
iconImg: iconMap[item.meta?.icon] || '',
label: item.meta?.label || item.name,
badge: item.meta?.badge,
sortOrder: item.sortOrder ?? 99,
accessible: item.accessible !== false, // 默认为 true
}))
.sort((a: MenuItem, b: MenuItem) => a.sortOrder - b.sortOrder)
})
/**
* 最终渲染的主菜单 = 静态菜单 + 动态菜单
* 最终渲染的主菜单 = 静态菜单 + 动态菜单,按 sortOrder 统一排序
* 如果后端返回了 Jobs 的数据,用后端数据覆盖静态菜单的 sortOrder
*/
const mainMenus = computed<MenuItem[]>(() => {
return [...staticMenus, ...dynamicMenuItems.value]
// 动态菜单中已包含后端返回的 Jobs(如果有的话),此时不再使用静态的
const dynamicNames = dynamicMenuItems.value.map(m => m.name)
const filteredStatic = staticMenus.filter(s => !dynamicNames.includes(s.name))
return [...filteredStatic, ...dynamicMenuItems.value].sort((a, b) => a.sortOrder - b.sortOrder)
})
const showShareDialog = ref(false)
const showMessageDialog = ref(false)
const showFeedbackDialog = ref(false)
/** 会员权限拦截弹窗 */
const showMemberAccessDialog = ref(false)
/** 会员购买弹窗 */
const showMemberDialog = ref(false)
const showSettingsDialog = computed({
get: () => store.state.showSettings,
set: (val: boolean) => store.commit('SET_SHOW_SETTINGS', val),
@@ -420,15 +443,25 @@ async function handleSettingsNav() {
/**
* 导航点击处理:
* - 静态页面(Jobs)直接跳转
* - accessible 为 false 的菜单 → 弹出会员权限拦截弹窗
* - 动态页面通过 checkLogin 接口验证,未登录则弹登录框
*/
const staticNames = staticMenus.map(m => m.name)
async function handleNav(item: MenuItem) {
// 无权限菜单 — 弹出会员权限拦截弹窗
if (!item.accessible) {
showMemberAccessDialog.value = true
return
}
// Jobs 等静态页面 — 无需登录验证,直接跳转
if (staticNames.includes(item.name)) {
router.push(item.path)
return
}
// 动态页面 — 需要登录验证
try {
const res = await checkLogin()
if (res.code === '0' && res.data === true) {
+7
View File
@@ -10,6 +10,7 @@ import type { MenuItemRaw } from '@/api/menu'
* 【新增页面时】在这里加一条映射即可,后端返回对应的 key 就能自动注册路由
*/
const componentMap: Record<string, () => Promise<any>> = {
'Jobs': () => import('@/views/Jobs.vue'),
'Agent': () => import('@/views/Agent.vue'),
'Profile': () => import('@/views/Profile.vue'),
'Resume': () => import('@/views/Resume.vue'),
@@ -19,11 +20,17 @@ const componentMap: Record<string, () => Promise<any>> = {
* 把后端返回的菜单数据转换成 vue-router 能识别的 RouteRecordRaw
*
* 如果后端返回了一个 component 字符串在映射表里找不到,会跳过该条路由并打印警告
* 已在静态路由中注册的路径(如 /jobs)不再重复注册
*/
const STATIC_PATHS = ['/', '/jobs', '/resume/:id', '/jobs/:id']
export function buildDynamicRoutes(menus: MenuItemRaw[]): RouteRecordRaw[] {
const routes: RouteRecordRaw[] = []
for (const item of menus) {
// 跳过已在静态路由中注册的路径
if (STATIC_PATHS.includes(item.path)) continue
const comp = componentMap[item.component]
if (!comp) {
console.warn(`[dynamicRoutes] 组件映射表中找不到 "${item.component}",已跳过`)
+6 -2
View File
@@ -37,10 +37,11 @@ const CHECK_INTERVAL = 5 * 60 * 1000 // 5 分钟
* 全局前置守卫 — 核心逻辑:
*
* 1. 动态路由未加载 → 先拉取并注册,再重新进入当前路由
* 2. 需要鉴权的路由 → 调 checkLogin 接口验证 Cookie 是否有效
* 2. 每次页面切换都刷新路由菜单数据(权限可能变化)
* 3. 需要鉴权的路由 → 调 checkLogin 接口验证 Cookie 是否有效
* - 有效:同步 isAuthenticated = true,放行
* - 无效:同步 isAuthenticated = false,弹登录框,阻止导航
* 3. 不需要鉴权的路由 → 静默同步登录状态(不阻止导航、不弹登录框),5 分钟内节流
* 4. 不需要鉴权的路由 → 静默同步登录状态(不阻止导航、不弹登录框),5 分钟内节流
*/
router.beforeEach(async (to, _from, next) => {
// 动态路由只需加载一次,与登录状态无关
@@ -55,6 +56,9 @@ router.beforeEach(async (to, _from, next) => {
return
}
// 每次页面切换时静默刷新路由菜单数据(更新权限状态),不阻塞导航
store.dispatch('refreshDynamicMenus')
// 需要鉴权的路由,每次都通过接口校验登录状态
if (to.meta?.requiresAuth) {
try {
+23
View File
@@ -93,6 +93,12 @@ export interface RootState {
*/
showSettings: boolean
settingsTab: string
/**
* 邀请码 — 从 URL 参数 invite_code 中提取
* 登录成功后自动清空,避免重复发送
*/
inviteCode: string
}
export default createStore<RootState>({
@@ -117,6 +123,7 @@ export default createStore<RootState>({
userInfo: null,
showSettings: false,
settingsTab: 'account',
inviteCode: '',
},
getters: {
getAppName: (state) => state.appName,
@@ -177,6 +184,9 @@ export default createStore<RootState>({
SET_SETTINGS_TAB(state, tab: string) {
state.settingsTab = tab
},
SET_INVITE_CODE(state, code: string) {
state.inviteCode = code
},
},
actions: {
updateAppName({ commit }, name: string) {
@@ -219,6 +229,19 @@ export default createStore<RootState>({
commit('SET_ROUTES_LOADED', true)
},
/**
* 静默刷新路由菜单数据(不重新注册路由,只更新菜单显示和权限状态)
* 每次页面切换时由路由守卫调用,用于更新 accessible 权限信息
*/
async refreshDynamicMenus({ commit }) {
try {
const menus = await fetchUserRoutes()
commit('SET_DYNAMIC_MENUS', menus)
} catch (err) {
console.error('[store] 刷新路由菜单数据失败', err)
}
},
/**
* 登出:重置状态(不再操作 localStorage,登录状态由 Cookie 管理)
* 注意:动态路由不清除,与登录状态无关
-14
View File
@@ -1,14 +0,0 @@
/**
* 从 document.cookie 中提取指定 name 的值
*/
export function getCookie(name: string): string | null {
const match = document.cookie.match(new RegExp(`(?:^|;\\s*)${name}=([^;]*)`))
return match ? decodeURIComponent(match[1]) : null
}
/**
* 判断是否已登录 — 通过 Cookie 中是否存在 Token 来判断
*/
export function isLoggedIn(): boolean {
return !!getCookie('Token')
}
+7
View File
@@ -566,6 +566,13 @@ function goSearchJobs() {
}
onMounted(() => {
// 检测 URL 中的邀请码参数 invite_code,存入全局状态
const urlParams = new URLSearchParams(window.location.search)
const inviteCode = urlParams.get('invite_code')
if (inviteCode && inviteCode.length === 10) {
store.commit('SET_INVITE_CODE', inviteCode)
}
// 加载公共工具数据(行业分类、岗位分类、地区分类等)
store.dispatch('loadCommonData')
})
+26 -1
View File
@@ -276,6 +276,11 @@
<!-- 岗位专属简历定制弹窗 -->
<JobResumeCustomDialog v-model="showResumeCustomDialog" :job-info="resumeCustomJobInfo" :job-id="jobId" @skip="handleSkipToApply" />
<!-- 会员权限拦截弹窗 -->
<MemberAccessDialog v-model="showMemberAccessDialog" @open-member="showMemberDialog = true" />
<!-- 会员购买弹窗 -->
<MemberDialog v-model="showMemberDialog" />
</div>
</template>
@@ -289,7 +294,10 @@ import JobPageHeader from '@/components/JobPageHeader.vue'
import JobDislikeDialog from '@/components/JobDislikeDialog.vue'
import JobFeedbackDialog from '@/components/JobFeedbackDialog.vue'
import JobResumeCustomDialog from '@/components/JobResumeCustomDialog.vue'
import MemberAccessDialog from '@/components/MemberAccessDialog.vue'
import MemberDialog from '@/components/MemberDialog.vue'
import { fetchJobDetail, toggleJobFavorite, removeJobFavorite, fetchSkillGap } from '@/api/jobs'
import { fetchMemberStatus } from '@/api/member'
import type { JobDetailData, SkillGapData } from '@/api/jobs'
// ==================== 路由相关 ====================
@@ -554,6 +562,11 @@ function handleReport() {
/** 简历定制弹窗显隐 */
const showResumeCustomDialog = ref(false)
/** 会员权限拦截弹窗 */
const showMemberAccessDialog = ref(false)
/** 会员购买弹窗 */
const showMemberDialog = ref(false)
/** 技能差距分析数据 */
const skillGapData = ref<SkillGapData | null>(null)
@@ -571,8 +584,20 @@ const resumeCustomJobInfo = computed(() => ({
defaultResume: skillGapData.value?.resume || null,
}))
/** 生成岗位专属简历 — 调用 skill-gap 接口后打开定制弹窗 */
/** 生成岗位专属简历 — 先检查会员状态,非会员弹权限拦截弹窗 */
async function handleGenerateResume() {
// 先检查会员状态
try {
const statusRes = await fetchMemberStatus()
if (statusRes.code === '0' && statusRes.data && !statusRes.data.isMember) {
// 非会员 — 弹出会员权限拦截弹窗
showMemberAccessDialog.value = true
return
}
} catch {
// 接口异常时不阻断,继续执行
}
const loadingInstance = ElLoading.service({
text: '正在分析岗位匹配度...',
background: 'rgba(0, 0, 0, 0.5)',
+26 -1
View File
@@ -412,6 +412,11 @@
</div>
</div>
</div>
<!-- 会员权限拦截弹窗 -->
<MemberAccessDialog v-model="showMemberAccessDialog" @open-member="showMemberDialog = true" />
<!-- 会员购买弹窗 -->
<MemberDialog v-model="showMemberDialog" />
</div>
</template>
@@ -424,6 +429,9 @@ import ResumeAnalysisReportDrawer from '@/components/ResumeAnalysisReportDrawer.
import ResumeIssueFixDrawer from '@/components/ResumeIssueFixDrawer.vue'
import ResumeExportDialog from '@/components/ResumeExportDialog.vue'
import ResumeEditNameDialog from '@/components/ResumeEditNameDialog.vue'
import MemberAccessDialog from '@/components/MemberAccessDialog.vue'
import MemberDialog from '@/components/MemberDialog.vue'
import { fetchMemberStatus } from '@/api/member'
import {
fetchResumeMain,
fetchResumeEducation,
@@ -463,6 +471,11 @@ import {
// ==================== 诊断数据相关 ====================
/** 会员权限拦截弹窗 */
const showMemberAccessDialog = ref(false)
/** 会员购买弹窗 */
const showMemberDialog = ref(false)
/** 问题类型枚举值 */
type IssueType = 'urgent' | 'optimize' | 'expression'
@@ -748,8 +761,20 @@ function handleClose_report() {
showReportDrawer.value = false
}
/** 重新诊断 / 开始诊断 — 调用 AI 诊断接口,完成后刷新诊断数据 */
/** 重新诊断 / 开始诊断 — 先检查会员状态,非会员弹权限拦截弹窗 */
async function handleDiagnose() {
// 先检查会员状态
try {
const statusRes = await fetchMemberStatus()
if (statusRes.code === '0' && statusRes.data && !statusRes.data.isMember) {
// 非会员 — 弹出会员权限拦截弹窗
showMemberAccessDialog.value = true
return
}
} catch {
// 接口异常时不阻断(可能是网络问题),继续执行诊断
}
// 全屏加载提示,AI 接口响应较慢
const loading = ElLoading.service({
lock: true,