1139 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			1139 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
| <template>
 | ||
| 	<view class="profile-page">
 | ||
| 		<!-- 移动设备顶部状态栏 -->
 | ||
| 		<view class="top_bar flex w-full" v-if="isMobile">
 | ||
| 			<view class="chat-header">
 | ||
| 				<view class="header-left">
 | ||
| 					<i class="fas fa-bell header-icon" @click="handleNotification"></i>
 | ||
| 				</view>
 | ||
| 				<view class="header-title">我的</view>
 | ||
| 				<view class="header-right">
 | ||
| 					<i class="fas fa-search header-icon" @click="handleSearch"></i>
 | ||
| 				</view>
 | ||
| 			</view>
 | ||
| 		</view>
 | ||
| 		<!-- 浏览器环境顶部导航栏 -->
 | ||
| 		<view class="unified-header" v-else>
 | ||
| 			<view class="header-content">
 | ||
| 				<view class="header-left">
 | ||
| 					<i class="fas fa-bell header-icon" @click="handleNotification"></i>
 | ||
| 				</view>
 | ||
| 				<view class="header-title">我的</view>
 | ||
| 				<view class="header-right">
 | ||
| 					<i class="fas fa-search header-icon" @click="handleSearch"></i>
 | ||
| 				</view>
 | ||
| 			</view>
 | ||
| 		</view>
 | ||
| 		
 | ||
| 		<!-- 页面内容 -->
 | ||
| 		<scroll-view scroll-y class="unified-content">
 | ||
| 			<!-- 个人信息卡片 -->
 | ||
| 			<view class="profile-card">
 | ||
| 				<!-- 已登录状态 -->
 | ||
| 				<view v-if="isLogin" class="profile-content">
 | ||
| 					<view class="profile-header" @click="editProfile">
 | ||
| 						<view class="profile-avatar">
 | ||
| 							<view class="avatar">
 | ||
| 								<text class="avatar-text">{{(userInfo.nickname || '未').charAt(0)}}</text>
 | ||
| 							</view>
 | ||
| 							<view class="edit-icon">
 | ||
| 								<i class="fas fa-camera"></i>
 | ||
| 							</view>
 | ||
| 						</view>
 | ||
| 						<view class="profile-info">
 | ||
| 							<text class="profile-name">{{userInfo.nickname || '未登录'}}</text>
 | ||
| 							<text class="profile-dept">{{userInfo.dept?.name || '未知部门'}}</text>
 | ||
| 							<text class="profile-id">工号:{{userInfo.employeeId || 'N/A'}}</text>
 | ||
| 						</view>
 | ||
| 						<view class="profile-arrow">
 | ||
| 							<i class="fas fa-chevron-right"></i>
 | ||
| 						</view>
 | ||
| 					</view>
 | ||
| 					<!-- {
 | ||
|     "id": 1,
 | ||
|     "username": "admin",
 | ||
|     "nickname": "江苏美天智能科技",
 | ||
|     "email": "admin@126.com",
 | ||
|     "mobile": "18888888888",
 | ||
|     "sex": 1,
 | ||
|     "avatar": "http://demo1.meteteme.top/admin-api/infra/file/4/get/5d6c4478d03b3f0971269e3f13ce8984221623f089248d1fb6069dbfcd8baf5e.png",
 | ||
|     "loginIp": "218.92.65.85",
 | ||
|     "loginDate": 1760610192000,
 | ||
|     "createTime": 1609837427000,
 | ||
|     "roles": [
 | ||
|         {
 | ||
|             "id": 1,
 | ||
|             "name": "超级管理员"
 | ||
|         },
 | ||
|         {
 | ||
|             "id": 2,
 | ||
|             "name": "普通角色"
 | ||
|         },
 | ||
|         {
 | ||
|             "id": 3,
 | ||
|             "name": "CRM 管理员"
 | ||
|         }
 | ||
|     ],
 | ||
|     "dept": {
 | ||
|         "id": 100,
 | ||
|         "name": "江苏美天智能科技有限公司",
 | ||
|         "parentId": 0
 | ||
|     },
 | ||
|     "posts": [
 | ||
|         {
 | ||
|             "id": 1,
 | ||
|             "name": "董事长"
 | ||
|         },
 | ||
|         {
 | ||
|             "id": 5,
 | ||
|             "name": "部门主管"
 | ||
|         }
 | ||
|     ],
 | ||
|     "socialUsers": []
 | ||
| } -->
 | ||
| 
 | ||
| 					<!-- 数据统计 -->
 | ||
| 					<view class="stats-grid">
 | ||
| 						<view class="stat-item" @click="goToAttendance">
 | ||
| 							<view class="stat-content">
 | ||
| 								<text class="stat-number">{{ userInfo.attendanceRate ?? 0 }}%</text>
 | ||
| 								<text class="stat-label">考勤率</text>
 | ||
| 							</view>
 | ||
| 						</view>
 | ||
| 						<view class="stat-item" @click="goToTasks">
 | ||
| 							<view class="stat-content">
 | ||
| 								<text class="stat-number">{{ userInfo.completedTasks ?? 0 }}</text>
 | ||
| 								<text class="stat-label">待完成</text>
 | ||
| 							</view>
 | ||
| 						</view>
 | ||
| 						<view class="stat-item" @click="goToReimbursement">
 | ||
| 							<view class="stat-content">
 | ||
| 								<text class="stat-number">¥{{ (userInfo.pendingAmount !== undefined && userInfo.pendingAmount !== null) ? userInfo.pendingAmount.toFixed(2) : '0.00' }}</text>
 | ||
| 								<text class="stat-label">待报销</text>
 | ||
| 							</view>
 | ||
| 						</view>
 | ||
| 					</view>
 | ||
| 				</view>
 | ||
| 
 | ||
| 				<!-- 未登录状态 -->
 | ||
