new-api/web/default/src/features/wallet/components/recharge-form-card.tsx
Calcium-Ion 8b2b03d276
feat(web/default): unified UI overhaul — Base UI migration, theme presets, rankings dashboard, and table toolbar refactor (#4633)
* 🎨 feat(web/default): add shadcn-style theme presets, radius prefs, and fix selection badges

Integrate the qn-platform–style OKLCH color system into the default frontend while keeping the existing blue-tinted dark tokens for the default theme. Add [data-theme-preset] palettes for seven named presets plus the default zinc-like scale, define [data-theme-radius] overrides so user radius beats preset --radius, and align the Tailwind @custom-variant dark helper with .dark usage.

Introduce ThemeCustomizationProvider to own preset and radius state, persist choices in cookies (theme-preset, theme-radius), and sync data-theme-preset / data-theme-radius on <html>. Wrap the tree in main.tsx.

Extend ConfigDrawer with theme preset swatches (scoped data-theme-preset) and radius previews wired to context; refactor swatch/card markup so selected CircleCheck badges sit outside clipped rows (remove outer overflow-hidden that hid the centered checkmark).

Add i18n keys for preset names, radius, and accessibility labels across en, zh, fr, ja, ru, vi.

* 🎨 fix(web): align segmented controls with theme radius tokens

- Replace hard-coded inner pill radii (rounded-[5px]) on dashboard chart
  toolbars with radius-md so the active state follows --radius when users
  change Radius in Theme Settings.
- Use nested radii consistent with TabsList/TabsTrigger: outer
  rounded-lg (var(--radius)) and inner rounded-md (calc(var(--radius) - 2px))
  so the track and active thumb stay concentric at small scales (e.g.
  0.3rem) instead of a squared “focus” block inside a rounded shell.
- Apply the same pattern to pricing SegmentedControl and the segmented
  groups in consumption-distribution-chart, model-charts, and user-charts.

Verified: bun run typecheck (web/default)

*  feat(pricing): enrich model details with uptime sparkline and API documentation

Add a compact 30-day uptime sparkline (OpenRouter-style bars + aggregate %) with
per-day tooltips, surface it in a status row under quick stats and in the
per-group performance table, and extend mock data so uptime series are stable
and optionally scoped by group.

Introduce an API tab with Shiki-highlighted code samples (cURL, Python,
TypeScript, JavaScript), endpoint-type switching, authentication guidance, a
supported-parameters table, and mock per-group RPM/TPM/RPD limits. Infer
vendor, tokenizer, license, and data-retention hints for a provider & data
privacy card on the Overview tab (capabilities/modalities stay with model
identity; rate limits stay with the API tab).

Update i18n for all new user-facing strings across en, zh, fr, ja, ru, and vi.

* 🏆 feat(rankings): add comprehensive rankings dashboard

Add a mock-data powered rankings experience with period tabs, model, app, and vendor leaderboards, market share and history charts, movers, new releases, and per-category sections while backend analytics are pending.

Link ranked models to pricing details and ranked vendors to filtered pricing results, and include localized copy for all supported frontend locales.

* fix(theme): correct theme preset selection state

- update Base UI Radio selectors to use data-checked/data-unchecked states.
- fix unchecked theme options still showing selected indicators.
- isolate the default theme preview tokens to prevent preset changes from leaking into it.

* fix(setup): correct usage mode radio state

- use Base UI data-checked/data-unchecked states for RadioGroup styling.
- hide radio indicators when options are unchecked to avoid setup page display issues.
- drive usage mode card and icon selection styles from Base UI state.

* fix(auth): submit sign-in and sign-up forms

* 🎨 refactor: Align default theme with shadcn Base Nova and prune legacy customization

Migrate shadcn UI to Base UI primitives via CLI (`base-nova` / `components.json`)
and reinstall full component registry with `--overwrite`, including Hugeicons-backed
widgets and newly added registry components.

- Remove custom multi-preset/theme-radius system (`ThemeCustomizationProvider`, cookies,
  preset UI from config drawer); rely on official semantic CSS tokens + light/dark only.
- Replace `theme.css` with shadcn’s documented neutral `:root`/`.dark` palette and
  `@theme inline` mappings (plus skeleton token vars for existing shimmer usage).
- Update global styles for Base UI: collapsible animation uses `--collapsible-panel-height`;
  clarify scroll-lock override comment.

Application compatibility:
- Keep minimal shims where app code diverged from official APIs (popover collision props,
  combobox legacy `options` callers, Spinner prop typing).
- Switch interactive styling from Radix-era `data-state` / `--radix-*` selectors to Base UI
  semantics (`data-open`, `data-popup-open`, `data-panel-open`, `--anchor-width`, etc.)

Tooling / docs / build:
- Rename Rsbuild vendor chunk grouping to `@base-ui` + transitive `@radix-ui`.
- Refresh AGENTS.md / CLAUDE.md / classic→default sync skill for Base UI stack.
- Bump `package.json` / lockfile for shadcn-postinstall deps (Hugeicons, chart stack, themes, etc.)

Verified: `bun run typecheck` passes.

Note: `bun run lint` still reports pre-existing hooks rule violations elsewhere;
not addressed in this change.

* 🎨 chore(web/default): unify table toolbar, relocate usage stats, refine filters

- Refactor DataTableToolbar to a single wrapping flex row with a
  right-aligned action cluster (Reset / Search / View / Expand) for a
  cleaner Ant Design Pro–style filter bar; remove the dedicated stats row
  and the toolbar `stats` prop.
- Move Common Logs summary badges (Usage / RPM / TPM) and the sensitive-
  data visibility toggle into the page header via CommonLogsHeaderActions
  and SectionPageLayout.Actions so the toolbar stays focused on filters.
- Slim CommonLogsFilterBar props (no stats / preActions eye control).
- Improve CompactDateTimeRangePicker: show minute-precision labels on the
  trigger (seconds omitted; aligns with datetime-local inputs); widen the
  trigger on sm+ breakpoints so the full range is visible without truncation;
  apply the same width in task logs filters.
- Simplify DataTableViewOptions: text-only “View” trigger, no sliders icon.
- Earlier layout tweak: extra top padding on SectionPageLayout scroll
  content so control focus rings are not clipped by overflow.

* feat(web/default): Base UI migration and component foundation

Migrate from Radix UI to Base UI, rewrite core UI primitives,
update dependencies (recharts, date-fns, next-themes), add
shadcn agent skill documentation, and refresh AI element components.

This is the foundational work from the v2/localmain lineage that
was not covered by the individual feature commits above.

---------

Co-authored-by: t0ng7u <dev@aiass.cc>
Co-authored-by: QuentinHsu <xuquentinyang@gmail.com>
2026-05-06 12:39:36 +08:00

476 lines
17 KiB
TypeScript
Vendored

import { useState, useEffect } from 'react'
import { Gift, ExternalLink, Loader2, Receipt, WalletCards } from 'lucide-react'
import { useTranslation } from 'react-i18next'
import { formatNumber } from '@/lib/format'
import { cn } from '@/lib/utils'
import { Alert, AlertDescription } from '@/components/ui/alert'
import { Button } from '@/components/ui/button'
import { Card, CardContent, CardHeader } from '@/components/ui/card'
import { Input } from '@/components/ui/input'
import { Label } from '@/components/ui/label'
import { Skeleton } from '@/components/ui/skeleton'
import { TitledCard } from '@/components/ui/titled-card'
import {
Tooltip,
TooltipContent,
TooltipProvider,
TooltipTrigger,
} from '@/components/ui/tooltip'
import {
formatCurrency,
getDiscountLabel,
getPaymentIcon,
getMinTopupAmount,
calculatePresetPricing,
} from '../lib'
import type {
PaymentMethod,
PresetAmount,
TopupInfo,
CreemProduct,
WaffoPayMethod,
} from '../types'
import { CreemProductsSection } from './creem-products-section'
interface RechargeFormCardProps {
topupInfo: TopupInfo | null
presetAmounts: PresetAmount[]
selectedPreset: number | null
onSelectPreset: (preset: PresetAmount) => void
topupAmount: number
onTopupAmountChange: (amount: number) => void
paymentAmount: number
calculating: boolean
onPaymentMethodSelect: (method: PaymentMethod) => void
paymentLoading: string | null
redemptionCode: string
onRedemptionCodeChange: (code: string) => void
onRedeem: () => void
redeeming: boolean
topupLink?: string
loading?: boolean
priceRatio?: number
usdExchangeRate?: number
onOpenBilling?: () => void
creemProducts?: CreemProduct[]
enableCreemTopup?: boolean
onCreemProductSelect?: (product: CreemProduct) => void
enableWaffoTopup?: boolean
waffoPayMethods?: WaffoPayMethod[]
waffoMinTopup?: number
onWaffoMethodSelect?: (method: WaffoPayMethod, index: number) => void
enableWaffoPancakeTopup?: boolean
}
export function RechargeFormCard({
topupInfo,
presetAmounts,
selectedPreset,
onSelectPreset,
topupAmount,
onTopupAmountChange,
paymentAmount,
calculating,
onPaymentMethodSelect,
paymentLoading,
redemptionCode,
onRedemptionCodeChange,
onRedeem,
redeeming,
topupLink,
loading,
priceRatio = 1,
usdExchangeRate = 1,
onOpenBilling,
creemProducts,
enableCreemTopup,
onCreemProductSelect,
enableWaffoTopup,
waffoPayMethods,
waffoMinTopup,
onWaffoMethodSelect,
enableWaffoPancakeTopup,
}: RechargeFormCardProps) {
const { t } = useTranslation()
const [localAmount, setLocalAmount] = useState(topupAmount.toString())
useEffect(() => {
setLocalAmount(topupAmount.toString())
}, [topupAmount])
const handleAmountChange = (value: string) => {
setLocalAmount(value)
const numValue = parseInt(value) || 0
if (numValue >= 0) {
onTopupAmountChange(numValue)
}
}
const hasConfigurableTopup =
topupInfo?.enable_online_topup ||
topupInfo?.enable_stripe_topup ||
enableWaffoTopup ||
enableWaffoPancakeTopup
const hasAnyTopup = hasConfigurableTopup || enableCreemTopup
const hasStandardPaymentMethods =
Array.isArray(topupInfo?.pay_methods) && topupInfo.pay_methods.length > 0
const hasWaffoPaymentMethods =
Array.isArray(waffoPayMethods) && waffoPayMethods.length > 0
const minTopup = getMinTopupAmount(topupInfo)
if (loading) {
return (
<Card className='gap-0 overflow-hidden py-0'>
<CardHeader className='border-b p-3 !pb-3 sm:p-5 sm:!pb-5'>
<Skeleton className='h-6 w-32' />
<Skeleton className='mt-2 h-4 w-48' />
</CardHeader>
<CardContent className='space-y-4 p-3 sm:space-y-6 sm:p-5'>
<div className='space-y-4 sm:space-y-6'>
{/* Preset Amounts Skeleton */}
<div className='space-y-3'>
<Skeleton className='h-3 w-16' />
<div className='grid grid-cols-2 gap-3 sm:grid-cols-4'>
{Array.from({ length: 8 }).map((_, i) => (
<Skeleton key={i} className='h-[72px] rounded-lg' />
))}
</div>
</div>
{/* Custom Amount Input Skeleton */}
<div className='space-y-3'>
<Skeleton className='h-3 w-28' />
<Skeleton className='h-[42px] w-full' />
</div>
{/* Payment Methods Skeleton */}
<div className='space-y-3'>
<Skeleton className='h-3 w-32' />
<div className='flex flex-wrap gap-3'>
{Array.from({ length: 3 }).map((_, i) => (
<Skeleton key={i} className='h-10 w-24 rounded-lg' />
))}
</div>
</div>
</div>
{/* Redemption Code Section Skeleton */}
<div className='space-y-3 border-t pt-8'>
<Skeleton className='h-3 w-24' />
<div className='flex gap-2'>
<Skeleton className='h-10 flex-1' />
<Skeleton className='h-10 w-20' />
</div>
</div>
</CardContent>
</Card>
)
}
return (
<TitledCard
title={t('Add Funds')}
description={t('Choose an amount and payment method')}
icon={<WalletCards className='h-4 w-4' />}
action={
onOpenBilling ? (
<Button
variant='outline'
size='sm'
onClick={onOpenBilling}
className='w-full gap-2 sm:w-auto'
>
<Receipt className='h-4 w-4' />
{t('Order History')}
</Button>
) : null
}
contentClassName='space-y-4 sm:space-y-6'
>
{/* Online Topup Section */}
{hasAnyTopup ? (
<div className='space-y-4 sm:space-y-6'>
{hasConfigurableTopup && (
<>
{presetAmounts.length > 0 && (
<div className='space-y-2.5 sm:space-y-3'>
<Label className='text-muted-foreground text-xs font-medium tracking-wider uppercase'>
{t('Amount')}
</Label>
<div className='grid grid-cols-2 gap-1.5 sm:gap-3 md:grid-cols-4'>
{presetAmounts.map((preset, index) => {
const discount =
preset.discount ||
topupInfo?.discount?.[preset.value] ||
1.0
const {
displayValue,
actualPrice,
savedAmount,
hasDiscount,
} = calculatePresetPricing(
preset.value,
priceRatio,
discount,
usdExchangeRate
)
return (
<Button
key={index}
variant='outline'
className={cn(
'hover:border-foreground flex min-h-16 flex-col items-start rounded-lg px-3 py-2.5 text-left whitespace-normal sm:min-h-[72px] sm:p-4',
selectedPreset === preset.value
? 'border-foreground bg-foreground/5'
: 'border-muted'
)}
onClick={() => onSelectPreset(preset)}
>
<div className='flex w-full items-center justify-between'>
<div className='text-base font-semibold sm:text-lg'>
{formatNumber(displayValue)}
</div>
{hasDiscount && (
<div className='text-xs font-medium text-green-600'>
{getDiscountLabel(discount)}
</div>
)}
</div>
<div className='text-muted-foreground mt-1.5 w-full text-xs sm:mt-2'>
Pay {formatCurrency(actualPrice)}
{hasDiscount && savedAmount > 0 && (
<span className='text-green-600'>
{' '}
Save {formatCurrency(savedAmount)}
</span>
)}
</div>
</Button>
)
})}
</div>
</div>
)}
<div className='space-y-2.5 sm:space-y-3'>
<Label
htmlFor='topup-amount'
className='text-muted-foreground text-xs font-medium tracking-wider uppercase'
>
{t('Custom Amount')}
</Label>
<div className='grid grid-cols-[minmax(0,1fr)_minmax(110px,0.55fr)] gap-2 lg:grid-cols-[minmax(0,1fr)_auto] lg:items-center'>
<Input
id='topup-amount'
type='number'
value={localAmount}
onChange={(e) => handleAmountChange(e.target.value)}
min={minTopup}
placeholder={`Minimum ${minTopup}`}
className='h-9 text-base sm:h-10 sm:text-lg'
/>
<div className='bg-muted/30 flex min-h-9 items-center justify-between gap-2 rounded-md border px-3 lg:min-w-52'>
<span className='text-muted-foreground truncate text-xs'>
{t('Amount to pay:')}
</span>
{calculating ? (
<Skeleton className='h-5 w-16' />
) : (
<span className='text-sm font-semibold'>
{formatCurrency(paymentAmount)}
</span>
)}
</div>
</div>
</div>
<div className='space-y-2.5 sm:space-y-3'>
<Label className='text-muted-foreground text-xs font-medium tracking-wider uppercase'>
{t('Payment Method')}
</Label>
{hasStandardPaymentMethods ? (
<div className='grid grid-cols-2 gap-1.5 sm:gap-3 lg:grid-cols-3'>
{topupInfo?.pay_methods?.map((method) => {
const minTopup = method.min_topup || 0
const disabled = minTopup > topupAmount
const button = (
<Button
key={method.type}
variant='outline'
onClick={() => onPaymentMethodSelect(method)}
disabled={disabled || !!paymentLoading}
className='h-9 min-w-0 justify-start gap-2 rounded-lg px-3'
>
{paymentLoading === method.type ? (
<Loader2 className='h-4 w-4 animate-spin' />
) : (
getPaymentIcon(
method.type,
'h-4 w-4',
method.icon,
method.name
)
)}
<span className='truncate'>{method.name}</span>
</Button>
)
return disabled ? (
<TooltipProvider key={method.type}>
<Tooltip>
<TooltipTrigger render={button}></TooltipTrigger>
<TooltipContent>
{t('Minimum topup amount: {{amount}}', {
amount: minTopup,
})}
</TooltipContent>
</Tooltip>
</TooltipProvider>
) : (
button
)
})}
</div>
) : hasWaffoPaymentMethods ? null : (
<Alert>
<AlertDescription>
{t(
'No payment methods available. Please contact administrator.'
)}
</AlertDescription>
</Alert>
)}
</div>
{enableWaffoTopup &&
hasWaffoPaymentMethods &&
onWaffoMethodSelect && (
<div className='space-y-2.5 sm:space-y-3'>
<Label className='text-muted-foreground text-xs font-medium tracking-wider uppercase'>
{t('Waffo Payment')}
</Label>
<div className='grid grid-cols-2 gap-1.5 sm:gap-3 lg:grid-cols-3'>
{waffoPayMethods?.map((method, index) => {
const loadingKey = `waffo-${index}`
const waffoMin = waffoMinTopup || 0
const belowMin = waffoMin > topupAmount
const button = (
<Button
key={`${method.name}-${index}`}
variant='outline'
onClick={() => onWaffoMethodSelect(method, index)}
disabled={belowMin || !!paymentLoading}
className='h-9 min-w-0 justify-start gap-2 rounded-lg px-3'
>
{paymentLoading === loadingKey ? (
<Loader2 className='h-4 w-4 animate-spin' />
) : method.icon ? (
<img
src={method.icon}
alt={method.name}
className='h-4 w-4 object-contain'
/>
) : (
getPaymentIcon('waffo')
)}
<span className='truncate'>{method.name}</span>
</Button>
)
return belowMin ? (
<TooltipProvider key={`${method.name}-${index}`}>
<Tooltip>
<TooltipTrigger render={button}></TooltipTrigger>
<TooltipContent>
{t('Minimum topup amount: {{amount}}', {
amount: waffoMin,
})}
</TooltipContent>
</Tooltip>
</TooltipProvider>
) : (
button
)
})}
</div>
</div>
)}
</>
)}
</div>
) : (
<Alert>
<AlertDescription>
{t(
'Online topup is not enabled. Please use redemption code or contact administrator.'
)}
</AlertDescription>
</Alert>
)}
{/* Creem Products Section */}
{enableCreemTopup &&
Array.isArray(creemProducts) &&
creemProducts.length > 0 &&
onCreemProductSelect && (
<div className='space-y-2.5 border-t pt-4 sm:space-y-3 sm:pt-6'>
<Label className='text-muted-foreground text-xs font-medium tracking-wider uppercase'>
{t('Creem Payment')}
</Label>
<CreemProductsSection
products={creemProducts}
onProductSelect={onCreemProductSelect}
/>
</div>
)}
{/* Redemption Code Section */}
<div className='space-y-2.5 border-t pt-4 sm:space-y-3 sm:pt-6'>
<div className='flex items-center gap-2'>
<Gift className='text-muted-foreground h-4 w-4' />
<Label
htmlFor='redemption-code'
className='text-muted-foreground text-xs font-medium tracking-wider uppercase'
>
{t('Have a Code?')}
</Label>
</div>
<div className='grid grid-cols-[minmax(0,1fr)_auto] gap-2'>
<Input
id='redemption-code'
value={redemptionCode}
onChange={(e) => onRedemptionCodeChange(e.target.value)}
placeholder={t('Enter your redemption code')}
className='h-9 min-w-0'
/>
<Button
onClick={onRedeem}
disabled={redeeming}
variant='outline'
className='h-9 px-4'
>
{redeeming && <Loader2 className='mr-2 h-4 w-4 animate-spin' />}
{t('Redeem')}
</Button>
</div>
{topupLink && (
<p className='text-muted-foreground text-xs'>
{t('Need a code?')}{' '}
<a
href={topupLink}
target='_blank'
rel='noopener noreferrer'
className='inline-flex items-center gap-1 underline-offset-4 hover:underline'
>
{t('Purchase here')}
<ExternalLink className='h-3 w-3' />
</a>
</p>
)}
</div>
</TitledCard>
)
}