✨ refactor(wallet): Top-up layout to embed subscription plans into the recharge card tabs
- Defaulting to subscriptions when available and avoiding initial flash when no plans exist. - Adjust the wide-screen layout to place wallet and invite sections side by side, simplify the subscription header and controls, and add padding to prevent card borders from clipping. - Update related i18n strings by adding the new tab label and removing the obsolete subscription blurb.
This commit is contained in:
parent
725473d3d5
commit
9942045b94
File diff suppressed because it is too large
Load Diff
@ -19,7 +19,6 @@ For commercial licensing, please contact support@quantumnous.com
|
|||||||
|
|
||||||
import React, { useMemo, useState } from 'react';
|
import React, { useMemo, useState } from 'react';
|
||||||
import {
|
import {
|
||||||
Avatar,
|
|
||||||
Badge,
|
Badge,
|
||||||
Button,
|
Button,
|
||||||
Card,
|
Card,
|
||||||
@ -33,7 +32,7 @@ import {
|
|||||||
} from '@douyinfe/semi-ui';
|
} from '@douyinfe/semi-ui';
|
||||||
import { API, showError, showSuccess, renderQuota } from '../../helpers';
|
import { API, showError, showSuccess, renderQuota } from '../../helpers';
|
||||||
import { getCurrencyConfig } from '../../helpers/render';
|
import { getCurrencyConfig } from '../../helpers/render';
|
||||||
import { Crown, RefreshCw, Sparkles } from 'lucide-react';
|
import { RefreshCw, Sparkles } from 'lucide-react';
|
||||||
import SubscriptionPurchaseModal from './modals/SubscriptionPurchaseModal';
|
import SubscriptionPurchaseModal from './modals/SubscriptionPurchaseModal';
|
||||||
import {
|
import {
|
||||||
formatSubscriptionDuration,
|
formatSubscriptionDuration,
|
||||||
@ -83,6 +82,7 @@ const SubscriptionPlansCard = ({
|
|||||||
activeSubscriptions = [],
|
activeSubscriptions = [],
|
||||||
allSubscriptions = [],
|
allSubscriptions = [],
|
||||||
reloadSubscriptionSelf,
|
reloadSubscriptionSelf,
|
||||||
|
withCard = true,
|
||||||
}) => {
|
}) => {
|
||||||
const [open, setOpen] = useState(false);
|
const [open, setOpen] = useState(false);
|
||||||
const [selectedPlan, setSelectedPlan] = useState(null);
|
const [selectedPlan, setSelectedPlan] = useState(null);
|
||||||
@ -241,33 +241,9 @@ const SubscriptionPlansCard = ({
|
|||||||
return Math.round((used / total) * 100);
|
return Math.round((used / total) * 100);
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
const cardContent = (
|
||||||
<Card className='!rounded-2xl shadow-sm border-0'>
|
<>
|
||||||
{/* 卡片头部 */}
|
{/* 卡片头部 */}
|
||||||
<div className='flex items-center justify-between mb-3'>
|
|
||||||
<div className='flex items-center'>
|
|
||||||
<Avatar size='small' color='violet' className='mr-3 shadow-md'>
|
|
||||||
<Crown size={16} />
|
|
||||||
</Avatar>
|
|
||||||
<div>
|
|
||||||
<Text className='text-lg font-medium'>{t('订阅套餐')}</Text>
|
|
||||||
<div className='text-xs'>{t('购买订阅获得模型额度/次数')}</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{/* 扣费策略 - 右上角 */}
|
|
||||||
<Select
|
|
||||||
value={billingPreference}
|
|
||||||
onChange={onChangeBillingPreference}
|
|
||||||
size='small'
|
|
||||||
optionList={[
|
|
||||||
{ value: 'subscription_first', label: t('优先订阅') },
|
|
||||||
{ value: 'wallet_first', label: t('优先钱包') },
|
|
||||||
{ value: 'subscription_only', label: t('仅用订阅') },
|
|
||||||
{ value: 'wallet_only', label: t('仅用钱包') },
|
|
||||||
]}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{loading ? (
|
{loading ? (
|
||||||
<div className='space-y-4'>
|
<div className='space-y-4'>
|
||||||
{/* 我的订阅骨架屏 */}
|
{/* 我的订阅骨架屏 */}
|
||||||
@ -281,7 +257,7 @@ const SubscriptionPlansCard = ({
|
|||||||
</div>
|
</div>
|
||||||
</Card>
|
</Card>
|
||||||
{/* 套餐列表骨架屏 */}
|
{/* 套餐列表骨架屏 */}
|
||||||
<div className='grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-2 xl:grid-cols-3 gap-5 w-full'>
|
<div className='grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-2 xl:grid-cols-3 gap-5 w-full px-1'>
|
||||||
{[1, 2, 3].map((i) => (
|
{[1, 2, 3].map((i) => (
|
||||||
<Card
|
<Card
|
||||||
key={i}
|
key={i}
|
||||||
@ -317,8 +293,8 @@ const SubscriptionPlansCard = ({
|
|||||||
<Space vertical style={{ width: '100%' }} spacing={8}>
|
<Space vertical style={{ width: '100%' }} spacing={8}>
|
||||||
{/* 当前订阅状态 */}
|
{/* 当前订阅状态 */}
|
||||||
<Card className='!rounded-xl w-full' bodyStyle={{ padding: '12px' }}>
|
<Card className='!rounded-xl w-full' bodyStyle={{ padding: '12px' }}>
|
||||||
<div className='flex items-center justify-between mb-2'>
|
<div className='flex items-center justify-between mb-2 gap-3'>
|
||||||
<div className='flex items-center gap-2'>
|
<div className='flex items-center gap-2 flex-1 min-w-0'>
|
||||||
<Text strong>{t('我的订阅')}</Text>
|
<Text strong>{t('我的订阅')}</Text>
|
||||||
{hasActiveSubscription ? (
|
{hasActiveSubscription ? (
|
||||||
<Tag
|
<Tag
|
||||||
@ -341,19 +317,32 @@ const SubscriptionPlansCard = ({
|
|||||||
</Tag>
|
</Tag>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
<Button
|
<div className='flex items-center gap-2'>
|
||||||
size='small'
|
<Select
|
||||||
theme='light'
|
value={billingPreference}
|
||||||
type='tertiary'
|
onChange={onChangeBillingPreference}
|
||||||
icon={
|
size='small'
|
||||||
<RefreshCw
|
optionList={[
|
||||||
size={12}
|
{ value: 'subscription_first', label: t('优先订阅') },
|
||||||
className={refreshing ? 'animate-spin' : ''}
|
{ value: 'wallet_first', label: t('优先钱包') },
|
||||||
/>
|
{ value: 'subscription_only', label: t('仅用订阅') },
|
||||||
}
|
{ value: 'wallet_only', label: t('仅用钱包') },
|
||||||
onClick={handleRefresh}
|
]}
|
||||||
loading={refreshing}
|
/>
|
||||||
/>
|
<Button
|
||||||
|
size='small'
|
||||||
|
theme='light'
|
||||||
|
type='tertiary'
|
||||||
|
icon={
|
||||||
|
<RefreshCw
|
||||||
|
size={12}
|
||||||
|
className={refreshing ? 'animate-spin' : ''}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
onClick={handleRefresh}
|
||||||
|
loading={refreshing}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{hasAnySubscription ? (
|
{hasAnySubscription ? (
|
||||||
@ -451,7 +440,7 @@ const SubscriptionPlansCard = ({
|
|||||||
|
|
||||||
{/* 可购买套餐 - 标准定价卡片 */}
|
{/* 可购买套餐 - 标准定价卡片 */}
|
||||||
{plans.length > 0 ? (
|
{plans.length > 0 ? (
|
||||||
<div className='grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-2 xl:grid-cols-3 gap-5 w-full'>
|
<div className='grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-2 xl:grid-cols-3 gap-5 w-full px-1'>
|
||||||
{plans.map((p, index) => {
|
{plans.map((p, index) => {
|
||||||
const plan = p?.plan;
|
const plan = p?.plan;
|
||||||
const totalAmount = Number(plan?.total_amount || 0);
|
const totalAmount = Number(plan?.total_amount || 0);
|
||||||
@ -482,9 +471,9 @@ const SubscriptionPlansCard = ({
|
|||||||
resetLabel ? { label: resetLabel } : null,
|
resetLabel ? { label: resetLabel } : null,
|
||||||
totalAmount > 0
|
totalAmount > 0
|
||||||
? {
|
? {
|
||||||
label: totalLabel,
|
label: totalLabel,
|
||||||
tooltip: `${t('原生额度')}:${totalAmount}`,
|
tooltip: `${t('原生额度')}:${totalAmount}`,
|
||||||
}
|
}
|
||||||
: { label: totalLabel },
|
: { label: totalLabel },
|
||||||
limitLabel ? { label: limitLabel } : null,
|
limitLabel ? { label: limitLabel } : null,
|
||||||
upgradeLabel ? { label: upgradeLabel } : null,
|
upgradeLabel ? { label: upgradeLabel } : null,
|
||||||
@ -493,9 +482,8 @@ const SubscriptionPlansCard = ({
|
|||||||
return (
|
return (
|
||||||
<Card
|
<Card
|
||||||
key={plan?.id}
|
key={plan?.id}
|
||||||
className={`!rounded-xl transition-all hover:shadow-lg w-full h-full ${
|
className={`!rounded-xl transition-all hover:shadow-lg w-full h-full ${isPopular ? 'ring-2 ring-purple-500' : ''
|
||||||
isPopular ? 'ring-2 ring-purple-500' : ''
|
}`}
|
||||||
}`}
|
|
||||||
bodyStyle={{ padding: 0 }}
|
bodyStyle={{ padding: 0 }}
|
||||||
>
|
>
|
||||||
<div className='p-4 h-full flex flex-col'>
|
<div className='p-4 h-full flex flex-col'>
|
||||||
@ -583,7 +571,7 @@ const SubscriptionPlansCard = ({
|
|||||||
const buttonEl = (
|
const buttonEl = (
|
||||||
<Button
|
<Button
|
||||||
theme='outline'
|
theme='outline'
|
||||||
type='tertiary'
|
type='primary'
|
||||||
block
|
block
|
||||||
disabled={reached}
|
disabled={reached}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
@ -614,6 +602,16 @@ const SubscriptionPlansCard = ({
|
|||||||
)}
|
)}
|
||||||
</Space>
|
</Space>
|
||||||
)}
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{withCard ? (
|
||||||
|
<Card className='!rounded-2xl shadow-sm border-0'>{cardContent}</Card>
|
||||||
|
) : (
|
||||||
|
<div className='space-y-3'>{cardContent}</div>
|
||||||
|
)}
|
||||||
|
|
||||||
{/* 购买确认弹窗 */}
|
{/* 购买确认弹窗 */}
|
||||||
<SubscriptionPurchaseModal
|
<SubscriptionPurchaseModal
|
||||||
@ -631,16 +629,16 @@ const SubscriptionPlansCard = ({
|
|||||||
purchaseLimitInfo={
|
purchaseLimitInfo={
|
||||||
selectedPlan?.plan?.id
|
selectedPlan?.plan?.id
|
||||||
? {
|
? {
|
||||||
limit: Number(selectedPlan?.plan?.max_purchase_per_user || 0),
|
limit: Number(selectedPlan?.plan?.max_purchase_per_user || 0),
|
||||||
count: getPlanPurchaseCount(selectedPlan?.plan?.id),
|
count: getPlanPurchaseCount(selectedPlan?.plan?.id),
|
||||||
}
|
}
|
||||||
: null
|
: null
|
||||||
}
|
}
|
||||||
onPayStripe={payStripe}
|
onPayStripe={payStripe}
|
||||||
onPayCreem={payCreem}
|
onPayCreem={payCreem}
|
||||||
onPayEpay={payEpay}
|
onPayEpay={payEpay}
|
||||||
/>
|
/>
|
||||||
</Card>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@ -35,7 +35,6 @@ import { StatusContext } from '../../context/Status';
|
|||||||
|
|
||||||
import RechargeCard from './RechargeCard';
|
import RechargeCard from './RechargeCard';
|
||||||
import InvitationCard from './InvitationCard';
|
import InvitationCard from './InvitationCard';
|
||||||
import SubscriptionPlansCard from './SubscriptionPlansCard';
|
|
||||||
import TransferModal from './modals/TransferModal';
|
import TransferModal from './modals/TransferModal';
|
||||||
import PaymentConfirmModal from './modals/PaymentConfirmModal';
|
import PaymentConfirmModal from './modals/PaymentConfirmModal';
|
||||||
import TopupHistoryModal from './modals/TopupHistoryModal';
|
import TopupHistoryModal from './modals/TopupHistoryModal';
|
||||||
@ -733,80 +732,58 @@ const TopUp = () => {
|
|||||||
</Modal>
|
</Modal>
|
||||||
|
|
||||||
{/* 主布局区域 */}
|
{/* 主布局区域 */}
|
||||||
<div className='grid grid-cols-1 lg:grid-cols-12 gap-6'>
|
<div className='grid grid-cols-1 lg:grid-cols-2 gap-6'>
|
||||||
{/* 左侧 - 订阅套餐(无套餐时隐藏整块卡片) */}
|
<RechargeCard
|
||||||
{(subscriptionLoading || subscriptionPlans.length > 0) && (
|
t={t}
|
||||||
<div className='lg:col-span-7'>
|
enableOnlineTopUp={enableOnlineTopUp}
|
||||||
<SubscriptionPlansCard
|
enableStripeTopUp={enableStripeTopUp}
|
||||||
t={t}
|
enableCreemTopUp={enableCreemTopUp}
|
||||||
loading={subscriptionLoading}
|
creemProducts={creemProducts}
|
||||||
plans={subscriptionPlans}
|
creemPreTopUp={creemPreTopUp}
|
||||||
payMethods={payMethods}
|
presetAmounts={presetAmounts}
|
||||||
enableOnlineTopUp={enableOnlineTopUp}
|
selectedPreset={selectedPreset}
|
||||||
enableStripeTopUp={enableStripeTopUp}
|
selectPresetAmount={selectPresetAmount}
|
||||||
enableCreemTopUp={enableCreemTopUp}
|
formatLargeNumber={formatLargeNumber}
|
||||||
billingPreference={billingPreference}
|
priceRatio={priceRatio}
|
||||||
onChangeBillingPreference={updateBillingPreference}
|
topUpCount={topUpCount}
|
||||||
activeSubscriptions={activeSubscriptions}
|
minTopUp={minTopUp}
|
||||||
allSubscriptions={allSubscriptions}
|
renderQuotaWithAmount={renderQuotaWithAmount}
|
||||||
reloadSubscriptionSelf={getSubscriptionSelf}
|
getAmount={getAmount}
|
||||||
/>
|
setTopUpCount={setTopUpCount}
|
||||||
</div>
|
setSelectedPreset={setSelectedPreset}
|
||||||
)}
|
renderAmount={renderAmount}
|
||||||
|
amountLoading={amountLoading}
|
||||||
{/* 右侧 - 账户充值 + 邀请奖励 */}
|
payMethods={payMethods}
|
||||||
<div
|
preTopUp={preTopUp}
|
||||||
className={
|
paymentLoading={paymentLoading}
|
||||||
subscriptionLoading || subscriptionPlans.length > 0
|
payWay={payWay}
|
||||||
? 'lg:col-span-5 flex flex-col gap-6'
|
redemptionCode={redemptionCode}
|
||||||
: 'lg:col-span-12 flex flex-col gap-6'
|
setRedemptionCode={setRedemptionCode}
|
||||||
}
|
topUp={topUp}
|
||||||
>
|
isSubmitting={isSubmitting}
|
||||||
<RechargeCard
|
topUpLink={topUpLink}
|
||||||
t={t}
|
openTopUpLink={openTopUpLink}
|
||||||
enableOnlineTopUp={enableOnlineTopUp}
|
userState={userState}
|
||||||
enableStripeTopUp={enableStripeTopUp}
|
renderQuota={renderQuota}
|
||||||
enableCreemTopUp={enableCreemTopUp}
|
statusLoading={statusLoading}
|
||||||
creemProducts={creemProducts}
|
topupInfo={topupInfo}
|
||||||
creemPreTopUp={creemPreTopUp}
|
onOpenHistory={handleOpenHistory}
|
||||||
presetAmounts={presetAmounts}
|
subscriptionLoading={subscriptionLoading}
|
||||||
selectedPreset={selectedPreset}
|
subscriptionPlans={subscriptionPlans}
|
||||||
selectPresetAmount={selectPresetAmount}
|
billingPreference={billingPreference}
|
||||||
formatLargeNumber={formatLargeNumber}
|
onChangeBillingPreference={updateBillingPreference}
|
||||||
priceRatio={priceRatio}
|
activeSubscriptions={activeSubscriptions}
|
||||||
topUpCount={topUpCount}
|
allSubscriptions={allSubscriptions}
|
||||||
minTopUp={minTopUp}
|
reloadSubscriptionSelf={getSubscriptionSelf}
|
||||||
renderQuotaWithAmount={renderQuotaWithAmount}
|
/>
|
||||||
getAmount={getAmount}
|
<InvitationCard
|
||||||
setTopUpCount={setTopUpCount}
|
t={t}
|
||||||
setSelectedPreset={setSelectedPreset}
|
userState={userState}
|
||||||
renderAmount={renderAmount}
|
renderQuota={renderQuota}
|
||||||
amountLoading={amountLoading}
|
setOpenTransfer={setOpenTransfer}
|
||||||
payMethods={payMethods}
|
affLink={affLink}
|
||||||
preTopUp={preTopUp}
|
handleAffLinkClick={handleAffLinkClick}
|
||||||
paymentLoading={paymentLoading}
|
/>
|
||||||
payWay={payWay}
|
|
||||||
redemptionCode={redemptionCode}
|
|
||||||
setRedemptionCode={setRedemptionCode}
|
|
||||||
topUp={topUp}
|
|
||||||
isSubmitting={isSubmitting}
|
|
||||||
topUpLink={topUpLink}
|
|
||||||
openTopUpLink={openTopUpLink}
|
|
||||||
userState={userState}
|
|
||||||
renderQuota={renderQuota}
|
|
||||||
statusLoading={statusLoading}
|
|
||||||
topupInfo={topupInfo}
|
|
||||||
onOpenHistory={handleOpenHistory}
|
|
||||||
/>
|
|
||||||
<InvitationCard
|
|
||||||
t={t}
|
|
||||||
userState={userState}
|
|
||||||
renderQuota={renderQuota}
|
|
||||||
setOpenTransfer={setOpenTransfer}
|
|
||||||
affLink={affLink}
|
|
||||||
handleAffLinkClick={handleAffLinkClick}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@ -2721,7 +2721,7 @@
|
|||||||
"绑定订阅套餐": "Bind Subscription Plan",
|
"绑定订阅套餐": "Bind Subscription Plan",
|
||||||
"绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。": "After binding, a user subscription is created immediately (no payment required); validity follows the plan configuration.",
|
"绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。": "After binding, a user subscription is created immediately (no payment required); validity follows the plan configuration.",
|
||||||
"订阅套餐": "Subscription Plans",
|
"订阅套餐": "Subscription Plans",
|
||||||
"购买订阅获得模型额度/次数": "Purchase a subscription to get model quota/usage",
|
"额度充值": "Quota Top-up",
|
||||||
"优先订阅": "Subscription first",
|
"优先订阅": "Subscription first",
|
||||||
"优先钱包": "Wallet first",
|
"优先钱包": "Wallet first",
|
||||||
"仅用订阅": "Subscription only",
|
"仅用订阅": "Subscription only",
|
||||||
|
|||||||
@ -2684,7 +2684,7 @@
|
|||||||
"绑定订阅套餐": "Lier un plan d'abonnement",
|
"绑定订阅套餐": "Lier un plan d'abonnement",
|
||||||
"绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。": "Après liaison, un abonnement utilisateur est créé immédiatement (sans paiement) ; la validité suit la configuration du plan.",
|
"绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。": "Après liaison, un abonnement utilisateur est créé immédiatement (sans paiement) ; la validité suit la configuration du plan.",
|
||||||
"订阅套餐": "Plans d'abonnement",
|
"订阅套餐": "Plans d'abonnement",
|
||||||
"购买订阅获得模型额度/次数": "Acheter un abonnement pour obtenir des quotas/usages de modèles",
|
"额度充值": "Recharge de quota",
|
||||||
"优先订阅": "Abonnement en priorité",
|
"优先订阅": "Abonnement en priorité",
|
||||||
"优先钱包": "Portefeuille en priorité",
|
"优先钱包": "Portefeuille en priorité",
|
||||||
"仅用订阅": "Abonnement uniquement",
|
"仅用订阅": "Abonnement uniquement",
|
||||||
|
|||||||
@ -2667,7 +2667,7 @@
|
|||||||
"绑定订阅套餐": "サブスクリプションプランを紐付け",
|
"绑定订阅套餐": "サブスクリプションプランを紐付け",
|
||||||
"绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。": "紐付け後、ユーザーサブスクリプションが即時に作成されます(支払い不要)。有効期限はプラン設定に従います。",
|
"绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。": "紐付け後、ユーザーサブスクリプションが即時に作成されます(支払い不要)。有効期限はプラン設定に従います。",
|
||||||
"订阅套餐": "サブスクリプションプラン",
|
"订阅套餐": "サブスクリプションプラン",
|
||||||
"购买订阅获得模型额度/次数": "サブスクリプション購入でモデルのクォータ/回数を取得",
|
"额度充值": "クォータ補充",
|
||||||
"优先订阅": "サブスクリプション優先",
|
"优先订阅": "サブスクリプション優先",
|
||||||
"优先钱包": "ウォレット優先",
|
"优先钱包": "ウォレット優先",
|
||||||
"仅用订阅": "サブスクリプションのみ",
|
"仅用订阅": "サブスクリプションのみ",
|
||||||
|
|||||||
@ -2697,7 +2697,7 @@
|
|||||||
"绑定订阅套餐": "Привязать план подписки",
|
"绑定订阅套餐": "Привязать план подписки",
|
||||||
"绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。": "После привязки подписка будет создана сразу (без оплаты); срок действия рассчитывается по настройкам плана.",
|
"绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。": "После привязки подписка будет создана сразу (без оплаты); срок действия рассчитывается по настройкам плана.",
|
||||||
"订阅套餐": "Планы подписки",
|
"订阅套餐": "Планы подписки",
|
||||||
"购买订阅获得模型额度/次数": "Купите подписку, чтобы получить лимит/количество использования моделей",
|
"额度充值": "Пополнение квоты",
|
||||||
"优先订阅": "Сначала подписка",
|
"优先订阅": "Сначала подписка",
|
||||||
"优先钱包": "Сначала кошелек",
|
"优先钱包": "Сначала кошелек",
|
||||||
"仅用订阅": "Только подписка",
|
"仅用订阅": "Только подписка",
|
||||||
|
|||||||
@ -3246,7 +3246,7 @@
|
|||||||
"绑定订阅套餐": "Liên kết gói đăng ký",
|
"绑定订阅套餐": "Liên kết gói đăng ký",
|
||||||
"绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。": "Sau khi liên kết, sẽ tạo đăng ký cho người dùng ngay (không cần thanh toán); thời hạn theo cấu hình gói.",
|
"绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。": "Sau khi liên kết, sẽ tạo đăng ký cho người dùng ngay (không cần thanh toán); thời hạn theo cấu hình gói.",
|
||||||
"订阅套餐": "Gói đăng ký",
|
"订阅套餐": "Gói đăng ký",
|
||||||
"购买订阅获得模型额度/次数": "Mua đăng ký để nhận hạn mức/lượt dùng mô hình",
|
"额度充值": "Nạp hạn mức",
|
||||||
"优先订阅": "Ưu tiên đăng ký",
|
"优先订阅": "Ưu tiên đăng ký",
|
||||||
"优先钱包": "Ưu tiên ví",
|
"优先钱包": "Ưu tiên ví",
|
||||||
"仅用订阅": "Chỉ dùng đăng ký",
|
"仅用订阅": "Chỉ dùng đăng ký",
|
||||||
|
|||||||
@ -2706,7 +2706,7 @@
|
|||||||
"绑定订阅套餐": "绑定订阅套餐",
|
"绑定订阅套餐": "绑定订阅套餐",
|
||||||
"绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。": "绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。",
|
"绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。": "绑定后会立即生成用户订阅(无需支付),有效期按套餐配置计算。",
|
||||||
"订阅套餐": "订阅套餐",
|
"订阅套餐": "订阅套餐",
|
||||||
"购买订阅获得模型额度/次数": "购买订阅获得模型额度/次数",
|
"额度充值": "额度充值",
|
||||||
"优先订阅": "优先订阅",
|
"优先订阅": "优先订阅",
|
||||||
"优先钱包": "优先钱包",
|
"优先钱包": "优先钱包",
|
||||||
"仅用订阅": "仅用订阅",
|
"仅用订阅": "仅用订阅",
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user