Files
ZDXX/zhandianxinxi/光谱数据监控/src/views/UserManagement.vue

482 lines
14 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<template>
<div class="user-manage-container">
<el-card shadow="never" class="main-card">
<template #header>
<div class="header-row">
<div class="left-panel">
<h2 class="sys-title">👥 用户与权限管理</h2>
</div>
<div class="header-actions">
<el-button @click="router.push('/dashboard')" icon="Back">返回监控</el-button>
<el-button type="primary" icon="Plus" @click="openCreateModal">新建用户</el-button>
</div>
</div>
</template>
<el-table :data="users" border style="width: 100%" v-loading="loading">
<el-table-column prop="id" label="ID" width="80" align="center" />
<el-table-column prop="username" label="用户名" min-width="150">
<template #default="{ row }">
<span style="font-weight: bold;">{{ row.username }}</span>
</template>
</el-table-column>
<el-table-column prop="role" label="角色身份" width="150" align="center">
<template #default="{ row }">
<el-tag v-if="row.role === 'admin'" type="danger" effect="dark">超级管理员</el-tag>
<el-tag v-else-if="row.role === 'engineer'" type="warning" effect="dark">设备工程师</el-tag>
<el-tag v-else type="info" effect="plain">普通客户</el-tag>
</template>
</el-table-column>
<el-table-column prop="created_at" label="创建时间" min-width="180">
<template #default="{ row }">
{{ new Date(row.created_at).toLocaleString() }}
</template>
</el-table-column>
<el-table-column label="关联设备数" min-width="120" align="center">
<template #default="{ row }">
<el-tag v-if="row.role === 'admin'" type="danger" effect="plain">全部权限</el-tag>
<el-tag v-else effect="plain" type="success">{{ row.allowed_device_ids?.length || 0 }} </el-tag>
</template>
</el-table-column>
<el-table-column label="操作" width="220" align="center" fixed="right">
<template #default="{ row }">
<el-button
v-if="row.role !== 'admin'"
type="primary"
link
icon="Setting"
@click="openPermissionModal(row)"
>
分配设备
</el-button>
<el-popconfirm
title="确定删除该用户吗?"
confirm-button-text="删除"
cancel-button-text="取消"
icon="Warning"
icon-color="red"
@confirm="deleteUser(row.id)"
>
<template #reference>
<el-button type="danger" link icon="Delete">删除</el-button>
</template>
</el-popconfirm>
</template>
</el-table-column>
</el-table>
</el-card>
<el-dialog v-model="showCreateModal" title="新建账号" width="400px">
<el-form :model="newUser" label-width="80px">
<el-form-item label="用户名">
<el-input v-model="newUser.username" placeholder="请输入登录名" />
</el-form-item>
<el-form-item label="密码">
<el-input v-model="newUser.password" type="password" placeholder="设置初始密码" show-password />
</el-form-item>
<el-form-item label="角色权限">
<el-select v-model="newUser.role" placeholder="请选择角色" style="width: 100%">
<el-option label="普通客户 (只读)" value="client" />
<el-option label="设备工程师 (可维护)" value="engineer" />
<el-option label="超级管理员 (Root权限)" value="admin" />
</el-select>
<div style="font-size: 12px; color: #999; margin-top: 5px; line-height: 1.2;">
<span v-if="newUser.role === 'admin'" style="color: #f56c6c;">* 拥有删除用户爬虫控制等最高权限</span>
<span v-else-if="newUser.role === 'engineer'" style="color: #e6a23c;">* 拥有修改设备地点写日志权限</span>
<span v-else>* 仅可查看被分配的设备数据</span>
</div>
</el-form-item>
</el-form>
<template #footer>
<span class="dialog-footer">
<el-button @click="showCreateModal = false">取消</el-button>
<el-button type="primary" @click="createUser" :loading="creating">确认创建</el-button>
</span>
</template>
</el-dialog>
<el-dialog
v-model="showPermissionModal"
:title="`分配设备 - [${currentUser?.username}]`"
width="720px"
top="5vh"
destroy-on-close
>
<div class="permission-wrapper">
<div class="selection-toolbar">
<el-input
v-model="deviceFilterKeyword"
placeholder="搜索设备名称或地点..."
prefix-icon="Search"
clearable
style="width: 240px"
/>
<div class="toolbar-stats">
<span>已选: <span class="highlight-count">{{ selectedDeviceIds.length }}</span> / {{ allDevices.length }}</span>
<el-divider direction="vertical" />
<el-checkbox v-model="showSelectedOnly" label="只看已选" size="small" />
</div>
<div class="toolbar-actions">
<el-button link type="primary" @click="selectAllDevices">全选</el-button>
<el-button link type="info" @click="clearAllDevices">清空</el-button>
</div>
</div>
<div class="device-grid-container">
<el-scrollbar max-height="450px">
<div class="device-grid">
<div
v-for="device in displayDevices"
:key="device.id"
class="device-card"
:class="{ 'is-active': selectedDeviceIds.includes(device.id) }"
@click="toggleDeviceSelection(device.id)"
>
<div class="card-content">
<div class="d-name">{{ device.name }}</div>
<div class="d-site">
<el-icon><Location /></el-icon> {{ device.install_site || '未分配地点' }}
</div>
</div>
<div class="check-mark" v-if="selectedDeviceIds.includes(device.id)">
<el-icon><Check /></el-icon>
</div>
</div>
<div v-if="displayDevices.length === 0" class="empty-tip">
<el-empty description="没有找到匹配的设备" :image-size="80" />
</div>
</div>
</el-scrollbar>
</div>
</div>
<template #footer>
<span class="dialog-footer">
<el-button @click="showPermissionModal = false">取消</el-button>
<el-button type="primary" @click="savePermissions">保存授权 ({{ selectedDeviceIds.length }})</el-button>
</span>
</template>
</el-dialog>
</div>
</template>
<script setup>
import { ref, onMounted, computed } from 'vue'
import { useRouter } from 'vue-router'
import request from '../utils/request'
import { ElMessage } from 'element-plus'
import { Back, Plus, Setting, Delete, Warning, Search, Location, Check } from '@element-plus/icons-vue'
const router = useRouter()
const loading = ref(false)
const creating = ref(false)
const users = ref([])
const rawDevices = ref([]) // 原始设备数据
const showCreateModal = ref(false)
const showPermissionModal = ref(false)
const newUser = ref({ username: '', password: '', role: 'client' })
const currentUser = ref(null)
// 🟢 新增/修改的状态变量
const selectedDeviceIds = ref([]) // 存储当前选中的ID数组
const deviceFilterKeyword = ref('') // 搜索关键词
const showSelectedOnly = ref(false) // 是否只看已选
// 统一设备列表数据源
const allDevices = computed(() => rawDevices.value)
// 🟢 核心计算逻辑:过滤设备列表
const displayDevices = computed(() => {
let list = allDevices.value
// 1. 关键词过滤
if (deviceFilterKeyword.value) {
const k = deviceFilterKeyword.value.toLowerCase()
list = list.filter(d =>
(d.name && d.name.toLowerCase().includes(k)) ||
(d.install_site && d.install_site.toLowerCase().includes(k))
)
}
// 2. "只看已选"过滤
if (showSelectedOnly.value) {
list = list.filter(d => selectedDeviceIds.value.includes(d.id))
}
return list
})
onMounted(async () => {
await fetchUsers()
await fetchAllDevices()
})
const fetchUsers = async () => {
loading.value = true
try {
const res = await request.get('/api/admin/users')
users.value = res.data.data || res.data
} catch (e) { console.error(e) } finally { loading.value = false }
}
const fetchAllDevices = async () => {
try {
const res = await request.get('/api/devices_overview')
rawDevices.value = res.data.data || res.data
} catch (e) { console.error(e) }
}
const openCreateModal = () => {
newUser.value = {username: '', password: '', role: 'client'}
showCreateModal.value = true
}
const createUser = async () => {
if (!newUser.value.username || !newUser.value.password) return ElMessage.warning('请填写完整')
creating.value = true
try {
await request.post('/api/admin/create_user', newUser.value)
ElMessage.success('用户创建成功')
showCreateModal.value = false
fetchUsers()
} catch (e) {
ElMessage.error(e.response?.data?.msg || '创建失败')
} finally {
creating.value = false
}
}
// 🟢 打开分配弹窗
const openPermissionModal = (user) => {
currentUser.value = user
// 确保是新数组,避免引用污染
selectedDeviceIds.value = user.allowed_device_ids ? [...user.allowed_device_ids] : []
// 重置筛选状态
deviceFilterKeyword.value = ''
showSelectedOnly.value = false
showPermissionModal.value = true
}
// 🟢 切换单个选中状态
const toggleDeviceSelection = (id) => {
const index = selectedDeviceIds.value.indexOf(id)
if (index > -1) {
selectedDeviceIds.value.splice(index, 1)
} else {
selectedDeviceIds.value.push(id)
}
}
// 🟢 全选(基于当前过滤后的列表)
const selectAllDevices = () => {
const currentIds = displayDevices.value.map(d => d.id)
// 将未选中的添加进去Set去重
const newSet = new Set([...selectedDeviceIds.value, ...currentIds])
selectedDeviceIds.value = Array.from(newSet)
}
// 🟢 清空(全部清空)
const clearAllDevices = () => {
selectedDeviceIds.value = []
}
const savePermissions = async () => {
try {
await request.post('/api/admin/assign_devices', {
user_id: currentUser.value.id,
device_ids: selectedDeviceIds.value
})
ElMessage.success('权限已更新')
showPermissionModal.value = false
fetchUsers()
} catch (e) {
ElMessage.error('保存失败')
}
}
const deleteUser = async (id) => {
try {
await request.post('/api/admin/delete_user', {user_id: id})
ElMessage.success('用户已删除')
fetchUsers()
} catch (e) {
ElMessage.error(e.response?.data?.msg || '删除失败')
}
}
</script>
<style scoped>
.user-manage-container {
padding: 10px;
background: #f5f7fa;
min-height: 100vh;
box-sizing: border-box;
}
.main-card {
border-radius: 8px;
min-height: 80vh;
}
.header-row {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 20px;
}
.sys-title {
margin: 0;
font-size: 20px;
color: #303133;
font-weight: 700;
}
.header-actions {
display: flex;
gap: 10px;
}
/* 🟢 新增:权限选择器样式 */
.permission-wrapper {
background: #fff;
border: 1px solid #e4e7ed;
border-radius: 6px;
overflow: hidden;
}
.selection-toolbar {
padding: 10px 15px;
background: #f5f7fa;
border-bottom: 1px solid #e4e7ed;
display: flex;
align-items: center;
gap: 15px;
flex-wrap: wrap;
}
.toolbar-stats {
font-size: 13px;
color: #606266;
display: flex;
align-items: center;
gap: 10px;
}
.highlight-count {
color: #409EFF;
font-weight: bold;
font-size: 15px;
}
.toolbar-actions {
margin-left: auto;
}
.device-grid-container {
padding: 15px;
background: #fff;
}
.device-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(160px, 1fr));
gap: 12px;
}
.device-card {
position: relative;
border: 1px solid #dcdfe6;
border-radius: 6px;
padding: 12px;
cursor: pointer;
transition: all 0.2s;
background: #fff;
display: flex;
flex-direction: column;
justify-content: center;
}
.device-card:hover {
border-color: #c6e2ff;
transform: translateY(-2px);
box-shadow: 0 2px 8px rgba(0,0,0,0.05);
}
.device-card.is-active {
border-color: #409EFF;
background-color: #ecf5ff;
}
.card-content {
pointer-events: none; /* 让点击穿透到底层div */
}
.d-name {
font-weight: bold;
color: #303133;
font-size: 14px;
margin-bottom: 6px;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.d-site {
font-size: 12px;
color: #909399;
display: flex;
align-items: center;
gap: 4px;
}
/* 右下角打钩图标 */
.check-mark {
position: absolute;
top: 0;
right: 0;
width: 0;
height: 0;
border-style: solid;
border-width: 0 28px 28px 0;
border-color: transparent #409EFF transparent transparent;
}
.check-mark .el-icon {
position: absolute;
top: 2px;
right: -26px;
color: #fff;
font-size: 12px;
font-weight: bold;
}
.empty-tip {
grid-column: 1 / -1;
text-align: center;
padding: 20px;
}
/* 移动端适配 */
@media screen and (max-width: 768px) {
.selection-toolbar {
flex-direction: column;
align-items: stretch;
gap: 10px;
}
.toolbar-actions {
margin-left: 0;
display: flex;
justify-content: flex-end;
}
.device-grid {
grid-template-columns: repeat(2, 1fr);
}
}
</style>