mirror of
https://github.com/Comfy-Org/ComfyUI_frontend.git
synced 2026-02-02 06:19:58 +00:00
Backport of #7899 to `cloud/1.36` ## Summary - Implement the new add credits (top up) dialog - Refactor the subscription dialog to make different credit types easier to understand ## Conflicts Resolved - `src/components/dialog/content/TopUpCreditsDialogContent.vue` - Took PR version (new UI) - `src/locales/en/main.json` - Added missing `usdAmount` translation key Automatically created by manual backport process. ┆Issue is synchronized with this [Notion page](https://www.notion.so/PR-7932-backport-cloud-1-36-Feat-cloud-new-top-up-dialog-2e36d73d3650812dba2cd7edcb43259b) by [Unito](https://www.unito.io)
443 lines
14 KiB
Vue
443 lines
14 KiB
Vue
<template>
|
|
<TabPanel value="PlanCredits" class="subscription-container h-full">
|
|
<div class="flex h-full flex-col gap-6">
|
|
<div class="flex items-center gap-2">
|
|
<span class="text-2xl font-inter font-semibold leading-tight">
|
|
{{
|
|
isActiveSubscription
|
|
? $t('subscription.title')
|
|
: $t('subscription.titleUnsubscribed')
|
|
}}
|
|
</span>
|
|
<div class="pt-1">
|
|
<CloudBadge
|
|
reverse-order
|
|
background-color="var(--p-dialog-background)"
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="grow overflow-auto">
|
|
<div class="rounded-2xl border border-interface-stroke p-6">
|
|
<div>
|
|
<div class="flex items-center justify-between gap-2">
|
|
<div class="flex flex-col gap-2">
|
|
<div class="text-sm font-bold text-text-primary">
|
|
{{ subscriptionTierName }}
|
|
</div>
|
|
<div class="flex items-baseline gap-1 font-inter font-semibold">
|
|
<span class="text-2xl">${{ tierPrice }}</span>
|
|
<span class="text-base">{{
|
|
$t('subscription.perMonth')
|
|
}}</span>
|
|
</div>
|
|
<div
|
|
v-if="isActiveSubscription"
|
|
class="text-sm text-text-secondary"
|
|
>
|
|
<template v-if="isCancelled">
|
|
{{
|
|
$t('subscription.expiresDate', {
|
|
date: formattedEndDate
|
|
})
|
|
}}
|
|
</template>
|
|
<template v-else>
|
|
{{
|
|
$t('subscription.renewsDate', {
|
|
date: formattedRenewalDate
|
|
})
|
|
}}
|
|
</template>
|
|
</div>
|
|
</div>
|
|
|
|
<Button
|
|
v-if="isActiveSubscription"
|
|
variant="secondary"
|
|
class="ml-auto rounded-lg px-4 py-2 text-sm font-normal text-text-primary bg-interface-menu-component-surface-selected"
|
|
@click="
|
|
async () => {
|
|
await authActions.accessBillingPortal()
|
|
}
|
|
"
|
|
>
|
|
{{ $t('subscription.manageSubscription') }}
|
|
</Button>
|
|
<Button
|
|
v-if="isActiveSubscription"
|
|
variant="primary"
|
|
class="rounded-lg px-4 py-2 text-sm font-normal text-text-primary"
|
|
@click="showSubscriptionDialog"
|
|
>
|
|
{{ $t('subscription.upgradePlan') }}
|
|
</Button>
|
|
|
|
<SubscribeButton
|
|
v-else
|
|
:label="$t('subscription.subscribeNow')"
|
|
size="sm"
|
|
:fluid="false"
|
|
class="text-xs"
|
|
@subscribed="handleRefresh"
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="flex flex-col lg:flex-row gap-6 pt-9">
|
|
<div class="flex flex-col shrink-0">
|
|
<div class="flex flex-col gap-3">
|
|
<div
|
|
:class="
|
|
cn(
|
|
'relative flex flex-col gap-6 rounded-2xl p-5',
|
|
'bg-modal-panel-background'
|
|
)
|
|
"
|
|
>
|
|
<Button
|
|
variant="muted-textonly"
|
|
size="icon-sm"
|
|
class="absolute top-4 right-4"
|
|
:loading="isLoadingBalance"
|
|
@click="handleRefresh"
|
|
>
|
|
<i class="pi pi-sync text-text-secondary text-sm" />
|
|
</Button>
|
|
|
|
<div class="flex flex-col gap-2">
|
|
<div class="text-sm text-muted">
|
|
{{ $t('subscription.totalCredits') }}
|
|
</div>
|
|
<Skeleton
|
|
v-if="isLoadingBalance"
|
|
width="8rem"
|
|
height="2rem"
|
|
/>
|
|
<div v-else class="text-2xl font-bold">
|
|
{{ totalCredits }}
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Credit Breakdown -->
|
|
<table class="text-sm text-muted">
|
|
<tbody>
|
|
<tr>
|
|
<td class="pr-4 font-bold text-left align-middle">
|
|
<Skeleton
|
|
v-if="isLoadingBalance"
|
|
width="5rem"
|
|
height="1rem"
|
|
/>
|
|
<span v-else>{{ includedCreditsDisplay }}</span>
|
|
</td>
|
|
<td class="align-middle" :title="creditsRemainingLabel">
|
|
{{ creditsRemainingLabel }}
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td class="pr-4 font-bold text-left align-middle">
|
|
<Skeleton
|
|
v-if="isLoadingBalance"
|
|
width="3rem"
|
|
height="1rem"
|
|
/>
|
|
<span v-else>{{ prepaidCredits }}</span>
|
|
</td>
|
|
<td
|
|
class="align-middle"
|
|
:title="$t('subscription.creditsYouveAdded')"
|
|
>
|
|
{{ $t('subscription.creditsYouveAdded') }}
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
|
|
<div class="flex items-center justify-between">
|
|
<a
|
|
href="https://platform.comfy.org/profile/usage"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
class="text-sm underline text-center text-muted"
|
|
>
|
|
{{ $t('subscription.viewUsageHistory') }}
|
|
</a>
|
|
<Button
|
|
v-if="isActiveSubscription"
|
|
variant="secondary"
|
|
class="p-2 min-h-8 rounded-lg text-sm font-normal text-text-primary bg-interface-menu-component-surface-selected"
|
|
@click="handleAddApiCredits"
|
|
>
|
|
{{ $t('subscription.addCredits') }}
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="flex flex-col gap-2">
|
|
<div class="text-sm text-text-primary">
|
|
{{ $t('subscription.yourPlanIncludes') }}
|
|
</div>
|
|
|
|
<div class="flex flex-col gap-0">
|
|
<div
|
|
v-for="benefit in tierBenefits"
|
|
:key="benefit.key"
|
|
class="flex items-center gap-2 py-2"
|
|
>
|
|
<i
|
|
v-if="benefit.type === 'feature'"
|
|
class="pi pi-check text-xs text-text-primary"
|
|
/>
|
|
<span
|
|
v-else-if="benefit.type === 'metric' && benefit.value"
|
|
class="text-sm font-normal whitespace-nowrap text-text-primary"
|
|
>
|
|
{{ benefit.value }}
|
|
</span>
|
|
<span class="text-sm text-muted">
|
|
{{ benefit.label }}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- View More Details - Outside main content -->
|
|
<div class="flex items-center gap-2 py-4">
|
|
<i class="pi pi-external-link text-muted"></i>
|
|
<a
|
|
href="https://www.comfy.org/cloud/pricing"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
class="text-sm underline hover:opacity-80 text-muted"
|
|
>
|
|
{{ $t('subscription.viewMoreDetailsPlans') }}
|
|
</a>
|
|
</div>
|
|
</div>
|
|
|
|
<div
|
|
class="flex items-center justify-between border-t border-interface-stroke pt-3"
|
|
>
|
|
<div class="flex gap-2">
|
|
<Button
|
|
variant="muted-textonly"
|
|
class="text-xs text-text-secondary"
|
|
@click="handleLearnMoreClick"
|
|
>
|
|
<i class="pi pi-question-circle text-text-secondary text-xs" />
|
|
{{ $t('subscription.learnMore') }}
|
|
</Button>
|
|
<Button
|
|
variant="muted-textonly"
|
|
class="text-xs text-text-secondary"
|
|
@click="handleOpenPartnerNodesInfo"
|
|
>
|
|
<i class="pi pi-question-circle text-text-secondary text-xs" />
|
|
{{ $t('subscription.partnerNodesCredits') }}
|
|
</Button>
|
|
<Button
|
|
variant="muted-textonly"
|
|
class="text-xs text-text-secondary"
|
|
:loading="isLoadingSupport"
|
|
@click="handleMessageSupport"
|
|
>
|
|
<i class="pi pi-comment text-text-secondary text-xs" />
|
|
{{ $t('subscription.messageSupport') }}
|
|
</Button>
|
|
</div>
|
|
|
|
<Button
|
|
variant="muted-textonly"
|
|
class="text-xs text-text-secondary"
|
|
@click="handleInvoiceHistory"
|
|
>
|
|
{{ $t('subscription.invoiceHistory') }}
|
|
<i class="pi pi-external-link text-text-secondary text-xs" />
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
</TabPanel>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import Skeleton from 'primevue/skeleton'
|
|
import TabPanel from 'primevue/tabpanel'
|
|
import { computed, onBeforeUnmount, onMounted } from 'vue'
|
|
import { useI18n } from 'vue-i18n'
|
|
|
|
import CloudBadge from '@/components/topbar/CloudBadge.vue'
|
|
import Button from '@/components/ui/button/Button.vue'
|
|
import { useFirebaseAuthActions } from '@/composables/auth/useFirebaseAuthActions'
|
|
import { useExternalLink } from '@/composables/useExternalLink'
|
|
import SubscribeButton from '@/platform/cloud/subscription/components/SubscribeButton.vue'
|
|
import { useSubscription } from '@/platform/cloud/subscription/composables/useSubscription'
|
|
import { useSubscriptionActions } from '@/platform/cloud/subscription/composables/useSubscriptionActions'
|
|
import { useSubscriptionCredits } from '@/platform/cloud/subscription/composables/useSubscriptionCredits'
|
|
import { useSubscriptionDialog } from '@/platform/cloud/subscription/composables/useSubscriptionDialog'
|
|
import {
|
|
DEFAULT_TIER_KEY,
|
|
TIER_TO_KEY,
|
|
getTierCredits,
|
|
getTierFeatures,
|
|
getTierPrice
|
|
} from '@/platform/cloud/subscription/constants/tierPricing'
|
|
import { cn } from '@/utils/tailwindUtil'
|
|
|
|
const { buildDocsUrl, docsPaths } = useExternalLink()
|
|
const authActions = useFirebaseAuthActions()
|
|
const { t, n } = useI18n()
|
|
|
|
const {
|
|
isActiveSubscription,
|
|
isCancelled,
|
|
formattedRenewalDate,
|
|
formattedEndDate,
|
|
subscriptionTier,
|
|
subscriptionTierName,
|
|
subscriptionStatus,
|
|
isYearlySubscription,
|
|
handleInvoiceHistory
|
|
} = useSubscription()
|
|
|
|
const { show: showSubscriptionDialog } = useSubscriptionDialog()
|
|
|
|
const tierKey = computed(() => {
|
|
const tier = subscriptionTier.value
|
|
if (!tier) return DEFAULT_TIER_KEY
|
|
return TIER_TO_KEY[tier] ?? DEFAULT_TIER_KEY
|
|
})
|
|
const tierPrice = computed(() =>
|
|
getTierPrice(tierKey.value, isYearlySubscription.value)
|
|
)
|
|
|
|
const refillsDate = computed(() => {
|
|
if (!subscriptionStatus.value?.renewal_date) return ''
|
|
const date = new Date(subscriptionStatus.value.renewal_date)
|
|
const day = String(date.getDate()).padStart(2, '0')
|
|
const month = String(date.getMonth() + 1).padStart(2, '0')
|
|
const year = String(date.getFullYear()).slice(-2)
|
|
return `${month}/${day}/${year}`
|
|
})
|
|
|
|
const creditsRemainingLabel = computed(() =>
|
|
isYearlySubscription.value
|
|
? t('subscription.creditsRemainingThisYear', {
|
|
date: refillsDate.value
|
|
})
|
|
: t('subscription.creditsRemainingThisMonth', {
|
|
date: refillsDate.value
|
|
})
|
|
)
|
|
|
|
const planTotalCredits = computed(() => {
|
|
const credits = getTierCredits(tierKey.value)
|
|
const total = isYearlySubscription.value ? credits * 12 : credits
|
|
return n(total)
|
|
})
|
|
|
|
const includedCreditsDisplay = computed(
|
|
() => `${monthlyBonusCredits.value} / ${planTotalCredits.value}`
|
|
)
|
|
|
|
// Tier benefits for v-for loop
|
|
type BenefitType = 'metric' | 'feature'
|
|
|
|
interface Benefit {
|
|
key: string
|
|
type: BenefitType
|
|
label: string
|
|
value?: string
|
|
}
|
|
|
|
const tierBenefits = computed((): Benefit[] => {
|
|
const key = tierKey.value
|
|
|
|
const benefits: Benefit[] = [
|
|
{
|
|
key: 'maxDuration',
|
|
type: 'metric',
|
|
value: t(`subscription.maxDuration.${key}`),
|
|
label: t('subscription.maxDurationLabel')
|
|
},
|
|
{
|
|
key: 'gpu',
|
|
type: 'feature',
|
|
label: t('subscription.gpuLabel')
|
|
},
|
|
{
|
|
key: 'addCredits',
|
|
type: 'feature',
|
|
label: t('subscription.addCreditsLabel')
|
|
}
|
|
]
|
|
|
|
if (getTierFeatures(key).customLoRAs) {
|
|
benefits.push({
|
|
key: 'customLoRAs',
|
|
type: 'feature',
|
|
label: t('subscription.customLoRAsLabel')
|
|
})
|
|
}
|
|
|
|
return benefits
|
|
})
|
|
|
|
const { totalCredits, monthlyBonusCredits, prepaidCredits, isLoadingBalance } =
|
|
useSubscriptionCredits()
|
|
|
|
const {
|
|
isLoadingSupport,
|
|
handleAddApiCredits,
|
|
handleMessageSupport,
|
|
handleRefresh,
|
|
handleLearnMoreClick
|
|
} = useSubscriptionActions()
|
|
|
|
// Focus-based polling: refresh balance when user returns from Stripe checkout
|
|
const PENDING_TOPUP_KEY = 'pending_topup_timestamp'
|
|
const TOPUP_EXPIRY_MS = 5 * 60 * 1000 // 5 minutes
|
|
|
|
function handleWindowFocus() {
|
|
const timestampStr = localStorage.getItem(PENDING_TOPUP_KEY)
|
|
if (!timestampStr) return
|
|
|
|
const timestamp = parseInt(timestampStr, 10)
|
|
|
|
// Clear expired tracking (older than 5 minutes)
|
|
if (Date.now() - timestamp > TOPUP_EXPIRY_MS) {
|
|
localStorage.removeItem(PENDING_TOPUP_KEY)
|
|
return
|
|
}
|
|
|
|
// Refresh and clear tracking to prevent repeated calls
|
|
void handleRefresh()
|
|
localStorage.removeItem(PENDING_TOPUP_KEY)
|
|
}
|
|
|
|
onMounted(() => {
|
|
window.addEventListener('focus', handleWindowFocus)
|
|
})
|
|
|
|
onBeforeUnmount(() => {
|
|
window.removeEventListener('focus', handleWindowFocus)
|
|
})
|
|
|
|
const handleOpenPartnerNodesInfo = () => {
|
|
window.open(
|
|
buildDocsUrl(docsPaths.partnerNodesPricing, { includeLocale: true }),
|
|
'_blank'
|
|
)
|
|
}
|
|
</script>
|
|
|
|
<style scoped>
|
|
:deep(.bg-comfy-menu-secondary) {
|
|
background-color: transparent;
|
|
}
|
|
</style>
|