feature/login-screen #4

Merged
zyb merged 3 commits from feature/login-screen into master 2025-06-30 19:17:33 +08:00
8 changed files with 1123 additions and 118 deletions

View File

@ -1,7 +1,7 @@
import axios from 'axios';
import { logoutUser } from '../utils/jwt'; // logoutUser会处理清除存储和重定向
const API_BASE_URL = 'http://zybdatasupport.online:8000';
const API_BASE_URL = 'https://zybdatasupport.online';
const axiosInstance = axios.create({
baseURL: API_BASE_URL,

View File

@ -75,4 +75,27 @@ export const deleteDemand = async (id) => {
console.error('删除需求失败:', error);
throw error;
}
};
};
/**
* 添加需求回复
* @param {number} id - 需求ID
* @param {Object} replyData - 回复数据
* @param {string} replyData.reply - 回复内容
* @returns {Promise<Object>} 返回添加回复的响应数据
* 说明会将回复内容与用户qq拼接在一起格式qq:内容|qq:内容
*/
export const addDemandReply = async (id, replyData) => {
try {
const payload = {
id: id,
reply: replyData.reply
};
console.log('添加需求回复的数据:', payload);
const response = await axiosInstance.put('/demands/reply', payload);
return response.data;
} catch (error) {
console.error('添加需求回复失败:', error);
throw error;
}
};

View File