| 				<view v-else class="login-prompt">
 | ||
| 					<view class="login-icon">
 | ||
| 						<i class="fas fa-user-circle"></i>
 | ||
| 					</view>
 | ||
| 					<view class="login-content">
 | ||
| 						<text class="login-title">请先登录</text>
 | ||
| 						<text class="login-desc">登录后查看个人信息和工作数据</text>
 | ||
| 					</view>
 | ||
| 					<view class="login-btn" @click="goToLogin">
 | ||
| 						<text class="login-btn-text">立即登录</text>
 | ||
| 						<i class="fas fa-arrow-right"></i>
 | ||
| 					</view>
 | ||
| 				</view>
 | ||
| 			</view>
 | ||
| 
 | ||
| 			<!-- 功能设置卡片 -->
 | ||
| 			<view class="functions-card">
 | ||
| 				<view class="card-header">
 | ||
| 					<text class="card-title">功能设置</text>
 | ||
| 				</view>
 | ||
| 				<view class="module-grid">
 | ||
| 					<view class="module-item" @click="toggleMessageSwitch">
 | ||
| 						<view class="module-icon" :style="getMessageIconStyle()">
 | ||
| 							<i :class="getMessageIcon()"></i>
 | ||
| 						</view>
 | ||
| 						<text class="module-name">消息提醒</text>
 | ||
| 					</view>
 | ||
| 
 | ||
| 					<view class="module-item" @click="themeSettings">
 | ||
| 						<view class="module-icon" :style="getThemeIconStyle()">
 | ||
| 							<i :class="getThemeIcon()"></i>
 | ||
| 						</view>
 | ||
| 						<text class="module-name">主题切换</text>
 | ||
| 					</view>
 | ||
| 
 | ||
| 					<view class="module-item" @click="changePassword">
 | ||
| 						<view class="module-icon" style="background: linear-gradient(135deg, #FF6B6B 0%, #e17055 100%)">
 | ||
| 							<i class="fas fa-lock"></i>
 | ||
| 						</view>
 | ||
| 						<text class="module-name">修改密码</text>
 | ||
| 					</view>
 | ||
| 
 | ||
| 					<view class="module-item" @click="bindPhone">
 | ||
| 						<view class="module-icon" style="background: linear-gradient(135deg, #4ECDC4 0%, #26c6da 100%)">
 | ||
| 							<i class="fas fa-mobile-alt"></i>
 | ||
| 						</view>
 | ||
| 						<text class="module-name">绑定手机</text>
 | ||
| 					</view>
 | ||
| 
 | ||
| 					<view class="module-item" @click="showCacheInfo">
 | ||
| 						<view class="module-icon" style="background: linear-gradient(135deg, #FECA57 0%, #f9d423 100%)">
 | ||
| 							<i class="fas fa-trash-alt"></i>
 | ||
| 						</view>
 | ||
| 						<text class="module-name">清除缓存</text>
 | ||
| 					</view>
 | ||
| 
 | ||
| 					<view class="module-item" @click="showVersionInfo">
 | ||
| 						<view class="module-icon" style="background: linear-gradient(135deg, #FF6B6B 0%, #e17055 100%)">
 | ||
| 							<i class="fas fa-sync-alt"></i>
 | ||
| 						</view>
 | ||
| 						<text class="module-name">版本更新</text>
 | ||
| 					</view>
 | ||
| 
 | ||
| 					<view class="module-item none"></view>
 | ||
| 					<view class="module-item none"></view>
 | ||
| 					<view class="module-item none"></view>
 | ||
| 					<view class="module-item none"></view>
 | ||
| 				</view>
 | ||
| 			</view>
 | ||
| 
 | ||
| 			<!-- 帮助与反馈卡片 -->
 | ||
| 			<view class="help-card">
 | ||
| 				<view class="card-header">
 | ||
| 					<text class="card-title">帮助与反馈</text>
 | ||
| 				</view>
 | ||
| 				<view class="module-grid">
 | ||
| 					<view class="module-item" @click="showFAQ">
 | ||
| 						<view class="module-icon" style="background: linear-gradient(135deg, #4ECDC4 0%, #26c6da 100%)">
 | ||
| 							<i class="fas fa-question-circle"></i>
 | ||
| 						</view>
 | ||
| 						<text class="module-name">常见问题</text>
 | ||
| 					</view>
 | ||
| 
 | ||
| 					<view class="module-item" @click="feedback">
 | ||
| 						<view class="module-icon" style="background: linear-gradient(135deg, #45B7D1 0%, #168aad 100%)">
 | ||
| 							<i class="fas fa-comment-dots"></i>
 | ||
| 						</view>
 | ||
| 						<text class="module-name">意见反馈</text>
 | ||
| 					</view>
 | ||
| 
 | ||
| 					<view class="module-item" @click="contactService">
 | ||
| 						<view class="module-icon" style="background: linear-gradient(135deg, #96CEB4 0%, #3ad29f 100%)">
 | ||
| 							<i class="fas fa-headset"></i>
 | ||
| 						</view>
 | ||
| 						<text class="module-name">客服联系</text>
 | ||
| 					</view>
 | ||
| 
 | ||
| 					<view class="module-item none"></view>
 | ||
| 					<view class="module-item none"></view>
 | ||
| 				</view>
 | ||
| 			</view>
 | ||
| 
 | ||
| 			<!-- 账号管理卡片 -->
 | ||
| 			<view class="account-card">
 | ||
| 				<view class="card-header">
 | ||
| 					<text class="card-title">账号管理</text>
 | ||
| 				</view>
 | ||
| 				<view class="module-grid">
 | ||
| 					<view class="module-item" @click="handleLogout">
 | ||
| 						<view class="module-icon" style="background: linear-gradient(135deg, #95a5a6 0%, #7f8c8d 100%)">
 | ||
| 							<i class="fas fa-sign-out-alt"></i>
 | ||
