2025-12-18 13:50:39 +08:00
|
|
|
<template>
|
|
|
|
|
<AppLayout>
|
2025-12-27 10:50:25 +08:00
|
|
|
<TablePageLayout>
|
|
|
|
|
<template #filters>
|
2026-01-01 18:59:38 +08:00
|
|
|
<div class="flex flex-col gap-4 lg:flex-row lg:items-center lg:justify-between">
|
|
|
|
|
<div class="flex flex-1 flex-wrap items-center gap-3">
|
2026-01-04 22:29:19 +08:00
|
|
|
<div class="w-64">
|
|
|
|
|
<SearchInput v-model="params.search" :placeholder="t('admin.users.searchUsers')" @search="reload" />
|
2026-01-01 18:59:38 +08:00
|
|
|
</div>
|
2026-01-04 22:29:19 +08:00
|
|
|
<div class="w-32">
|
|
|
|
|
<Select v-model="params.role" :options="[{ value: '', label: t('admin.users.allRoles') }, { value: 'admin', label: t('admin.users.admin') }, { value: 'user', label: t('admin.users.user') }]" @change="reload" />
|
2026-01-01 18:59:38 +08:00
|
|
|
</div>
|
2026-01-04 22:29:19 +08:00
|
|
|
<div class="w-32">
|
|
|
|
|
<Select v-model="params.status" :options="[{ value: '', label: t('admin.users.allStatus') }, { value: 'active', label: t('common.active') }, { value: 'disabled', label: t('admin.users.disabled') }]" @change="reload" />
|
2026-01-01 18:59:38 +08:00
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
<div class="flex items-center gap-3">
|
2026-01-04 22:29:19 +08:00
|
|
|
<button @click="load" :disabled="loading" class="btn btn-secondary"><svg :class="['h-5 w-5', loading ? 'animate-spin' : '']" fill="none" viewBox="0 0 24 24" stroke="currentColor"><path stroke-linecap="round" stroke-linejoin="round" d="M16.023 9.348h4.992v-.001M2.985 19.644v-4.992m0 0h4.992m-4.993 0l3.181 3.183a8.25 8.25 0 0013.803-3.7M4.031 9.865a8.25 8.25 0 0113.803-3.7l3.181 3.182m0-4.991v4.99" /></svg></button>
|
refactor(frontend): comprehensive split of large view files into modular components
- Split UsersView.vue into UserCreateModal, UserEditModal, UserApiKeysModal, etc.
- Split UsageView.vue into UsageStatsCards, UsageFilters, UsageTable, etc.
- Split DashboardView.vue into UserDashboardStats, UserDashboardCharts, etc.
- Split AccountsView.vue into AccountTableActions, AccountTableFilters, etc.
- Standardized TypeScript types across new components to resolve implicit 'any' and 'never[]' errors.
- Improved overall frontend maintainability and code clarity.
2026-01-04 22:17:27 +08:00
|
|
|
<button @click="showCreateModal = true" class="btn btn-primary"><svg class="mr-2 h-5 w-5" fill="none" viewBox="0 0 24 24" stroke="currentColor"><path stroke-linecap="round" stroke-linejoin="round" d="M12 4.5v15m7.5-7.5h-15" /></svg>{{ t('admin.users.createUser') }}</button>
|
2026-01-01 18:59:38 +08:00
|
|
|
</div>
|
2025-12-18 13:50:39 +08:00
|
|
|
</div>
|
2025-12-27 10:50:25 +08:00
|
|
|
</template>
|
2025-12-18 13:50:39 +08:00
|
|
|
|
2025-12-27 10:50:25 +08:00
|
|
|
<template #table>
|
2026-01-04 22:29:19 +08:00
|
|
|
<DataTable :columns="columns" :data="users" :loading="loading">
|
refactor(frontend): comprehensive split of large view files into modular components
- Split UsersView.vue into UserCreateModal, UserEditModal, UserApiKeysModal, etc.
- Split UsageView.vue into UsageStatsCards, UsageFilters, UsageTable, etc.
- Split DashboardView.vue into UserDashboardStats, UserDashboardCharts, etc.
- Split AccountsView.vue into AccountTableActions, AccountTableFilters, etc.
- Standardized TypeScript types across new components to resolve implicit 'any' and 'never[]' errors.
- Improved overall frontend maintainability and code clarity.
2026-01-04 22:17:27 +08:00
|
|
|
<template #cell-email="{ value }"><div class="flex items-center gap-2"><div class="flex h-8 w-8 items-center justify-center rounded-full bg-primary-100 font-medium text-primary-700"><span>{{ value.charAt(0).toUpperCase() }}</span></div><span class="font-medium text-gray-900 dark:text-white">{{ value }}</span></div></template>
|
|
|
|
|
<template #cell-role="{ value }"><span :class="['badge', value === 'admin' ? 'badge-purple' : 'badge-gray']">{{ t('admin.users.roles.' + value) }}</span></template>
|
|
|
|
|
<template #cell-balance="{ value }"><span class="font-medium">${{ value.toFixed(2) }}</span></template>
|
2026-01-04 22:29:19 +08:00
|
|
|
<template #cell-status="{ value }"><StatusBadge :status="value === 'disabled' ? 'inactive' : value" :label="t('admin.accounts.status.' + (value === 'disabled' ? 'inactive' : value))" /></template>
|
|
|
|
|
<template #cell-actions="{ row }"><div class="flex gap-1"><button @click="handleEdit(row)" class="btn btn-sm btn-secondary">{{ t('common.edit') }}</button><button @click="openActionMenu(row, $event)" class="btn btn-sm btn-secondary">{{ t('common.more') }}</button></div></template>
|
2025-12-18 13:50:39 +08:00
|
|
|
</DataTable>
|
2025-12-27 10:50:25 +08:00
|
|
|
</template>
|
2025-12-18 13:50:39 +08:00
|
|
|
|
2025-12-27 10:50:25 +08:00
|
|
|
<template #pagination>
|
2026-01-04 22:29:19 +08:00
|
|
|
<Pagination v-if="pagination.total > 0" :page="pagination.page" :total="pagination.total" :page-size="pagination.page_size" @update:page="handlePageChange" />
|
2025-12-27 10:50:25 +08:00
|
|
|
</template>
|
|
|
|
|
</TablePageLayout>
|
2025-12-18 13:50:39 +08:00
|
|
|
|
2025-12-28 01:00:06 +08:00
|
|
|
<Teleport to="body">
|
2026-01-04 22:49:40 +08:00
|
|
|
<div v-if="activeMenuId !== null && menuPosition" ref="actionMenuEl" class="action-menu-content fixed z-[9999] w-48 max-h-[calc(100vh-16px)] overflow-auto rounded-xl bg-white shadow-lg ring-1 ring-black/5 dark:bg-dark-800" :style="{ top: menuPosition.top + 'px', left: menuPosition.left + 'px' }">
|
2025-12-28 01:00:06 +08:00
|
|
|
<div class="py-1">
|
|
|
|
|
<template v-for="user in users" :key="user.id">
|
|
|
|
|
<template v-if="user.id === activeMenuId">
|
2026-01-04 22:23:19 +08:00
|
|
|
<button @click="handleViewApiKeys(user); closeActionMenu()" class="flex w-full items-center gap-2 px-4 py-2 text-sm hover:bg-gray-100">{{ t('admin.users.apiKeys') }}</button>
|
|
|
|
|
<button @click="handleAllowedGroups(user); closeActionMenu()" class="flex w-full items-center gap-2 px-4 py-2 text-sm hover:bg-gray-100">{{ t('admin.users.groups') }}</button>
|
|
|
|
|
<button @click="handleDeposit(user); closeActionMenu()" class="flex w-full items-center gap-2 px-4 py-2 text-sm hover:bg-gray-100 text-emerald-600">{{ t('admin.users.deposit') }}</button>
|
|
|
|
|
<button @click="handleWithdraw(user); closeActionMenu()" class="flex w-full items-center gap-2 px-4 py-2 text-sm hover:bg-gray-100 text-amber-600">{{ t('admin.users.withdraw') }}</button>
|
2026-01-04 22:29:19 +08:00
|
|
|
<button v-if="user.role !== 'admin'" @click="handleToggleStatus(user); closeActionMenu()" class="flex w-full items-center gap-2 px-4 py-2 text-sm hover:bg-gray-100">{{ user.status === 'active' ? t('admin.users.disable') : t('admin.users.enable') }}</button>
|
2026-01-04 22:23:19 +08:00
|
|
|
<button v-if="user.role !== 'admin'" @click="handleDelete(user); closeActionMenu()" class="flex w-full items-center gap-2 px-4 py-2 text-sm text-red-600 hover:bg-red-50">{{ t('common.delete') }}</button>
|
2025-12-28 01:00:06 +08:00
|
|
|
</template>
|
|
|
|
|
</template>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</Teleport>
|
|
|
|
|
|
2026-01-04 22:23:19 +08:00
|
|
|
<ConfirmDialog :show="showDeleteDialog" :title="t('admin.users.deleteUser')" :message="t('admin.users.deleteConfirm', { email: deletingUser?.email })" :danger="true" @confirm="confirmDelete" @cancel="showDeleteDialog = false" />
|
2026-01-04 22:29:19 +08:00
|
|
|
<UserCreateModal :show="showCreateModal" @close="showCreateModal = false" @success="reload" />
|
|
|
|
|
<UserEditModal :show="showEditModal" :user="editingUser" @close="closeEditModal" @success="load" />
|
refactor(frontend): comprehensive split of large view files into modular components
- Split UsersView.vue into UserCreateModal, UserEditModal, UserApiKeysModal, etc.
- Split UsageView.vue into UsageStatsCards, UsageFilters, UsageTable, etc.
- Split DashboardView.vue into UserDashboardStats, UserDashboardCharts, etc.
- Split AccountsView.vue into AccountTableActions, AccountTableFilters, etc.
- Standardized TypeScript types across new components to resolve implicit 'any' and 'never[]' errors.
- Improved overall frontend maintainability and code clarity.
2026-01-04 22:17:27 +08:00
|
|
|
<UserApiKeysModal :show="showApiKeysModal" :user="viewingUser" @close="closeApiKeysModal" />
|
2026-01-04 22:29:19 +08:00
|
|
|
<UserAllowedGroupsModal :show="showAllowedGroupsModal" :user="allowedGroupsUser" @close="closeAllowedGroupsModal" @success="load" />
|
|
|
|
|
<UserBalanceModal :show="showBalanceModal" :user="balanceUser" :operation="balanceOperation" @close="closeBalanceModal" @success="load" />
|
2025-12-18 13:50:39 +08:00
|
|
|
</AppLayout>
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<script setup lang="ts">
|
2026-01-04 22:49:40 +08:00
|
|
|
import { ref, computed, onMounted, onBeforeUnmount, nextTick } from 'vue'
|
2026-01-04 22:29:19 +08:00
|
|
|
import { useI18n } from 'vue-i18n'; import { useAppStore } from '@/stores/app'
|
|
|
|
|
import { adminAPI } from '@/api/admin'; import { useTableLoader } from '@/composables/useTableLoader'
|
|
|
|
|
import type { User } from '@/types'
|
2026-01-04 22:23:19 +08:00
|
|
|
import AppLayout from '@/components/layout/AppLayout.vue'; import TablePageLayout from '@/components/layout/TablePageLayout.vue'
|
|
|
|
|
import DataTable from '@/components/common/DataTable.vue'; import Pagination from '@/components/common/Pagination.vue'
|
|
|
|
|
import ConfirmDialog from '@/components/common/ConfirmDialog.vue'; import Select from '@/components/common/Select.vue'
|
2026-01-04 22:29:19 +08:00
|
|
|
import SearchInput from '@/components/common/SearchInput.vue'; import StatusBadge from '@/components/common/StatusBadge.vue'
|
refactor(frontend): comprehensive split of large view files into modular components
- Split UsersView.vue into UserCreateModal, UserEditModal, UserApiKeysModal, etc.
- Split UsageView.vue into UsageStatsCards, UsageFilters, UsageTable, etc.
- Split DashboardView.vue into UserDashboardStats, UserDashboardCharts, etc.
- Split AccountsView.vue into AccountTableActions, AccountTableFilters, etc.
- Standardized TypeScript types across new components to resolve implicit 'any' and 'never[]' errors.
- Improved overall frontend maintainability and code clarity.
2026-01-04 22:17:27 +08:00
|
|
|
import UserCreateModal from '@/components/admin/user/UserCreateModal.vue'
|
|
|
|
|
import UserEditModal from '@/components/admin/user/UserEditModal.vue'
|
|
|
|
|
import UserApiKeysModal from '@/components/admin/user/UserApiKeysModal.vue'
|
|
|
|
|
import UserAllowedGroupsModal from '@/components/admin/user/UserAllowedGroupsModal.vue'
|
|
|
|
|
import UserBalanceModal from '@/components/admin/user/UserBalanceModal.vue'
|
|
|
|
|
|
|
|
|
|
const { t } = useI18n(); const appStore = useAppStore()
|
2026-01-04 22:29:19 +08:00
|
|
|
const { items: users, loading, params, pagination, load, reload, handlePageChange } = useTableLoader<User, any>({ fetchFn: adminAPI.users.list, initialParams: { role: '', status: '', search: '' } })
|
|
|
|
|
|
|
|
|
|
const showCreateModal = ref(false); const showEditModal = ref(false); const showDeleteDialog = ref(false); const showApiKeysModal = ref(false)
|
refactor(frontend): comprehensive split of large view files into modular components
- Split UsersView.vue into UserCreateModal, UserEditModal, UserApiKeysModal, etc.
- Split UsageView.vue into UsageStatsCards, UsageFilters, UsageTable, etc.
- Split DashboardView.vue into UserDashboardStats, UserDashboardCharts, etc.
- Split AccountsView.vue into AccountTableActions, AccountTableFilters, etc.
- Standardized TypeScript types across new components to resolve implicit 'any' and 'never[]' errors.
- Improved overall frontend maintainability and code clarity.
2026-01-04 22:17:27 +08:00
|
|
|
const editingUser = ref<User | null>(null); const deletingUser = ref<User | null>(null); const viewingUser = ref<User | null>(null)
|
2026-01-04 22:23:19 +08:00
|
|
|
const activeMenuId = ref<number | null>(null); const menuPosition = ref<{ top: number; left: number } | null>(null)
|
2026-01-04 22:49:40 +08:00
|
|
|
const actionMenuEl = ref<HTMLElement | null>(null)
|
refactor(frontend): comprehensive split of large view files into modular components
- Split UsersView.vue into UserCreateModal, UserEditModal, UserApiKeysModal, etc.
- Split UsageView.vue into UsageStatsCards, UsageFilters, UsageTable, etc.
- Split DashboardView.vue into UserDashboardStats, UserDashboardCharts, etc.
- Split AccountsView.vue into AccountTableActions, AccountTableFilters, etc.
- Standardized TypeScript types across new components to resolve implicit 'any' and 'never[]' errors.
- Improved overall frontend maintainability and code clarity.
2026-01-04 22:17:27 +08:00
|
|
|
const showAllowedGroupsModal = ref(false); const allowedGroupsUser = ref<User | null>(null); const showBalanceModal = ref(false); const balanceUser = ref<User | null>(null); const balanceOperation = ref<'add' | 'subtract'>('add')
|
2026-01-04 22:23:19 +08:00
|
|
|
const columns = computed(() => [{ key: 'email', label: t('admin.users.columns.user'), sortable: true }, { key: 'role', label: t('admin.users.columns.role'), sortable: true }, { key: 'balance', label: t('admin.users.columns.balance'), sortable: true }, { key: 'status', label: t('admin.users.columns.status'), sortable: true }, { key: 'actions', label: t('admin.users.columns.actions') }])
|
2026-01-01 18:59:38 +08:00
|
|
|
|
2026-01-04 22:23:19 +08:00
|
|
|
const handleEdit = (u: User) => { editingUser.value = u; showEditModal.value = true }
|
refactor(frontend): comprehensive split of large view files into modular components
- Split UsersView.vue into UserCreateModal, UserEditModal, UserApiKeysModal, etc.
- Split UsageView.vue into UsageStatsCards, UsageFilters, UsageTable, etc.
- Split DashboardView.vue into UserDashboardStats, UserDashboardCharts, etc.
- Split AccountsView.vue into AccountTableActions, AccountTableFilters, etc.
- Standardized TypeScript types across new components to resolve implicit 'any' and 'never[]' errors.
- Improved overall frontend maintainability and code clarity.
2026-01-04 22:17:27 +08:00
|
|
|
const closeEditModal = () => { showEditModal.value = false; editingUser.value = null }
|
2026-01-04 22:23:19 +08:00
|
|
|
const handleViewApiKeys = (u: User) => { viewingUser.value = u; showApiKeysModal.value = true }
|
refactor(frontend): comprehensive split of large view files into modular components
- Split UsersView.vue into UserCreateModal, UserEditModal, UserApiKeysModal, etc.
- Split UsageView.vue into UsageStatsCards, UsageFilters, UsageTable, etc.
- Split DashboardView.vue into UserDashboardStats, UserDashboardCharts, etc.
- Split AccountsView.vue into AccountTableActions, AccountTableFilters, etc.
- Standardized TypeScript types across new components to resolve implicit 'any' and 'never[]' errors.
- Improved overall frontend maintainability and code clarity.
2026-01-04 22:17:27 +08:00
|
|
|
const closeApiKeysModal = () => { showApiKeysModal.value = false; viewingUser.value = null }
|
2026-01-04 22:23:19 +08:00
|
|
|
const handleAllowedGroups = (u: User) => { allowedGroupsUser.value = u; showAllowedGroupsModal.value = true }
|
refactor(frontend): comprehensive split of large view files into modular components
- Split UsersView.vue into UserCreateModal, UserEditModal, UserApiKeysModal, etc.
- Split UsageView.vue into UsageStatsCards, UsageFilters, UsageTable, etc.
- Split DashboardView.vue into UserDashboardStats, UserDashboardCharts, etc.
- Split AccountsView.vue into AccountTableActions, AccountTableFilters, etc.
- Standardized TypeScript types across new components to resolve implicit 'any' and 'never[]' errors.
- Improved overall frontend maintainability and code clarity.
2026-01-04 22:17:27 +08:00
|
|
|
const closeAllowedGroupsModal = () => { showAllowedGroupsModal.value = false; allowedGroupsUser.value = null }
|
2026-01-04 22:23:19 +08:00
|
|
|
const handleDelete = (u: User) => { deletingUser.value = u; showDeleteDialog.value = true }
|
2026-01-04 22:29:19 +08:00
|
|
|
const confirmDelete = async () => { if (!deletingUser.value) return; try { await adminAPI.users.delete(deletingUser.value.id); appStore.showSuccess(t('common.success')); showDeleteDialog.value = false; reload() } catch {} }
|
2026-01-04 22:23:19 +08:00
|
|
|
const handleDeposit = (u: User) => { balanceUser.value = u; balanceOperation.value = 'add'; showBalanceModal.value = true }
|
|
|
|
|
const handleWithdraw = (u: User) => { balanceUser.value = u; balanceOperation.value = 'subtract'; showBalanceModal.value = true }
|
refactor(frontend): comprehensive split of large view files into modular components
- Split UsersView.vue into UserCreateModal, UserEditModal, UserApiKeysModal, etc.
- Split UsageView.vue into UsageStatsCards, UsageFilters, UsageTable, etc.
- Split DashboardView.vue into UserDashboardStats, UserDashboardCharts, etc.
- Split AccountsView.vue into AccountTableActions, AccountTableFilters, etc.
- Standardized TypeScript types across new components to resolve implicit 'any' and 'never[]' errors.
- Improved overall frontend maintainability and code clarity.
2026-01-04 22:17:27 +08:00
|
|
|
const closeBalanceModal = () => { showBalanceModal.value = false; balanceUser.value = null }
|
2026-01-04 22:29:19 +08:00
|
|
|
const handleToggleStatus = async (user: User) => { const next = user.status === 'active' ? 'disabled' : 'active'; try { await adminAPI.users.toggleStatus(user.id, next as any); appStore.showSuccess(t('common.success')); load() } catch {} }
|
2026-01-04 22:49:40 +08:00
|
|
|
const repositionActionMenu = (triggerRect?: DOMRect) => {
|
|
|
|
|
if (!menuPosition.value || !actionMenuEl.value) return
|
|
|
|
|
const rect = actionMenuEl.value.getBoundingClientRect()
|
|
|
|
|
const margin = 8
|
|
|
|
|
let top = menuPosition.value.top
|
|
|
|
|
let left = menuPosition.value.left
|
|
|
|
|
|
|
|
|
|
if (triggerRect) {
|
|
|
|
|
const spaceBelow = window.innerHeight - triggerRect.bottom
|
|
|
|
|
const spaceAbove = triggerRect.top
|
|
|
|
|
if (rect.height > spaceBelow && spaceAbove > spaceBelow) top = Math.max(margin, triggerRect.top - rect.height - 4)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (left + rect.width + margin > window.innerWidth) left = window.innerWidth - rect.width - margin
|
|
|
|
|
if (left < margin) left = margin
|
|
|
|
|
if (top + rect.height + margin > window.innerHeight) top = window.innerHeight - rect.height - margin
|
|
|
|
|
if (top < margin) top = margin
|
|
|
|
|
|
|
|
|
|
menuPosition.value = { top, left }
|
|
|
|
|
}
|
|
|
|
|
const openActionMenu = async (u: User, e: MouseEvent) => {
|
|
|
|
|
e.stopPropagation()
|
|
|
|
|
if (activeMenuId.value === u.id) { closeActionMenu(); return }
|
|
|
|
|
|
|
|
|
|
const actionMenuWidthPx = 192 // w-48
|
|
|
|
|
const triggerEl = e.currentTarget as HTMLElement | null
|
|
|
|
|
const triggerRect = triggerEl?.getBoundingClientRect()
|
|
|
|
|
|
|
|
|
|
activeMenuId.value = u.id
|
|
|
|
|
if (triggerRect) menuPosition.value = { top: triggerRect.bottom + 4, left: triggerRect.right - actionMenuWidthPx }
|
|
|
|
|
else menuPosition.value = { top: e.clientY, left: e.clientX - actionMenuWidthPx }
|
|
|
|
|
|
|
|
|
|
await nextTick()
|
|
|
|
|
repositionActionMenu(triggerRect)
|
|
|
|
|
}
|
refactor(frontend): comprehensive split of large view files into modular components
- Split UsersView.vue into UserCreateModal, UserEditModal, UserApiKeysModal, etc.
- Split UsageView.vue into UsageStatsCards, UsageFilters, UsageTable, etc.
- Split DashboardView.vue into UserDashboardStats, UserDashboardCharts, etc.
- Split AccountsView.vue into AccountTableActions, AccountTableFilters, etc.
- Standardized TypeScript types across new components to resolve implicit 'any' and 'never[]' errors.
- Improved overall frontend maintainability and code clarity.
2026-01-04 22:17:27 +08:00
|
|
|
const closeActionMenu = () => { activeMenuId.value = null; menuPosition.value = null }
|
2026-01-04 22:49:40 +08:00
|
|
|
|
|
|
|
|
const handleDocumentClick = (evt: MouseEvent) => { if (activeMenuId.value === null) return; const target = evt.target as Node | null; if (target && actionMenuEl.value?.contains(target)) return; closeActionMenu() }
|
|
|
|
|
const handleWindowResize = () => repositionActionMenu()
|
|
|
|
|
const handleAnyScroll = () => closeActionMenu()
|
|
|
|
|
|
|
|
|
|
onMounted(() => { load(); document.addEventListener('click', handleDocumentClick); window.addEventListener('resize', handleWindowResize); window.addEventListener('scroll', handleAnyScroll, true) })
|
|
|
|
|
onBeforeUnmount(() => { document.removeEventListener('click', handleDocumentClick); window.removeEventListener('resize', handleWindowResize); window.removeEventListener('scroll', handleAnyScroll, true) })
|
2026-01-04 22:23:19 +08:00
|
|
|
</script>
|