@ -1,54 +1,6 @@
import axiosInstance from './axiosConfig';
import { loginSuccess } from '../utils/jwt';
// const API_BASE_URL = 'http://zybdatasupport.online:8000' // 不再需要
// // 创建 axios 实例 // 不再需要
// const axiosInstance = axios.create({
// baseURL: API_BASE_URL,
// headers: {
// 'Content-Type': 'application/json',
// 'Accept': 'application/json',
// 'X-Requested-With': 'XMLHttpRequest'
// },
// timeout: 10000 // 添加超时设置
// })
// // 设置请求拦截器,自动添加 token // 不再需要
// axiosInstance.interceptors.request.use(
// config => {
// const token = localStorage.getItem('access_token')
// if (token) {
// config.headers.Authorization = `bearer ${token}`
// }
// return config
// },
// error => {
// return Promise.reject(error)
// }
// )
// // 添加响应拦截器 // 不再需要
// axiosInstance.interceptors.response.use(
// response => response,
// error => {
// if (error.response) {
// // 服务器返回错误状态码
// console.error('请求错误:', {
// status: error.response.status,
// data: error.response.data,
// config: error.config
// })
// } else if (error.request) {
// // 请求已发出但没有收到响应
// console.error('网络错误:', error.request)
// } else {
// // 请求配置出错
// console.error('请求配置错误:', error.message)
// }
// return Promise.reject(error)
// }
// )
/**
* 获取验证码

View File

@ -0,0 +1,964 @@
<template>
<div class="affair-management">
<div class="page-header">
<h1>事项管理</h1>
<div class="header-actions">
<button class="btn-common btn-gradient" @click="openAddModal">添加需求</button>
<button class="btn-common btn-light" @click="onRefresh">刷新</button>
<button class="btn-common btn-gradient" @click="showDeletedItems = !showDeletedItems">
{{ showDeletedItems ? '隐藏已删除' : '显示已删除' }}
</button>
</div>
</div>
<div v-if="error" class="error-message">
{{ error }}
</div>
<div class="table-container">
<table class="maps-table">
<thead>
<tr>
<th>ID</th>
<th>请求者</th>
<th>QQ号</th>
<th>请求内容</th>
<th>悬赏金额</th>
<th>需求创建时间</th>
<th>回复数量</th>
<th>状态</th>
<th>操作</th>
</tr>
</thead>
<tbody>
<tr v-for="(demand, index) in displayDemands" :key="demand.id" class="table-row">
<td class="id" data-label="ID">{{ index + 1 }}</td>
<td class="requester" data-label="请求者">
<span v-if="!demand.requester" class="tag no-reward">匿名</span>
<span v-else>{{ demand.requester }}</span>
</td>
<td class="name" data-label="QQ号">
<span v-if="!demand.qq_code" class="tag no-reward">匿名</span>
<span v-else>{{ demand.qq_code }}</span>
</td>
<td class="content" data-label="请求内容">{{ getDisplayContent(demand.content) }}</td>
<td class="reward" data-label="悬赏金额">
<span :class="['tag', isNoReward(demand.reward) ? 'no-reward' : 'has-reward']">
{{ isNoReward(demand.reward) ? '无赏金' : demand.reward }}
</span>
</td>
<td class="date" data-label="创建时间">{{ formatDate(demand.date) }}</td>
<td class="reply-count" data-label="回复数量">{{ getReplyCount(demand.sendcontent) }}</td>
<td class="status" data-label="状态">
<span :class="['tag', isDeleted(demand.content) ? 'deleted' : 'active']">
{{ isDeleted(demand.content) ? '已删除' : '正常' }}
</span>
</td>
<td class="actions" data-label="操作">
<button class="btn-common btn-small btn-gradient" @click="showDetail(demand)">查看</button>
<button v-if="!isDeleted(demand.content)" class="btn-common btn-small btn-warning" @click="handleHide(demand)">隐藏</button>
<button v-else class="btn-common btn-small btn-restore" @click="handleRestore(demand)">恢复</button>
<button v-if="showDeletedItems" class="btn-common btn-small btn-danger" @click="handleDelete(demand)">删除</button>
</td>
</tr>
</tbody>
</table>
</div>
<!-- 详情弹窗 -->
<div v-if="showModal" class="modal-overlay" @click="closeModal">
<div class="modal-content" @click.stop>
<div class="modal-header">
<h2>需求详情</h2>
<button class="close-btn" @click="closeModal">&times;</button>
</div>
<div class="modal-body">
<div class="detail-item">
<span class="label">QQ号</span>
<span v-if="!selectedDemand?.qq_code" class="tag no-reward">匿名</span>
<span v-else>{{ selectedDemand?.qq_code }}</span>
</div>
<div class="detail-item">
<span class="label">请求者</span>
<span v-if="!selectedDemand?.requester" class="tag no-reward">匿名</span>
<span v-else>{{ selectedDemand?.requester }}</span>
</div>
<div class="detail-item">
<span class="label">需求内容</span>
<span class="value">{{ getDisplayContent(selectedDemand?.content) }}</span>
</div>
<div class="detail-item">
<span class="label">赏金</span>
<span :class="['tag', isNoReward(selectedDemand?.reward) ? 'no-reward' : 'has-reward']">
{{ isNoReward(selectedDemand?.reward) ? '无赏金' : selectedDemand?.reward }}
</span>
</div>
<div class="detail-item">
<span class="label">发布时间</span>
<span class="value">{{ formatDate(selectedDemand?.date) }}</span>
</div>
<div class="detail-item">
<span class="label">状态</span>
<span :class="['tag', isDeleted(selectedDemand?.content) ? 'deleted' : 'active']">
{{ isDeleted(selectedDemand?.content) ? '已删除' : '正常' }}
</span>
</div>
<div class="reply-section">
<h3>回复内容</h3>
<div class="reply-list">
<div v-if="selectedDemand?.sendcontent">
<div v-for="(reply, index) in selectedDemand.sendcontent.split('|')" :key="index" class="reply-content">
<div class="reply-with-avatar">
<img :src="`https://q1.qlogo.cn/g?b=qq&nk=${extractQQ(reply)}&s=40`" alt="User Avatar" class="reply-avatar" />
<div class="reply-text">
<b>{{ parseReplyText(reply).user }}</b> {{ parseReplyText(reply).content }}
</div>
</div>
</div>
</div>
<div v-else class="no-reply">
暂无回复
</div>
</div>
</div>
</div>
</div>
</div>
<!-- 删除确认弹窗 -->
<div v-if="showDeleteConfirm" class="modal-overlay" style="z-index:2000;">
<div class="modal-content" style="max-width:350px;text-align:center;">
<div class="modal-header">
<h2 style="color:#F56C6C;">{{ currentOperation === 'hide' ? '隐藏确认' : '删除确认' }}</h2>
</div>
<div class="modal-body" style="font-size:16px;">
{{ currentOperation === 'hide' ? '确定要隐藏该需求吗?' : '确定要删除该需求吗?此操作不可恢复。' }}
</div>
<div class="delete-dialog-footer" style="display:flex;justify-content:center;gap:18px;margin:18px 0 8px 0;">
<button class="confirm-button" @click="confirmOperation">
{{ currentOperation === 'hide' ? '确认隐藏' : '确认删除' }}
</button>
<button class="cancel-button" @click="cancelDelete">取消</button>
</div>
</div>
</div>
<!-- 恢复确认弹窗 -->
<div v-if="showRestoreConfirm" class="modal-overlay" style="z-index:2000;">
<div class="modal-content" style="max-width:350px;text-align:center;">
<div class="modal-header">
<h2 style="color:#10B981;">恢复确认</h2>
</div>
<div class="modal-body" style="font-size:16px;">确定要恢复该需求吗</div>
<div class="delete-dialog-footer" style="display:flex;justify-content:center;gap:18px;margin:18px 0 8px 0;">
<button class="confirm-button restore" @click="confirmRestore">确认恢复</button>
<button class="cancel-button" @click="cancelRestore">取消</button>
</div>
</div>
</div>
<!-- 添加需求弹窗 -->
<div v-if="showAddModal" class="modal-overlay" @click="closeAddModal">
<div class="modal-content" @click.stop>
<div class="modal-header">
<h2>添加需求</h2>
<button class="close-btn" @click="closeAddModal">&times;</button>
</div>
<div class="modal-body">
<form class="add-modal-form" @submit.prevent="submitAddForm">
<div class="form-row">
<span class="label">请求者</span>
<input v-model="addForm.requester" class="input" placeholder="可选" />
</div>
<div class="form-row">
<span class="label">QQ号</span>
<input v-model="addForm.qq_code" class="input" placeholder="可选" />
</div>
<div class="form-row">
<span class="label">需求内容</span>
<textarea
v-model="addForm.content"
class="input"
placeholder="请输入需求内容"
rows="3"
required
></textarea>
</div>
<div class="form-row">
<span class="label">赏金</span>
<input v-model="addForm.reward" class="input" placeholder="可选" />
</div>
<div v-if="addError" class="error">{{ addError }}</div>
<button class="btn-common btn-gradient submit-btn" :disabled="addLoading">
{{ addLoading ? '提交中...' : '提交' }}
</button>
</form>
</div>
</div>
</div>
</div>
</template>
<script setup>
import { ref, onMounted, computed } from 'vue'
import { getDemandsList, updateDemand, deleteDemand, addDemand } from '../../api/demands'
//
const demands = ref([])
const loading = ref(false)
const error = ref(null)
const showModal = ref(false)
const selectedDemand = ref(null)
const showDeleteConfirm = ref(false)
const showRestoreConfirm = ref(false)
const pendingDeleteId = ref(null)
const pendingRestoreId = ref(null)
const showDeletedItems = ref(false)
const currentOperation = ref('') // 'hide' 'delete'
const showAddModal = ref(false)
const addForm = ref({
requester: '',
content: '',
reward: '',
qq_code: ''
})
const addError = ref('')
const addLoading = ref(false)
//
const displayDemands = computed(() => {
if (showDeletedItems.value) {
return demands.value
} else {
return demands.value.filter(demand => !isDeleted(demand.content))
}
})
//
const isNoReward = (reward) => {
return !reward || reward === '无赏金'
}
//
const isDeleted = (content) => {
return content?.startsWith('&DEL')
}
//
const getDisplayContent = (content) => {
if (!content) return ''
return content.replace(/^&DEL/, '')
}
//
const getReplyCount = (sendcontent) => {
if (!sendcontent) return 0
return sendcontent.split('|').length
}
//
const formatDate = (dateString) => {
if (!dateString || dateString === 'Test_date') return '日期未提供'
try {
const date = new Date(dateString)
if (isNaN(date.getTime())) {
return dateString
}
const pad = num => num.toString().padStart(2, '0')
return `${date.getFullYear()}-${pad(date.getMonth() + 1)}-${pad(date.getDate())} ${pad(date.getHours())}:${pad(date.getMinutes())}:${pad(date.getSeconds())}`
} catch (e) {
return dateString
}
}
//
const fetchDemands = async () => {
loading.value = true
error.value = null
try {
const data = await getDemandsList()
demands.value = data
} catch (err) {
console.error('加载需求列表失败:', err)
if (err.response?.status === 403) {
error.value = '权限不足,请确认您有管理员权限'
} else if (err.response?.status === 401) {
error.value = '登录已过期,请重新登录'
} else {
error.value = `加载失败: ${err.response?.data?.detail || err.message}`
}
} finally {
loading.value = false
}
}
//
const showDetail = (demand) => {
selectedDemand.value = demand
showModal.value = true
}
//
const closeModal = () => {
showModal.value = false
selectedDemand.value = null
}
//
const handleHide = (demand) => {
pendingDeleteId.value = demand.id
currentOperation.value = 'hide'
showDeleteConfirm.value = true
}
//
const confirmOperation = async () => {
try {
if (currentOperation.value === 'hide') {
const demand = demands.value.find(d => d.id === pendingDeleteId.value)
if (!demand) return
const updatedContent = `&DEL${demand.content}`
await updateDemand(pendingDeleteId.value, {
requester: demand.requester,
qq_code: demand.qq_code,
content: updatedContent,
reward: demand.reward,
date: demand.date,
sendcontent: demand.sendcontent
})
} else if (currentOperation.value === 'delete') {
await deleteDemand(pendingDeleteId.value)
}
fetchDemands()
} catch (e) {
console.error('操作失败:', e)
} finally {
showDeleteConfirm.value = false
pendingDeleteId.value = null
currentOperation.value = ''
}
}
//
const handleDelete = (demand) => {
pendingDeleteId.value = demand.id
currentOperation.value = 'delete'
showDeleteConfirm.value = true
}
//
const cancelDelete = () => {
showDeleteConfirm.value = false
pendingDeleteId.value = null
}
//
const handleRestore = (demand) => {
pendingRestoreId.value = demand.id
showRestoreConfirm.value = true
}
//
const confirmRestore = async () => {
try {
const demand = demands.value.find(d => d.id === pendingRestoreId.value)
if (!demand) return
const restoredContent = demand.content.replace(/^&DEL/, '')
await updateDemand(pendingRestoreId.value, {
requester: demand.requester,
qq_code: demand.qq_code,
content: restoredContent,
reward: demand.reward,
date: demand.date,
sendcontent: demand.sendcontent
})
fetchDemands()
} catch (e) {
console.error('恢复失败:', e)
} finally {
showRestoreConfirm.value = false
pendingRestoreId.value = null
}
}
//
const cancelRestore = () => {
showRestoreConfirm.value = false
pendingRestoreId.value = null
}
//
const onRefresh = () => {
fetchDemands()
}
//
const openAddModal = () => {
addForm.value = {
requester: '',
content: '',
reward: '',
qq_code: ''
}
addError.value = ''
showAddModal.value = true
}
//
const closeAddModal = () => {
showAddModal.value = false
addError.value = ''
}
//
const submitAddForm = async () => {
if (!addForm.value.content?.trim()) {
addError.value = '需求内容不能为空'
return
}
addLoading.value = true
addError.value = ''
try {
const now = new Date()
const dateStr = `${now.getFullYear()}-${(now.getMonth()+1).toString().padStart(2,'0')}-${now.getDate().toString().padStart(2,'0')} ${now.getHours().toString().padStart(2,'0')}:${now.getMinutes().toString().padStart(2,'0')}:${now.getSeconds().toString().padStart(2,'0')}`
const payload = {
requester: addForm.value.requester || '',
sendcontent: '',
content: addForm.value.content,
reward: addForm.value.reward || '',
date: dateStr,
qq_code: addForm.value.qq_code || ''
}
console.log('提交的数据:', payload)
await addDemand(payload)
showAddModal.value = false
fetchDemands() //
} catch (e) {
console.error('提交失败:', e)
addError.value = e.response?.data?.detail || '提交失败,请稍后重试'
} finally {
addLoading.value = false
}
}
//
function parseReplyText(str) {
// qq:
const qqMatch = str.match(/^(\d+):(.+)$/)
if (qqMatch) {
return {
user: qqMatch[1],
content: qqMatch[2].trim()
}
}
// QQ
const match = str.match(/^(.+?[(][1-9][0-9]{4,}[)])(.*)$/)
if (match) {
return {
user: match[1].trim(),
content: match[2].replace(/^|^:/, '').trim()
}
}
// fallback
return {
user: '',
content: str
}
}
// QQ
function extractQQ(str) {
// qq:
const qqMatch = str.match(/^(\d+):(.+)$/)
if (qqMatch) {
return qqMatch[1]
}
// QQ
const match = str.match(/[(]([1-9][0-9]{4,})[)]/)
return match ? match[1] : ''
}
//
onMounted(() => {
fetchDemands()
})
</script>
<style scoped>
.affair-management {
padding: 20px;
}
.page-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 20px;
}
.page-header h1 {
color: #1a237e;
margin: 0;
}
.header-actions {
display: flex;
gap: 10px;
}
.btn-common {
display: inline-block;
padding: 8px 22px;
font-size: 15px;
font-weight: 500;
border-radius: 6px;
border: 1px solid #b6d2ff;
cursor: pointer;
transition: background 0.2s, color 0.2s, border 0.2s;
outline: none;
box-shadow: none;
}
.btn-gradient {
background: linear-gradient(90deg, #71eaeb 0%, #416bdf 100%);
color: #fff;
border: 1px solid #71eaeb;
}
.btn-gradient:hover {
background: linear-gradient(90deg, #416bdf 0%, #71eaeb 100%);
color: #fff;
border: 1.5px solid #416bdf;
}
.btn-light {
background: linear-gradient(90deg, #e3f0ff 0%, #f7fbff 100%);
color: #2563eb;
border: 1px solid #b6d2ff;
}
.btn-light:hover {
background: linear-gradient(90deg, #d0e7ff 0%, #eaf4ff 100%);
color: #174ea6;
border: 1.5px solid #2563eb;
}
.btn-small {
padding: 5px 15px;
font-size: 14px;
}
.btn-danger {
background: #ef4444;
color: #fff;
border: 1px solid #ef4444;
}
.btn-danger:hover {
background: #dc2626;
border: 1.5px solid #dc2626;
}
.btn-warning {
background: #f59e0b;
color: #fff;
border: 1px solid #f59e0b;
}
.btn-warning:hover {
background: #d97706;
border: 1.5px solid #d97706;
}
.btn-restore {
background: #10b981;
color: #fff;
border: 1px solid #10b981;
}
.btn-restore:hover {
background: #059669;
border: 1.5px solid #059669;
}
.table-container {
background: white;
border-radius: 8px;
overflow: hidden;
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1);
}
.maps-table {
width: 100%;
border-collapse: collapse;
}
.maps-table th,
.maps-table td {
text-align: center;
vertical-align: middle;
padding: 12px 8px;
border-bottom: 1px solid #f0f0f0;
}
.maps-table th {
background: #f8fafc;
font-weight: 600;
color: #1a237e;
}
.maps-table tr:hover {
background: #f8fafc;
}
.content {
max-width: 300px;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.tag {
display: inline-block;
padding: 2px 10px;
border-radius: 12px;
font-size: 13px;
font-weight: 500;
line-height: 1.2;
background: none;
box-shadow: none;
border: none;
margin: 0;
vertical-align: middle;
}
.has-reward {
background: #fff7e6;
color: #d46b08;
border: 1px solid #ffd591;
}
.no-reward {
background: #f5f5f5;
color: #8c8c8c;
border: 1px solid #d9d9d9;
}
.active {
background: #f0f9ff;
color: #0369a1;
border: 1px solid #7dd3fc;
}
.deleted {
background: #fef2f2;
color: #dc2626;
border: 1px solid #fca5a5;
}
.actions {
display: flex;
gap: 5px;
justify-content: center;
}
/* 弹窗样式 */
.modal-overlay {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: rgba(0, 0, 0, 0.5);
display: flex;
justify-content: center;
align-items: flex-start;
padding-top: 80px;
z-index: 1000;
}
.modal-content {
background: white;
border-radius: 8px;
width: 90%;
max-width: 800px;
max-height: 80vh;
overflow-y: auto;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
}
.modal-header {
padding: 20px;
border-bottom: 1px solid #eee;
display: flex;
justify-content: space-between;
align-items: center;
}
.modal-header h2 {
margin: 0;
color: #1a237e;
font-size: 1.5rem;
}
.close-btn {
background: none;
border: none;
font-size: 24px;
color: #666;
cursor: pointer;
padding: 0;
line-height: 1;
}
.close-btn:hover {
color: #1a237e;
}
.modal-body {
padding: 20px;
}
.detail-item {
margin-bottom: 15px;
}
.detail-item:last-child {
margin-bottom: 0;
}
.label {
font-weight: 600;
color: #1a237e;
margin-right: 10px;
}
.value {
color: #333;
}
.reply-section {
margin-top: 20px;
padding-top: 20px;
border-top: 1px solid #eee;
}
.reply-section h3 {
color: #1a237e;
margin-bottom: 15px;
font-size: 1.2rem;
}
.reply-list {
background: none;
border-radius: 0;
padding: 0;
}
.reply-content {
text-align: left;
padding: 10px 0;
color: #333;
border-left: 3px solid #2563eb;
padding-left: 15px;
margin: 10px 0;
font-size: 14px;
line-height: 1.6;
background: transparent;
box-shadow: none;
}
.no-reply {
text-align: center;
color: #666;
padding: 20px;
background: #f7faff;
border-radius: 8px;
}
.reply-with-avatar {
display: flex;
align-items: flex-start;
gap: 8px;
margin-bottom: 16px;
}
.reply-with-avatar:last-child {
margin-bottom: 0;
}
.reply-avatar {
width: 40px;
height: 40px;
border-radius: 50%;
border: none;
object-fit: cover;
flex-shrink: 0;
background: #f3f6fa;
}
.reply-text {
flex: 1;
display: flex;
align-items: center;
min-height: 40px;
font-size: 15px;
color: #222;
line-height: 1.7;
padding-left: 14px;
background: none;
border-radius: 0;
box-shadow: none;
}
.reply-text b {
font-weight: 600;
margin-right: 6px;
color: #2563eb;
font-size: 15px;
}
.confirm-button {
background-color: #f56c6c;
color: white;
border: none;
padding: 8px 24px;
border-radius: 4px;
font-size: 14px;
cursor: pointer;
transition: background-color 0.3s;
}
.confirm-button:hover {
background-color: #f78989;
}
.confirm-button.restore {
background-color: #10b981;
}
.confirm-button.restore:hover {
background-color: #059669;
}
.cancel-button {
background-color: #f0f0f0;
color: #333;
border: none;
padding: 8px 24px;
border-radius: 4px;
font-size: 14px;
cursor: pointer;
}
.cancel-button:hover {
background-color: #e0e0e0;
}
.add-modal-form {
display: flex;
flex-direction: column;
gap: 15px;
}
.form-row {
display: flex;
align-items: center;
gap: 10px;
}
.label {
font-weight: 600;
color: #1a237e;
width: 100px;
}
.input {
flex: 1;
padding: 8px;
border: 1px solid #b6d2ff;
border-radius: 4px;
}
.error {
color: #f56c6c;
font-size: 14px;
margin-top: 5px;
}
.submit-btn {
background-color: #416bdf;
color: white;
border: none;
padding: 8px 24px;
border-radius: 4px;
font-size: 14px;
cursor: pointer;
transition: background-color 0.3s;
}
.submit-btn:hover {
background-color: #71eaeb;
}
.submit-btn:disabled {
background-color: #b6d2ff;
cursor: not-allowed;
}
.error-message {
color: #f56c6c;
font-size: 14px;
margin-bottom: 10px;
}
/* --- Mobile Responsiveness: Horizontal Scroll Table --- */
@media (max-width: 768px) {
/* Header actions */
.page-header {
flex-direction: column;
align-items: flex-start;
gap: 15px;
}
.header-actions {
display: flex;
flex-wrap: wrap;
gap: 10px;
width: 100%;
}
/* Make the table container scrollable */
.table-container {
overflow-x: auto;
-webkit-overflow-scrolling: touch; /* for smooth scrolling on iOS */
border: 1px solid #e2e8f0;
border-radius: 8px;
background: #fff;
}
/* Give the table a minimum width to ensure scrolling is needed */
.maps-table {
min-width: 900px;
width: 100%;
}
/* Modal Windows */
.modal-content {
width: calc(100% - 2rem);
margin: 1rem;
}
.modal-body {
max-height: 70vh;
}
}
</style>