| 						</view>
 | ||
| 						<text class="module-name">退出登录</text>
 | ||
| 					</view>
 | ||
| 
 | ||
| 					<view class="module-item" @click="accountLogout">
 | ||
| 						<view class="module-icon" style="background: linear-gradient(135deg, #e74c3c 0%, #c0392b 100%)">
 | ||
| 							<i class="fas fa-user-slash"></i>
 | ||
| 						</view>
 | ||
| 						<text class="module-name">账号注销</text>
 | ||
| 					</view>
 | ||
| 
 | ||
| 					<view class="module-item none"></view>
 | ||
| 					<view class="module-item none"></view>
 | ||
| 					<view class="module-item none"></view>
 | ||
| 				</view>
 | ||
| 			</view>
 | ||
| 		</scroll-view>
 | ||
| 	</view>
 | ||
| </template>
 | ||
| 
 | ||
| <script>
 | ||
| import { ref, reactive, computed, onMounted } from 'vue'
 | ||
| import { useAuthStore } from '../../src/store/authStore.js'
 | ||
| import { redirectAfterLogout } from '../../src/utils/routeGuard.js'
 | ||
| 
 | ||
| export default {
 | ||
| 	setup() {
 | ||
| 		const authStore = useAuthStore()
 | ||
| 		
 | ||
| 		// 响应式数据
 | ||
| 		const unreadCount = ref(2)
 | ||
| 		const messageEnabled = ref(true)
 | ||
| 		const currentTheme = ref('浅色模式')
 | ||
| 		const cacheSize = ref('12.5MB')
 | ||
| 		const appVersion = ref('v1.0.0')
 | ||
| 
 | ||
| 		// 用户信息 - 从 Pinia store 获取
 | ||
| 		const userInfo = computed(() => {
 | ||
| 			return authStore.userInfo || {
 | ||
| 				nickname: '未登录',
 | ||
| 				name: '未登录',
 | ||
| 				department: '未知部门',
 | ||
| 				dept: { name: '未知部门' },
 | ||
| 				employeeId: 'N/A',
 | ||
| 				avatar: 'static\\imgs\\default_avatar.png',
 | ||
| 				attendanceRate: 0,
 | ||
| 				completedTasks: 0,
 | ||
| 				pendingAmount: '0'
 | ||
| 			}
 | ||
| 		})
 | ||
| 
 | ||
| 		// 登录状态
 | ||
| 		const isLogin = computed(() => {
 | ||
| 			return authStore.isAuthenticated
 | ||
| 		})
 | ||
| 
 | ||
| 		// 设备检测
 | ||
| 		const isMobile = computed(() => {
 | ||
| 			return getApp().globalData.isMobile
 | ||
| 		})
 | ||
| 
 | ||
| 		// 方法
 | ||
| 		const handleSearch = () => {
 | ||
| 			uni.showToast({
 | ||
| 				title: '搜索功能',
 | ||
| 				icon: 'none'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const handleNotification = () => {
 | ||
| 			uni.switchTab({
 | ||
| 				url: '/pages/message/message'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const editProfile = () => {
 | ||
| 			// 跳转到编辑个人资料页面
 | ||
| 			uni.navigateTo({
 | ||
| 				url: '/pages/profile/editprofile'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const goToAttendance = () => {
 | ||
| 			uni.showToast({
 | ||
| 				title: '查看考勤详情',
 | ||
| 				icon: 'none'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const goToTasks = () => {
 | ||
| 			uni.showToast({
 | ||
| 				title: '查看任务详情',
 | ||
| 				icon: 'none'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const goToReimbursement = () => {
 | ||
| 			uni.showToast({
 | ||
| 				title: '查看报销详情',
 | ||
| 				icon: 'none'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const changePassword = () => {
 | ||
| 			uni.showToast({
 | ||
| 				title: '修改密码',
 | ||
| 				icon: 'none'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const bindPhone = () => {
 | ||
| 			uni.showToast({
 | ||
| 				title: '绑定手机',
 | ||
| 				icon: 'none'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const accountLogout = () => {
 | ||
| 			uni.showModal({
 | ||
| 				title: '确认注销',
 | ||
| 				content: '注销后账号将无法恢复,确定要注销吗?',
 | ||
| 				success: (res) => {
 | ||
| 					if (res.confirm) {
 | ||
| 						uni.showToast({
 | ||
| 							title: '账号注销成功',
 | ||
| 							icon: 'none'
 | ||
| 						})
 | ||
| 					}
 | ||
| 				}
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		// 登出功能
 | ||
| 		const handleLogout = () => {
 | ||
| 			uni.showModal({
 | ||
| 				title: '确认登出',
 | ||
| 				content: '确定要退出登录吗?',
 | ||
| 				success: (res) => {
 | ||
| 					if (res.confirm) {
 | ||
| 						// 使用 Pinia store 登出
 | ||
| 						authStore.logout()
 | ||
| 						
 | ||
| 						uni.showToast({
 | ||
| 							title: '已退出登录',
 | ||
| 							icon: 'success'
 | ||
| 						})
 | ||
| 						
 | ||
| 						// 跳转到登录页面
 | ||
| 						setTimeout(() => {
 | ||
| 							redirectAfterLogout()
 | ||
| 						}, 500)
 | ||
| 					}
 | ||
| 				}
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		// 跳转到登录页面
 | ||
| 		const goToLogin = () => {
 | ||
| 			uni.navigateTo({
 | ||
| 				url: '/pages/login/index'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const messageSettings = () => {
 | ||
| 			uni.showToast({
 | ||
| 				title: '消息设置',
 | ||
| 				icon: 'none'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const toggleMessage = (value) => {
 | ||
| 			uni.showToast({
 | ||
| 				title: value ? '消息提醒已开启' : '消息提醒已关闭',
 | ||
| 				icon: 'none'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const themeSettings = () => {
 | ||
| 			uni.showActionSheet({
 | ||
| 				itemList: ['浅色模式', '深色模式', '跟随系统'],
 | ||
| 				success: (res) => {
 | ||
| 					const themes = ['浅色模式', '深色模式', '跟随系统']
 | ||
| 					currentTheme.value = themes[res.tapIndex]
 | ||
| 					uni.showToast({
 | ||
| 						title: `已切换到${themes[res.tapIndex]}`,
 | ||
| 						icon: 'none'
 | ||
| 					})
 | ||
| 				}
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const clearCache = () => {
 | ||
| 			uni.showModal({
 | ||
| 				title: '清除缓存',
 | ||
| 				content: '确定要清除应用缓存吗?',
 | ||
| 				success: (res) => {
 | ||
| 					if (res.confirm) {
 | ||
| 						cacheSize.value = '0MB'
 | ||
| 						uni.showToast({
 | ||
| 							title: '缓存清除成功',
 | ||
| 							icon: 'none'
 | ||
| 						})
 | ||
| 					}
 | ||
| 				}
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const checkUpdate = () => {
 | ||
| 			uni.showToast({
 | ||
| 				title: '检查更新中...',
 | ||
| 				icon: 'none'
 | ||
| 			})
 | ||
| 			setTimeout(() => {
 | ||
| 				uni.showToast({
 | ||
| 					title: '已是最新版本',
 | ||
| 					icon: 'none'
 | ||
| 				})
 | ||
| 			}, 2000)
 | ||
| 		}
 | ||
| 
 | ||
| 		const showFAQ = () => {
 | ||
| 			uni.showToast({
 | ||
| 				title: '常见问题',
 | ||
| 				icon: 'none'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const feedback = () => {
 | ||
| 			uni.showToast({
 | ||
| 				title: '意见反馈',
 | ||
| 				icon: 'none'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const contactService = () => {
 | ||
| 			uni.showToast({
 | ||
| 				title: '客服联系',
 | ||
| 				icon: 'none'
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		const toggleSwitch = () => {
 | ||
| 			messageEnabled.value = !messageEnabled.value
 | ||
| 			toggleMessage(messageEnabled.value)
 | ||
| 		}
 | ||
| 
 | ||
| 		// 获取主题图标
 | ||
| 		const getThemeIcon = () => {
 | ||
| 			const themeIcons = {
 | ||
| 				'浅色模式': 'fas fa-sun',
 | ||
| 				'深色模式': 'fas fa-moon',
 | ||
| 				'跟随系统': 'fas fa-desktop'
 | ||
| 			}
 | ||
| 			return themeIcons[currentTheme.value] || 'fas fa-palette'
 | ||
| 		}
 | ||
| 
 | ||
| 		// 获取主题图标背景样式
 | ||
| 		const getThemeIconStyle = () => {
 | ||
| 			const themeStyles = {
 | ||
| 				'浅色模式': 'background: linear-gradient(135deg, #FFD700 0%, #FFA500 100%)', // 金黄色渐变
 | ||
| 				'深色模式': 'background: linear-gradient(135deg, #4A4A4A 0%, #2C2C2C 100%)', // 深灰色渐变
 | ||
| 				'跟随系统': 'background: linear-gradient(135deg, #96CEB4 0%, #3ad29f 100%)' // 绿色渐变
 | ||
| 			}
 | ||
| 			return themeStyles[currentTheme.value] || 'background: linear-gradient(135deg, #96CEB4 0%, #3ad29f 100%)'
 | ||
| 		}
 | ||
| 
 | ||
| 		// 获取消息提醒图标
 | ||
| 		const getMessageIcon = () => {
 | ||
| 			return messageEnabled.value ? 'fas fa-bell' : 'fas fa-bell-slash'
 | ||
| 		}
 | ||
| 
 | ||
| 		// 获取消息提醒图标背景样式
 | ||
| 		const getMessageIconStyle = () => {
 | ||
| 			return messageEnabled.value 
 | ||
| 				? 'background: linear-gradient(135deg, #45B7D1 0%, #168aad 100%)' // 开启状态:蓝色渐变
 | ||
| 				: 'background: linear-gradient(135deg, #95a5a6 0%, #7f8c8d 100%)' // 关闭状态:灰色渐变
 | ||
| 		}
 | ||
| 
 | ||
| 		// 切换消息提醒状态
 | ||
| 		const toggleMessageSwitch = () => {
 | ||
| 			messageEnabled.value = !messageEnabled.value
 | ||
| 			toggleMessage(messageEnabled.value)
 | ||
| 		}
 | ||
| 
 | ||
| 		// 显示缓存信息弹窗
 | ||
| 		const showCacheInfo = () => {
 | ||
| 			uni.showModal({
 | ||
| 				title: '缓存信息',
 | ||
| 				content: `当前缓存大小:${cacheSize.value}\n\n点击确定清除缓存`,
 | ||
| 				confirmText: '清除缓存',
 | ||
| 				cancelText: '取消',
 | ||
| 				success: (res) => {
 | ||
| 					if (res.confirm) {
 | ||
| 						clearCache()
 | ||
| 					}
 | ||
| 				}
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		// 显示版本信息弹窗
 | ||
| 		const showVersionInfo = () => {
 | ||
| 			uni.showModal({
 | ||
| 				title: '版本信息',
 | ||
| 				content: `当前版本:${appVersion.value}\n\n点击确定检查更新`,
 | ||
| 				confirmText: '检查更新',
 | ||
| 				cancelText: '取消',
 | ||
| 				success: (res) => {
 | ||
| 					if (res.confirm) {
 | ||
| 						checkUpdate()
 | ||
| 					}
 | ||
| 				}
 | ||
| 			})
 | ||
| 		}
 | ||
| 
 | ||
| 		return {
 | ||
| 			unreadCount,
 | ||
| 			messageEnabled,
 | ||
| 			currentTheme,
 | ||
| 			cacheSize,
 | ||
| 			appVersion,
 | ||
| 			userInfo,
 | ||
| 			isLogin,
 | ||
| 			isMobile,
 | ||
| 			handleSearch,
 | ||
| 			handleNotification,
 | ||
| 			editProfile,
 | ||
| 			goToAttendance,
 | ||
| 			goToTasks,
 | ||
| 			goToReimbursement,
 | ||
| 			changePassword,
 | ||
| 			bindPhone,
 | ||
| 			accountLogout,
 | ||
| 			handleLogout,
 | ||
| 			goToLogin,
 | ||
| 			messageSettings,
 | ||
| 			toggleMessage,
 | ||
| 			themeSettings,
 | ||
| 			clearCache,
 | ||
| 			checkUpdate,
 | ||
| 			showFAQ,
 | ||
| 			feedback,
 | ||
| 			contactService,
 | ||
| 			toggleSwitch,
 | ||
| 			getThemeIcon,
 | ||
| 			getThemeIconStyle,
 | ||
| 			getMessageIcon,
 | ||
| 			getMessageIconStyle,
 | ||
| 			toggleMessageSwitch,
 | ||
| 			showCacheInfo,
 | ||
| 			showVersionInfo
 | ||
| 		}
 | ||
| 	}
 | ||
| }
 | ||
| </script>
 | ||
| 
 | ||
| <style lang="scss" scoped>
 | ||
| .profile-page {
 | ||
| 	min-height: 100vh;
 | ||
| 	width: 100%;
 | ||
| 	background-color: var(--background);
 | ||
| 	box-sizing: border-box;
 | ||
| }
 | ||
| 
 | ||
| /* 浏览器环境下的顶部间距 */
 | ||
| .unified-header + .unified-content {
 | ||
| 	margin-top: calc(var(--status-bar-height) + 88rpx);
 | ||
| 	padding-top: 30rpx;
 | ||
| }
 | ||
| 
 | ||
| /* 支持安全区域的设备 - 移动设备不需要额外的 padding-top,因为使用了 top_bar */
 | ||
| 
 | ||
| /* 移动设备顶部状态栏 */
 | ||
| .top_bar {
 | ||
| 	background: var(--gradient-primary);
 | ||
| 	box-shadow: var(--shadow-lg);
 | ||
| 	z-index: 9999;
 | ||
| 	position: fixed;
 | ||
| 	top: 0;
 | ||
| 	left: 0;
 | ||
| 	right: 0;
 | ||
| 	height: calc(var(--status-bar-height) + 88rpx);
 | ||
| 	display: flex;
 | ||
| 	align-items: flex-end;
 | ||
| 	padding-top: var(--status-bar-height);
 | ||
| 	box-sizing: border-box;
 | ||
| }
 | ||
| 
 | ||
| /* 支持安全区域的设备 */
 | ||
| @supports (padding: max(0px)) {
 | ||
|   .top_bar {
 | ||
|     height: calc(var(--status-bar-height) + 88rpx + env(safe-area-inset-top));
 | ||
|     padding-top: calc(var(--status-bar-height) + env(safe-area-inset-top));
 | ||
|   }
 | ||
|   
 | ||
|   .top_bar + .unified-content {
 | ||
|     margin-top: calc(var(--status-bar-height) + 88rpx + env(safe-area-inset-top));
 | ||
|     padding-top: 30rpx;
 | ||
|   }
 | ||
| }
 | ||
| 
 | ||
| /* 移动设备下的导航栏样式 */
 | ||
| .top_bar .chat-header {
 | ||
| 	background: transparent;
 | ||
| 	border-bottom: none;
 | ||
| 	box-shadow: none;
 | ||
| 	width: 100%;
 | ||
| 	height: 88rpx;
 | ||
| 	padding: 0 20rpx;
 | ||
| 	box-sizing: border-box;
 | ||
| 	display: flex;
 | ||
| 	justify-content: space-between;
 | ||
| 	align-items: center;
 | ||
| }
 | ||
| 
 | ||
| .top_bar .header-title {
 | ||
| 	color: var(--white);
 | ||
| 	font-weight: 600;
 | ||
| 	font-size: 36rpx;
 | ||
| }
 | ||
| 
 | ||
| .top_bar .header-left i,
 | ||
| .top_bar .header-right i {
 | ||
| 	color: var(--white);
 | ||
| 	font-size: 32rpx;
 | ||
| }
 | ||
| 
 | ||
| .top_bar .header-icon {
 | ||
| 	font-size: 32rpx;
 | ||
| 	color: var(--white);
 | ||
| }
 | ||
| 
 | ||
| .header-left,
 | ||
| .header-right {
 | ||
| 	width: 80rpx;
 | ||
| 	height: 88rpx;
 | ||
| 	display: flex;
 | ||
| 	align-items: center;
 | ||
| 	justify-content: center;
 | ||
| 	transition: background-color 0.2s ease;
 | ||
| }
 | ||
| 
 | ||
| .top_bar .header-left:active,
 | ||
| .top_bar .header-right:active {
 | ||
| 	background-color: rgba(255, 255, 255, 0.2);
 | ||
| 	border-radius: 8rpx;
 | ||
| }
 | ||
| 
 | ||
| /* 移动设备下为内容添加顶部间距 */
 | ||
| .top_bar + .unified-content {
 | ||
| 	margin-top: calc(var(--status-bar-height) + 88rpx);
 | ||
| 	padding-top: 30rpx;
 | ||
| }
 | ||
| 
 | ||
| /* 浏览器环境顶部导航栏 */
 | ||
| .unified-header {
 | ||
| 	background: var(--gradient-primary);
 | ||
| 	box-shadow: var(--shadow-lg);
 | ||
| 	position: fixed;
 | ||
| 	top: 0;
 | ||
| 	left: 0;
 | ||
| 	right: 0;
 | ||
| 	z-index: 9999;
 | ||
| 	height: calc(var(--status-bar-height) + 88rpx);
 | ||
| }
 | ||
| 
 | ||
| .unified-header .header-content {
 | ||
| 	padding: 20rpx 30rpx;
 | ||
| 	padding-top: calc(var(--status-bar-height) + 20rpx);
 | ||
| 	display: flex;
 | ||
| 	align-items: center;
 | ||
| 	justify-content: space-between;
 | ||
| 	height: 100%;
 | ||
| 	box-sizing: border-box;
 | ||
| }
 | ||
| 
 | ||
| .unified-header .header-title {
 | ||
| 	flex: 1;
 | ||
| 	text-align: center;
 | ||
| 	color: var(--white);
 | ||
| 	font-size: 32rpx;
 | ||
| 	font-weight: 600;
 | ||
| 	margin: 0 20rpx;
 | ||
| }
 | ||
| 
 | ||
| .unified-header .header-left,
 | ||
| .unified-header .header-right {
 | ||
| 	display: flex;
 | ||
| 	align-items: center;
 | ||
| 	min-width: 80rpx;
 | ||
| }
 | ||
| 
 | ||
| .unified-header .header-right {
 | ||
| 	justify-content: flex-end;
 | ||
| }
 | ||
| 
 | ||
| .unified-header .header-icon {
 | ||
| 	color: var(--white);
 | ||
| 	font-size: 36rpx;
 | ||
| 	padding: 10rpx;
 | ||
| 	transition: all 0.3s ease;
 | ||
| }
 | ||
| 
 | ||
| .unified-header .header-icon:active {
 | ||
| 	background: rgba(255, 255, 255, 0.2);
 | ||
| 	transform: scale(0.95);
 | ||
| }
 | ||
| 
 | ||
| /* 支持安全区域的设备 */
 | ||
| @supports (padding: max(0px)) {
 | ||
|   .unified-header {
 | ||
|     height: calc(var(--status-bar-height) + 88rpx + env(safe-area-inset-top));
 | ||
|   }
 | ||
|   
 | ||
|   .unified-header .header-content {
 | ||
|     padding-top: calc(var(--status-bar-height) + 20rpx + env(safe-area-inset-top));
 | ||
|   }
 | ||
|   
 | ||
|   .unified-header + .unified-content {
 | ||
|     margin-top: calc(var(--status-bar-height) + 88rpx + env(safe-area-inset-top));
 | ||
|   }
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| .custom-navbar {
 | ||
| 	background: var(--gradient-primary);
 | ||
| 	padding: 20rpx 30rpx;
 | ||
| 	padding-top: calc(var(--status-bar-height) + 20rpx);
 | ||
| }
 | ||
| 
 | ||
| .navbar-content {
 | ||
| 	display: flex;
 | ||
| 	align-items: center;
 | ||
| 	justify-content: space-between;
 | ||
| }
 | ||
| 
 | ||
| .search-box {
 | ||
| 	flex: 1;
 | ||
| 	background-color: rgba(255, 255, 255, 0.2);
 | ||
| 	border-radius: 25rpx;
 | ||
| 	padding: 15rpx 20rpx;
 | ||
| 	margin-right: 20rpx;
 | ||
| 	display: flex;
 | ||
| 	align-items: center;
 | ||
| }
 | ||
| 
 | ||
| .search-placeholder {
 | ||
| 	color: rgba(255, 255, 255, 0.8);
 | ||
| 	font-size: 28rpx;
 | ||
| 	margin-left: 10rpx;
 | ||
| }
 | ||
| 
 | ||
| .notification {
 | ||
| 	position: relative;
 | ||
| 	padding: 10rpx;
 | ||
| }
 | ||
| 
 | ||
| .page-content {
 | ||
| 	height: 100vh;
 | ||
| 	padding: 20rpx;
 | ||
| 	box-sizing: border-box;
 | ||
| 	width: 100%;
 | ||
| }
 | ||
| 
 | ||
| /* 卡片基础样式 */
 | ||
| .profile-card, .functions-card, .help-card, .account-card {
 | ||
| 	background: var(--white);
 | ||
| 	border-radius: 20rpx;
 | ||
| 	margin-bottom: 20rpx;
 | ||
| 	box-shadow: var(--shadow-md);
 | ||
| 	overflow: hidden;
 | ||
| 	width: 100%;
 | ||
| 	box-sizing: border-box;
 | ||
| }
 | ||
| 
 | ||
| /* 个人信息卡片 */
 | ||
| .profile-card {
 | ||
| 	background: var(--gradient-primary);
 | ||
| 	color: var(--white);
 | ||
| }
 | ||
| 
 | ||
| .profile-content {
 | ||
| 	padding: 40rpx;
 | ||
| }
 | ||
| 
 | ||
| .profile-header {
 | ||
| 	display: flex;
 | ||
| 	align-items: center;
 | ||
| 	margin-bottom: 30rpx;
 | ||
| }
 | ||
| 
 | ||
| .profile-avatar {
 | ||
| 	position: relative;
 | ||
| 	margin-right: 30rpx;
 | ||
| }
 | ||
| 
 | ||
| .avatar {
 | ||
| 	width: 120rpx;
 | ||
| 	height: 120rpx;
 | ||
| 	border-radius: 60rpx;
 | ||
| 	background: rgba(255, 255, 255, 0.2);
 | ||
| 	display: flex;
 | ||
| 	align-items: center;
 | ||
| 	justify-content: center;
 | ||
| 	backdrop-filter: blur(10rpx);
 | ||
| 	border: 2rpx solid rgba(255, 255, 255, 0.3);
 | ||
| }
 | ||
| 
 | ||
| .avatar-text {
 | ||
| 	color: var(--white);
 | ||
| 	font-size: 48rpx;
 | ||
| 	font-weight: 600;
 | ||
| }
 | ||
| 
 | ||
| .edit-icon {
 | ||
| 	position: absolute;
 | ||
| 	bottom: 0;
 | ||
| 	right: 0;
 | ||
| 	width: 36rpx;
 | ||
| 	height: 36rpx;
 | ||
| 	background: rgba(255, 255, 255, 0.9);
 | ||
| 	border-radius: 50%;
 | ||
| 	display: flex;
 | ||
| 	align-items: center;
 | ||
| 	justify-content: center;
 | ||
| 	font-size: 20rpx;
 | ||
| 	color: #667eea;
 | ||
| }
 | ||
| 
 | ||
| .profile-info {
 | ||
| 	flex: 1;
 | ||
| }
 | ||
| 
 | ||
| .profile-name {
 | ||
| 	font-size: 36rpx;
 | ||
| 	font-weight: 700;
 | ||
| 	color: var(--white);
 | ||
| 	margin-bottom: 8rpx;
 | ||
| 	display: block;
 | ||
| }
 | ||
| 
 | ||
| .profile-dept {
 | ||
| 	font-size: 28rpx;
 | ||
| 	color: rgba(255, 255, 255, 0.8);
 | ||
| 	margin-bottom: 6rpx;
 | ||
| 	display: block;
 | ||
| }
 | ||
| 
 | ||
| .profile-id {
 | ||
| 	font-size: 24rpx;
 | ||
| 	color: rgba(255, 255, 255, 0.7);
 | ||
| }
 | ||
| 
 | ||
| .profile-arrow {
 | ||
| 	margin-left: 20rpx;
 | ||
| 	font-size: 24rpx;
 | ||
| 	color: rgba(255, 255, 255, 0.7);
 | ||
| }
 | ||
| 
 | ||
| /* 数据统计网格 */
 | ||
| .stats-grid {
 | ||
| 	display: flex;
 | ||
| 	gap: 20rpx;
 | ||
| }
 | ||
| 
 | ||
| .stat-item {
 | ||
| 	flex: 1;
 | ||
| 	background: rgba(255, 255, 255, 0.15);
 | ||
| 	border-radius: 16rpx;
 | ||
| 	padding: 24rpx;
 | ||
| 	display: flex;
 | ||
| 	align-items: center;
 | ||
| 	backdrop-filter: blur(10rpx);
 | ||
| 	border: 1rpx solid rgba(255, 255, 255, 0.2);
 | ||
| 	transition: all 0.3s ease;
 | ||
| }
 | ||
| 
 | ||
| .stat-item:active {
 | ||
| 	transform: scale(0.95);
 | ||
| 	background: rgba(255, 255, 255, 0.25);
 | ||
| }
 | ||
| 
 | ||
| .stat-icon {
 | ||
| 	width: 48rpx;
 | ||
| 	height: 48rpx;
 | ||
| 	border-radius: 24rpx;
 | ||
| 	background: rgba(255, 255, 255, 0.3);
 | ||
| 	display: flex;
 | ||
| 	align-items: center;
 | ||
| 	justify-content: center;
 | ||
| 	margin-right: 16rpx;
 | ||
| 	font-size: 24rpx;
 | ||
| 	color: #fff;
 | ||
| }
 | ||
| 
 | ||
| .stat-content {
 | ||
| 	flex: 1;
 | ||
| }
 | ||
| 
 | ||
| .stat-number {
 | ||
| 	font-size: 28rpx;
 | ||
| 	font-weight: 700;
 | ||
| 	color: var(--white);
 | ||
| 	display: block;
 | ||
| 	margin-bottom: 4rpx;
 | ||
| }
 | ||
| 
 | ||
| .stat-label {
 | ||
| 	font-size: 22rpx;
 | ||
| 	color: rgba(255, 255, 255, 0.8);
 | ||
| }
 | ||
| 
 | ||
| /* 未登录提示 */
 | ||
| .login-prompt {
 | ||
| 	display: flex;
 | ||
| 	align-items: center;
 | ||
| 	padding: 40rpx;
 | ||
| 	background: rgba(255, 255, 255, 0.1);
 | ||
| 	border-radius: 16rpx;
 | ||
| 	border: 2rpx dashed rgba(255, 255, 255, 0.3);
 | ||
| 	backdrop-filter: blur(10rpx);
 | ||
| }
 | ||
| 
 | ||
| .login-icon {
 | ||
| 	font-size: 80rpx;
 | ||
| 	color: rgba(255, 255, 255, 0.8);
 | ||
| 	margin-right: 24rpx;
 | ||
| }
 | ||
| 
 | ||
| .login-content {
 | ||
| 	flex: 1;
 | ||
| 	margin-right: 20rpx;
 | ||
| }
 | ||
| 
 | ||
| .login-title {
 | ||
| 	display: block;
 | ||
| 	font-size: 32rpx;
 | ||
| 	font-weight: 600;
 | ||
| 	color: var(--white);
 | ||
| 	margin-bottom: 8rpx;
 | ||
| }
 | ||
| 
 | ||
| .login-desc {
 | ||
| 	font-size: 26rpx;
 | ||
| 	color: rgba(255, 255, 255, 0.8);
 | ||
| 	line-height: 1.4;
 | ||
| }
 | ||
| 
 | ||
| .login-btn {
 | ||
| 	display: flex;
 | ||
| 	align-items: center;
 | ||
| 	gap: 8rpx;
 | ||
| 	padding: 16rpx 24rpx;
 | ||
| 	background: rgba(255, 255, 255, 0.2);
 | ||
| 	border-radius: 25rpx;
 | ||
| 	border: 1rpx solid rgba(255, 255, 255, 0.3);
 | ||
| 	transition: all 0.3s ease;
 | ||
| }
 | ||
| 
 | ||
| .login-btn:active {
 | ||
| 	transform: scale(0.95);
 | ||
| 	background: rgba(255, 255, 255, 0.3);
 | ||
| }
 | ||
| 
 | ||
| .login-btn-text {
 | ||
| 	font-size: 26rpx;
 | ||
| 	font-weight: 600;
 | ||
| 	color: var(--white);
 | ||
| }
 | ||
| 
 | ||
| .login-btn i {
 | ||
| 	font-size: 20rpx;
 | ||
| 	color: var(--white);
 | ||
| }
 | ||
| 
 | ||
| /* 卡片头部 */
 | ||
| .card-header {
 | ||
| 	padding: 30rpx 30rpx 20rpx;
 | ||
| 	border-bottom: 1rpx solid var(--border-light);
 | ||
| }
 | ||
| 
 | ||
| .card-title {
 | ||
| 	font-size: 32rpx;
 | ||
| 	font-weight: 600;
 | ||
| 	color: var(--text-color);
 | ||
| }
 | ||
| 
 | ||
| /* 模块网格布局 */
 | ||
| .module-grid {
 | ||
| 	display: flex;
 | ||
| 	flex-wrap: wrap;
 | ||
| 	padding: 20rpx;
 | ||
| 	gap: 16rpx;
 | ||
| 	align-items: flex-start;
 | ||
| }
 | ||
| 
 | ||
| .module-item {
 | ||
| 	width: calc(20% - 12.8rpx);
 | ||
| 	display: flex;
 | ||
| 	flex-direction: column;
 | ||
| 	align-items: center;
 | ||
| 	justify-content: flex-start;
 | ||
| 	background: var(--gray-lighter);
 | ||
| 	border-radius: 16rpx;
 | ||
| 	padding: 20rpx 12rpx;
 | ||
| 	transition: all 0.3s ease;
 | ||
| 	position: relative;
 | ||
| 	min-height: auto;
 | ||
| 	height: auto;
 | ||
| }
 | ||
| 
 | ||
| .module-item:active {
 | ||
| 	transform: scale(0.95);
 | ||
| 	background: var(--gray);
 | ||
| }
 | ||
| 
 | ||
| .module-item.none {
 | ||
| 	background: transparent;
 | ||
| 	pointer-events: none;
 | ||
| }
 | ||
| 
 | ||
| .module-icon {
 | ||
| 	width: 56rpx;
 | ||
| 	height: 56rpx;
 | ||
| 	border-radius: 14rpx;
 | ||
| 	display: flex;
 | ||
| 	align-items: center;
 | ||
| 	justify-content: center;
 | ||
| 	margin-bottom: 12rpx;
 | ||
| 	font-size: 28rpx;
 | ||
| 	color: #fff;
 | ||
| 	box-shadow: 0 4rpx 12rpx rgba(0, 0, 0, 0.15);
 | ||
| 	flex-shrink: 0;
 | ||
| }
 | ||
| 
 | ||
| .module-name {
 | ||
| 	font-size: 20rpx;
 | ||
| 	color: var(--text-color);
 | ||
| 	font-weight: 500;
 | ||
| 	text-align: center;
 | ||
| 	line-height: 1.3;
 | ||
| 	margin-bottom: 8rpx;
 | ||
| 	word-break: break-all;
 | ||
| 	hyphens: auto;
 | ||
| }
 | ||
| 
 | ||
| .module-value {
 | ||
| 	font-size: 20rpx;
 | ||
| 	color: var(--text-secondary);
 | ||
| 	text-align: center;
 | ||
| 	line-height: 1.3;
 | ||
| 	margin-bottom: 8rpx;
 | ||
| 	word-break: break-all;
 | ||
| }
 | ||
| 
 | ||
| .module-switch {
 | ||
| 	margin-top: 4rpx;
 | ||
| 	flex-shrink: 0;
 | ||
| }
 | ||
| 
 | ||
| /* 开关样式 */
 | ||
| .switch {
 | ||
| 	width: 48rpx;
 | ||
| 	height: 24rpx;
 | ||
| 	background-color: #e0e0e0;
 | ||
| 	border-radius: 12rpx;
 | ||
| 	position: relative;
 | ||
| 	transition: all 0.3s ease;
 | ||
| }
 | ||
| 
 | ||
| .switch.active {
 | ||
| 	background-color: #3498db;
 | ||
| }
 | ||
| 
 | ||
| .switch-handle {
 | ||
| 	width: 20rpx;
 | ||
| 	height: 20rpx;
 | ||
| 	background-color: #fff;
 | ||
| 	border-radius: 50%;
 | ||
| 	position: absolute;
 | ||
| 	top: 2rpx;
 | ||
| 	left: 2rpx;
 | ||
| 	transition: all 0.3s ease;
 | ||
| 	box-shadow: 0 1rpx 3rpx rgba(0, 0, 0, 0.2);
 | ||
| }
 | ||
| 
 | ||
| .switch.active .switch-handle {
 | ||
| 	transform: translateX(24rpx);
 | ||
| }
 | ||
| 
 | ||
| .search-icon, .notification-icon {
 | ||
| 	font-size: 32rpx;
 | ||
| 	margin-right: 10rpx;
 | ||
| 	color: var(--white);
 | ||
| }
 | ||
| 
 | ||
| .badge {
 | ||
| 	position: absolute;
 | ||
| 	top: 0rpx;
 | ||
| 	right: 15rpx;
 | ||
| 	background-color: var(--error);
 | ||
| 	color: var(--white);
 | ||
| 	font-size: 20rpx;
 | ||
| 	padding: 2rpx 8rpx;
 | ||
| 	border-radius: 50%;
 | ||
| 	// min-width: 30rpx;
 | ||
| 	text-align: center;
 | ||
| 	line-height: 1.2;
 | ||
| }
 | ||
| </style>
 |