This commit is contained in:
Sakurasan
2025-04-16 18:14:53 +08:00
parent ffb4496fd8
commit 15f17f4e8d
55 changed files with 7654 additions and 0 deletions
+320
View File
@@ -0,0 +1,320 @@
<template>
<div class="min-h-screen bg-base-100 p-4">
<!-- Breadcrumb and Title -->
<BreadcrumbHeader />
<!-- Search and Controls -->
<!-- <div v-if="userStore.loading" class="loading loading-spinner loading-lg"></div> -->
<div class="flex flex-wrap gap-2 mb-4">
<div class="flex flex-1 items-center space-x-2">
<input
class="flex rounded-md border border-input bg-transparent px-3 py-1 text-sm shadow-sm transition-colors placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:cursor-not-allowed disabled:opacity-50 h-8 w-[150px] lg:w-[250px]"
placeholder="Filter" value="">
<div class="dropdown">
<label tabindex="0"
class="inline-flex items-center justify-center flex gap-2 whitespace-nowrap font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 border border-input bg-background shadow-sm hover:bg-accent hover:text-accent-foreground rounded-md px-3 text-xs h-8 border-dashed">
<svg viewBox="0 0 15 15" width="1.2em" height="1.2em" class="mr-2 h-4 w-4">
<path fill="currentColor" fill-rule="evenodd"
d="M7.5.877a6.623 6.623 0 1 0 0 13.246A6.623 6.623 0 0 0 7.5.877M1.827 7.5a5.673 5.673 0 1 1 11.346 0a5.673 5.673 0 0 1-11.346 0M7.5 4a.5.5 0 0 1 .5.5V7h2.5a.5.5 0 1 1 0 1H8v2.5a.5.5 0 0 1-1 0V8H4.5a.5.5 0 0 1 0-1H7V4.5a.5.5 0 0 1 .5-.5"
clip-rule="evenodd"></path>
</svg>
Status
</label>
<ul tabindex="0" class="dropdown-content z-[1] menu shadow-lg bg-base-100 rounded-none w-24 px-0">
<li v-for="status in statusOptions" :key="status" class="px-0 mx-0">
<a class="px-2 mx-0 hover:rounded-none">
<input type="checkbox" :checked="selectedStatuses.some(item => item.status === status)"
@change="toggleStatusFilter(status)" class="checkbox checkbox-xs" />
{{ status }}
</a>
</li>
</ul>
</div>
</div>
<button class="btn btn-outline btn-success btn-sm gap-1" onclick="myModal.showModal()">
<PlusIcon class="w-3.5 h-3.5" />New
</button>
<dialog id="myModal" class="modal" ref="modalRef">
<div class="modal-box w-11/12 max-w-5xl h-screen">
<form method="dialog">
<button class="btn btn-sm btn-circle btn-ghost absolute right-2 top-2"></button>
</form>
<UserNew @closeModal="closeModal" />
</div>
<form method="dialog" class="modal-backdrop">
<button>关闭</button>
</form>
</dialog>
<div class="dropdown dropdown-end dropdown-hover">
<div tabindex="0" role="button" class="btn btn-ghost btn-sm p-1 h-8 w-8">
<Settings2Icon class="w-6 h-6" />
</div>
<ul tabindex="0" class="dropdown-content menu p-2 shadow bg-base-100 rounded-box z-[1] w-20 text-sm">
<li>
<div class="btn btn-xs p-1 text-xs hover:bg-rose-100 hover:text-rose-600 transition-colors"
@click="handleBatchAction('delete')">
<TrashIcon class="w-4 h-4" />删除
</div>
</li>
<hr class="-mx-2 my-1 border-base-content/10">
<li>
<div class="btn btn-xs p-1 text-xs hover:bg-rose-100 hover:text-rose-600 transition-colors"
@click="handleBatchAction('disable')">
<BadgeXIcon class="w-4 h-4" />禁用
</div>
</li>
<li>
<div class="btn btn-xs p-1 text-xs hover:bg-green-100 hover:text-green-600 transition-colors"
@click="handleBatchAction('enable')">
<BadgeCheckIcon class="w-4 h-4" />启用
</div>
</li>
</ul>
</div>
</div>
<!-- Table -->
<div class="card bg-base-100 shadow-xs overflow-x-auto dark:bg-base-200">
<table class="table table-sm">
<!-- Table Header -->
<thead>
<tr>
<th>
<input type="checkbox" class="checkbox checkbox-xs" v-model="selectAll" @change="toggleSelectAll" />
</th>
<th>ID</th>
<th>Name</th>
<th>Active</th>
<th>Quota</th>
<th>UsedQuota</th>
<th></th>
</tr>
</thead>
<!-- Table Body -->
<tbody>
<tr v-for="user in users" :key="user.id"
class="hover:bg-gray-500/50 dark:hover:bg-neutral-600 transition-colors">
<td>
<input type="checkbox" class="checkbox checkbox-xs" v-model="user.selected"
@change="toggleUserSelection(user)" />
</td>
<td class="text-xs dark:text-white">{{ user.id }}</td>
<td class="text-xs dark:text-white">{{ user.username }}</td>
<td>
<input type="checkbox" class="toggle toggle-xs" :class="user.active ? 'toggle-success' : 'toggle-error'"
v-model="user.active" @change="updateStatus(user)" />
</td>
<td class="text-xs dark:text-white">
<template v-if="user.unlimited_quota">
<Infinity />
</template>
<template v-else>{{ user.quota }}</template>
</td>
<td class="text-xs dark:text-white">{{ user.used_quota }}</td>
<td>
<div class="flex gap-1">
<div class="lg:tooltip lg:tooltip-top lg:tooltip-open" data-tip="预览">
<button class="btn btn-ghost btn-xs btn-square " @click="viewUser(user)">
<EyeIcon class="w-3.5 h-3.5 dark:text-white" />
</button>
</div>
<div class="lg:tooltip lg:tooltip-top lg:tooltip-open" data-tip="删除" v-if="user.role<20">
<button class="btn btn-ghost btn-xs btn-square text-error hover:bg-error/30"
@click="confirmDeleteUser(user)">
<TrashIcon class="w-3.5 h-3.5 dark:text-white" />
</button>
</div>
</div>
</td>
</tr>
</tbody>
</table>
</div>
<!-- Pagination -->
<Pagination :currentPage="currentPage" :totalItems="totalItems" :pageSize="pageSize"
:pageSizeOptions="[10, 20, 50, 100]" @changePage="changePage" @changePageSize="changePageSize" />
</div>
</template>
<script setup>
import { ref, reactive, onMounted, inject, computed } from 'vue';
import { useRouter } from 'vue-router';
import BreadcrumbHeader from '@/components/dashboard/BreadcrumbHeader.vue';
import Pagination from '@/components/Pagination.vue';
import UserNew from '@/views/dashboard/UserNew.vue';
import { useUserStore } from '@/stores/user';
import {
BadgeXIcon, BadgeCheckIcon, EyeIcon, PlusIcon, Settings2Icon,
TrashIcon, Infinity
} from 'lucide-vue-next';
const router = useRouter();
const userStore = useUserStore();
const users = computed(() => userStore.users);
const { setToast } = inject('toast');
// 用户数据
const currentPage = ref(1);
const pageSize = ref(10);
const totalItems = computed(() => userStore.totalUsers);
// 封装公共的用户列表获取方法
const listUsers = async (size = pageSize.value, page = currentPage.value, active=selectedStatuses.map(status => status.value)) => {
currentPage.value = page || currentPage.value;
// console.log('pagesize', pageSize.value, 'page', currentPage.value, 'active', selectedStatuses.map(status => status.value));
await userStore.listUser(size, page, active);
};
// 组件挂载时加载用户数据
onMounted(() => {
listUsers();
});
// 分页与页面大小变化
const changePage = async (page, size) => {
if (page == currentPage.value && size == pageSize.value) {
return
}
currentPage.value = page;
pageSize.value = size;
await listUsers();
};
const changePageSize = changePage;
// 复选框选择状态
const selectAll = ref(false)
const selectedUsers = ref([])
const toggleSelectAll = () => {
users.value.forEach(key => key.selected = selectAll.value)
if (selectAll.value) {
// Select all users on the current page
selectedUsers.value = users.value.map(user => user)
} else {
// Clear all selections
selectedUsers.value = []
}
}
const toggleUserSelection = (user) => {
if (selectedUsers.value.includes(user)) {
selectedUsers.value = selectedUsers.value.filter(selectedUser => selectedUser !== user);
} else {
selectedUsers.value.push(user);
}
selectAll.value = selectedUsers.value.length === users.value.length;
};
// 状态筛选
const statusOptions = ['Active', 'Inactive'];
const selectedStatuses = reactive([]);
const toggleStatusFilter = async (status) => {
const statusValue = status === 'Active';
const index = selectedStatuses.findIndex(item => item.status === status);
if (index > -1) {
selectedStatuses.splice(index, 1);
} else {
selectedStatuses.push({ status, value: statusValue });
}
await listUsers(undefined,1,undefined);
};
// 处理批量操作
const handleBatchAction = async (action) => {
if (selectedUsers.value.length === 0) {
return setToast('请选择用户', 'error');
}
if (!['enable', 'disable', 'delete'].includes(action)) {
return setToast('无效的操作 ${action}', 'error');
}
if (selectedUsers.value.length === 0) {
return setToast('请选择用户', 'error');
}
try {
const res = await userStore.userOption(action, selectedUsers.value.map(user => user.id));
if (res.data?.code === 200) {
setToast(`Users ${action} Success`, 'success');
} else {
setToast(res.error || `${action} Failed`, 'error');
}
selectedUsers.value = [];
selectAll.value = false;
await listUsers();
} catch (error) {
console.error(`批量操作 ${action} 失败:`, error);
setToast('批量操作失败', 'error');
}
};
// 更新用户状态
const updateStatus = async (user) => {
try {
const action = user.active ? 'enable' : 'disable';
const res = await userStore.userOption(action, [user.id]);
if (res.data?.code === 200) {
setToast(`User ${user.name} has been ${action}`, 'success');
} else {
setToast(res.error || `用户 ${user.id} ${action} 失败`, 'error');
}
await listUsers();
} catch (error) {
console.error('状态更新失败:', error);
setToast('状态更新失败', 'error');
}
};
const viewUser = (user) => {
router.push({ name: 'UserView', query: { id: user.id } });
}
const confirmDeleteUser = (user) => {
if (confirm(`确认删除 ${user.username}?`)) {
deleteUser(user);
}
};
// 删除用户
const deleteUser = async (user) => {
try {
const res = await userStore.userOption('delete', [user.id]);
if (res.data?.code === 200) {
setToast('用户删除成功', 'success');
} else {
setToast(res.error || '删除失败', 'error');
}
await listUsers();
} catch (error) {
console.error('删除失败:', error);
setToast('删除失败', 'error');
}
};
// 关闭模态框
const modalRef = ref(null);
const closeModal = async () => {
if (modalRef.value) {
modalRef.value.close();
}
await listUsers();
};
</script>