| <template> |
| <AppLayout> |
| <TablePageLayout> |
| |
| <template #filters> |
| <div class="flex flex-wrap items-center gap-3"> |
| |
| <div class="flex flex-1 flex-wrap items-center gap-3"> |
| |
| <div class="relative w-full md:w-64"> |
| <Icon |
| name="search" |
| size="md" |
| class="absolute left-3 top-1/2 -translate-y-1/2 text-gray-400" |
| /> |
| <input |
| v-model="searchQuery" |
| type="text" |
| :placeholder="t('admin.users.searchUsers')" |
| class="input pl-10" |
| @input="handleSearch" |
| /> |
| </div> |
| |
| |
| <div v-if="visibleFilters.has('role')" class="w-full sm:w-32"> |
| <Select |
| v-model="filters.role" |
| :options="[ |
| { value: '', label: t('admin.users.allRoles') }, |
| { value: 'admin', label: t('admin.users.admin') }, |
| { value: 'user', label: t('admin.users.user') } |
| ]" |
| @change="applyFilter" |
| /> |
| </div> |
| |
| |
| <div v-if="visibleFilters.has('status')" class="w-full sm:w-32"> |
| <Select |
| v-model="filters.status" |
| :options="[ |
| { value: '', label: t('admin.users.allStatus') }, |
| { value: 'active', label: t('common.active') }, |
| { value: 'disabled', label: t('admin.users.disabled') } |
| ]" |
| @change="applyFilter" |
| /> |
| </div> |
| |
| |
| <div v-if="visibleFilters.has('group')" class="w-full sm:w-44"> |
| <Select |
| v-model="filters.group" |
| :options="groupFilterOptions" |
| searchable |
| creatable |
| :creatable-prefix="t('admin.users.fuzzySearch')" |
| :search-placeholder="t('admin.users.searchGroups')" |
| @change="applyFilter" |
| /> |
| </div> |
| |
| |
| <template v-for="(value, attrId) in activeAttributeFilters" :key="attrId"> |
| <div |
| v-if="visibleFilters.has(`attr_${attrId}`)" |
| class="relative w-full sm:w-36" |
| > |
| |
| <input |
| v-if="['text', 'textarea', 'email', 'url', 'date'].includes(getAttributeDefinition(Number(attrId))?.type || 'text')" |
| :value="value" |
| @input="(e) => updateAttributeFilter(Number(attrId), (e.target as HTMLInputElement).value)" |
| @keyup.enter="applyFilter" |
| :placeholder="getAttributeDefinitionName(Number(attrId))" |
| class="input w-full" |
| /> |
| |
| <input |
| v-else-if="getAttributeDefinition(Number(attrId))?.type === 'number'" |
| :value="value" |
| type="number" |
| @input="(e) => updateAttributeFilter(Number(attrId), (e.target as HTMLInputElement).value)" |
| @keyup.enter="applyFilter" |
| :placeholder="getAttributeDefinitionName(Number(attrId))" |
| class="input w-full" |
| /> |
| |
| <template v-else-if="['select', 'multi_select'].includes(getAttributeDefinition(Number(attrId))?.type || '')"> |
| <div class="w-full"> |
| <Select |
| :model-value="value" |
| :options="[ |
| { value: '', label: getAttributeDefinitionName(Number(attrId)) }, |
| ...(getAttributeDefinition(Number(attrId))?.options || []) |
| ]" |
| @update:model-value="(val) => { updateAttributeFilter(Number(attrId), String(val ?? '')); applyFilter() }" |
| /> |
| </div> |
| </template> |
| |
| <input |
| v-else |
| :value="value" |
| @input="(e) => updateAttributeFilter(Number(attrId), (e.target as HTMLInputElement).value)" |
| @keyup.enter="applyFilter" |
| :placeholder="getAttributeDefinitionName(Number(attrId))" |
| class="input w-full" |
| /> |
| </div> |
| </template> |
| </div> |
| |
| |
| <div class="flex flex-wrap items-center justify-end gap-2"> |
| |
| <div class="flex items-center gap-2 md:contents"> |
| |
| <button |
| @click="loadUsers" |
| :disabled="loading" |
| class="btn btn-secondary px-2 md:px-3" |
| :title="t('common.refresh')" |
| > |
| <Icon name="refresh" size="md" :class="loading ? 'animate-spin' : ''" /> |
| </button> |
| |
| <div class="relative" ref="filterDropdownRef"> |
| <button |
| @click="showFilterDropdown = !showFilterDropdown" |
| class="btn btn-secondary px-2 md:px-3" |
| :title="t('admin.users.filterSettings')" |
| > |
| <Icon name="filter" size="sm" class="md:mr-1.5" /> |
| <span class="hidden md:inline">{{ t('admin.users.filterSettings') }}</span> |
| </button> |
| |
| <div |
| v-if="showFilterDropdown" |
| class="absolute right-0 top-full z-50 mt-1 w-48 rounded-lg border border-gray-200 bg-white py-1 shadow-lg dark:border-dark-600 dark:bg-dark-800" |
| > |
| |
| <button |
| v-for="filter in builtInFilters" |
| :key="filter.key" |
| @click="toggleBuiltInFilter(filter.key)" |
| class="flex w-full items-center justify-between px-4 py-2 text-left text-sm text-gray-700 hover:bg-gray-100 dark:text-gray-300 dark:hover:bg-dark-700" |
| > |
| <span>{{ filter.name }}</span> |
| <Icon |
| v-if="visibleFilters.has(filter.key)" |
| name="check" |
| size="sm" |
| class="text-primary-500" |
| :stroke-width="2" |
| /> |
| </button> |
| |
| <div |
| v-if="filterableAttributes.length > 0" |
| class="my-1 border-t border-gray-100 dark:border-dark-700" |
| ></div> |
| |
| <button |
| v-for="attr in filterableAttributes" |
| :key="attr.id" |
| @click="toggleAttributeFilter(attr)" |
| class="flex w-full items-center justify-between px-4 py-2 text-left text-sm text-gray-700 hover:bg-gray-100 dark:text-gray-300 dark:hover:bg-dark-700" |
| > |
| <span>{{ attr.name }}</span> |
| <Icon |
| v-if="visibleFilters.has(`attr_${attr.id}`)" |
| name="check" |
| size="sm" |
| class="text-primary-500" |
| :stroke-width="2" |
| /> |
| </button> |
| </div> |
| </div> |
| <!-- Column Settings Dropdown --> |
| <div class="relative" ref="columnDropdownRef"> |
| <button |
| @click="showColumnDropdown = !showColumnDropdown" |
| class="btn btn-secondary px-2 md:px-3" |
| :title="t('admin.users.columnSettings')" |
| > |
| <svg class="h-4 w-4 md:mr-1.5" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5"> |
| <path stroke-linecap="round" stroke-linejoin="round" d="M9 4.5v15m6-15v15m-10.875 0h15.75c.621 0 1.125-.504 1.125-1.125V5.625c0-.621-.504-1.125-1.125-1.125H4.125C3.504 4.5 3 5.004 3 5.625v12.75c0 .621.504 1.125 1.125 1.125z" /> |
| </svg> |
| <span class="hidden md:inline">{{ t('admin.users.columnSettings') }}</span> |
| </button> |
| <!-- Dropdown menu --> |
| <div |
| v-if="showColumnDropdown" |
| class="absolute right-0 top-full z-50 mt-1 max-h-80 w-48 overflow-y-auto rounded-lg border border-gray-200 bg-white py-1 shadow-lg dark:border-dark-600 dark:bg-dark-800" |
| > |
| <button |
| v-for="col in toggleableColumns" |
| :key="col.key" |
| @click="toggleColumn(col.key)" |
| class="flex w-full items-center justify-between px-4 py-2 text-left text-sm text-gray-700 hover:bg-gray-100 dark:text-gray-300 dark:hover:bg-dark-700" |
| > |
| <span>{{ col.label }}</span> |
| <Icon |
| v-if="isColumnVisible(col.key)" |
| name="check" |
| size="sm" |
| class="text-primary-500" |
| :stroke-width="2" |
| /> |
| </button> |
| </div> |
| </div> |
| <!-- Attributes Config Button --> |
| <button |
| @click="showAttributesModal = true" |
| class="btn btn-secondary px-2 md:px-3" |
| :title="t('admin.users.attributes.configButton')" |
| > |
| <Icon name="cog" size="sm" class="md:mr-1.5" /> |
| <span class="hidden md:inline">{{ t('admin.users.attributes.configButton') }}</span> |
| </button> |
| </div> |
|
|
| <!-- Create User Button (full width on mobile, auto width on desktop) --> |
| <button @click="showCreateModal = true" class="btn btn-primary flex-1 md:flex-initial"> |
| <Icon name="plus" size="md" class="mr-2" /> |
| {{ t('admin.users.createUser') }} |
| </button> |
| </div> |
| </div> |
| </template> |
|
|
| <!-- Users Table --> |
| <template #table> |
| <DataTable :columns="columns" :data="users" :loading="loading" :actions-count="7"> |
| <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 dark:bg-primary-900/30" |
| > |
| <span class="text-sm font-medium text-primary-700 dark:text-primary-300"> |
| {{ value.charAt(0).toUpperCase() }} |
| </span> |
| </div> |
| <span class="font-medium text-gray-900 dark:text-white">{{ value }}</span> |
| </div> |
| </template> |
|
|
| <template #cell-username="{ value }"> |
| <span class="text-sm text-gray-700 dark:text-gray-300">{{ value || '-' }}</span> |
| </template> |
|
|
| <template #cell-notes="{ value }"> |
| <div class="max-w-xs"> |
| <span |
| v-if="value" |
| :title="value.length > 30 ? value : undefined" |
| class="block truncate text-sm text-gray-600 dark:text-gray-400" |
| > |
| {{ value.length > 30 ? value.substring(0, 25) + '...' : value }} |
| </span> |
| <span v-else class="text-sm text-gray-400">-</span> |
| </div> |
| </template> |
|
|
| <!-- Dynamic attribute columns --> |
| <template |
| v-for="def in attributeDefinitions.filter(d => d.enabled)" |
| :key="def.id" |
| #[`cell-attr_${def.id}`]="{ row }" |
| > |
| <div class="max-w-xs"> |
| <span |
| class="block truncate text-sm text-gray-700 dark:text-gray-300" |
| :title="getAttributeValue(row.id, def.id)" |
| > |
| {{ getAttributeValue(row.id, def.id) }} |
| </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-groups="{ row }"> |
| <div v-if="allGroups.length > 0" class="flex flex-col gap-1"> |
| |
| <span |
| v-if="getUserGroups(row).exclusive.length > 0" |
| class="group/ex relative inline-flex cursor-pointer items-center gap-1 whitespace-nowrap text-xs" |
| @click.stop="toggleExpandedGroup(row.id)" |
| > |
| <Icon name="shield" size="xs" class="h-3.5 w-3.5 text-purple-500 dark:text-purple-400" /> |
| <span class="font-medium text-purple-600 dark:text-purple-400">{{ getUserGroups(row).exclusive.length }}</span> |
| <span class="text-gray-500 dark:text-dark-400">{{ t('admin.users.exclusiveLabel') }}</span> |
| |
| <div |
| v-if="expandedGroupUserId !== row.id" |
| class="pointer-events-none absolute left-0 top-full z-50 mt-1.5 rounded bg-gray-900 px-2.5 py-1.5 text-xs text-white opacity-0 shadow-lg transition-opacity duration-75 group-hover/ex:opacity-100 dark:bg-dark-600" |
| > |
| <div class="absolute left-4 bottom-full border-4 border-transparent border-b-gray-900 dark:border-b-dark-600"></div> |
| <div class="flex flex-col gap-0.5 whitespace-nowrap"> |
| <span v-for="g in getUserGroups(row).exclusive" :key="g.id">{{ g.name }}</span> |
| </div> |
| </div> |
| |
| <div |
| v-if="expandedGroupUserId === row.id" |
| class="absolute left-0 top-full z-50 mt-1.5 min-w-[160px] overflow-hidden rounded-lg border border-gray-200 bg-white py-1 text-xs shadow-xl dark:border-dark-600 dark:bg-dark-700" |
| > |
| <div class="border-b border-gray-100 px-3 py-1.5 text-[10px] font-medium uppercase tracking-wider text-gray-400 dark:border-dark-600 dark:text-dark-400"> |
| {{ t('admin.users.clickToReplace') }} |
| </div> |
| <div |
| v-for="g in getUserGroups(row).exclusive" |
| :key="g.id" |
| class="flex cursor-pointer items-center gap-2 px-3 py-2 text-gray-700 transition-colors hover:bg-primary-50 hover:text-primary-600 dark:text-dark-200 dark:hover:bg-primary-900/30 dark:hover:text-primary-400" |
| @click.stop="openGroupReplace(row, g)" |
| > |
| <Icon name="swap" size="xs" class="h-3.5 w-3.5 flex-shrink-0 opacity-50" /> |
| <span class="flex-1">{{ g.name }}</span> |
| </div> |
| </div> |
| </span> |
| <!-- 公开分组行 --> |
| <span |
| v-if="getUserGroups(row).publicGroups.length > 0" |
| class="group/pub relative inline-flex cursor-default items-center gap-1 whitespace-nowrap text-xs" |
| > |
| <Icon name="globe" size="xs" class="h-3.5 w-3.5 text-gray-400 dark:text-dark-500" /> |
| <span class="font-medium text-gray-600 dark:text-dark-300">{{ getUserGroups(row).publicGroups.length }}</span> |
| <span class="text-gray-400 dark:text-dark-500">{{ t('admin.users.publicLabel') }}</span> |
| <!-- Tooltip: 向下弹出 --> |
| <div class="pointer-events-none absolute left-0 top-full z-50 mt-1.5 rounded bg-gray-900 px-2.5 py-1.5 text-xs text-white opacity-0 shadow-lg transition-opacity duration-75 group-hover/pub:opacity-100 dark:bg-dark-600"> |
| <div class="absolute left-4 bottom-full border-4 border-transparent border-b-gray-900 dark:border-b-dark-600"></div> |
| <div class="flex flex-col gap-0.5 whitespace-nowrap"> |
| <span v-for="g in getUserGroups(row).publicGroups" :key="g.id">{{ g.name }}</span> |
| </div> |
| </div> |
| </span> |
| <!-- 都没有 --> |
| <span |
| v-if="getUserGroups(row).exclusive.length === 0 && getUserGroups(row).publicGroups.length === 0" |
| class="text-xs text-gray-400 dark:text-dark-500" |
| >-</span> |
| </div> |
| <span v-else class="text-xs text-gray-400 dark:text-dark-500">-</span> |
| </template> |
|
|
| <template #cell-subscriptions="{ row }"> |
| <div |
| v-if="row.subscriptions && row.subscriptions.length > 0" |
| class="flex flex-wrap gap-1.5" |
| > |
| <GroupBadge |
| v-for="sub in row.subscriptions" |
| :key="sub.id" |
| :name="sub.group?.name || ''" |
| :platform="sub.group?.platform" |
| :subscription-type="sub.group?.subscription_type" |
| :rate-multiplier="sub.group?.rate_multiplier" |
| :days-remaining="sub.expires_at ? getDaysRemaining(sub.expires_at) : null" |
| :title="sub.expires_at ? formatDateTime(sub.expires_at) : ''" |
| /> |
| </div> |
| <span |
| v-else |
| class="inline-flex items-center gap-1.5 rounded-md bg-gray-50 px-2 py-1 text-xs text-gray-400 dark:bg-dark-700/50 dark:text-dark-500" |
| > |
| <Icon name="ban" size="xs" class="h-3.5 w-3.5" /> |
| <span>{{ t('admin.users.noSubscription') }}</span> |
| </span> |
| </template> |
|
|
| <template #cell-balance="{ value, row }"> |
| <div class="flex items-center gap-2"> |
| <div class="group relative"> |
| <button |
| class="font-medium text-gray-900 underline decoration-dashed decoration-gray-300 underline-offset-4 transition-colors hover:text-primary-600 dark:text-white dark:decoration-dark-500 dark:hover:text-primary-400" |
| @click="handleBalanceHistory(row)" |
| > |
| ${{ value.toFixed(2) }} |
| </button> |
| |
| <div class="pointer-events-none absolute bottom-full left-1/2 z-50 mb-1.5 -translate-x-1/2 whitespace-nowrap rounded bg-gray-900 px-2 py-1 text-xs text-white opacity-0 shadow-lg transition-opacity duration-75 group-hover:opacity-100 dark:bg-dark-600"> |
| {{ t('admin.users.balanceHistoryTip') }} |
| <div class="absolute left-1/2 top-full -translate-x-1/2 border-4 border-transparent border-t-gray-900 dark:border-t-dark-600"></div> |
| </div> |
| </div> |
| <button |
| @click.stop="handleDeposit(row)" |
| class="rounded px-2 py-0.5 text-xs font-medium text-emerald-600 transition-colors hover:bg-emerald-50 dark:text-emerald-400 dark:hover:bg-emerald-900/20" |
| :title="t('admin.users.deposit')" |
| > |
| {{ t('admin.users.deposit') }} |
| </button> |
| </div> |
| </template> |
|
|
| <template #cell-usage="{ row }"> |
| <div class="text-sm"> |
| <div class="flex items-center gap-1.5"> |
| <span class="text-gray-500 dark:text-gray-400">{{ t('admin.users.today') }}:</span> |
| <span class="font-medium text-gray-900 dark:text-white"> |
| ${{ (usageStats[row.id]?.today_actual_cost ?? 0).toFixed(4) }} |
| </span> |
| </div> |
| <div class="mt-0.5 flex items-center gap-1.5"> |
| <span class="text-gray-500 dark:text-gray-400">{{ t('admin.users.total') }}:</span> |
| <span class="font-medium text-gray-900 dark:text-white"> |
| ${{ (usageStats[row.id]?.total_actual_cost ?? 0).toFixed(4) }} |
| </span> |
| </div> |
| </div> |
| </template> |
|
|
| <template #cell-concurrency="{ row }"> |
| <UserConcurrencyCell |
| :current="row.current_concurrency ?? 0" |
| :max="row.concurrency" |
| /> |
| </template> |
|
|
| <template #cell-status="{ value }"> |
| <div class="flex items-center gap-1.5"> |
| <span |
| :class="[ |
| 'inline-block h-2 w-2 rounded-full', |
| value === 'active' ? 'bg-green-500' : 'bg-red-500' |
| ]" |
| ></span> |
| <span class="text-sm text-gray-700 dark:text-gray-300"> |
| {{ value === 'active' ? t('common.active') : t('admin.users.disabled') }} |
| </span> |
| </div> |
| </template> |
|
|
| <template #cell-created_at="{ value }"> |
| <span class="text-sm text-gray-500 dark:text-dark-400">{{ formatDateTime(value) }}</span> |
| </template> |
|
|
| <template #cell-actions="{ row }"> |
| <div class="flex items-center gap-1"> |
| |
| <button |
| @click="handleEdit(row)" |
| class="flex flex-col items-center gap-0.5 rounded-lg p-1.5 text-gray-500 transition-colors hover:bg-gray-100 hover:text-primary-600 dark:hover:bg-dark-700 dark:hover:text-primary-400" |
| > |
| <Icon name="edit" size="sm" /> |
| <span class="text-xs">{{ t('common.edit') }}</span> |
| </button> |
| |
| |
| <button |
| v-if="row.role !== 'admin'" |
| @click="handleToggleStatus(row)" |
| :class="[ |
| 'flex flex-col items-center gap-0.5 rounded-lg p-1.5 text-gray-500 transition-colors', |
| row.status === 'active' |
| ? 'hover:bg-orange-50 hover:text-orange-600 dark:hover:bg-orange-900/20 dark:hover:text-orange-400' |
| : 'hover:bg-green-50 hover:text-green-600 dark:hover:bg-green-900/20 dark:hover:text-green-400' |
| ]" |
| > |
| <Icon v-if="row.status === 'active'" name="ban" size="sm" /> |
| <Icon v-else name="checkCircle" size="sm" /> |
| <span class="text-xs">{{ row.status === 'active' ? t('admin.users.disable') : t('admin.users.enable') }}</span> |
| </button> |
|
|
| <!-- More Actions Menu Trigger --> |
| <button |
| @click="openActionMenu(row, $event)" |
| class="action-menu-trigger flex flex-col items-center gap-0.5 rounded-lg p-1.5 text-gray-500 transition-colors hover:bg-gray-100 hover:text-gray-900 dark:hover:bg-dark-700 dark:hover:text-white" |
| :class="{ 'bg-gray-100 text-gray-900 dark:bg-dark-700 dark:text-white': activeMenuId === row.id }" |
| > |
| <Icon name="more" size="sm" /> |
| <span class="text-xs">{{ t('common.more') }}</span> |
| </button> |
| </div> |
| </template> |
|
|
| <template #empty> |
| <EmptyState |
| :title="t('admin.users.noUsersYet')" |
| :description="t('admin.users.createFirstUser')" |
| :action-text="t('admin.users.createUser')" |
| @action="showCreateModal = true" |
| /> |
| </template> |
| </DataTable> |
| </template> |
|
|
| <!-- Pagination --> |
| <template #pagination> |
| <Pagination |
| v-if="pagination.total > 0" |
| :page="pagination.page" |
| :total="pagination.total" |
| :page-size="pagination.page_size" |
| @update:page="handlePageChange" |
| @update:pageSize="handlePageSizeChange" |
| /> |
| </template> |
| </TablePageLayout> |
|
|
| <!-- Action Menu (Teleported) --> |
| <Teleport to="body"> |
| <div |
| v-if="activeMenuId !== null && menuPosition" |
| class="action-menu-content fixed z-[9999] w-48 overflow-hidden rounded-xl bg-white shadow-lg ring-1 ring-black/5 dark:bg-dark-800 dark:ring-white/10" |
| :style="{ top: menuPosition.top + 'px', left: menuPosition.left + 'px' }" |
| > |
| <div class="py-1"> |
| <template v-for="user in users" :key="user.id"> |
| <template v-if="user.id === activeMenuId"> |
| |
| <button |
| @click="handleViewApiKeys(user); closeActionMenu()" |
| class="flex w-full items-center gap-2 px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 dark:text-gray-300 dark:hover:bg-dark-700" |
| > |
| <Icon name="key" size="sm" class="text-gray-400" :stroke-width="2" /> |
| {{ t('admin.users.apiKeys') }} |
| </button> |
| |
| |
| <button |
| @click="handleAllowedGroups(user); closeActionMenu()" |
| class="flex w-full items-center gap-2 px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 dark:text-gray-300 dark:hover:bg-dark-700" |
| > |
| <Icon name="users" size="sm" class="text-gray-400" :stroke-width="2" /> |
| {{ t('admin.users.groups') }} |
| </button> |
| |
| <div class="my-1 border-t border-gray-100 dark:border-dark-700"></div> |
| |
| |
| <button |
| @click="handleDeposit(user); closeActionMenu()" |
| class="flex w-full items-center gap-2 px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 dark:text-gray-300 dark:hover:bg-dark-700" |
| > |
| <Icon name="plus" size="sm" class="text-emerald-500" :stroke-width="2" /> |
| {{ t('admin.users.deposit') }} |
| </button> |
| |
| |
| <button |
| @click="handleWithdraw(user); closeActionMenu()" |
| class="flex w-full items-center gap-2 px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 dark:text-gray-300 dark:hover:bg-dark-700" |
| > |
| <svg class="h-4 w-4 text-amber-500" fill="none" viewBox="0 0 24 24" stroke="currentColor"> |
| <path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M20 12H4" /> |
| </svg> |
| {{ t('admin.users.withdraw') }} |
| </button> |
|
|
| <!-- Balance History --> |
| <button |
| @click="handleBalanceHistory(user); closeActionMenu()" |
| class="flex w-full items-center gap-2 px-4 py-2 text-sm text-gray-700 hover:bg-gray-100 dark:text-gray-300 dark:hover:bg-dark-700" |
| > |
| <Icon name="dollar" size="sm" class="text-gray-400" :stroke-width="2" /> |
| {{ t('admin.users.balanceHistory') }} |
| </button> |
|
|
| <div class="my-1 border-t border-gray-100 dark:border-dark-700"></div> |
|
|
| <!-- Delete (not for admin) --> |
| <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 dark:text-red-400 dark:hover:bg-red-900/20" |
| > |
| <Icon name="trash" size="sm" :stroke-width="2" /> |
| {{ t('common.delete') }} |
| </button> |
| </template> |
| </template> |
| </div> |
| </div> |
| </Teleport> |
|
|
| <ConfirmDialog :show="showDeleteDialog" :title="t('admin.users.deleteUser')" :message="t('admin.users.deleteConfirm', { email: deletingUser?.email })" :danger="true" @confirm="confirmDelete" @cancel="showDeleteDialog = false" /> |
| <UserCreateModal :show="showCreateModal" @close="showCreateModal = false" @success="loadUsers" /> |
| <UserEditModal :show="showEditModal" :user="editingUser" @close="closeEditModal" @success="loadUsers" /> |
| <UserApiKeysModal :show="showApiKeysModal" :user="viewingUser" @close="closeApiKeysModal" /> |
| <UserAllowedGroupsModal :show="showAllowedGroupsModal" :user="allowedGroupsUser" @close="closeAllowedGroupsModal" @success="loadUsers" /> |
| <UserBalanceModal :show="showBalanceModal" :user="balanceUser" :operation="balanceOperation" @close="closeBalanceModal" @success="loadUsers" /> |
| <UserBalanceHistoryModal :show="showBalanceHistoryModal" :user="balanceHistoryUser" @close="closeBalanceHistoryModal" @deposit="handleDepositFromHistory" @withdraw="handleWithdrawFromHistory" /> |
| <GroupReplaceModal :show="showGroupReplaceModal" :user="groupReplaceUser" :old-group="groupReplaceOldGroup" :all-groups="allGroups" @close="closeGroupReplaceModal" @success="loadUsers" /> |
| <UserAttributesConfigModal :show="showAttributesModal" @close="handleAttributesModalClose" /> |
| </AppLayout> |
| </template> |
|
|
| <script setup lang="ts"> |
| import { ref, reactive, computed, onMounted, onUnmounted } from 'vue' |
| import { useI18n } from 'vue-i18n' |
| import { useAppStore } from '@/stores/app' |
| import { getPersistedPageSize } from '@/composables/usePersistedPageSize' |
| import { formatDateTime } from '@/utils/format' |
| import Icon from '@/components/icons/Icon.vue' |
|
|
| const { t } = useI18n() |
| import { adminAPI } from '@/api/admin' |
| import type { AdminUser, AdminGroup, UserAttributeDefinition } from '@/types' |
| import type { BatchUserUsageStats } from '@/api/admin/dashboard' |
| import type { Column } from '@/components/common/types' |
| 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 EmptyState from '@/components/common/EmptyState.vue' |
| import GroupBadge from '@/components/common/GroupBadge.vue' |
| import Select from '@/components/common/Select.vue' |
| import UserAttributesConfigModal from '@/components/user/UserAttributesConfigModal.vue' |
| import UserConcurrencyCell from '@/components/user/UserConcurrencyCell.vue' |
| 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' |
| import UserBalanceHistoryModal from '@/components/admin/user/UserBalanceHistoryModal.vue' |
| import GroupReplaceModal from '@/components/admin/user/GroupReplaceModal.vue' |
|
|
| const appStore = useAppStore() |
|
|
| // Generate dynamic attribute columns from enabled definitions |
| const attributeColumns = computed<Column[]>(() => |
| attributeDefinitions.value |
| .filter(def => def.enabled) |
| .map(def => ({ |
| key: `attr_${def.id}`, |
| label: def.name, |
| sortable: false |
| })) |
| ) |
|
|
| // Get formatted attribute value for display in table |
| const getAttributeValue = (userId: number, attrId: number): string => { |
| const userAttrs = userAttributeValues.value[userId] |
| if (!userAttrs) return '-' |
| const value = userAttrs[attrId] |
| if (!value) return '-' |
|
|
| // Find definition for this attribute |
| const def = attributeDefinitions.value.find(d => d.id === attrId) |
| if (!def) return value |
|
|
| // Format based on type |
| if (def.type === 'multi_select' && value) { |
| try { |
| const arr = JSON.parse(value) |
| if (Array.isArray(arr)) { |
| // Map values to labels |
| return arr.map(v => { |
| const opt = def.options?.find(o => o.value === v) |
| return opt?.label || v |
| }).join(', ') |
| } |
| } catch { |
| return value |
| } |
| } |
|
|
| if (def.type === 'select' && value && def.options) { |
| const opt = def.options.find(o => o.value === value) |
| return opt?.label || value |
| } |
|
|
| return value |
| } |
|
|
| // All possible columns (for column settings) |
| const allColumns = computed<Column[]>(() => [ |
| { key: 'email', label: t('admin.users.columns.user'), sortable: true }, |
| { key: 'id', label: 'ID', sortable: true }, |
| { key: 'username', label: t('admin.users.columns.username'), sortable: true }, |
| { key: 'notes', label: t('admin.users.columns.notes'), sortable: false }, |
| // Dynamic attribute columns |
| ...attributeColumns.value, |
| { key: 'role', label: t('admin.users.columns.role'), sortable: true }, |
| { key: 'groups', label: t('admin.users.columns.groups'), sortable: false }, |
| { key: 'subscriptions', label: t('admin.users.columns.subscriptions'), sortable: false }, |
| { key: 'balance', label: t('admin.users.columns.balance'), sortable: true }, |
| { key: 'usage', label: t('admin.users.columns.usage'), sortable: false }, |
| { key: 'concurrency', label: t('admin.users.columns.concurrency'), sortable: true }, |
| { key: 'status', label: t('admin.users.columns.status'), sortable: true }, |
| { key: 'created_at', label: t('admin.users.columns.created'), sortable: true }, |
| { key: 'actions', label: t('admin.users.columns.actions'), sortable: false } |
| ]) |
|
|
| // Columns that can be toggled (exclude email and actions which are always visible) |
| const toggleableColumns = computed(() => |
| allColumns.value.filter(col => col.key !== 'email' && col.key !== 'actions') |
| ) |
|
|
| // Hidden columns (stored in Set - columns NOT in this set are visible) |
| // This way, new columns are visible by default |
| const hiddenColumns = reactive<Set<string>>(new Set()) |
| |
| // Default hidden columns (columns hidden by default on first load) |
| const DEFAULT_HIDDEN_COLUMNS = ['notes', 'groups', 'subscriptions', 'usage', 'concurrency'] |
| |
| // localStorage key for column settings |
| const HIDDEN_COLUMNS_KEY = 'user-hidden-columns' |
| |
| // Load saved column settings |
| const loadSavedColumns = () => { |
| try { |
| const saved = localStorage.getItem(HIDDEN_COLUMNS_KEY) |
| if (saved) { |
| const parsed = JSON.parse(saved) as string[] |
| parsed.forEach(key => hiddenColumns.add(key)) |
| } else { |
| // Use default hidden columns on first load |
| DEFAULT_HIDDEN_COLUMNS.forEach(key => hiddenColumns.add(key)) |
| } |
| } catch (e) { |
| console.error('Failed to load saved columns:', e) |
| DEFAULT_HIDDEN_COLUMNS.forEach(key => hiddenColumns.add(key)) |
| } |
| } |
| |
| // Save column settings to localStorage |
| const saveColumnsToStorage = () => { |
| try { |
| localStorage.setItem(HIDDEN_COLUMNS_KEY, JSON.stringify([...hiddenColumns])) |
| } catch (e) { |
| console.error('Failed to save columns:', e) |
| } |
| } |
| |
| // Toggle column visibility |
| const toggleColumn = (key: string) => { |
| const wasHidden = hiddenColumns.has(key) |
| if (hiddenColumns.has(key)) { |
| hiddenColumns.delete(key) |
| } else { |
| hiddenColumns.add(key) |
| } |
| saveColumnsToStorage() |
| if (wasHidden && (key === 'usage' || key.startsWith('attr_'))) { |
| refreshCurrentPageSecondaryData() |
| } |
| if (key === 'subscriptions') { |
| loadUsers() |
| } |
| if (wasHidden && key === 'groups') { |
| loadAllGroups() |
| } |
| } |
| |
| // Check if column is visible (not in hidden set) |
| const isColumnVisible = (key: string) => !hiddenColumns.has(key) |
| const hasVisibleUsageColumn = computed(() => !hiddenColumns.has('usage')) |
| const hasVisibleSubscriptionsColumn = computed(() => !hiddenColumns.has('subscriptions')) |
| const hasVisibleGroupsColumn = computed(() => !hiddenColumns.has('groups')) |
| const hasVisibleAttributeColumns = computed(() => |
| attributeDefinitions.value.some((def) => def.enabled && !hiddenColumns.has(`attr_${def.id}`)) |
| ) |
| |
| // Filtered columns based on visibility |
| const columns = computed<Column[]>(() => |
| allColumns.value.filter(col => |
| col.key === 'email' || col.key === 'actions' || !hiddenColumns.has(col.key) |
| ) |
| ) |
| |
| const users = ref<AdminUser[]>([]) |
| const loading = ref(false) |
| const searchQuery = ref('') |
| |
| // Groups data for the groups column |
| const allGroups = ref<AdminGroup[]>([]) |
| const loadAllGroups = async () => { |
| if (allGroups.value.length > 0) return |
| try { |
| allGroups.value = await adminAPI.groups.getAll() |
| } catch (e) { |
| console.error('Failed to load groups:', e) |
| } |
| } |
| // Resolve user's accessible groups: exclusive groups first, then public groups |
| const getUserGroups = (user: AdminUser) => { |
| const exclusive: AdminGroup[] = [] |
| const publicGroups: AdminGroup[] = [] |
| for (const g of allGroups.value) { |
| if (g.status !== 'active' || g.subscription_type !== 'standard') continue |
| if (g.is_exclusive) { |
| if (user.allowed_groups?.includes(g.id)) { |
| exclusive.push(g) |
| } |
| } else { |
| publicGroups.push(g) |
| } |
| } |
| return { exclusive, publicGroups } |
| } |
| |
| // Group filter options: "All Groups" + active exclusive groups (value = group name for fuzzy match) |
| const groupFilterOptions = computed(() => { |
| const options: { value: string; label: string }[] = [ |
| { value: '', label: t('admin.users.allGroups') } |
| ] |
| for (const g of allGroups.value) { |
| if (g.status !== 'active' || !g.is_exclusive || g.subscription_type !== 'standard') continue |
| options.push({ value: g.name, label: g.name }) |
| } |
| return options |
| }) |
| |
| // Filter values (role, status, and custom attributes) |
| const filters = reactive({ |
| role: '', |
| status: '', |
| group: '' // group name for fuzzy match, '' = all |
| }) |
| const activeAttributeFilters = reactive<Record<number, string>>({}) |
| |
| // Visible filters tracking (which filters are shown in the UI) |
| // Keys: 'role', 'status', 'attr_${id}' |
| const visibleFilters = reactive<Set<string>>(new Set()) |
| |
| // Dropdown states |
| const showFilterDropdown = ref(false) |
| const showColumnDropdown = ref(false) |
| |
| // Dropdown refs for click outside detection |
| const filterDropdownRef = ref<HTMLElement | null>(null) |
| const columnDropdownRef = ref<HTMLElement | null>(null) |
| |
| // localStorage keys |
| const FILTER_VALUES_KEY = 'user-filter-values' |
| const VISIBLE_FILTERS_KEY = 'user-visible-filters' |
| |
| // All filterable attribute definitions (enabled attributes) |
| const filterableAttributes = computed(() => |
| attributeDefinitions.value.filter(def => def.enabled) |
| ) |
| |
| // Built-in filter definitions |
| const builtInFilters = computed(() => [ |
| { key: 'role', name: t('admin.users.columns.role'), type: 'select' as const }, |
| { key: 'status', name: t('admin.users.columns.status'), type: 'select' as const }, |
| { key: 'group', name: t('admin.users.columns.groups'), type: 'select' as const } |
| ]) |
| |
| // Load saved filters from localStorage |
| const loadSavedFilters = () => { |
| try { |
| // Load visible filters |
| const savedVisible = localStorage.getItem(VISIBLE_FILTERS_KEY) |
| if (savedVisible) { |
| const parsed = JSON.parse(savedVisible) as string[] |
| parsed.forEach(key => visibleFilters.add(key)) |
| } |
| // Load filter values |
| const savedValues = localStorage.getItem(FILTER_VALUES_KEY) |
| if (savedValues) { |
| const parsed = JSON.parse(savedValues) |
| if (parsed.role) filters.role = parsed.role |
| if (parsed.status) filters.status = parsed.status |
| if (parsed.group) filters.group = parsed.group |
| if (parsed.attributes) { |
| Object.assign(activeAttributeFilters, parsed.attributes) |
| } |
| } |
| } catch (e) { |
| console.error('Failed to load saved filters:', e) |
| } |
| } |
| |
| // Save filters to localStorage |
| const saveFiltersToStorage = () => { |
| try { |
| // Save visible filters |
| localStorage.setItem(VISIBLE_FILTERS_KEY, JSON.stringify([...visibleFilters])) |
| // Save filter values |
| const values = { |
| role: filters.role, |
| status: filters.status, |
| group: filters.group, |
| attributes: activeAttributeFilters |
| } |
| localStorage.setItem(FILTER_VALUES_KEY, JSON.stringify(values)) |
| } catch (e) { |
| console.error('Failed to save filters:', e) |
| } |
| } |
| |
| // Get attribute definition by ID |
| const getAttributeDefinition = (attrId: number): UserAttributeDefinition | undefined => { |
| return attributeDefinitions.value.find(d => d.id === attrId) |
| } |
| const usageStats = ref<Record<string, BatchUserUsageStats>>({}) |
| // User attribute definitions and values |
| const attributeDefinitions = ref<UserAttributeDefinition[]>([]) |
| const userAttributeValues = ref<Record<number, Record<number, string>>>({}) |
| const pagination = reactive({ |
| page: 1, |
| page_size: getPersistedPageSize(), |
| total: 0, |
| pages: 0 |
| }) |
| |
| const showCreateModal = ref(false) |
| const showEditModal = ref(false) |
| const showDeleteDialog = ref(false) |
| const showApiKeysModal = ref(false) |
| const showAttributesModal = ref(false) |
| const editingUser = ref<AdminUser | null>(null) |
| const deletingUser = ref<AdminUser | null>(null) |
| const viewingUser = ref<AdminUser | null>(null) |
| let abortController: AbortController | null = null |
| let secondaryDataSeq = 0 |
| |
| const loadUsersSecondaryData = async ( |
| userIds: number[], |
| signal?: AbortSignal, |
| expectedSeq?: number |
| ) => { |
| if (userIds.length === 0) return |
| |
| const tasks: Promise<void>[] = [] |
| |
| if (hasVisibleUsageColumn.value) { |
| tasks.push( |
| (async () => { |
| try { |
| const usageResponse = await adminAPI.dashboard.getBatchUsersUsage(userIds) |
| if (signal?.aborted) return |
| if (typeof expectedSeq === 'number' && expectedSeq !== secondaryDataSeq) return |
| usageStats.value = usageResponse.stats |
| } catch (e) { |
| if (signal?.aborted) return |
| console.error('Failed to load usage stats:', e) |
| } |
| })() |
| ) |
| } |
| |
| if (attributeDefinitions.value.length > 0 && hasVisibleAttributeColumns.value) { |
| tasks.push( |
| (async () => { |
| try { |
| const attrResponse = await adminAPI.userAttributes.getBatchUserAttributes(userIds) |
| if (signal?.aborted) return |
| if (typeof expectedSeq === 'number' && expectedSeq !== secondaryDataSeq) return |
| userAttributeValues.value = attrResponse.attributes |
| } catch (e) { |
| if (signal?.aborted) return |
| console.error('Failed to load user attribute values:', e) |
| } |
| })() |
| ) |
| } |
| |
| if (tasks.length > 0) { |
| await Promise.allSettled(tasks) |
| } |
| } |
| |
| const refreshCurrentPageSecondaryData = () => { |
| const userIds = users.value.map((u) => u.id) |
| if (userIds.length === 0) return |
| const seq = ++secondaryDataSeq |
| void loadUsersSecondaryData(userIds, undefined, seq) |
| } |
| |
| // Action Menu State |
| const activeMenuId = ref<number | null>(null) |
| const menuPosition = ref<{ top: number; left: number } | null>(null) |
| |
| const openActionMenu = (user: AdminUser, e: MouseEvent) => { |
| if (activeMenuId.value === user.id) { |
| closeActionMenu() |
| } else { |
| const target = e.currentTarget as HTMLElement |
| if (!target) { |
| closeActionMenu() |
| return |
| } |
| |
| const rect = target.getBoundingClientRect() |
| const menuWidth = 200 |
| const menuHeight = 240 |
| const padding = 8 |
| const viewportWidth = window.innerWidth |
| const viewportHeight = window.innerHeight |
| |
| let left, top |
| |
| if (viewportWidth < 768) { |
| // 居中显示,水平位置 |
| left = Math.max(padding, Math.min( |
| rect.left + rect.width / 2 - menuWidth / 2, |
| viewportWidth - menuWidth - padding |
| )) |
| |
| // 优先显示在按钮下方 |
| top = rect.bottom + 4 |
| |
| // 如果下方空间不够,显示在上方 |
| if (top + menuHeight > viewportHeight - padding) { |
| top = rect.top - menuHeight - 4 |
| // 如果上方也不够,就贴在视口顶部 |
| if (top < padding) { |
| top = padding |
| } |
| } |
| } else { |
| left = Math.max(padding, Math.min( |
| e.clientX - menuWidth, |
| viewportWidth - menuWidth - padding |
| )) |
| top = e.clientY |
| if (top + menuHeight > viewportHeight - padding) { |
| top = viewportHeight - menuHeight - padding |
| } |
| } |
| |
| menuPosition.value = { top, left } |
| activeMenuId.value = user.id |
| } |
| } |
| |
| const closeActionMenu = () => { |
| activeMenuId.value = null |
| menuPosition.value = null |
| } |
| |
| // Close menu when clicking outside |
| const handleClickOutside = (event: MouseEvent) => { |
| const target = event.target as HTMLElement |
| if (!target.closest('.action-menu-trigger') && !target.closest('.action-menu-content')) { |
| closeActionMenu() |
| } |
| // Close filter dropdown when clicking outside |
| if (filterDropdownRef.value && !filterDropdownRef.value.contains(target)) { |
| showFilterDropdown.value = false |
| } |
| // Close column dropdown when clicking outside |
| if (columnDropdownRef.value && !columnDropdownRef.value.contains(target)) { |
| showColumnDropdown.value = false |
| } |
| // Close expanded group dropdown when clicking outside |
| if (expandedGroupUserId.value !== null) { |
| expandedGroupUserId.value = null |
| } |
| } |
| |
| // Allowed groups modal state |
| const showAllowedGroupsModal = ref(false) |
| const allowedGroupsUser = ref<AdminUser | null>(null) |
| |
| // Expanded group dropdown state (click to show exclusive groups list) |
| const expandedGroupUserId = ref<number | null>(null) |
| const toggleExpandedGroup = (userId: number) => { |
| expandedGroupUserId.value = expandedGroupUserId.value === userId ? null : userId |
| } |
| |
| // Group replace modal state |
| const showGroupReplaceModal = ref(false) |
| const groupReplaceUser = ref<AdminUser | null>(null) |
| const groupReplaceOldGroup = ref<{ id: number; name: string } | null>(null) |
| |
| // Balance (Deposit/Withdraw) modal state |
| const showBalanceModal = ref(false) |
| const balanceUser = ref<AdminUser | null>(null) |
| const balanceOperation = ref<'add' | 'subtract'>('add') |
| |
| // Balance History modal state |
| const showBalanceHistoryModal = ref(false) |
| const balanceHistoryUser = ref<AdminUser | null>(null) |
| |
| // 计算剩余天数 |
| const getDaysRemaining = (expiresAt: string): number => { |
| const now = new Date() |
| const expires = new Date(expiresAt) |
| const diffMs = expires.getTime() - now.getTime() |
| return Math.ceil(diffMs / (1000 * 60 * 60 * 24)) |
| } |
| |
| const loadAttributeDefinitions = async () => { |
| try { |
| attributeDefinitions.value = await adminAPI.userAttributes.listEnabledDefinitions() |
| } catch (e) { |
| console.error('Failed to load attribute definitions:', e) |
| } |
| } |
| |
| // Handle attributes modal close - reload definitions and users |
| const handleAttributesModalClose = async () => { |
| showAttributesModal.value = false |
| await loadAttributeDefinitions() |
| loadUsers() |
| } |
| |
| const loadUsers = async () => { |
| abortController?.abort() |
| const currentAbortController = new AbortController() |
| abortController = currentAbortController |
| const { signal } = currentAbortController |
| loading.value = true |
| try { |
| // Build attribute filters from active filters |
| const attrFilters: Record<number, string> = {} |
| for (const [attrId, value] of Object.entries(activeAttributeFilters)) { |
| if (value) { |
| attrFilters[Number(attrId)] = value |
| } |
| } |
| |
| const response = await adminAPI.users.list( |
| pagination.page, |
| pagination.page_size, |
| { |
| role: filters.role as any, |
| status: filters.status as any, |
| search: searchQuery.value || undefined, |
| group_name: filters.group || undefined, |
| attributes: Object.keys(attrFilters).length > 0 ? attrFilters : undefined, |
| include_subscriptions: hasVisibleSubscriptionsColumn.value |
| }, |
| { signal } |
| ) |
| if (signal.aborted) { |
| return |
| } |
| users.value = response.items |
| pagination.total = response.total |
| pagination.pages = response.pages |
| usageStats.value = {} |
| userAttributeValues.value = {} |
| |
| // Defer heavy secondary data so table can render first. |
| if (response.items.length > 0) { |
| const userIds = response.items.map((u) => u.id) |
| const seq = ++secondaryDataSeq |
| window.setTimeout(() => { |
| if (signal.aborted || seq !== secondaryDataSeq) return |
| void loadUsersSecondaryData(userIds, signal, seq) |
| }, 50) |
| } |
| } catch (error: any) { |
| const errorInfo = error as { name?: string; code?: string } |
| if (errorInfo?.name === 'AbortError' || errorInfo?.name === 'CanceledError' || errorInfo?.code === 'ERR_CANCELED') { |
| return |
| } |
| const message = error.response?.data?.detail || error.message || t('admin.users.failedToLoad') |
| appStore.showError(message) |
| console.error('Error loading users:', error) |
| } finally { |
| if (abortController === currentAbortController) { |
| loading.value = false |
| } |
| } |
| } |
| |
| let searchTimeout: ReturnType<typeof setTimeout> |
| const handleSearch = () => { |
| clearTimeout(searchTimeout) |
| searchTimeout = setTimeout(() => { |
| pagination.page = 1 |
| loadUsers() |
| }, 300) |
| } |
| |
| const handlePageChange = (page: number) => { |
| // 确保页码在有效范围内 |
| const validPage = Math.max(1, Math.min(page, pagination.pages || 1)) |
| pagination.page = validPage |
| loadUsers() |
| } |
| |
| const handlePageSizeChange = (pageSize: number) => { |
| pagination.page_size = pageSize |
| pagination.page = 1 |
| loadUsers() |
| } |
| |
| // Filter helpers |
| const getAttributeDefinitionName = (attrId: number): string => { |
| const def = attributeDefinitions.value.find(d => d.id === attrId) |
| return def?.name || String(attrId) |
| } |
| |
| // Toggle a built-in filter (role/status) |
| const toggleBuiltInFilter = (key: string) => { |
| if (visibleFilters.has(key)) { |
| visibleFilters.delete(key) |
| if (key === 'role') filters.role = '' |
| if (key === 'status') filters.status = '' |
| if (key === 'group') filters.group = '' |
| } else { |
| visibleFilters.add(key) |
| if (key === 'group') loadAllGroups() |
| } |
| saveFiltersToStorage() |
| pagination.page = 1 |
| loadUsers() |
| } |
| |
| // Toggle a custom attribute filter |
| const toggleAttributeFilter = (attr: UserAttributeDefinition) => { |
| const key = `attr_${attr.id}` |
| if (visibleFilters.has(key)) { |
| visibleFilters.delete(key) |
| delete activeAttributeFilters[attr.id] |
| } else { |
| visibleFilters.add(key) |
| activeAttributeFilters[attr.id] = '' |
| } |
| saveFiltersToStorage() |
| pagination.page = 1 |
| loadUsers() |
| } |
| |
| const updateAttributeFilter = (attrId: number, value: string) => { |
| activeAttributeFilters[attrId] = value |
| } |
| |
| // Apply filter and save to localStorage |
| const applyFilter = () => { |
| saveFiltersToStorage() |
| loadUsers() |
| } |
| |
| const handleEdit = (user: AdminUser) => { |
| editingUser.value = user |
| showEditModal.value = true |
| } |
| |
| const closeEditModal = () => { |
| showEditModal.value = false |
| editingUser.value = null |
| } |
| |
| const handleToggleStatus = async (user: AdminUser) => { |
| const newStatus = user.status === 'active' ? 'disabled' : 'active' |
| try { |
| await adminAPI.users.toggleStatus(user.id, newStatus) |
| appStore.showSuccess( |
| newStatus === 'active' ? t('admin.users.userEnabled') : t('admin.users.userDisabled') |
| ) |
| loadUsers() |
| } catch (error: any) { |
| appStore.showError(error.response?.data?.detail || t('admin.users.failedToToggle')) |
| console.error('Error toggling user status:', error) |
| } |
| } |
| |
| const handleViewApiKeys = (user: AdminUser) => { |
| viewingUser.value = user |
| showApiKeysModal.value = true |
| } |
| |
| const closeApiKeysModal = () => { |
| showApiKeysModal.value = false |
| viewingUser.value = null |
| } |
| |
| const handleAllowedGroups = (user: AdminUser) => { |
| allowedGroupsUser.value = user |
| showAllowedGroupsModal.value = true |
| } |
| |
| const closeAllowedGroupsModal = () => { |
| showAllowedGroupsModal.value = false |
| allowedGroupsUser.value = null |
| } |
| |
| const openGroupReplace = (user: AdminUser, group: { id: number; name: string }) => { |
| expandedGroupUserId.value = null |
| groupReplaceUser.value = user |
| groupReplaceOldGroup.value = group |
| showGroupReplaceModal.value = true |
| } |
| |
| const closeGroupReplaceModal = () => { |
| showGroupReplaceModal.value = false |
| groupReplaceUser.value = null |
| groupReplaceOldGroup.value = null |
| } |
| |
| const handleDelete = (user: AdminUser) => { |
| deletingUser.value = user |
| showDeleteDialog.value = true |
| } |
| |
| const confirmDelete = async () => { |
| if (!deletingUser.value) return |
| try { |
| await adminAPI.users.delete(deletingUser.value.id) |
| appStore.showSuccess(t('common.success')) |
| showDeleteDialog.value = false |
| deletingUser.value = null |
| loadUsers() |
| } catch (error: any) { |
| appStore.showError(error.response?.data?.detail || t('admin.users.failedToDelete')) |
| console.error('Error deleting user:', error) |
| } |
| } |
| |
| const handleDeposit = (user: AdminUser) => { |
| balanceUser.value = user |
| balanceOperation.value = 'add' |
| showBalanceModal.value = true |
| } |
| |
| const handleWithdraw = (user: AdminUser) => { |
| balanceUser.value = user |
| balanceOperation.value = 'subtract' |
| showBalanceModal.value = true |
| } |
| |
| const closeBalanceModal = () => { |
| showBalanceModal.value = false |
| balanceUser.value = null |
| } |
| |
| const handleBalanceHistory = (user: AdminUser) => { |
| balanceHistoryUser.value = user |
| showBalanceHistoryModal.value = true |
| } |
| |
| const closeBalanceHistoryModal = () => { |
| showBalanceHistoryModal.value = false |
| balanceHistoryUser.value = null |
| } |
| |
| // Handle deposit from balance history modal |
| const handleDepositFromHistory = () => { |
| if (balanceHistoryUser.value) { |
| handleDeposit(balanceHistoryUser.value) |
| } |
| } |
| |
| // Handle withdraw from balance history modal |
| const handleWithdrawFromHistory = () => { |
| if (balanceHistoryUser.value) { |
| handleWithdraw(balanceHistoryUser.value) |
| } |
| } |
| |
| // 滚动时关闭菜单 |
| const handleScroll = () => { |
| closeActionMenu() |
| } |
| |
| onMounted(async () => { |
| await loadAttributeDefinitions() |
| loadSavedFilters() |
| loadSavedColumns() |
| loadUsers() |
| if (hasVisibleGroupsColumn.value || visibleFilters.has('group')) { |
| loadAllGroups() |
| } |
| document.addEventListener('click', handleClickOutside) |
| window.addEventListener('scroll', handleScroll, true) |
| }) |
| |
| onUnmounted(() => { |
| document.removeEventListener('click', handleClickOutside) |
| window.removeEventListener('scroll', handleScroll, true) |
| clearTimeout(searchTimeout) |
| abortController?.abort() |
| }) |
| </script> |
| |