Pricing popover: - Teleport to body + fixed-position re-measuring on hover/scroll/resize so it escapes the card's overflow-hidden clip that was chopping off the lower half of the panel. - Header + border adopt the platform theme via platformBorderClass / platformBadgeLightClass so each model card reads at a glance. Accessible groups: - Backend AvailableGroupRef / user DTO now expose subscription_type, rate_multiplier and is_exclusive. User-specific rates continue to come from /groups/rates (same pattern as the API keys page). - Table renders groups through the shared GroupBadge, which already deepens subscription-type colors and shows default vs custom rate as <s>default</s> <b>custom</b>. - Exclusive groups are labelled with a purple shield row, public groups with a grey globe row so admins and users can tell at a glance which groups they got via explicit grant vs. public access. i18n keys for exclusive / public / their tooltips added to zh + en.
176 lines
6.8 KiB
Vue
176 lines
6.8 KiB
Vue
<template>
|
||
<div class="card">
|
||
<!-- 表头 -->
|
||
<div
|
||
class="grid items-center rounded-t-lg border-b border-gray-100 bg-gray-50/50 px-4 py-3 text-xs font-medium uppercase tracking-wide text-gray-500 dark:border-dark-700 dark:bg-dark-800/50 dark:text-gray-400"
|
||
:style="gridStyle"
|
||
>
|
||
<div>{{ columns.name }}</div>
|
||
<div>{{ columns.platform }}</div>
|
||
<div>{{ columns.groups }}</div>
|
||
<div>{{ columns.supportedModels }}</div>
|
||
</div>
|
||
|
||
<div v-if="loading" class="flex justify-center py-10">
|
||
<Icon name="refresh" size="lg" class="animate-spin text-gray-400" />
|
||
</div>
|
||
|
||
<div v-else-if="rows.length === 0" class="flex flex-col items-center py-12">
|
||
<Icon name="inbox" size="xl" class="mb-3 h-12 w-12 text-gray-400" />
|
||
<p class="text-sm text-gray-500 dark:text-gray-400">{{ emptyLabel }}</p>
|
||
</div>
|
||
|
||
<!-- 渠道分组:每个渠道一个 section,内部按 platform 顺序铺开。
|
||
外层无 overflow-hidden,避免裁掉 SupportedModelChip 的价格浮层。 -->
|
||
<div
|
||
v-else
|
||
v-for="(channel, chIdx) in rows"
|
||
:key="`${channel.name}-${chIdx}`"
|
||
class="border-b border-gray-100 last:rounded-b-lg last:border-b-0 dark:border-dark-700"
|
||
>
|
||
<div
|
||
v-for="(section, secIdx) in channel.platforms"
|
||
:key="`${channel.name}-${section.platform}`"
|
||
class="grid items-center px-4 py-3 transition-colors hover:bg-gray-50/40 dark:hover:bg-dark-800/40"
|
||
:class="{ 'border-t border-gray-100/70 dark:border-dark-700/50': secIdx > 0 }"
|
||
:style="gridStyle"
|
||
>
|
||
<!-- 渠道名:仅第一行渲染,后续行留空(视觉上的 rowspan) -->
|
||
<div>
|
||
<template v-if="secIdx === 0">
|
||
<div class="font-medium text-gray-900 dark:text-white">{{ channel.name }}</div>
|
||
<div
|
||
v-if="channel.description"
|
||
class="mt-0.5 text-xs text-gray-500 dark:text-gray-400"
|
||
>
|
||
{{ channel.description }}
|
||
</div>
|
||
</template>
|
||
</div>
|
||
|
||
<!-- 平台徽章 -->
|
||
<div>
|
||
<span
|
||
:class="[
|
||
'inline-flex items-center gap-1 rounded-md border px-2 py-0.5 text-[11px] font-medium uppercase',
|
||
platformBadgeClass(section.platform),
|
||
]"
|
||
>
|
||
<PlatformIcon :platform="section.platform as GroupPlatform" size="xs" />
|
||
{{ section.platform }}
|
||
</span>
|
||
</div>
|
||
|
||
<!-- 分组:专属分组在前(紫色 shield 行),公开分组在后(灰色 globe 行)。
|
||
订阅分组由 GroupBadge 内部按 subscription_type 自动加深背景。 -->
|
||
<div class="flex flex-col gap-1.5">
|
||
<div
|
||
v-if="exclusiveGroups(section).length > 0"
|
||
class="flex flex-wrap items-center gap-1.5"
|
||
>
|
||
<span
|
||
class="inline-flex items-center gap-0.5 text-[10px] font-medium uppercase text-purple-600 dark:text-purple-400"
|
||
:title="t('availableChannels.exclusiveTooltip')"
|
||
>
|
||
<Icon name="shield" size="xs" class="h-3 w-3" />
|
||
{{ t('availableChannels.exclusive') }}
|
||
</span>
|
||
<GroupBadge
|
||
v-for="g in exclusiveGroups(section)"
|
||
:key="`ex-${g.id}`"
|
||
:name="g.name"
|
||
:platform="g.platform as GroupPlatform"
|
||
:subscription-type="(g.subscription_type || 'standard') as SubscriptionType"
|
||
:rate-multiplier="g.rate_multiplier"
|
||
:user-rate-multiplier="userGroupRates[g.id] ?? null"
|
||
/>
|
||
</div>
|
||
<div
|
||
v-if="publicGroups(section).length > 0"
|
||
class="flex flex-wrap items-center gap-1.5"
|
||
>
|
||
<span
|
||
class="inline-flex items-center gap-0.5 text-[10px] font-medium uppercase text-gray-500 dark:text-gray-400"
|
||
:title="t('availableChannels.publicTooltip')"
|
||
>
|
||
<Icon name="globe" size="xs" class="h-3 w-3" />
|
||
{{ t('availableChannels.public') }}
|
||
</span>
|
||
<GroupBadge
|
||
v-for="g in publicGroups(section)"
|
||
:key="`pub-${g.id}`"
|
||
:name="g.name"
|
||
:platform="g.platform as GroupPlatform"
|
||
:subscription-type="(g.subscription_type || 'standard') as SubscriptionType"
|
||
:rate-multiplier="g.rate_multiplier"
|
||
:user-rate-multiplier="userGroupRates[g.id] ?? null"
|
||
/>
|
||
</div>
|
||
<span v-if="section.groups.length === 0" class="text-xs text-gray-400">-</span>
|
||
</div>
|
||
|
||
<!-- 支持模型 -->
|
||
<div class="flex flex-wrap gap-1">
|
||
<SupportedModelChip
|
||
v-for="m in section.supported_models"
|
||
:key="`${section.platform}-${m.name}`"
|
||
:model="m"
|
||
:pricing-key-prefix="pricingKeyPrefix"
|
||
:no-pricing-label="noPricingLabel"
|
||
:show-platform="false"
|
||
:platform-hint="section.platform"
|
||
/>
|
||
<span v-if="section.supported_models.length === 0" class="text-xs text-gray-400">
|
||
{{ noModelsLabel }}
|
||
</span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script setup lang="ts">
|
||
import { useI18n } from 'vue-i18n'
|
||
import Icon from '@/components/icons/Icon.vue'
|
||
import PlatformIcon from '@/components/common/PlatformIcon.vue'
|
||
import GroupBadge from '@/components/common/GroupBadge.vue'
|
||
import SupportedModelChip from './SupportedModelChip.vue'
|
||
import type { UserAvailableChannel, UserAvailableGroup, UserChannelPlatformSection } from '@/api/channels'
|
||
import type { GroupPlatform, SubscriptionType } from '@/types'
|
||
import { platformBadgeClass } from '@/utils/platformColors'
|
||
|
||
/** 四列 grid 的 template-columns;与表头、每个 section 行共享。 */
|
||
const gridStyle = 'grid-template-columns: 220px 140px minmax(240px, 1fr) minmax(280px, 2fr); display: grid;'
|
||
|
||
const props = defineProps<{
|
||
columns: {
|
||
name: string
|
||
platform: string
|
||
groups: string
|
||
supportedModels: string
|
||
}
|
||
rows: UserAvailableChannel[]
|
||
loading: boolean
|
||
pricingKeyPrefix: string
|
||
noPricingLabel: string
|
||
noModelsLabel: string
|
||
emptyLabel: string
|
||
/** 用户专属倍率(group_id → multiplier);无专属时由 GroupBadge 仅显示默认倍率。 */
|
||
userGroupRates: Record<number, number>
|
||
}>()
|
||
|
||
// Suppress unused warning — props is accessed via template automatically but
|
||
// the explicit reference here keeps the linter from flagging userGroupRates.
|
||
void props.userGroupRates
|
||
|
||
const { t } = useI18n()
|
||
|
||
function exclusiveGroups(section: UserChannelPlatformSection): UserAvailableGroup[] {
|
||
return section.groups.filter((g) => g.is_exclusive)
|
||
}
|
||
|
||
function publicGroups(section: UserChannelPlatformSection): UserAvailableGroup[] {
|
||
return section.groups.filter((g) => !g.is_exclusive)
|
||
}
|
||
</script>
|