485 lines
10 KiB
Vue
485 lines
10 KiB
Vue
|
<script setup>
|
|||
|
import {ref} from 'vue'
|
|||
|
import AuthService from "../../../services/auth.js";
|
|||
|
import router from "../../router/index.js";
|
|||
|
import store from "../../store/index.js";
|
|||
|
import api from "../../utils/axios.js";
|
|||
|
import swal from "../../utils/sweetalert.js";
|
|||
|
|
|||
|
|
|||
|
const permissionLevel = ref({
|
|||
|
"0" : "普通用户",
|
|||
|
"1" : "管理员"
|
|||
|
})
|
|||
|
|
|||
|
const isAvatarHover = ref(false)
|
|||
|
const isEditingUsername = ref(false)
|
|||
|
|
|||
|
const tempUsername = ref(store.state.userInfo.username)
|
|||
|
|
|||
|
const isEditingIntro = ref(false)
|
|||
|
|
|||
|
const userIntro = ref('')
|
|||
|
|
|||
|
const tempIntro = ref(userIntro.value)
|
|||
|
|
|||
|
const fileInput = ref(null);
|
|||
|
const openFileDialog = () => {
|
|||
|
fileInput.value.click();
|
|||
|
}
|
|||
|
|
|||
|
const handleFileChange = async (event) => {
|
|||
|
const file = event.target.files[0];
|
|||
|
if (file) {
|
|||
|
fileInput.value = file;
|
|||
|
store.commit('startLoading', '正在上传图片...');
|
|||
|
try {
|
|||
|
const formData = new FormData();
|
|||
|
formData.append("image", file);
|
|||
|
const jsonData = JSON.stringify({TOKEN: store.state.token});
|
|||
|
formData.append("jsondata", jsonData);
|
|||
|
const response = await api.post("/changeprofile", formData, {
|
|||
|
headers: {
|
|||
|
"Content-Type": "multipart/form-data",
|
|||
|
},
|
|||
|
});
|
|||
|
|
|||
|
if (response.code === 0) {
|
|||
|
await AuthService.setSelfInfo();
|
|||
|
swal.tip('success', '头像上传成功! ');
|
|||
|
} else {
|
|||
|
swal.tip('error', '未知错误...');
|
|||
|
}
|
|||
|
} catch (error) {
|
|||
|
console.error("败:", error);
|
|||
|
swal.tip('error', '网络连接错误');
|
|||
|
}
|
|||
|
store.commit('stopLoading');
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
function editUsername() {
|
|||
|
isEditingUsername.value = true
|
|||
|
tempUsername.value = store.state.userInfo.username
|
|||
|
}
|
|||
|
|
|||
|
function confirmEditUsername() {
|
|||
|
swal.tip('error', '不让改');
|
|||
|
isEditingUsername.value = false
|
|||
|
}
|
|||
|
|
|||
|
function cancelEditUsername() {
|
|||
|
isEditingUsername.value = false
|
|||
|
}
|
|||
|
|
|||
|
function logout() {
|
|||
|
AuthService.logout();
|
|||
|
router.replace('/login')
|
|||
|
}
|
|||
|
|
|||
|
// 进入编辑简介
|
|||
|
function editIntro() {
|
|||
|
isEditingIntro.value = true
|
|||
|
tempIntro.value = userIntro.value
|
|||
|
}
|
|||
|
|
|||
|
// 取消编辑简介
|
|||
|
function cancelEditIntro() {
|
|||
|
isEditingIntro.value = false
|
|||
|
tempIntro.value = userIntro.value
|
|||
|
}
|
|||
|
|
|||
|
// 保存编辑简介
|
|||
|
function saveIntro() {
|
|||
|
userIntro.value = tempIntro.value
|
|||
|
isEditingIntro.value = false
|
|||
|
}
|
|||
|
</script>
|
|||
|
|
|||
|
<template>
|
|||
|
<div class="container">
|
|||
|
<div class="profile-box">
|
|||
|
<!-- 头像 -->
|
|||
|
<div
|
|||
|
class="avatar-wrapper"
|
|||
|
@mouseenter="isAvatarHover = true"
|
|||
|
@mouseleave="isAvatarHover = false"
|
|||
|
>
|
|||
|
<img
|
|||
|
:src="store.getters.profileImage"
|
|||
|
alt="用户头像"
|
|||
|
class="avatar-image"
|
|||
|
/>
|
|||
|
<div :style="isAvatarHover ? 'opacity: 1;' : 'opacity: 0;'" class="avatar-overlay" @click="openFileDialog">
|
|||
|
可修改头像
|
|||
|
</div>
|
|||
|
<input
|
|||
|
type="file"
|
|||
|
ref="fileInput"
|
|||
|
accept="image/png, image/jpg, image/jpeg"
|
|||
|
style="display: none"
|
|||
|
@change="handleFileChange"
|
|||
|
/>
|
|||
|
</div>
|
|||
|
|
|||
|
<!-- 信息 -->
|
|||
|
<div class="user-info">
|
|||
|
<div class="username-row">
|
|||
|
<label>用户名:</label>
|
|||
|
<span v-if="!isEditingUsername" class="username-text">
|
|||
|
{{ store.state.userInfo.username }}
|
|||
|
</span>
|
|||
|
|
|||
|
<input
|
|||
|
v-else
|
|||
|
type="text"
|
|||
|
class="username-input"
|
|||
|
v-model="tempUsername"
|
|||
|
/>
|
|||
|
|
|||
|
<!-- 改名 -->
|
|||
|
<span
|
|||
|
v-if="!isEditingUsername"
|
|||
|
class="edit-emoji"
|
|||
|
@click="editUsername"
|
|||
|
>
|
|||
|
🖊
|
|||
|
</span>
|
|||
|
<div v-if="isEditingUsername">
|
|||
|
<button
|
|||
|
class="confirm-btn"
|
|||
|
@click="confirmEditUsername"
|
|||
|
v-if="tempUsername !== store.state.userInfo.username"
|
|||
|
>
|
|||
|
√
|
|||
|
</button>
|
|||
|
<button
|
|||
|
class="cancel-btn"
|
|||
|
@click="cancelEditUsername"
|
|||
|
>
|
|||
|
×
|
|||
|
</button>
|
|||
|
</div>
|
|||
|
|
|||
|
</div>
|
|||
|
|
|||
|
<div class="permission-row">
|
|||
|
<label>账号权限:</label>
|
|||
|
<span>{{ permissionLevel[store.state.userInfo.role_id] }}</span>
|
|||
|
</div>
|
|||
|
|
|||
|
<div class="register-date">
|
|||
|
<label>注册日期:</label>
|
|||
|
<span>{{ store.state.userInfo.birth?.split(' ')[0] || '未知' }}</span>
|
|||
|
</div>
|
|||
|
</div>
|
|||
|
|
|||
|
</div>
|
|||
|
<button
|
|||
|
class="logout-btn"
|
|||
|
@click="logout"
|
|||
|
>登出
|
|||
|
</button>
|
|||
|
<div class="halving-line" />
|
|||
|
<label class="color-mode-box">
|
|||
|
<input type="radio" v-model="store.state.theme" value="light">亮色模式
|
|||
|
</label>
|
|||
|
<label class="color-mode-box">
|
|||
|
<input type="radio" v-model="store.state.theme" value="dark">深色模式
|
|||
|
</label>
|
|||
|
|
|||
|
<!-- -->
|
|||
|
<!-- <div class="intro-box">-->
|
|||
|
<!-- <label>个人简介:</label>-->
|
|||
|
<!-- <textarea-->
|
|||
|
<!-- class="intro-textarea"-->
|
|||
|
<!-- :readonly="!isEditingIntro"-->
|
|||
|
<!-- v-model="tempIntro"-->
|
|||
|
<!-- placeholder="空空如也~"-->
|
|||
|
<!-- />-->
|
|||
|
<!-- <!– 修改/保存/取消 按钮组 –>-->
|
|||
|
<!-- <div class="intro-actions">-->
|
|||
|
<!-- <!– 默认只显示“修改” –>-->
|
|||
|
<!-- <button-->
|
|||
|
<!-- v-if="!isEditingIntro"-->
|
|||
|
<!-- class="intro-edit-btn"-->
|
|||
|
<!-- @click="editIntro"-->
|
|||
|
<!-- >-->
|
|||
|
<!-- 修改-->
|
|||
|
<!-- </button>-->
|
|||
|
<!-- <!– 编辑状态下,显示“保存”和“取消” –>-->
|
|||
|
<!-- <div v-else class="intro-edit-group">-->
|
|||
|
<!-- <button class="intro-save-btn" @click="saveIntro">保存</button>-->
|
|||
|
<!-- <button class="intro-cancel-btn" @click="cancelEditIntro">取消</button>-->
|
|||
|
<!-- </div>-->
|
|||
|
<!-- </div>-->
|
|||
|
|
|||
|
<!-- </div>-->
|
|||
|
</div>
|
|||
|
</template>
|
|||
|
|
|||
|
<style scoped>
|
|||
|
/* ========== 公共容器区 ========== */
|
|||
|
.container {
|
|||
|
//height: calc(100vh - 60px);
|
|||
|
width: 100%;
|
|||
|
overflow-y: auto;
|
|||
|
display: flex;
|
|||
|
flex-direction: column;
|
|||
|
align-items: center;
|
|||
|
color: #f5f6f7;
|
|||
|
padding: 20px 0;
|
|||
|
transition: background-color 0.3s ease, color 0.3s ease;
|
|||
|
animation: fadeIn 0.3s ease-in-out;
|
|||
|
}
|
|||
|
|
|||
|
.theme-light .container {
|
|||
|
color: #333333;
|
|||
|
}
|
|||
|
|
|||
|
input, textarea {
|
|||
|
color: white;
|
|||
|
background: #000000;
|
|||
|
}
|
|||
|
|
|||
|
.theme-light input, .theme-light textarea {
|
|||
|
color: #2a2a2a;
|
|||
|
background: #ffffff;
|
|||
|
}
|
|||
|
|
|||
|
/* ========== 个人信息区 ========== */
|
|||
|
.profile-box {
|
|||
|
display: flex;
|
|||
|
flex-direction: row;
|
|||
|
gap: 20px;
|
|||
|
width: 80%;
|
|||
|
max-width: 800px;
|
|||
|
margin-bottom: 30px;
|
|||
|
}
|
|||
|
|
|||
|
.avatar-wrapper {
|
|||
|
position: relative;
|
|||
|
width: 100px;
|
|||
|
height: 100px;
|
|||
|
}
|
|||
|
|
|||
|
.avatar-image {
|
|||
|
width: 100%;
|
|||
|
height: 100%;
|
|||
|
object-fit: cover;
|
|||
|
border-radius: 50%;
|
|||
|
cursor: pointer;
|
|||
|
box-shadow: 0 0 10px rgba(255, 255, 255, 0.4);
|
|||
|
}
|
|||
|
|
|||
|
.theme-light .avatar-image {
|
|||
|
box-shadow: 0 0 10px rgba(0, 0, 0, 0.4);
|
|||
|
}
|
|||
|
|
|||
|
.avatar-overlay {
|
|||
|
position: absolute;
|
|||
|
top: 0;
|
|||
|
left: 0;
|
|||
|
width: 100%;
|
|||
|
height: 100%;
|
|||
|
border-radius: 50%;
|
|||
|
background-color: rgba(0, 0, 0, 0.4);
|
|||
|
display: flex;
|
|||
|
align-items: center;
|
|||
|
justify-content: center;
|
|||
|
color: #f5f6f7;
|
|||
|
font-size: 12px;
|
|||
|
transition: all 0.3s ease;
|
|||
|
cursor: pointer;
|
|||
|
}
|
|||
|
|
|||
|
.user-info {
|
|||
|
display: flex;
|
|||
|
flex-direction: column;
|
|||
|
justify-content: space-evenly;
|
|||
|
}
|
|||
|
|
|||
|
.username-row,
|
|||
|
.permission-row,
|
|||
|
.register-date {
|
|||
|
display: flex;
|
|||
|
align-items: center;
|
|||
|
gap: 8px;
|
|||
|
margin: 6px 0;
|
|||
|
}
|
|||
|
|
|||
|
.username-text {
|
|||
|
margin: 2px;
|
|||
|
font-weight: bold;
|
|||
|
}
|
|||
|
|
|||
|
.halving-line {
|
|||
|
width: 80%;
|
|||
|
height: 1px;
|
|||
|
background: linear-gradient(to bottom, rgba(217, 217, 217, 0.6), rgba(114, 114, 114, 0.6));
|
|||
|
margin: 20px 0;
|
|||
|
border-radius: 2px;
|
|||
|
}
|
|||
|
|
|||
|
/* ========== 编辑用户名输入框 ========== */
|
|||
|
.username-input {
|
|||
|
padding: 4px;
|
|||
|
font-size: 14px;
|
|||
|
border: 1px solid #ccc;
|
|||
|
border-radius: 4px;
|
|||
|
outline: none;
|
|||
|
transition: border-color 0.3s ease;
|
|||
|
}
|
|||
|
|
|||
|
/* ========== 编辑笔按钮 ========== */
|
|||
|
.edit-emoji {
|
|||
|
cursor: pointer;
|
|||
|
opacity: 0.7;
|
|||
|
transition: opacity 0.2s ease;
|
|||
|
}
|
|||
|
|
|||
|
.edit-emoji:hover {
|
|||
|
opacity: 1;
|
|||
|
}
|
|||
|
|
|||
|
/* ========== 确定按钮 ========== */
|
|||
|
.confirm-btn, .cancel-btn {
|
|||
|
margin: 0 3px;
|
|||
|
padding: 4px 8px;
|
|||
|
border: none;
|
|||
|
background-color: #3b6ea8;
|
|||
|
color: #f5f6f7;
|
|||
|
border-radius: 4px;
|
|||
|
cursor: pointer;
|
|||
|
transition: background-color 0.3s ease;
|
|||
|
}
|
|||
|
|
|||
|
.confirm-btn:hover {
|
|||
|
background-color: #2f5687;
|
|||
|
}
|
|||
|
|
|||
|
/* 亮色模式下的确定按钮 */
|
|||
|
.theme-light .confirm-btn, .theme-light .cancel-btn {
|
|||
|
background-color: #ffc107;
|
|||
|
color: #333333;
|
|||
|
}
|
|||
|
|
|||
|
.theme-light .confirm-btn:hover, .theme-light .cancel-btn:hover {
|
|||
|
background-color: #e0a806;
|
|||
|
}
|
|||
|
|
|||
|
/* ========== 颜色模式区 ========== */
|
|||
|
|
|||
|
.color-mode-box {
|
|||
|
width: 80%;
|
|||
|
max-width: 800px;
|
|||
|
display: flex;
|
|||
|
align-items: center;
|
|||
|
font-size: 18px;
|
|||
|
margin: 5px;
|
|||
|
}
|
|||
|
|
|||
|
.color-mode-box input {
|
|||
|
width: 20px;
|
|||
|
height: 20px;
|
|||
|
accent-color: #007bff;
|
|||
|
margin-right: 10px;
|
|||
|
transition: transform 0.2s ease;
|
|||
|
}
|
|||
|
|
|||
|
.color-mode-box input:checked {
|
|||
|
transform: scale(1.2);
|
|||
|
}
|
|||
|
|
|||
|
.theme-light .color-mode-box input {
|
|||
|
accent-color: #ffb74d;
|
|||
|
}
|
|||
|
|
|||
|
/* ========== 个人简介区 ========== */
|
|||
|
.intro-box {
|
|||
|
width: 80%;
|
|||
|
max-width: 800px;
|
|||
|
display: flex;
|
|||
|
flex-direction: column;
|
|||
|
}
|
|||
|
|
|||
|
.intro-textarea {
|
|||
|
width: 100%;
|
|||
|
min-height: 100px;
|
|||
|
margin-top: 5px;
|
|||
|
resize: vertical;
|
|||
|
padding: 8px;
|
|||
|
font-size: 14px;
|
|||
|
border: 1px solid #ccc;
|
|||
|
border-radius: 4px;
|
|||
|
outline: none;
|
|||
|
transition: background-color 0.3s ease;
|
|||
|
}
|
|||
|
|
|||
|
.intro-actions {
|
|||
|
margin-top: 10px;
|
|||
|
}
|
|||
|
|
|||
|
.intro-edit-group {
|
|||
|
display: flex;
|
|||
|
gap: 10px;
|
|||
|
}
|
|||
|
|
|||
|
/* ========== 修改/保存/取消/登出 按钮 ========== */
|
|||
|
.intro-edit-btn,
|
|||
|
.intro-save-btn,
|
|||
|
.intro-cancel-btn {
|
|||
|
padding: 6px 12px;
|
|||
|
border: none;
|
|||
|
border-radius: 4px;
|
|||
|
cursor: pointer;
|
|||
|
background-color: #3b6ea8;
|
|||
|
color: #f5f6f7;
|
|||
|
transition: background-color 0.3s ease;
|
|||
|
}
|
|||
|
|
|||
|
.intro-edit-btn:hover,
|
|||
|
.intro-save-btn:hover,
|
|||
|
.intro-cancel-btn:hover {
|
|||
|
background-color: #2f5687;
|
|||
|
}
|
|||
|
|
|||
|
.theme-light .intro-edit-btn,
|
|||
|
.theme-light .intro-save-btn,
|
|||
|
.theme-light .intro-cancel-btn {
|
|||
|
background-color: #ffc107;
|
|||
|
color: #333333;
|
|||
|
}
|
|||
|
|
|||
|
.theme-light .intro-edit-btn:hover,
|
|||
|
.theme-light .intro-save-btn:hover,
|
|||
|
.theme-light .intro-cancel-btn:hover {
|
|||
|
background-color: #e0a806;
|
|||
|
}
|
|||
|
|
|||
|
.logout-btn {
|
|||
|
padding: 6px 12px;
|
|||
|
border: none;
|
|||
|
border-radius: 4px;
|
|||
|
cursor: pointer;
|
|||
|
background-color: orangered;
|
|||
|
color: white;
|
|||
|
transition: all 0.1s ease;
|
|||
|
}
|
|||
|
|
|||
|
.logout-btn:hover {
|
|||
|
background-color: #be3300;
|
|||
|
}
|
|||
|
|
|||
|
@keyframes fadeIn {
|
|||
|
0% {
|
|||
|
opacity: 0;
|
|||
|
}
|
|||
|
100% {
|
|||
|
opacity: 1;
|
|||
|
}
|
|||
|
}
|
|||
|
</style>
|