refactor(keys): 优化分组选项显示与代码复用

- 删除冗余的 vite.config.js,统一使用 TypeScript 配置
- 创建 GroupOptionItem 组件封装分组选项 UI 逻辑(GroupBadge + 描述 + 勾选状态)
- 在密钥页面的分组选择器中显示分组描述文字
- 添加选中状态的勾选图标,提升交互体验
- 优化描述文字左对齐和截断显示效果
- 消除代码重复,简化维护成本
This commit is contained in:
墨颜
2026-01-04 21:34:38 +08:00
parent 9d3ec9e627
commit 6708f40005
3 changed files with 64 additions and 50 deletions

View File

@@ -0,0 +1,52 @@
<template>
<div class="flex min-w-0 flex-1 items-center justify-between gap-2">
<div
class="flex min-w-0 flex-1 flex-col items-start gap-1"
:title="description || undefined"
>
<GroupBadge
:name="name"
:platform="platform"
:subscription-type="subscriptionType"
:rate-multiplier="rateMultiplier"
/>
<span
v-if="description"
class="w-full truncate text-left text-xs text-gray-500 dark:text-gray-400"
>
{{ description }}
</span>
</div>
<svg
v-if="showCheckmark && selected"
class="h-4 w-4 shrink-0 text-primary-600 dark:text-primary-400"
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>
</div>
</template>
<script setup lang="ts">
import GroupBadge from './GroupBadge.vue'
import type { SubscriptionType, GroupPlatform } from '@/types'
interface Props {
name: string
platform: GroupPlatform
subscriptionType?: SubscriptionType
rateMultiplier?: number
description?: string | null
selected?: boolean
showCheckmark?: boolean
}
withDefaults(defineProps<Props>(), {
subscriptionType: 'standard',
selected: false,
showCheckmark: true
})
</script>

View File

@@ -335,12 +335,14 @@
/> />
<span v-else class="text-gray-400">{{ t('keys.selectGroup') }}</span> <span v-else class="text-gray-400">{{ t('keys.selectGroup') }}</span>
</template> </template>
<template #option="{ option }"> <template #option="{ option, selected }">
<GroupBadge <GroupOptionItem
:name="(option as unknown as GroupOption).label" :name="(option as unknown as GroupOption).label"
:platform="(option as unknown as GroupOption).platform" :platform="(option as unknown as GroupOption).platform"
:subscription-type="(option as unknown as GroupOption).subscriptionType" :subscription-type="(option as unknown as GroupOption).subscriptionType"
:rate-multiplier="(option as unknown as GroupOption).rate" :rate-multiplier="(option as unknown as GroupOption).rate"
:description="(option as unknown as GroupOption).description"
:selected="selected"
/> />
</template> </template>
</Select> </Select>
@@ -516,26 +518,19 @@
? 'bg-primary-50 dark:bg-primary-900/20' ? 'bg-primary-50 dark:bg-primary-900/20'
: 'hover:bg-gray-100 dark:hover:bg-dark-700' : 'hover:bg-gray-100 dark:hover:bg-dark-700'
]" ]"
:title="option.description || undefined"
> >
<GroupBadge <GroupOptionItem
:name="option.label" :name="option.label"
:platform="option.platform" :platform="option.platform"
:subscription-type="option.subscriptionType" :subscription-type="option.subscriptionType"
:rate-multiplier="option.rate" :rate-multiplier="option.rate"
/> :description="option.description"
<svg :selected="
v-if="
selectedKeyForGroup?.group_id === option.value || selectedKeyForGroup?.group_id === option.value ||
(!selectedKeyForGroup?.group_id && option.value === null) (!selectedKeyForGroup?.group_id && option.value === null)
" "
class="h-4 w-4 shrink-0 text-primary-600 dark:text-primary-400" />
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>
</button> </button>
</div> </div>
</div> </div>
@@ -562,6 +557,7 @@ import EmptyState from '@/components/common/EmptyState.vue'
import Select from '@/components/common/Select.vue' import Select from '@/components/common/Select.vue'
import UseKeyModal from '@/components/keys/UseKeyModal.vue' import UseKeyModal from '@/components/keys/UseKeyModal.vue'
import GroupBadge from '@/components/common/GroupBadge.vue' import GroupBadge from '@/components/common/GroupBadge.vue'
import GroupOptionItem from '@/components/common/GroupOptionItem.vue'
import type { ApiKey, Group, PublicSettings, SubscriptionType, GroupPlatform } from '@/types' import type { ApiKey, Group, PublicSettings, SubscriptionType, GroupPlatform } from '@/types'
import type { Column } from '@/components/common/types' import type { Column } from '@/components/common/types'
import type { BatchApiKeyUsageStats } from '@/api/usage' import type { BatchApiKeyUsageStats } from '@/api/usage'
@@ -570,6 +566,7 @@ import { formatDateTime } from '@/utils/format'
interface GroupOption { interface GroupOption {
value: number value: number
label: string label: string
description: string | null
rate: number rate: number
subscriptionType: SubscriptionType subscriptionType: SubscriptionType
platform: GroupPlatform platform: GroupPlatform
@@ -665,6 +662,7 @@ const groupOptions = computed(() =>
groups.value.map((group) => ({ groups.value.map((group) => ({
value: group.id, value: group.id,
label: group.name, label: group.name,
description: group.description,
rate: group.rate_multiplier, rate: group.rate_multiplier,
subscriptionType: group.subscription_type, subscriptionType: group.subscription_type,
platform: group.platform platform: group.platform

View File

@@ -1,36 +0,0 @@
import { defineConfig } from 'vite';
import vue from '@vitejs/plugin-vue';
import checker from 'vite-plugin-checker';
import { resolve } from 'path';
export default defineConfig({
plugins: [
vue(),
checker({
typescript: true,
vueTsc: true
})
],
resolve: {
alias: {
'@': resolve(__dirname, 'src')
}
},
build: {
outDir: '../backend/internal/web/dist',
emptyOutDir: true
},
server: {
host: '0.0.0.0',
port: 3000,
proxy: {
'/api': {
target: 'http://localhost:8080',
changeOrigin: true
},
'/setup': {
target: 'http://localhost:8080',
changeOrigin: true
}
}
}
});