View File

@ -69,6 +69,7 @@
</div>
<div class="admin-main-content">
<AdminEditUserPrivilege v-if="currentAdminView === 'admin-edit-user-privilege'" />
<AffairManagement v-if="currentAdminView === 'affair-management'" />
</div>
</div>
</div>
@ -79,6 +80,7 @@ import { ref, onMounted, computed, onUnmounted } from 'vue'
import { useRouter } from 'vue-router'
import { getUserInfo } from '@/utils/jwt'
import AdminEditUserPrivilege from '@/components/backend/AdminEditUserPrivilege.vue'
import AffairManagement from '@/components/backend/AffairManagement.vue'
const router = useRouter()
const hasToken = ref(false)
@ -91,7 +93,11 @@ const dropdownOpen3 = ref(false)
let privilegeCheckTimer = null
const isAdmin = computed(() => {
return currentUserData.value && currentUserData.value.privilege === 'lv-admin';
return currentUserData.value && (
currentUserData.value.privilege === 'lv-admin' ||
currentUserData.value.privilege === 'lv-user' ||
currentUserData.value.privilege === 'admin'
);
})
async function checkPrivilege() {
@ -106,7 +112,7 @@ async function checkPrivilege() {
try {
const userInfo = await getUserInfo();
currentUserData.value = userInfo;
if (!userInfo || userInfo.privilege !== 'lv-admin') {
if (!userInfo || (userInfo.privilege !== 'lv-admin' && userInfo.privilege !== 'lv-user' && userInfo.privilege !== 'admin')) {
// 退
localStorage.removeItem('access_token')
currentUserData.value = null

View File

@ -602,4 +602,84 @@ const showCompetitionMenu = computed(() => showCompetition.value)
background-color: #f5f5f5;
color: #416bdf;
}
@media (max-width: 768px) {
/* 整体导航栏布局 */
.nav-container {
flex-direction: column;
align-items: stretch;
}
.nav-brand {
text-align: center;
margin-right: 0;
}
/* 固定汉堡菜单按钮到右上角 */
.mobile-menu-toggle {
position: fixed;
top: 10px;
right: 12px;
z-index: 1002;
background-color: rgba(0, 0, 0, 0.1);
border-radius: 50%;
padding: 8px;
height: 40px;
width: 40px;
display: flex;
align-items: center;
justify-content: center;
}
/* 移动端菜单展开后的容器样式 */
.nav-left.active,
.nav-right.active {
display: flex !important; /* 由脚本控制,我们只定义样式 */
flex-direction: column;
width: 100%;
gap: 5px;
padding: 10px 0;
}
.nav-right {
margin-left: 0;
}
/* 调整下拉菜单在一级菜单内的定位方式 */
.nav-dropdown {
position: static;
}
/* 下拉菜单内容区的移动端样式 */
.dropdown-content {
position: static; /* 覆盖桌面端的 absolute 定位 */
background-color: transparent;
box-shadow: none;
border-radius: 0;
padding: 8px 0 8px 15px; /* 子菜单缩进 */
margin-top: 5px;
border-left: 2px solid rgba(255, 255, 255, 0.15);
min-width: unset;
width: 100%;
}
/* 下拉菜单中的链接在移动端的样式 */
.dropdown-content .nav-link {
color: rgba(255, 255, 255, 0.9);
font-weight: normal;
}
.dropdown-content .nav-link:hover {
color: #fff;
background-color: rgba(255, 255, 255, 0.1);
}
/* 用户登录/信息区域的移动端样式 */
.login-btn,
.user-info-nav {
width: 100%;
justify-content: center;
margin: 5px 0;
}
}
</style>

View File

@ -22,7 +22,7 @@
</tr>
</thead>
<tbody>
<tr v-for="(demand, index) in demands" :key="demand.id" class="table-row" @click="showDetail(demand)">
<tr v-for="(demand, index) in filteredDemands" :key="demand.id" class="table-row" @click="showDetail(demand)">
<td class="id">{{ index + 1 }}</td>
<td class="requester">
<span v-if="!demand.requester" class="tag no-reward">匿名</span>
@ -55,13 +55,6 @@
</div>
<div class="modal-body">
<form class="add-modal-form" @submit.prevent="submitReply">
<!-- <div class="form-row">-->
<!-- <span class="label">回复消息</span>-->
<!-- <select v-model="addForm.replyTo" class="input">-->
<!-- <option value=""></option>-->
<!-- <option v-for="msg in replyOptions" :key="msg.id" :value="msg.id">{{ msg.text }}</option>-->
<!-- </select>-->
<!-- </div>-->
<div class="form-row">
<span class="label">昵称</span>
<input
@ -132,7 +125,7 @@
</div>
<div class="detail-item">
<span class="label">需求内容</span>
<span class="value">{{ selectedDemand?.content }}</span>
<span class="value">{{ selectedDemand?.content?.replace(/^&DEL/, '') || selectedDemand?.content }}</span>
</div>
<div class="detail-item">
<span class="label">赏金</span>
@ -263,11 +256,11 @@
<div v-if="showDeleteConfirm" class="modal-overlay" style="z-index:2000;">
<div class="modal-content" style="max-width:350px;text-align:center;">
<div class="modal-header">
<h2 style="color:#F56C6C;">删除确认</h2>
<h2 style="color:#F56C6C;">隐藏并删除</h2>
</div>
<div class="modal-body" style="font-size:16px;">确定要删除该需求吗此操作不可恢复</div>
<div class="modal-body" style="font-size:16px;">确定要隐藏并删除该需求吗此操作不可恢复</div>
<div class="delete-dialog-footer" style="display:flex;justify-content:center;gap:18px;margin:18px 0 8px 0;">
<button class="confirm-button" @click="confirmDelete">确认删除</button>
<button class="confirm-button" @click="confirmDelete">确认</button>
<button class="cancel-button" @click="cancelDelete">取消</button>
</div>
</div>
@ -278,7 +271,7 @@
<script setup>
import { ref, onMounted, nextTick, computed } from 'vue'
import { getDemandsList, addDemand, updateDemand, deleteDemand } from '../../api/demands'
import { getDemandsList, addDemand, updateDemand, deleteDemand, addDemandReply } from '../../api/demands'
import { getStoredUser } from '@/utils/jwt'
import ErrorDialog from '@/components/ErrorDialog.vue'
@ -322,6 +315,12 @@ const pendingDeleteId = ref(null)
const isNoReward = (reward) => {
return !reward || reward === '无赏金'
}
// &DEL
const filteredDemands = computed(() => {
return demands.value.filter(demand => !demand.content?.startsWith('&DEL'))
})
//
const formatDate = (dateString) => {
if (!dateString || dateString === 'Test_date') return '日期未提供'
@ -363,24 +362,6 @@ const fetchDemands = async () => {
const showReply = (demand) => {
reply.value = demand
replyModal.value = true
// sendcontent
if (demand.sendcontent) {
replyOptions.value = demand.sendcontent.split('|').map((msg, idx) => {
// qq
// qq (qq)
let match = msg.match(/^(.+?)[(](\d+)[)]/)
let nickname = match ? match[1] : ''
let qq = match ? match[2] : ''
return {
id: idx,
text: msg,
nickname,
qq
}
})
} else {
replyOptions.value = []
}
resetReplyForm();
}
@ -397,8 +378,7 @@ const resetReplyForm = () => {
addForm.value = {
sendcontent: '',
author: '',
author_contact: user && user.qq_code ? user.qq_code : '',
replyTo: ''
author_contact: user && user.qq_code ? user.qq_code : ''
};
addError.value = '';
//
@ -430,8 +410,7 @@ const openAddModal = () => {
qq_code: user && user.qq_code ? user.qq_code : '',
sendcontent: '',
author: '',
author_contact: '',
replyTo: ''
author_contact: ''
};
addError.value = '';
showAddModal.value = true;
@ -506,36 +485,12 @@ const submitReply = async () => {
addLoading.value = true;
addError.value = '';
try {
const now = new Date();
const dateStr = `${now.getFullYear()}-${(now.getMonth()+1).toString().padStart(2,'0')}-${now.getDate().toString().padStart(2,'0')} ${now.getHours().toString().padStart(2,'0')}:${now.getMinutes().toString().padStart(2,'0')}:${now.getSeconds().toString().padStart(2,'0')}`;
//
let newReply = '';
if (addForm.value.replyTo !== '' && replyOptions.value.length > 0) {
const targetMsg = replyOptions.value.find(msg => String(msg.id) === String(addForm.value.replyTo));
let targetName = targetMsg && targetMsg.nickname ? targetMsg.nickname : '';
newReply = `${addForm.value.author}${addForm.value.author_contact})回复${targetName ? ' ' + targetName : ''}${addForm.value.sendcontent}`;
} else {
newReply = `${addForm.value.author}${addForm.value.author_contact}${addForm.value.sendcontent}`;
}
const formattedContent = reply.value.sendcontent
? `${reply.value.sendcontent}|${newReply}`
: newReply;
const payload = {
requester: reply.value.requester || '',
sendcontent: formattedContent,
content: reply.value.content,
reward: reply.value.reward || '',
date: dateStr,
qq_code: reply.value.qq_code || '',
author: addForm.value.author,
author_contact: addForm.value.author_contact
const replyData = {
reply: addForm.value.sendcontent
};
console.log('提交的回复数据:', payload);
await updateDemand(reply.value.id, payload);
console.log('提交的回复数据:', replyData);
await addDemandReply(reply.value.id, replyData);
replyModal.value = false;
resetReplyForm();
fetchDemands(); //
@ -563,12 +518,28 @@ function autoResize() {
}
function extractQQ(str) {
// qq:
const qqMatch = str.match(/^(\d+):(.+)$/)
if (qqMatch) {
return qqMatch[1]
}
// QQ
const match = str.match(/[(]([1-9][0-9]{4,})[)]/)
return match ? match[1] : ''
}
function parseReplyText(str) {
// /
// qq:
const qqMatch = str.match(/^(\d+):(.+)$/)
if (qqMatch) {
return {
user: qqMatch[1],
content: qqMatch[2].trim()
}
}
// QQ
const match = str.match(/^(.+?[(][1-9][0-9]{4,}[)])(.*)$/)
if (match) {
// match[1] QQmatch[2] " xxx "
@ -634,7 +605,16 @@ function handleDeleteDemand() {
async function confirmDelete() {
try {
await deleteDemand(pendingDeleteId.value)
// 使updateDemand APIcontent&DEL
const updatedContent = `&DEL${selectedDemand.value.content}`
await updateDemand(pendingDeleteId.value, {
requester: selectedDemand.value.requester,
qq_code: selectedDemand.value.qq_code,
content: updatedContent,
reward: selectedDemand.value.reward,
date: selectedDemand.value.date,
sendcontent: selectedDemand.value.sendcontent
})
fetchDemands()
closeModal()
} catch (e) {

View File

@ -74,7 +74,7 @@
error: null,
currentPage: 1,
itemsPerPage: 100,
apiBaseUrl: 'http://zybdatasupport.online:8000',
apiBaseUrl: 'https://zybdatasupport.online',
categoryList: [],
selectedCategory: '全部',
};