## 功能特性 ### 前端 - 在用户列表操作列添加充值和退款按钮 - 实现充值/退款对话框,支持输入金额和备注 - 从编辑用户表单中移除余额字段,防止直接修改 - 添加余额不足验证,实时显示操作后余额 - 优化备注提示词,提供多种场景示例 ### 后端 - 为 redeem_codes 表添加 notes 字段(迁移文件) - 在 UpdateUserBalance 接口添加 notes 参数支持 - 添加余额验证:金额必须大于0,操作后余额不能为负 - UpdateUser 接口移除 balance 字段处理,防止误操作 - 完整的审计日志和缓存管理 ## 安全保护 - 前端:余额不足时禁用提交按钮,实时提示 - 后端:双重验证(输入金额 > 0 + 结果余额 >= 0) - 权限:仅管理员可访问(AdminAuth 中间件) - 审计:所有操作记录到 redeem_codes 表 ## 修改文件 后端: - backend/migrations/004_add_redeem_code_notes.sql - backend/internal/model/redeem_code.go - backend/internal/service/admin_service.go - backend/internal/handler/admin/user_handler.go 前端: - frontend/src/views/admin/UsersView.vue - frontend/src/api/admin/users.ts - frontend/src/i18n/locales/zh.ts - frontend/src/i18n/locales/en.ts 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-authored-by: Claude Sonnet 4.5 <noreply@anthropic.com>
1334 lines
54 KiB
Vue
1334 lines
54 KiB
Vue
<template>
|
||
<AppLayout>
|
||
<div class="space-y-6">
|
||
<!-- Page Header Actions -->
|
||
<div class="flex justify-end gap-3">
|
||
<button
|
||
@click="loadUsers"
|
||
:disabled="loading"
|
||
class="btn btn-secondary"
|
||
:title="t('common.refresh')"
|
||
>
|
||
<svg
|
||
:class="['w-5 h-5', loading ? 'animate-spin' : '']"
|
||
fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="1.5"
|
||
>
|
||
<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>
|
||
<button
|
||
@click="showCreateModal = true"
|
||
class="btn btn-primary"
|
||
>
|
||
<svg class="w-5 h-5 mr-2" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M12 4.5v15m7.5-7.5h-15" />
|
||
</svg>
|
||
{{ t('admin.users.createUser') }}
|
||
</button>
|
||
</div>
|
||
|
||
<!-- Search and Filters -->
|
||
<div class="flex flex-col gap-4 lg:flex-row lg:items-center lg:justify-between">
|
||
<div class="relative flex-1 max-w-md">
|
||
<svg class="absolute left-3 top-1/2 -translate-y-1/2 w-5 h-5 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M21 21l-5.197-5.197m0 0A7.5 7.5 0 105.196 5.196a7.5 7.5 0 0010.607 10.607z" />
|
||
</svg>
|
||
<input
|
||
v-model="searchQuery"
|
||
type="text"
|
||
:placeholder="t('admin.users.searchUsers')"
|
||
class="input pl-10"
|
||
@input="handleSearch"
|
||
/>
|
||
</div>
|
||
<div class="flex flex-wrap gap-3">
|
||
<Select
|
||
v-model="filters.role"
|
||
:options="roleOptions"
|
||
:placeholder="t('admin.users.allRoles')"
|
||
class="w-36"
|
||
@change="loadUsers"
|
||
/>
|
||
<Select
|
||
v-model="filters.status"
|
||
:options="statusOptions"
|
||
:placeholder="t('admin.users.allStatus')"
|
||
class="w-36"
|
||
@change="loadUsers"
|
||
/>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Users Table -->
|
||
<div class="card overflow-hidden">
|
||
<DataTable :columns="columns" :data="users" :loading="loading">
|
||
<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-wechat="{ 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="text-sm text-gray-600 dark:text-gray-400 block truncate"
|
||
>
|
||
{{ value.length > 30 ? value.substring(0, 25) + '...' : value }}
|
||
</span>
|
||
<span v-else class="text-sm text-gray-400">-</span>
|
||
</div>
|
||
</template>
|
||
|
||
<template #cell-role="{ value }">
|
||
<span
|
||
:class="[
|
||
'badge',
|
||
value === 'admin' ? 'badge-purple' : 'badge-gray'
|
||
]"
|
||
>
|
||
{{ value }}
|
||
</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 ? formatExpiresAt(sub.expires_at) : ''"
|
||
/>
|
||
</div>
|
||
<span v-else class="inline-flex items-center gap-1.5 px-2 py-1 rounded-md text-xs text-gray-400 dark:text-dark-500 bg-gray-50 dark:bg-dark-700/50">
|
||
<svg class="w-3.5 h-3.5" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M18.364 18.364A9 9 0 005.636 5.636m12.728 12.728A9 9 0 015.636 5.636m12.728 12.728L5.636 5.636" />
|
||
</svg>
|
||
<span>{{ t('admin.users.noSubscription') }}</span>
|
||
</span>
|
||
</template>
|
||
|
||
<template #cell-balance="{ value }">
|
||
<span class="font-medium text-gray-900 dark:text-white">${{ value.toFixed(2) }}</span>
|
||
</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="flex items-center gap-1.5 mt-0.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="{ value }">
|
||
<span class="text-sm text-gray-700 dark:text-gray-300">{{ value }}</span>
|
||
</template>
|
||
|
||
<template #cell-status="{ value }">
|
||
<span
|
||
:class="[
|
||
'badge',
|
||
value === 'active' ? 'badge-success' : 'badge-danger'
|
||
]"
|
||
>
|
||
{{ value }}
|
||
</span>
|
||
</template>
|
||
|
||
<template #cell-created_at="{ value }">
|
||
<span class="text-sm text-gray-500 dark:text-dark-400">{{ formatDate(value) }}</span>
|
||
</template>
|
||
|
||
<template #cell-actions="{ row }">
|
||
<div class="flex items-center gap-1">
|
||
<!-- Toggle Status (hidden for admin users) -->
|
||
<button
|
||
v-if="row.role !== 'admin'"
|
||
@click="handleToggleStatus(row)"
|
||
:class="[
|
||
'p-2 rounded-lg transition-colors',
|
||
row.status === 'active'
|
||
? 'hover:bg-orange-50 dark:hover:bg-orange-900/20 text-gray-500 hover:text-orange-600 dark:hover:text-orange-400'
|
||
: 'hover:bg-green-50 dark:hover:bg-green-900/20 text-gray-500 hover:text-green-600 dark:hover:text-green-400'
|
||
]"
|
||
:title="row.status === 'active' ? t('admin.users.disableUser') : t('admin.users.enableUser')"
|
||
>
|
||
<svg v-if="row.status === 'active'" class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M18.364 18.364A9 9 0 005.636 5.636m12.728 12.728A9 9 0 015.636 5.636m12.728 12.728L5.636 5.636" />
|
||
</svg>
|
||
<svg v-else class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M9 12.75L11.25 15 15 9.75M21 12a9 9 0 11-18 0 9 9 0 0118 0z" />
|
||
</svg>
|
||
</button>
|
||
<!-- Allowed Groups -->
|
||
<button
|
||
@click="handleAllowedGroups(row)"
|
||
class="p-2 rounded-lg hover:bg-blue-50 dark:hover:bg-blue-900/20 text-gray-500 hover:text-blue-600 dark:hover:text-blue-400 transition-colors"
|
||
:title="t('admin.users.setAllowedGroups')"
|
||
>
|
||
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M18 18.72a9.094 9.094 0 003.741-.479 3 3 0 00-4.682-2.72m.94 3.198l.001.031c0 .225-.012.447-.037.666A11.944 11.944 0 0112 21c-2.17 0-4.207-.576-5.963-1.584A6.062 6.062 0 016 18.719m12 0a5.971 5.971 0 00-.941-3.197m0 0A5.995 5.995 0 0012 12.75a5.995 5.995 0 00-5.058 2.772m0 0a3 3 0 00-4.681 2.72 8.986 8.986 0 003.74.477m.94-3.197a5.971 5.971 0 00-.94 3.197M15 6.75a3 3 0 11-6 0 3 3 0 016 0zm6 3a2.25 2.25 0 11-4.5 0 2.25 2.25 0 014.5 0zm-13.5 0a2.25 2.25 0 11-4.5 0 2.25 2.25 0 014.5 0z" />
|
||
</svg>
|
||
</button>
|
||
<!-- View API Keys -->
|
||
<button
|
||
@click="handleViewApiKeys(row)"
|
||
class="p-2 rounded-lg hover:bg-purple-50 dark:hover:bg-purple-900/20 text-gray-500 hover:text-purple-600 dark:hover:text-purple-400 transition-colors"
|
||
:title="t('admin.users.viewApiKeys')"
|
||
>
|
||
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M15.75 5.25a3 3 0 013 3m3 0a6 6 0 01-7.029 5.912c-.563-.097-1.159.026-1.563.43L10.5 17.25H8.25v2.25H6v2.25H2.25v-2.818c0-.597.237-1.17.659-1.591l6.499-6.499c.404-.404.527-1 .43-1.563A6 6 0 1121.75 8.25z" />
|
||
</svg>
|
||
</button>
|
||
<!-- Deposit -->
|
||
<button
|
||
@click="handleDeposit(row)"
|
||
class="p-2 rounded-lg hover:bg-emerald-50 dark:hover:bg-emerald-900/20 text-gray-500 hover:text-emerald-600 dark:hover:text-emerald-400 transition-colors"
|
||
:title="t('admin.users.deposit')"
|
||
>
|
||
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M12 4.5v15m7.5-7.5h-15" />
|
||
</svg>
|
||
</button>
|
||
<!-- Withdraw -->
|
||
<button
|
||
@click="handleWithdraw(row)"
|
||
class="p-2 rounded-lg hover:bg-red-50 dark:hover:bg-red-900/20 text-gray-500 hover:text-red-600 dark:hover:text-red-400 transition-colors"
|
||
:title="t('admin.users.withdraw')"
|
||
>
|
||
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M5 12h14" />
|
||
</svg>
|
||
</button>
|
||
<!-- Edit -->
|
||
<button
|
||
@click="handleEdit(row)"
|
||
class="p-2 rounded-lg hover:bg-gray-100 dark:hover:bg-dark-700 text-gray-500 hover:text-primary-600 dark:hover:text-primary-400 transition-colors"
|
||
:title="t('common.edit')"
|
||
>
|
||
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round"
|
||
d="M16.862 4.487l1.687-1.688a1.875 1.875 0 112.652 2.652L10.582 16.07a4.5 4.5 0 01-1.897 1.13L6 18l.8-2.685a4.5 4.5 0 011.13-1.897l8.932-8.931zm0 0L19.5 7.125M18 14v4.75A2.25 2.25 0 0115.75 21H5.25A2.25 2.25 0 013 18.75V8.25A2.25 2.25 0 015.25 6H10" />
|
||
</svg>
|
||
</button>
|
||
<!-- Delete (hidden for admin users) -->
|
||
<button
|
||
v-if="row.role !== 'admin'"
|
||
@click="handleDelete(row)"
|
||
class="p-2 rounded-lg hover:bg-red-50 dark:hover:bg-red-900/20 text-gray-500 hover:text-red-600 dark:hover:text-red-400 transition-colors"
|
||
:title="t('common.delete')"
|
||
>
|
||
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round"
|
||
d="M14.74 9l-.346 9m-4.788 0L9.26 9m9.968-3.21c.342.052.682.107 1.022.166m-1.022-.165L18.16 19.673a2.25 2.25 0 01-2.244 2.077H8.084a2.25 2.25 0 01-2.244-2.077L4.772 5.79m14.456 0a48.108 48.108 0 00-3.478-.397m-12 .562c.34-.059.68-.114 1.022-.165m0 0a48.11 48.11 0 013.478-.397m7.5 0v-.916c0-1.18-.91-2.164-2.09-2.201a51.964 51.964 0 00-3.32 0c-1.18.037-2.09 1.022-2.09 2.201v.916m7.5 0a48.667 48.667 0 00-7.5 0" />
|
||
</svg>
|
||
</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>
|
||
</div>
|
||
|
||
<!-- Pagination -->
|
||
<Pagination
|
||
v-if="pagination.total > 0"
|
||
:page="pagination.page"
|
||
:total="pagination.total"
|
||
:page-size="pagination.page_size"
|
||
@update:page="handlePageChange"
|
||
/>
|
||
</div>
|
||
|
||
<!-- Create User Modal -->
|
||
<Modal
|
||
:show="showCreateModal"
|
||
:title="t('admin.users.createUser')"
|
||
size="lg"
|
||
@close="closeCreateModal"
|
||
>
|
||
<form @submit.prevent="handleCreateUser" class="space-y-5">
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.email') }}</label>
|
||
<input
|
||
v-model="createForm.email"
|
||
type="email"
|
||
required
|
||
class="input"
|
||
:placeholder="t('admin.users.enterEmail')"
|
||
/>
|
||
</div>
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.password') }}</label>
|
||
<div class="flex gap-2">
|
||
<div class="relative flex-1">
|
||
<input
|
||
v-model="createForm.password"
|
||
type="text"
|
||
required
|
||
class="input pr-10"
|
||
:placeholder="t('admin.users.enterPassword')"
|
||
/>
|
||
<!-- Copy Password Button -->
|
||
<button
|
||
v-if="createForm.password"
|
||
type="button"
|
||
@click="copyPassword"
|
||
class="absolute right-2 top-1/2 -translate-y-1/2 p-1 rounded-lg hover:bg-gray-100 dark:hover:bg-dark-700 transition-colors"
|
||
:class="passwordCopied ? 'text-green-500' : 'text-gray-400 hover:text-gray-600 dark:hover:text-gray-300'"
|
||
:title="passwordCopied ? t('keys.copied') : t('admin.users.copyPassword')"
|
||
>
|
||
<svg v-if="passwordCopied" class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M5 13l4 4L19 7" />
|
||
</svg>
|
||
<svg v-else class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round"
|
||
d="M15.666 3.888A2.25 2.25 0 0013.5 2.25h-3c-1.03 0-1.9.693-2.166 1.638m7.332 0c.055.194.084.4.084.612v0a.75.75 0 01-.75.75H9a.75.75 0 01-.75-.75v0c0-.212.03-.418.084-.612m7.332 0c.646.049 1.288.11 1.927.184 1.1.128 1.907 1.077 1.907 2.185V19.5a2.25 2.25 0 01-2.25 2.25H6.75A2.25 2.25 0 014.5 19.5V6.257c0-1.108.806-2.057 1.907-2.185a48.208 48.208 0 011.927-.184" />
|
||
</svg>
|
||
</button>
|
||
</div>
|
||
<!-- Generate Random Password Button -->
|
||
<button
|
||
type="button"
|
||
@click="generateRandomPassword"
|
||
class="btn btn-secondary px-3"
|
||
:title="t('admin.users.generatePassword')"
|
||
>
|
||
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<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>
|
||
</div>
|
||
</div>
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.username') }}</label>
|
||
<input
|
||
v-model="createForm.username"
|
||
type="text"
|
||
class="input"
|
||
:placeholder="t('admin.users.enterUsername')"
|
||
/>
|
||
</div>
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.wechat') }}</label>
|
||
<input
|
||
v-model="createForm.wechat"
|
||
type="text"
|
||
class="input"
|
||
:placeholder="t('admin.users.enterWechat')"
|
||
/>
|
||
</div>
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.notes') }}</label>
|
||
<textarea
|
||
v-model="createForm.notes"
|
||
rows="3"
|
||
class="input"
|
||
:placeholder="t('admin.users.enterNotes')"
|
||
></textarea>
|
||
<p class="input-hint">{{ t('admin.users.notesHint') }}</p>
|
||
</div>
|
||
<div class="grid grid-cols-2 gap-4">
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.columns.balance') }}</label>
|
||
<input
|
||
v-model.number="createForm.balance"
|
||
type="number"
|
||
step="any"
|
||
class="input"
|
||
/>
|
||
</div>
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.columns.concurrency') }}</label>
|
||
<input
|
||
v-model.number="createForm.concurrency"
|
||
type="number"
|
||
class="input"
|
||
/>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="flex justify-end gap-3 pt-4">
|
||
<button
|
||
@click="closeCreateModal"
|
||
type="button"
|
||
class="btn btn-secondary"
|
||
>
|
||
{{ t('common.cancel') }}
|
||
</button>
|
||
<button
|
||
type="submit"
|
||
:disabled="submitting"
|
||
class="btn btn-primary"
|
||
>
|
||
<svg
|
||
v-if="submitting"
|
||
class="animate-spin -ml-1 mr-2 h-4 w-4"
|
||
fill="none"
|
||
viewBox="0 0 24 24"
|
||
>
|
||
<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
|
||
<path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
||
</svg>
|
||
{{ submitting ? t('admin.users.creating') : t('common.create') }}
|
||
</button>
|
||
</div>
|
||
</form>
|
||
</Modal>
|
||
|
||
<!-- Edit User Modal -->
|
||
<Modal
|
||
:show="showEditModal"
|
||
:title="t('admin.users.editUser')"
|
||
size="lg"
|
||
@close="closeEditModal"
|
||
>
|
||
<form v-if="editingUser" @submit.prevent="handleUpdateUser" class="space-y-5">
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.email') }}</label>
|
||
<input
|
||
v-model="editForm.email"
|
||
type="email"
|
||
class="input"
|
||
/>
|
||
</div>
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.password') }}</label>
|
||
<p class="text-xs text-gray-500 dark:text-dark-400 mb-1">{{ t('admin.users.leaveEmptyToKeep') }}</p>
|
||
<div class="flex gap-2">
|
||
<div class="relative flex-1">
|
||
<input
|
||
v-model="editForm.password"
|
||
type="text"
|
||
class="input pr-10"
|
||
:placeholder="t('admin.users.enterNewPassword')"
|
||
/>
|
||
<!-- Copy Password Button -->
|
||
<button
|
||
v-if="editForm.password"
|
||
type="button"
|
||
@click="copyEditPassword"
|
||
class="absolute right-2 top-1/2 -translate-y-1/2 p-1 rounded-lg hover:bg-gray-100 dark:hover:bg-dark-700 transition-colors"
|
||
:class="editPasswordCopied ? 'text-green-500' : 'text-gray-400 hover:text-gray-600 dark:hover:text-gray-300'"
|
||
:title="editPasswordCopied ? t('keys.copied') : t('admin.users.copyPassword')"
|
||
>
|
||
<svg v-if="editPasswordCopied" class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M5 13l4 4L19 7" />
|
||
</svg>
|
||
<svg v-else class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round"
|
||
d="M15.666 3.888A2.25 2.25 0 0013.5 2.25h-3c-1.03 0-1.9.693-2.166 1.638m7.332 0c.055.194.084.4.084.612v0a.75.75 0 01-.75.75H9a.75.75 0 01-.75-.75v0c0-.212.03-.418.084-.612m7.332 0c.646.049 1.288.11 1.927.184 1.1.128 1.907 1.077 1.907 2.185V19.5a2.25 2.25 0 01-2.25 2.25H6.75A2.25 2.25 0 014.5 19.5V6.257c0-1.108.806-2.057 1.907-2.185a48.208 48.208 0 011.927-.184" />
|
||
</svg>
|
||
</button>
|
||
</div>
|
||
<!-- Generate Random Password Button -->
|
||
<button
|
||
type="button"
|
||
@click="generateEditPassword"
|
||
class="btn btn-secondary px-3"
|
||
:title="t('admin.users.generatePassword')"
|
||
>
|
||
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<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>
|
||
</div>
|
||
</div>
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.username') }}</label>
|
||
<input
|
||
v-model="editForm.username"
|
||
type="text"
|
||
class="input"
|
||
:placeholder="t('admin.users.enterUsername')"
|
||
/>
|
||
</div>
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.wechat') }}</label>
|
||
<input
|
||
v-model="editForm.wechat"
|
||
type="text"
|
||
class="input"
|
||
:placeholder="t('admin.users.enterWechat')"
|
||
/>
|
||
</div>
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.notes') }}</label>
|
||
<textarea
|
||
v-model="editForm.notes"
|
||
rows="3"
|
||
class="input"
|
||
:placeholder="t('admin.users.enterNotes')"
|
||
></textarea>
|
||
<p class="input-hint">{{ t('admin.users.notesHint') }}</p>
|
||
</div>
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.columns.concurrency') }}</label>
|
||
<input
|
||
v-model.number="editForm.concurrency"
|
||
type="number"
|
||
class="input"
|
||
/>
|
||
</div>
|
||
|
||
<div class="flex justify-end gap-3 pt-4">
|
||
<button
|
||
@click="closeEditModal"
|
||
type="button"
|
||
class="btn btn-secondary"
|
||
>
|
||
{{ t('common.cancel') }}
|
||
</button>
|
||
<button
|
||
type="submit"
|
||
:disabled="submitting"
|
||
class="btn btn-primary"
|
||
>
|
||
<svg
|
||
v-if="submitting"
|
||
class="animate-spin -ml-1 mr-2 h-4 w-4"
|
||
fill="none"
|
||
viewBox="0 0 24 24"
|
||
>
|
||
<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
|
||
<path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
||
</svg>
|
||
{{ submitting ? t('admin.users.updating') : t('common.update') }}
|
||
</button>
|
||
</div>
|
||
</form>
|
||
</Modal>
|
||
|
||
<!-- View API Keys Modal -->
|
||
<Modal
|
||
:show="showApiKeysModal"
|
||
:title="t('admin.users.userApiKeys')"
|
||
size="xl"
|
||
@close="closeApiKeysModal"
|
||
>
|
||
<div v-if="viewingUser" class="space-y-4">
|
||
<!-- User Info Header -->
|
||
<div class="flex items-center gap-3 p-4 rounded-xl bg-gray-50 dark:bg-dark-700">
|
||
<div class="flex h-10 w-10 items-center justify-center rounded-full bg-primary-100 dark:bg-primary-900/30">
|
||
<span class="text-lg font-medium text-primary-700 dark:text-primary-300">
|
||
{{ viewingUser.email.charAt(0).toUpperCase() }}
|
||
</span>
|
||
</div>
|
||
<div>
|
||
<p class="font-medium text-gray-900 dark:text-white">{{ viewingUser.email }}</p>
|
||
<p class="text-sm text-gray-500 dark:text-dark-400">{{ viewingUser.username }}</p>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- API Keys List -->
|
||
<div v-if="loadingApiKeys" class="flex justify-center py-8">
|
||
<svg class="animate-spin h-8 w-8 text-primary-500" fill="none" viewBox="0 0 24 24">
|
||
<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
|
||
<path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
||
</svg>
|
||
</div>
|
||
<div v-else-if="userApiKeys.length === 0" class="text-center py-8">
|
||
<svg class="mx-auto h-12 w-12 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M15.75 5.25a3 3 0 013 3m3 0a6 6 0 01-7.029 5.912c-.563-.097-1.159.026-1.563.43L10.5 17.25H8.25v2.25H6v2.25H2.25v-2.818c0-.597.237-1.17.659-1.591l6.499-6.499c.404-.404.527-1 .43-1.563A6 6 0 1121.75 8.25z" />
|
||
</svg>
|
||
<p class="mt-2 text-sm text-gray-500 dark:text-dark-400">{{ t('admin.users.noApiKeys') }}</p>
|
||
</div>
|
||
<div v-else class="space-y-3 max-h-96 overflow-y-auto">
|
||
<div
|
||
v-for="key in userApiKeys"
|
||
:key="key.id"
|
||
class="p-4 rounded-xl border border-gray-200 dark:border-dark-600 bg-white dark:bg-dark-800"
|
||
>
|
||
<div class="flex items-start justify-between">
|
||
<div class="flex-1 min-w-0">
|
||
<div class="flex items-center gap-2 mb-1">
|
||
<span class="font-medium text-gray-900 dark:text-white">{{ key.name }}</span>
|
||
<span
|
||
:class="[
|
||
'badge text-xs',
|
||
key.status === 'active' ? 'badge-success' : 'badge-danger'
|
||
]"
|
||
>
|
||
{{ key.status }}
|
||
</span>
|
||
</div>
|
||
<p class="text-sm font-mono text-gray-500 dark:text-dark-400 truncate">
|
||
{{ key.key.substring(0, 20) }}...{{ key.key.substring(key.key.length - 8) }}
|
||
</p>
|
||
</div>
|
||
</div>
|
||
<div class="mt-3 flex flex-wrap gap-4 text-xs text-gray-500 dark:text-dark-400">
|
||
<div class="flex items-center gap-1">
|
||
<svg class="w-3.5 h-3.5" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M18 18.72a9.094 9.094 0 003.741-.479 3 3 0 00-4.682-2.72m.94 3.198l.001.031c0 .225-.012.447-.037.666A11.944 11.944 0 0112 21c-2.17 0-4.207-.576-5.963-1.584A6.062 6.062 0 016 18.719m12 0a5.971 5.971 0 00-.941-3.197m0 0A5.995 5.995 0 0012 12.75a5.995 5.995 0 00-5.058 2.772m0 0a3 3 0 00-4.681 2.72 8.986 8.986 0 003.74.477m.94-3.197a5.971 5.971 0 00-.94 3.197M15 6.75a3 3 0 11-6 0 3 3 0 016 0zm6 3a2.25 2.25 0 11-4.5 0 2.25 2.25 0 014.5 0zm-13.5 0a2.25 2.25 0 11-4.5 0 2.25 2.25 0 014.5 0z" />
|
||
</svg>
|
||
<span>{{ t('admin.users.group') }}: {{ key.group?.name || t('admin.users.none') }}</span>
|
||
</div>
|
||
<div class="flex items-center gap-1">
|
||
<svg class="w-3.5 h-3.5" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M6.75 3v2.25M17.25 3v2.25M3 18.75V7.5a2.25 2.25 0 012.25-2.25h13.5A2.25 2.25 0 0121 7.5v11.25m-18 0A2.25 2.25 0 005.25 21h13.5A2.25 2.25 0 0021 18.75m-18 0v-7.5A2.25 2.25 0 015.25 9h13.5A2.25 2.25 0 0121 11.25v7.5" />
|
||
</svg>
|
||
<span>{{ t('admin.users.columns.created') }}: {{ formatDate(key.created_at) }}</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<template #footer>
|
||
<div class="flex justify-end">
|
||
<button
|
||
@click="closeApiKeysModal"
|
||
type="button"
|
||
class="btn btn-secondary"
|
||
>
|
||
{{ t('common.cancel') }}
|
||
</button>
|
||
</div>
|
||
</template>
|
||
</Modal>
|
||
|
||
<!-- Allowed Groups Modal -->
|
||
<Modal
|
||
:show="showAllowedGroupsModal"
|
||
:title="t('admin.users.setAllowedGroups')"
|
||
size="lg"
|
||
@close="closeAllowedGroupsModal"
|
||
>
|
||
<div v-if="allowedGroupsUser" class="space-y-4">
|
||
<!-- User Info Header -->
|
||
<div class="flex items-center gap-3 p-4 rounded-xl bg-gray-50 dark:bg-dark-700">
|
||
<div class="flex h-10 w-10 items-center justify-center rounded-full bg-primary-100 dark:bg-primary-900/30">
|
||
<span class="text-lg font-medium text-primary-700 dark:text-primary-300">
|
||
{{ allowedGroupsUser.email.charAt(0).toUpperCase() }}
|
||
</span>
|
||
</div>
|
||
<div>
|
||
<p class="font-medium text-gray-900 dark:text-white">{{ allowedGroupsUser.email }}</p>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Loading State -->
|
||
<div v-if="loadingGroups" class="flex justify-center py-8">
|
||
<svg class="animate-spin h-8 w-8 text-primary-500" fill="none" viewBox="0 0 24 24">
|
||
<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
|
||
<path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
||
</svg>
|
||
</div>
|
||
|
||
<!-- Groups Selection -->
|
||
<div v-else>
|
||
<p class="text-sm text-gray-600 dark:text-dark-400 mb-3">
|
||
{{ t('admin.users.allowedGroupsHint') }}
|
||
</p>
|
||
|
||
<!-- Empty State -->
|
||
<div v-if="standardGroups.length === 0" class="text-center py-6">
|
||
<svg class="mx-auto h-12 w-12 text-gray-400" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M18 18.72a9.094 9.094 0 003.741-.479 3 3 0 00-4.682-2.72m.94 3.198l.001.031c0 .225-.012.447-.037.666A11.944 11.944 0 0112 21c-2.17 0-4.207-.576-5.963-1.584A6.062 6.062 0 016 18.719m12 0a5.971 5.971 0 00-.941-3.197m0 0A5.995 5.995 0 0012 12.75a5.995 5.995 0 00-5.058 2.772m0 0a3 3 0 00-4.681 2.72 8.986 8.986 0 003.74.477m.94-3.197a5.971 5.971 0 00-.94 3.197M15 6.75a3 3 0 11-6 0 3 3 0 016 0zm6 3a2.25 2.25 0 11-4.5 0 2.25 2.25 0 014.5 0zm-13.5 0a2.25 2.25 0 11-4.5 0 2.25 2.25 0 014.5 0z" />
|
||
</svg>
|
||
<p class="mt-2 text-sm text-gray-500 dark:text-dark-400">{{ t('admin.users.noStandardGroups') }}</p>
|
||
</div>
|
||
|
||
<!-- Groups List -->
|
||
<div v-else class="space-y-2 max-h-64 overflow-y-auto">
|
||
<label
|
||
v-for="group in standardGroups"
|
||
:key="group.id"
|
||
class="flex items-center gap-3 p-3 rounded-lg border border-gray-200 dark:border-dark-600 hover:bg-gray-50 dark:hover:bg-dark-700 cursor-pointer transition-colors"
|
||
:class="{ 'bg-primary-50 dark:bg-primary-900/20 border-primary-300 dark:border-primary-700': selectedGroupIds.includes(group.id) }"
|
||
>
|
||
<input
|
||
type="checkbox"
|
||
:value="group.id"
|
||
v-model="selectedGroupIds"
|
||
class="h-4 w-4 rounded border-gray-300 text-primary-600 focus:ring-primary-500"
|
||
/>
|
||
<div class="flex-1 min-w-0">
|
||
<p class="font-medium text-gray-900 dark:text-white">{{ group.name }}</p>
|
||
<p v-if="group.description" class="text-sm text-gray-500 dark:text-dark-400 truncate">{{ group.description }}</p>
|
||
</div>
|
||
<div class="flex items-center gap-2">
|
||
<span class="badge badge-gray text-xs">{{ group.platform }}</span>
|
||
<span v-if="group.is_exclusive" class="badge badge-purple text-xs">{{ t('admin.groups.exclusive') }}</span>
|
||
</div>
|
||
</label>
|
||
</div>
|
||
|
||
<!-- Clear Selection -->
|
||
<div class="mt-4 pt-4 border-t border-gray-200 dark:border-dark-600">
|
||
<label class="flex items-center gap-3 p-3 rounded-lg border border-gray-200 dark:border-dark-600 hover:bg-gray-50 dark:hover:bg-dark-700 cursor-pointer transition-colors"
|
||
:class="{ 'bg-green-50 dark:bg-green-900/20 border-green-300 dark:border-green-700': selectedGroupIds.length === 0 }"
|
||
>
|
||
<input
|
||
type="radio"
|
||
:checked="selectedGroupIds.length === 0"
|
||
@change="selectedGroupIds = []"
|
||
class="h-4 w-4 border-gray-300 text-green-600 focus:ring-green-500"
|
||
/>
|
||
<div class="flex-1">
|
||
<p class="font-medium text-gray-900 dark:text-white">{{ t('admin.users.allowAllGroups') }}</p>
|
||
<p class="text-sm text-gray-500 dark:text-dark-400">{{ t('admin.users.allowAllGroupsHint') }}</p>
|
||
</div>
|
||
</label>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<template #footer>
|
||
<div class="flex justify-end gap-3">
|
||
<button
|
||
@click="closeAllowedGroupsModal"
|
||
type="button"
|
||
class="btn btn-secondary"
|
||
>
|
||
{{ t('common.cancel') }}
|
||
</button>
|
||
<button
|
||
@click="handleSaveAllowedGroups"
|
||
:disabled="savingAllowedGroups"
|
||
class="btn btn-primary"
|
||
>
|
||
<svg
|
||
v-if="savingAllowedGroups"
|
||
class="animate-spin -ml-1 mr-2 h-4 w-4"
|
||
fill="none"
|
||
viewBox="0 0 24 24"
|
||
>
|
||
<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
|
||
<path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
||
</svg>
|
||
{{ savingAllowedGroups ? t('common.saving') : t('common.save') }}
|
||
</button>
|
||
</div>
|
||
</template>
|
||
</Modal>
|
||
|
||
<!-- Deposit/Withdraw Modal -->
|
||
<Modal
|
||
:show="showBalanceModal"
|
||
:title="balanceOperation === 'add' ? t('admin.users.deposit') : t('admin.users.withdraw')"
|
||
size="md"
|
||
@close="closeBalanceModal"
|
||
>
|
||
<form v-if="balanceUser" @submit.prevent="handleBalanceSubmit" class="space-y-5">
|
||
<div class="flex items-center gap-3 p-4 rounded-xl bg-gray-50 dark:bg-dark-700">
|
||
<div class="flex h-10 w-10 items-center justify-center rounded-full bg-primary-100 dark:bg-primary-900/30">
|
||
<span class="text-lg font-medium text-primary-700 dark:text-primary-300">
|
||
{{ balanceUser.email.charAt(0).toUpperCase() }}
|
||
</span>
|
||
</div>
|
||
<div class="flex-1">
|
||
<p class="font-medium text-gray-900 dark:text-white">{{ balanceUser.email }}</p>
|
||
<p class="text-sm text-gray-500 dark:text-dark-400">
|
||
{{ t('admin.users.currentBalance') }}: ${{ balanceUser.balance.toFixed(2) }}
|
||
</p>
|
||
</div>
|
||
</div>
|
||
|
||
<div>
|
||
<label class="input-label">
|
||
{{ balanceOperation === 'add' ? t('admin.users.depositAmount') : t('admin.users.withdrawAmount') }}
|
||
</label>
|
||
<div class="relative">
|
||
<div class="absolute left-3 top-1/2 -translate-y-1/2 text-gray-500 dark:text-dark-400 font-medium">
|
||
$
|
||
</div>
|
||
<input
|
||
v-model.number="balanceForm.amount"
|
||
type="number"
|
||
step="0.01"
|
||
min="0.01"
|
||
required
|
||
class="input pl-8"
|
||
:placeholder="balanceOperation === 'add' ? '10.00' : '5.00'"
|
||
/>
|
||
</div>
|
||
<p class="input-hint">
|
||
{{ t('admin.users.amountHint') }}
|
||
</p>
|
||
</div>
|
||
|
||
<div>
|
||
<label class="input-label">{{ t('admin.users.notes') }}</label>
|
||
<textarea
|
||
v-model="balanceForm.notes"
|
||
rows="3"
|
||
class="input"
|
||
:placeholder="balanceOperation === 'add'
|
||
? t('admin.users.depositNotesPlaceholder')
|
||
: t('admin.users.withdrawNotesPlaceholder')"
|
||
></textarea>
|
||
<p class="input-hint">{{ t('admin.users.notesOptional') }}</p>
|
||
</div>
|
||
|
||
<div v-if="balanceForm.amount > 0" class="p-4 rounded-xl bg-blue-50 dark:bg-blue-900/20 border border-blue-200 dark:border-blue-800/50">
|
||
<div class="flex items-center justify-between text-sm">
|
||
<span class="text-blue-700 dark:text-blue-300">{{ t('admin.users.newBalance') }}:</span>
|
||
<span class="font-bold text-blue-900 dark:text-blue-100">
|
||
${{ calculateNewBalance().toFixed(2) }}
|
||
</span>
|
||
</div>
|
||
</div>
|
||
|
||
<div v-if="balanceOperation === 'subtract' && calculateNewBalance() < 0" class="p-4 rounded-xl bg-red-50 dark:bg-red-900/20 border border-red-200 dark:border-red-800/50">
|
||
<div class="flex items-center gap-2 text-sm text-red-700 dark:text-red-300">
|
||
<svg class="w-5 h-5 flex-shrink-0" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="1.5">
|
||
<path stroke-linecap="round" stroke-linejoin="round" d="M12 9v3.75m-9.303 3.376c-.866 1.5.217 3.374 1.948 3.374h14.71c1.73 0 2.813-1.874 1.948-3.374L13.949 3.378c-.866-1.5-3.032-1.5-3.898 0L2.697 16.126zM12 15.75h.007v.008H12v-.008z" />
|
||
</svg>
|
||
<span>{{ t('admin.users.insufficientBalance') }}</span>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="flex justify-end gap-3 pt-4">
|
||
<button
|
||
@click="closeBalanceModal"
|
||
type="button"
|
||
class="btn btn-secondary"
|
||
>
|
||
{{ t('common.cancel') }}
|
||
</button>
|
||
<button
|
||
type="submit"
|
||
:disabled="balanceSubmitting || !balanceForm.amount || balanceForm.amount <= 0 || (balanceOperation === 'subtract' && calculateNewBalance() < 0)"
|
||
class="btn"
|
||
:class="balanceOperation === 'add' ? 'bg-emerald-600 hover:bg-emerald-700 text-white' : 'btn-danger'"
|
||
>
|
||
<svg
|
||
v-if="balanceSubmitting"
|
||
class="animate-spin -ml-1 mr-2 h-4 w-4"
|
||
fill="none"
|
||
viewBox="0 0 24 24"
|
||
>
|
||
<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
|
||
<path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
||
</svg>
|
||
{{ balanceSubmitting
|
||
? (balanceOperation === 'add' ? t('admin.users.depositing') : t('admin.users.withdrawing'))
|
||
: (balanceOperation === 'add' ? t('admin.users.confirmDeposit') : t('admin.users.confirmWithdraw'))
|
||
}}
|
||
</button>
|
||
</div>
|
||
</form>
|
||
</Modal>
|
||
|
||
<!-- Delete Confirmation Dialog -->
|
||
<ConfirmDialog
|
||
:show="showDeleteDialog"
|
||
:title="t('admin.users.deleteUser')"
|
||
:message="t('admin.users.deleteConfirm', { email: deletingUser?.email })"
|
||
:confirm-text="t('common.delete')"
|
||
:cancel-text="t('common.cancel')"
|
||
:danger="true"
|
||
@confirm="confirmDelete"
|
||
@cancel="showDeleteDialog = false"
|
||
/>
|
||
</AppLayout>
|
||
</template>
|
||
|
||
<script setup lang="ts">
|
||
import { ref, reactive, computed, onMounted } from 'vue'
|
||
import { useI18n } from 'vue-i18n'
|
||
import { useAppStore } from '@/stores/app'
|
||
|
||
const { t } = useI18n()
|
||
import { adminAPI } from '@/api/admin'
|
||
import type { User, ApiKey, Group } from '@/types'
|
||
import type { BatchUserUsageStats } from '@/api/admin/dashboard'
|
||
import type { Column } from '@/components/common/types'
|
||
import AppLayout from '@/components/layout/AppLayout.vue'
|
||
import DataTable from '@/components/common/DataTable.vue'
|
||
import Pagination from '@/components/common/Pagination.vue'
|
||
import Modal from '@/components/common/Modal.vue'
|
||
import ConfirmDialog from '@/components/common/ConfirmDialog.vue'
|
||
import EmptyState from '@/components/common/EmptyState.vue'
|
||
import Select from '@/components/common/Select.vue'
|
||
import GroupBadge from '@/components/common/GroupBadge.vue'
|
||
|
||
const appStore = useAppStore()
|
||
|
||
const columns = computed<Column[]>(() => [
|
||
{ key: 'email', label: t('admin.users.columns.user'), sortable: true },
|
||
{ key: 'username', label: t('admin.users.columns.username'), sortable: true },
|
||
{ key: 'wechat', label: t('admin.users.columns.wechat'), sortable: false },
|
||
{ key: 'notes', label: t('admin.users.columns.notes'), sortable: false },
|
||
{ key: 'role', label: t('admin.users.columns.role'), sortable: true },
|
||
{ 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 }
|
||
])
|
||
|
||
// Filter options
|
||
const roleOptions = computed(() => [
|
||
{ value: '', label: t('admin.users.allRoles') },
|
||
{ value: 'admin', label: t('admin.users.admin') },
|
||
{ value: 'user', label: t('admin.users.user') }
|
||
])
|
||
|
||
const statusOptions = computed(() => [
|
||
{ value: '', label: t('admin.users.allStatus') },
|
||
{ value: 'active', label: t('common.active') },
|
||
{ value: 'disabled', label: t('admin.users.disabled') }
|
||
])
|
||
|
||
|
||
const users = ref<User[]>([])
|
||
const loading = ref(false)
|
||
const searchQuery = ref('')
|
||
const filters = reactive({
|
||
role: '',
|
||
status: ''
|
||
})
|
||
const usageStats = ref<Record<string, BatchUserUsageStats>>({})
|
||
const pagination = reactive({
|
||
page: 1,
|
||
page_size: 20,
|
||
total: 0,
|
||
pages: 0
|
||
})
|
||
|
||
const showCreateModal = ref(false)
|
||
const showEditModal = ref(false)
|
||
const showDeleteDialog = ref(false)
|
||
const showApiKeysModal = ref(false)
|
||
const submitting = ref(false)
|
||
const editingUser = ref<User | null>(null)
|
||
const deletingUser = ref<User | null>(null)
|
||
const viewingUser = ref<User | null>(null)
|
||
const userApiKeys = ref<ApiKey[]>([])
|
||
const loadingApiKeys = ref(false)
|
||
const passwordCopied = ref(false)
|
||
|
||
// Allowed groups modal state
|
||
const showAllowedGroupsModal = ref(false)
|
||
const allowedGroupsUser = ref<User | null>(null)
|
||
const standardGroups = ref<Group[]>([])
|
||
const selectedGroupIds = ref<number[]>([])
|
||
const loadingGroups = ref(false)
|
||
const savingAllowedGroups = ref(false)
|
||
|
||
// Balance (Deposit/Withdraw) modal state
|
||
const showBalanceModal = ref(false)
|
||
const balanceUser = ref<User | null>(null)
|
||
const balanceOperation = ref<'add' | 'subtract'>('add')
|
||
const balanceSubmitting = ref(false)
|
||
const balanceForm = reactive({
|
||
amount: 0,
|
||
notes: ''
|
||
})
|
||
|
||
const createForm = reactive({
|
||
email: '',
|
||
password: '',
|
||
username: '',
|
||
wechat: '',
|
||
notes: '',
|
||
balance: 0,
|
||
concurrency: 1
|
||
})
|
||
|
||
const editForm = reactive({
|
||
email: '',
|
||
password: '',
|
||
username: '',
|
||
wechat: '',
|
||
notes: '',
|
||
concurrency: 1
|
||
})
|
||
const editPasswordCopied = ref(false)
|
||
|
||
const formatDate = (dateString: string): string => {
|
||
const date = new Date(dateString)
|
||
return date.toLocaleDateString('en-US', {
|
||
year: 'numeric',
|
||
month: 'short',
|
||
day: 'numeric'
|
||
})
|
||
}
|
||
|
||
// 计算剩余天数
|
||
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))
|
||
}
|
||
|
||
// 格式化过期时间(用于 tooltip)
|
||
const formatExpiresAt = (expiresAt: string): string => {
|
||
const date = new Date(expiresAt)
|
||
return date.toLocaleString()
|
||
}
|
||
|
||
const generateRandomPasswordStr = () => {
|
||
const chars = 'ABCDEFGHJKLMNPQRSTUVWXYZabcdefghjkmnpqrstuvwxyz23456789!@#$%^&*'
|
||
let password = ''
|
||
for (let i = 0; i < 16; i++) {
|
||
password += chars.charAt(Math.floor(Math.random() * chars.length))
|
||
}
|
||
return password
|
||
}
|
||
|
||
const generateRandomPassword = () => {
|
||
createForm.password = generateRandomPasswordStr()
|
||
}
|
||
|
||
const generateEditPassword = () => {
|
||
editForm.password = generateRandomPasswordStr()
|
||
}
|
||
|
||
const copyPassword = async () => {
|
||
if (!createForm.password) return
|
||
try {
|
||
await navigator.clipboard.writeText(createForm.password)
|
||
passwordCopied.value = true
|
||
setTimeout(() => {
|
||
passwordCopied.value = false
|
||
}, 2000)
|
||
} catch (error) {
|
||
appStore.showError(t('common.copyFailed'))
|
||
}
|
||
}
|
||
|
||
const copyEditPassword = async () => {
|
||
if (!editForm.password) return
|
||
try {
|
||
await navigator.clipboard.writeText(editForm.password)
|
||
editPasswordCopied.value = true
|
||
setTimeout(() => {
|
||
editPasswordCopied.value = false
|
||
}, 2000)
|
||
} catch (error) {
|
||
appStore.showError(t('common.copyFailed'))
|
||
}
|
||
}
|
||
|
||
const loadUsers = async () => {
|
||
loading.value = true
|
||
try {
|
||
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
|
||
}
|
||
)
|
||
users.value = response.items
|
||
pagination.total = response.total
|
||
pagination.pages = response.pages
|
||
|
||
// Load usage stats for all users in the list
|
||
if (response.items.length > 0) {
|
||
const userIds = response.items.map(u => u.id)
|
||
try {
|
||
const usageResponse = await adminAPI.dashboard.getBatchUsersUsage(userIds)
|
||
usageStats.value = usageResponse.stats
|
||
} catch (e) {
|
||
console.error('Failed to load usage stats:', e)
|
||
}
|
||
}
|
||
} catch (error) {
|
||
appStore.showError(t('admin.users.failedToLoad'))
|
||
console.error('Error loading users:', error)
|
||
} finally {
|
||
loading.value = false
|
||
}
|
||
}
|
||
|
||
let searchTimeout: ReturnType<typeof setTimeout>
|
||
const handleSearch = () => {
|
||
clearTimeout(searchTimeout)
|
||
searchTimeout = setTimeout(() => {
|
||
pagination.page = 1
|
||
loadUsers()
|
||
}, 300)
|
||
}
|
||
|
||
const handlePageChange = (page: number) => {
|
||
pagination.page = page
|
||
loadUsers()
|
||
}
|
||
|
||
const closeCreateModal = () => {
|
||
showCreateModal.value = false
|
||
createForm.email = ''
|
||
createForm.password = ''
|
||
createForm.username = ''
|
||
createForm.wechat = ''
|
||
createForm.notes = ''
|
||
createForm.balance = 0
|
||
createForm.concurrency = 1
|
||
passwordCopied.value = false
|
||
}
|
||
|
||
const handleCreateUser = async () => {
|
||
submitting.value = true
|
||
try {
|
||
await adminAPI.users.create(createForm)
|
||
appStore.showSuccess(t('admin.users.userCreated'))
|
||
closeCreateModal()
|
||
loadUsers()
|
||
} catch (error: any) {
|
||
appStore.showError(error.response?.data?.detail || t('admin.users.failedToCreate'))
|
||
console.error('Error creating user:', error)
|
||
} finally {
|
||
submitting.value = false
|
||
}
|
||
}
|
||
|
||
const handleEdit = (user: User) => {
|
||
editingUser.value = user
|
||
editForm.email = user.email
|
||
editForm.password = ''
|
||
editForm.username = user.username || ''
|
||
editForm.wechat = user.wechat || ''
|
||
editForm.notes = user.notes || ''
|
||
editForm.concurrency = user.concurrency
|
||
editPasswordCopied.value = false
|
||
showEditModal.value = true
|
||
}
|
||
|
||
const closeEditModal = () => {
|
||
showEditModal.value = false
|
||
editingUser.value = null
|
||
editForm.password = ''
|
||
editPasswordCopied.value = false
|
||
}
|
||
|
||
const handleUpdateUser = async () => {
|
||
if (!editingUser.value) return
|
||
|
||
submitting.value = true
|
||
try {
|
||
const updateData: Record<string, any> = {
|
||
email: editForm.email,
|
||
username: editForm.username,
|
||
wechat: editForm.wechat,
|
||
notes: editForm.notes,
|
||
concurrency: editForm.concurrency
|
||
}
|
||
if (editForm.password.trim()) {
|
||
updateData.password = editForm.password.trim()
|
||
}
|
||
|
||
await adminAPI.users.update(editingUser.value.id, updateData)
|
||
appStore.showSuccess(t('admin.users.userUpdated'))
|
||
closeEditModal()
|
||
loadUsers()
|
||
} catch (error: any) {
|
||
appStore.showError(error.response?.data?.detail || t('admin.users.failedToUpdate'))
|
||
console.error('Error updating user:', error)
|
||
} finally {
|
||
submitting.value = false
|
||
}
|
||
}
|
||
|
||
const handleToggleStatus = async (user: User) => {
|
||
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 = async (user: User) => {
|
||
viewingUser.value = user
|
||
showApiKeysModal.value = true
|
||
loadingApiKeys.value = true
|
||
userApiKeys.value = []
|
||
|
||
try {
|
||
const response = await adminAPI.users.getUserApiKeys(user.id)
|
||
userApiKeys.value = response.items || []
|
||
} catch (error: any) {
|
||
appStore.showError(error.response?.data?.detail || t('admin.users.failedToLoadApiKeys'))
|
||
console.error('Error loading user API keys:', error)
|
||
} finally {
|
||
loadingApiKeys.value = false
|
||
}
|
||
}
|
||
|
||
const closeApiKeysModal = () => {
|
||
showApiKeysModal.value = false
|
||
viewingUser.value = null
|
||
userApiKeys.value = []
|
||
}
|
||
|
||
// Allowed Groups functions
|
||
const handleAllowedGroups = async (user: User) => {
|
||
allowedGroupsUser.value = user
|
||
showAllowedGroupsModal.value = true
|
||
loadingGroups.value = true
|
||
standardGroups.value = []
|
||
selectedGroupIds.value = user.allowed_groups ? [...user.allowed_groups] : []
|
||
|
||
try {
|
||
const allGroups = await adminAPI.groups.getAll()
|
||
// Only show standard type groups (subscription type groups are managed in /admin/subscriptions)
|
||
standardGroups.value = allGroups.filter(g => g.subscription_type === 'standard' && g.status === 'active')
|
||
} catch (error: any) {
|
||
appStore.showError(error.response?.data?.detail || t('admin.users.failedToLoadGroups'))
|
||
console.error('Error loading groups:', error)
|
||
} finally {
|
||
loadingGroups.value = false
|
||
}
|
||
}
|
||
|
||
const closeAllowedGroupsModal = () => {
|
||
showAllowedGroupsModal.value = false
|
||
allowedGroupsUser.value = null
|
||
standardGroups.value = []
|
||
selectedGroupIds.value = []
|
||
}
|
||
|
||
const handleSaveAllowedGroups = async () => {
|
||
if (!allowedGroupsUser.value) return
|
||
|
||
savingAllowedGroups.value = true
|
||
try {
|
||
// null means allow all non-exclusive groups, empty array also means allow all
|
||
const allowedGroups = selectedGroupIds.value.length > 0 ? selectedGroupIds.value : null
|
||
await adminAPI.users.update(allowedGroupsUser.value.id, { allowed_groups: allowedGroups })
|
||
appStore.showSuccess(t('admin.users.allowedGroupsUpdated'))
|
||
closeAllowedGroupsModal()
|
||
loadUsers()
|
||
} catch (error: any) {
|
||
appStore.showError(error.response?.data?.detail || t('admin.users.failedToUpdateAllowedGroups'))
|
||
console.error('Error updating allowed groups:', error)
|
||
} finally {
|
||
savingAllowedGroups.value = false
|
||
}
|
||
}
|
||
|
||
const handleDelete = (user: User) => {
|
||
deletingUser.value = user
|
||
showDeleteDialog.value = true
|
||
}
|
||
|
||
const confirmDelete = async () => {
|
||
if (!deletingUser.value) return
|
||
|
||
try {
|
||
await adminAPI.users.delete(deletingUser.value.id)
|
||
appStore.showSuccess(t('admin.users.userDeleted'))
|
||
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: User) => {
|
||
balanceUser.value = user
|
||
balanceOperation.value = 'add'
|
||
balanceForm.amount = 0
|
||
balanceForm.notes = ''
|
||
showBalanceModal.value = true
|
||
}
|
||
|
||
const handleWithdraw = (user: User) => {
|
||
balanceUser.value = user
|
||
balanceOperation.value = 'subtract'
|
||
balanceForm.amount = 0
|
||
balanceForm.notes = ''
|
||
showBalanceModal.value = true
|
||
}
|
||
|
||
const closeBalanceModal = () => {
|
||
showBalanceModal.value = false
|
||
balanceUser.value = null
|
||
balanceForm.amount = 0
|
||
balanceForm.notes = ''
|
||
}
|
||
|
||
const calculateNewBalance = () => {
|
||
if (!balanceUser.value) return 0
|
||
if (balanceOperation.value === 'add') {
|
||
return balanceUser.value.balance + balanceForm.amount
|
||
} else {
|
||
return balanceUser.value.balance - balanceForm.amount
|
||
}
|
||
}
|
||
|
||
const handleBalanceSubmit = async () => {
|
||
if (!balanceUser.value || balanceForm.amount <= 0) return
|
||
|
||
balanceSubmitting.value = true
|
||
try {
|
||
await adminAPI.users.updateBalance(
|
||
balanceUser.value.id,
|
||
balanceForm.amount,
|
||
balanceOperation.value,
|
||
balanceForm.notes
|
||
)
|
||
|
||
const successMsg = balanceOperation.value === 'add'
|
||
? t('admin.users.depositSuccess')
|
||
: t('admin.users.withdrawSuccess')
|
||
|
||
appStore.showSuccess(successMsg)
|
||
closeBalanceModal()
|
||
loadUsers()
|
||
} catch (error: any) {
|
||
const errorMsg = balanceOperation.value === 'add'
|
||
? t('admin.users.failedToDeposit')
|
||
: t('admin.users.failedToWithdraw')
|
||
|
||
appStore.showError(error.response?.data?.detail || errorMsg)
|
||
console.error('Error updating balance:', error)
|
||
} finally {
|
||
balanceSubmitting.value = false
|
||
}
|
||
}
|
||
|
||
onMounted(() => {
|
||
loadUsers()
|
||
})
|
||
</script>
|