app / src /components /dashboard /CurrentAccount.tsx
AZILS's picture
Upload 323 files
a21c316 verified
import { CheckCircle, Mail, Diamond, Gem, Circle, Tag, Lock } from 'lucide-react';
import { Account } from '../../types/account';
import { formatTimeRemaining } from '../../utils/format';
interface CurrentAccountProps {
account: Account | null;
onSwitch?: () => void;
}
import { useTranslation } from 'react-i18next';
function CurrentAccount({ account, onSwitch }: CurrentAccountProps) {
const { t } = useTranslation();
if (!account) {
return (
<div className="bg-white dark:bg-base-100 rounded-xl p-4 shadow-sm border border-gray-100 dark:border-base-200">
<h2 className="text-base font-semibold text-gray-900 dark:text-base-content mb-2 flex items-center gap-2">
<CheckCircle className="w-4 h-4 text-green-500" />
{t('dashboard.current_account')}
</h2>
<div className="text-center py-4 text-gray-400 dark:text-gray-500 text-sm">
{t('dashboard.no_active_account')}
</div>
</div>
);
}
const geminiProModel = account.quota?.models
.filter(m =>
m.name.toLowerCase() === 'gemini-3-pro-high'
|| m.name.toLowerCase() === 'gemini-3-pro-low'
|| m.name.toLowerCase() === 'gemini-3.1-pro-high'
|| m.name.toLowerCase() === 'gemini-3.1-pro-low'
)
.sort((a, b) => (a.percentage || 0) - (b.percentage || 0))[0];
const geminiFlashModel = account.quota?.models.find(m => m.name.toLowerCase() === 'gemini-3-flash');
const geminiImageModel = account.quota?.models.find(m => m.name.toLowerCase() === 'gemini-3-pro-image');
const claudeGroupNames = [
'claude-opus-4-6-thinking',
'claude'
];
const claudeModel = account.quota?.models
.filter(m => claudeGroupNames.includes(m.name.toLowerCase()))
.sort((a, b) => (a.percentage || 0) - (b.percentage || 0))[0];
return (
<div className="bg-white dark:bg-base-100 rounded-xl p-4 shadow-sm border border-gray-100 dark:border-base-200 h-full flex flex-col">
<h2 className="text-base font-semibold text-gray-900 dark:text-base-content mb-3 flex items-center gap-2">
<CheckCircle className="w-4 h-4 text-green-500" />
{t('dashboard.current_account')}
</h2>
<div className="space-y-4 flex-1">
<div className="flex items-center gap-3 mb-1">
<div className="flex items-center gap-2 flex-1 min-w-0">
<Mail className="w-3.5 h-3.5 text-gray-400" />
<span className="text-sm font-medium text-gray-700 dark:text-gray-300 truncate">{account.email}</span>
</div>
{/* 订阅类型 */}
{account.quota?.subscription_tier && (() => {
const tier = account.quota.subscription_tier.toLowerCase();
if (tier.includes('ultra')) {
return (
<span className="flex items-center gap-1 px-2 py-0.5 rounded-md bg-gradient-to-r from-purple-600 to-pink-600 text-white text-[10px] font-bold shadow-sm shrink-0">
<Gem className="w-2.5 h-2.5 fill-current" />
ULTRA
</span>
);
} else if (tier.includes('pro')) {
return (
<span className="flex items-center gap-1 px-2 py-0.5 rounded-md bg-gradient-to-r from-blue-600 to-indigo-600 text-white text-[10px] font-bold shadow-sm shrink-0">
<Diamond className="w-2.5 h-2.5 fill-current" />
PRO
</span>
);
} else {
return (
<span className="flex items-center gap-1 px-2 py-0.5 rounded-md bg-gray-100 dark:bg-white/10 text-gray-500 dark:text-gray-400 text-[10px] font-bold shadow-sm border border-gray-200 dark:border-white/10 shrink-0">
<Circle className="w-2.5 h-2.5" />
FREE
</span>
);
}
})()}
{/* 自定义标签 */}
{account.custom_label && (
<span className="flex items-center gap-1 px-2 py-0.5 rounded-md bg-orange-100 dark:bg-orange-900/30 text-orange-600 dark:text-orange-400 text-[10px] font-bold shadow-sm shrink-0">
<Tag className="w-2.5 h-2.5" />
{account.custom_label}
</span>
)}
</div>
{/* Gemini Pro 配额 */}
{geminiProModel && (
<div className="space-y-1.5">
<div className="flex justify-between items-baseline">
<span className="text-xs font-medium text-gray-600 dark:text-gray-400 flex items-center gap-1">
{(account.protected_models?.includes('gemini-3-pro-high') || account.protected_models?.includes('gemini-3.1-pro-high')) && <Lock className="w-2.5 h-2.5 text-rose-500" />}
Gemini 3.1 Pro
</span>
<div className="flex items-center gap-2">
<span className="text-[10px] text-gray-400 dark:text-gray-500" title={`${t('accounts.reset_time')}: ${new Date(geminiProModel.reset_time).toLocaleString()}`}>
{geminiProModel.reset_time ? `R: ${formatTimeRemaining(geminiProModel.reset_time)}` : t('common.unknown')}
</span>
<span className={`text-xs font-bold ${geminiProModel.percentage >= 50 ? 'text-emerald-600 dark:text-emerald-400' :
geminiProModel.percentage >= 20 ? 'text-amber-600 dark:text-amber-400' : 'text-rose-600 dark:text-rose-400'
}`}>
{geminiProModel.percentage}%
</span>
</div>
</div>
<div className="w-full bg-gray-100 dark:bg-base-300 rounded-full h-1.5 overflow-hidden">
<div
className={`h-full rounded-full transition-all duration-700 ${geminiProModel.percentage >= 50 ? 'bg-gradient-to-r from-emerald-400 to-emerald-500' :
geminiProModel.percentage >= 20 ? 'bg-gradient-to-r from-amber-400 to-amber-500' :
'bg-gradient-to-r from-rose-400 to-rose-500'
}`}
style={{ width: `${geminiProModel.percentage}%` }}
></div>
</div>
</div>
)}
{/* Gemini 3 Pro Image 配额 */}
{geminiImageModel && (
<div className="space-y-1.5">
<div className="flex justify-between items-baseline">
<span className="text-xs font-medium text-gray-600 dark:text-gray-400 flex items-center gap-1">
{account.protected_models?.includes('gemini-3-pro-image') && <Lock className="w-2.5 h-2.5 text-rose-500" />}
Gemini 3 Pro Image
</span>
<div className="flex items-center gap-2">
<span className="text-[10px] text-gray-400 dark:text-gray-500" title={`${t('accounts.reset_time')}: ${new Date(geminiImageModel.reset_time).toLocaleString()}`}>
{geminiImageModel.reset_time ? `R: ${formatTimeRemaining(geminiImageModel.reset_time)}` : t('common.unknown')}
</span>
<span className={`text-xs font-bold ${geminiImageModel.percentage >= 50 ? 'text-emerald-600 dark:text-emerald-400' :
geminiImageModel.percentage >= 20 ? 'text-amber-600 dark:text-amber-400' : 'text-rose-600 dark:text-rose-400'
}`}>
{geminiImageModel.percentage}%
</span>
</div>
</div>
<div className="w-full bg-gray-100 dark:bg-base-300 rounded-full h-1.5 overflow-hidden">
<div
className={`h-full rounded-full transition-all duration-700 ${geminiImageModel.percentage >= 50 ? 'bg-gradient-to-r from-emerald-400 to-emerald-500' :
geminiImageModel.percentage >= 20 ? 'bg-gradient-to-r from-amber-400 to-amber-500' :
'bg-gradient-to-r from-rose-400 to-rose-500'
}`}
style={{ width: `${geminiImageModel.percentage}%` }}
></div>
</div>
</div>
)}
{/* Gemini Flash 配额 */}
{geminiFlashModel && (
<div className="space-y-1.5">
<div className="flex justify-between items-baseline">
<span className="text-xs font-medium text-gray-600 dark:text-gray-400 flex items-center gap-1">
{account.protected_models?.includes('gemini-3-flash') && <Lock className="w-2.5 h-2.5 text-rose-500" />}
Gemini 3 Flash
</span>
<div className="flex items-center gap-2">
<span className="text-[10px] text-gray-400 dark:text-gray-500" title={`${t('accounts.reset_time')}: ${new Date(geminiFlashModel.reset_time).toLocaleString()}`}>
{geminiFlashModel.reset_time ? `R: ${formatTimeRemaining(geminiFlashModel.reset_time)}` : t('common.unknown')}
</span>
<span className={`text-xs font-bold ${geminiFlashModel.percentage >= 50 ? 'text-emerald-600 dark:text-emerald-400' :
geminiFlashModel.percentage >= 20 ? 'text-amber-600 dark:text-amber-400' : 'text-rose-600 dark:text-rose-400'
}`}>
{geminiFlashModel.percentage}%
</span>
</div>
</div>
<div className="w-full bg-gray-100 dark:bg-base-300 rounded-full h-1.5 overflow-hidden">
<div
className={`h-full rounded-full transition-all duration-700 ${geminiFlashModel.percentage >= 50 ? 'bg-gradient-to-r from-emerald-400 to-emerald-500' :
geminiFlashModel.percentage >= 20 ? 'bg-gradient-to-r from-amber-400 to-amber-500' :
'bg-gradient-to-r from-rose-400 to-rose-500'
}`}
style={{ width: `${geminiFlashModel.percentage}%` }}
></div>
</div>
</div>
)}
{/* Claude 配额 */}
{claudeModel && (
<div className="space-y-1.5">
<div className="flex justify-between items-baseline">
<span className="text-xs font-medium text-gray-600 dark:text-gray-400 flex items-center gap-1">
{account.protected_models?.includes('claude') && <Lock className="w-2.5 h-2.5 text-rose-500" />}
Claude 系列
</span>
<div className="flex items-center gap-2">
<span className="text-[10px] text-gray-400 dark:text-gray-500" title={`${t('accounts.reset_time')}: ${new Date(claudeModel.reset_time).toLocaleString()}`}>
{claudeModel.reset_time ? `R: ${formatTimeRemaining(claudeModel.reset_time)}` : t('common.unknown')}
</span>
<span className={`text-xs font-bold ${claudeModel.percentage >= 50 ? 'text-cyan-600 dark:text-cyan-400' :
claudeModel.percentage >= 20 ? 'text-orange-600 dark:text-orange-400' : 'text-rose-600 dark:text-rose-400'
}`}>
{claudeModel.percentage}%
</span>
</div>
</div>
<div className="w-full bg-gray-100 dark:bg-base-300 rounded-full h-1.5 overflow-hidden">
<div
className={`h-full rounded-full transition-all duration-700 ${claudeModel.percentage >= 50 ? 'bg-gradient-to-r from-cyan-400 to-cyan-500' :
claudeModel.percentage >= 20 ? 'bg-gradient-to-r from-orange-400 to-orange-500' :
'bg-gradient-to-r from-rose-400 to-rose-500'
}`}
style={{ width: `${claudeModel.percentage}%` }}
></div>
</div>
</div>
)}
</div>
{onSwitch && (
<div className="mt-auto pt-3">
<button
className="w-full px-3 py-1.5 text-xs text-gray-700 dark:text-gray-300 border border-gray-200 dark:border-base-300 rounded-lg hover:bg-gray-50 dark:hover:bg-base-200 transition-colors"
onClick={onSwitch}
>
{t('dashboard.switch_account')}
</button>
</div>
)}
</div>
);
}
export default CurrentAccount;