Skip to content

chore(clerk-js,types): Update checkout flow to support free trials #6494

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Open
wants to merge 26 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
b228074
chore(clerk-js,types): Update PricingTable with trial info
panteliselef Aug 6, 2025
9eda6fa
chore(clerk-js,types): Update checkout to handle trials
panteliselef Aug 8, 2025
1c6b47c
chore(clerk-js,types): Update checkout to handle trials
panteliselef Aug 8, 2025
e729316
handle checkout confirming step
panteliselef Aug 12, 2025
0388c4a
respect fapi
panteliselef Aug 12, 2025
11e6953
add tests
panteliselef Aug 12, 2025
1cb9c20
Merge branch 'refs/heads/main' into elef/com-1121-update-pricingtable…
panteliselef Aug 12, 2025
4b3d116
patch
panteliselef Aug 12, 2025
13f3e53
add localizations
panteliselef Aug 12, 2025
eb68681
patch type in react test
panteliselef Aug 12, 2025
ae0d7e6
Merge branch 'main' into elef/com-1121-update-pricingtable-plan-card-…
panteliselef Aug 12, 2025
252f581
Merge branch 'refs/heads/main' into elef/com-1123-update-checkout-flo…
panteliselef Aug 12, 2025
c01e7bb
Merge branch 'elef/com-1121-update-pricingtable-plan-card-with-free-t…
panteliselef Aug 12, 2025
4fe6457
handle conflicts
panteliselef Aug 12, 2025
9505525
Merge branch 'main' into elef/com-1123-update-checkout-flow-to-suppor…
panteliselef Aug 12, 2025
7c875ab
address duplicate
panteliselef Aug 12, 2025
f35f64c
handle localizations
panteliselef Aug 12, 2025
4122331
address review
panteliselef Aug 12, 2025
afc4836
add checkout tests
panteliselef Aug 12, 2025
cfa14a1
Merge branch 'main' into elef/com-1123-update-checkout-flow-to-suppor…
panteliselef Aug 12, 2025
bac7ba5
revert localization key changes
panteliselef Aug 12, 2025
14a1c1b
bump bundlewatch.config.json
panteliselef Aug 13, 2025
4a877c4
fix tests
panteliselef Aug 13, 2025
bbf4534
mock to avoid logging errors
panteliselef Aug 13, 2025
f71b276
add changeset
panteliselef Aug 13, 2025
fa5a912
Merge branch 'main' into elef/com-1123-update-checkout-flow-to-suppor…
panteliselef Aug 13, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
9 changes: 9 additions & 0 deletions .changeset/rotten-lines-cough.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
---
'@clerk/localizations': minor
'@clerk/clerk-js': minor
'@clerk/shared': minor
'@clerk/types': minor
---

Add support for trials in `<Checkout/>`
- Added `freeTrialEndsAt` property to `CommerceCheckoutResource` interface.
4 changes: 2 additions & 2 deletions packages/clerk-js/bundlewatch.config.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"files": [
{ "path": "./dist/clerk.js", "maxSize": "622KB" },
{ "path": "./dist/clerk.js", "maxSize": "622.1KB" },
{ "path": "./dist/clerk.browser.js", "maxSize": "76KB" },
{ "path": "./dist/clerk.legacy.browser.js", "maxSize": "117KB" },
{ "path": "./dist/clerk.headless*.js", "maxSize": "58KB" },
Expand All @@ -23,7 +23,7 @@
{ "path": "./dist/waitlist*.js", "maxSize": "1.5KB" },
{ "path": "./dist/keylessPrompt*.js", "maxSize": "6.5KB" },
{ "path": "./dist/pricingTable*.js", "maxSize": "4.02KB" },
{ "path": "./dist/checkout*.js", "maxSize": "8.5KB" },
{ "path": "./dist/checkout*.js", "maxSize": "8.75KB" },
{ "path": "./dist/up-billing-page*.js", "maxSize": "3.0KB" },
{ "path": "./dist/op-billing-page*.js", "maxSize": "3.0KB" },
{ "path": "./dist/up-plans-page*.js", "maxSize": "1.0KB" },
Expand Down
16 changes: 15 additions & 1 deletion packages/clerk-js/jest.setup.ts
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@ if (typeof window !== 'undefined') {
})),
});

//@ts-expect-error
//@ts-expect-error - JSDOM doesn't provide IntersectionObserver, so we mock it for testing
global.IntersectionObserver = class IntersectionObserver {
constructor() {}

Expand All @@ -53,4 +53,18 @@ if (typeof window !== 'undefined') {
return null;
}
};

// Mock HTMLCanvasElement.prototype.getContext to prevent errors
HTMLCanvasElement.prototype.getContext = jest.fn().mockImplementation(((contextType: string) => {
if (contextType === '2d') {
return {
fillRect: jest.fn(),
getImageData: jest.fn(() => ({ data: new Uint8ClampedArray([255, 255, 255, 255]) }) as unknown as ImageData),
} as unknown as CanvasRenderingContext2D;
}
if (contextType === 'webgl' || contextType === 'webgl2') {
return {} as unknown as WebGLRenderingContext;
}
return null;
}) as any) as jest.MockedFunction<HTMLCanvasElement['getContext']>;
}
4 changes: 4 additions & 0 deletions packages/clerk-js/src/core/resources/CommerceCheckout.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@ import type {
ConfirmCheckoutParams,
} from '@clerk/types';

import { unixEpochToDate } from '@/utils/date';

import { commerceTotalsFromJSON } from '../../utils';
import { BaseResource, CommercePaymentSource, CommercePlan, isClerkAPIResponseError } from './internal';

Expand All @@ -21,6 +23,7 @@ export class CommerceCheckout extends BaseResource implements CommerceCheckoutRe
status!: 'needs_confirmation' | 'completed';
totals!: CommerceCheckoutTotals;
isImmediatePlanChange!: boolean;
freeTrialEndsAt!: Date | null;

constructor(data: CommerceCheckoutJSON, orgId?: string) {
super();
Expand All @@ -43,6 +46,7 @@ export class CommerceCheckout extends BaseResource implements CommerceCheckoutRe
this.status = data.status;
this.totals = commerceTotalsFromJSON(data.totals);
this.isImmediatePlanChange = data.is_immediate_plan_change;
this.freeTrialEndsAt = data.free_trial_ends_at ? unixEpochToDate(data.free_trial_ends_at) : null;
return this;
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -52,7 +52,7 @@ export class CommerceSubscription extends BaseResource implements CommerceSubscr
}
: null;
this.subscriptionItems = (data.subscription_items || []).map(item => new CommerceSubscriptionItem(item));
this.eligibleForFreeTrial = data.eligible_for_free_trial;
this.eligibleForFreeTrial = this.withDefault(data.eligible_for_free_trial, false);
return this;
}
}
Expand Down
21 changes: 15 additions & 6 deletions packages/clerk-js/src/ui/components/Checkout/CheckoutComplete.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ export const CheckoutComplete = () => {
const { setIsOpen } = useDrawerContext();
const { newSubscriptionRedirectUrl } = useCheckoutContext();
const { checkout } = useCheckout();
const { totals, paymentSource, planPeriodStart } = checkout;
const { totals, paymentSource, planPeriodStart, freeTrialEndsAt } = checkout;
const [mousePosition, setMousePosition] = useState({ x: 256, y: 256 });
const [currentPosition, setCurrentPosition] = useState({ x: 256, y: 256 });

Expand Down Expand Up @@ -310,9 +310,11 @@ export const CheckoutComplete = () => {
as='h2'
textVariant='h2'
localizationKey={
totals.totalDueNow.amount > 0
? localizationKeys('commerce.checkout.title__paymentSuccessful')
: localizationKeys('commerce.checkout.title__subscriptionSuccessful')
freeTrialEndsAt
? localizationKeys('commerce.checkout.title__trialSuccess')
: totals.totalDueNow.amount > 0
? localizationKeys('commerce.checkout.title__paymentSuccessful')
: localizationKeys('commerce.checkout.title__subscriptionSuccessful')
}
sx={t => ({
opacity: 0,
Expand Down Expand Up @@ -399,17 +401,24 @@ export const CheckoutComplete = () => {
<LineItems.Title title={localizationKeys('commerce.checkout.lineItems.title__totalPaid')} />
<LineItems.Description text={`${totals.totalDueNow.currencySymbol}${totals.totalDueNow.amountFormatted}`} />
</LineItems.Group>

{freeTrialEndsAt ? (
<LineItems.Group variant='secondary'>
<LineItems.Title title={localizationKeys('commerce.checkout.lineItems.title__freeTrialEndsAt')} />
<LineItems.Description text={formatDate(freeTrialEndsAt)} />
</LineItems.Group>
) : null}
<LineItems.Group variant='secondary'>
<LineItems.Title
title={
totals.totalDueNow.amount > 0
totals.totalDueNow.amount > 0 || freeTrialEndsAt !== null
? localizationKeys('commerce.checkout.lineItems.title__paymentMethod')
: localizationKeys('commerce.checkout.lineItems.title__subscriptionBegins')
}
/>
<LineItems.Description
text={
totals.totalDueNow.amount > 0
totals.totalDueNow.amount > 0 || freeTrialEndsAt !== null
? paymentSource
? paymentSource.paymentMethod !== 'card'
? `${capitalize(paymentSource.paymentMethod)}`
Expand Down
77 changes: 50 additions & 27 deletions packages/clerk-js/src/ui/components/Checkout/CheckoutForm.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ import { Box, Button, Col, descriptors, Flex, Form, localizationKeys, Text } fro
import { ChevronUpDown, InformationCircle } from '../../icons';
import * as AddPaymentSource from '../PaymentSources/AddPaymentSource';
import { PaymentSourceRow } from '../PaymentSources/PaymentSourceRow';
import { SubscriptionBadge } from '../Subscriptions/badge';

type PaymentMethodSource = 'existing' | 'new';

Expand All @@ -25,7 +26,7 @@ const capitalize = (name: string) => name[0].toUpperCase() + name.slice(1);
export const CheckoutForm = withCardStateProvider(() => {
const { checkout } = useCheckout();

const { id, plan, totals, isImmediatePlanChange, planPeriod } = checkout;
const { id, plan, totals, isImmediatePlanChange, planPeriod, freeTrialEndsAt } = checkout;

if (!id) {
return null;
Expand Down Expand Up @@ -53,6 +54,11 @@ export const CheckoutForm = withCardStateProvider(() => {
<LineItems.Title
title={plan.name}
description={planPeriod === 'annual' ? localizationKeys('commerce.billedAnnually') : undefined}
badge={
plan.freeTrialEnabled && freeTrialEndsAt ? (
<SubscriptionBadge subscription={{ status: 'free_trial' }} />
) : null
}
/>
<LineItems.Description
prefix={planPeriod === 'annual' ? 'x12' : undefined}
Expand Down Expand Up @@ -87,6 +93,20 @@ export const CheckoutForm = withCardStateProvider(() => {
<LineItems.Description text={`${totals.pastDue?.currencySymbol}${totals.pastDue?.amountFormatted}`} />
</LineItems.Group>
)}

{!!freeTrialEndsAt && !!plan.freeTrialDays && (
<LineItems.Group variant='tertiary'>
<LineItems.Title
title={localizationKeys('commerce.checkout.totalDueAfterTrial', {
days: plan.freeTrialDays,
})}
/>
<LineItems.Description
text={`${totals.grandTotal?.currencySymbol}${totals.grandTotal?.amountFormatted}`}
/>
</LineItems.Group>
)}

<LineItems.Group borderTop>
<LineItems.Title title={localizationKeys('commerce.totalDueToday')} />
<LineItems.Description text={`${totals.totalDueNow.currencySymbol}${totals.totalDueNow.amountFormatted}`} />
Expand Down Expand Up @@ -278,15 +298,32 @@ export const PayWithTestPaymentSource = () => {
);
};

const AddPaymentSourceForCheckout = withCardStateProvider(() => {
const { addPaymentSourceAndPay } = useCheckoutMutations();
const useSubmitLabel = () => {
const { checkout } = useCheckout();
const { status, totals } = checkout;
const { status, freeTrialEndsAt, totals } = checkout;

if (status === 'needs_initialization') {
return null;
throw new Error('Clerk: Invalid state');
}

if (freeTrialEndsAt) {
return localizationKeys('commerce.startFreeTrial');
}

if (totals.totalDueNow.amount > 0) {
return localizationKeys('commerce.pay', {
amount: `${totals.totalDueNow.currencySymbol}${totals.totalDueNow.amountFormatted}`,
});
}

return localizationKeys('commerce.subscribe');
};

const AddPaymentSourceForCheckout = withCardStateProvider(() => {
const { addPaymentSourceAndPay } = useCheckoutMutations();
const submitLabel = useSubmitLabel();
const { checkout } = useCheckout();

return (
<AddPaymentSource.Root
onSuccess={addPaymentSourceAndPay}
Expand All @@ -296,15 +333,7 @@ const AddPaymentSourceForCheckout = withCardStateProvider(() => {
<PayWithTestPaymentSource />
</DevOnly>

{totals.totalDueNow.amount > 0 ? (
<AddPaymentSource.FormButton
text={localizationKeys('commerce.pay', {
amount: `${totals.totalDueNow.currencySymbol}${totals.totalDueNow.amountFormatted}`,
})}
/>
) : (
<AddPaymentSource.FormButton text={localizationKeys('commerce.subscribe')} />
)}
<AddPaymentSource.FormButton text={submitLabel} />
</AddPaymentSource.Root>
);
});
Expand All @@ -317,8 +346,9 @@ const ExistingPaymentSourceForm = withCardStateProvider(
totalDueNow: CommerceMoneyAmount;
paymentSources: CommercePaymentSourceResource[];
}) => {
const submitLabel = useSubmitLabel();
const { checkout } = useCheckout();
const { paymentSource } = checkout;
const { paymentSource, freeTrialEndsAt } = checkout;

const { payWithExistingPaymentSource } = useCheckoutMutations();
const card = useCardState();
Expand All @@ -340,6 +370,8 @@ const ExistingPaymentSourceForm = withCardStateProvider(
});
}, [paymentSources]);

const isSchedulePayment = totalDueNow.amount > 0 && !freeTrialEndsAt;

return (
<Form
onSubmit={payWithExistingPaymentSource}
Expand All @@ -349,7 +381,7 @@ const ExistingPaymentSourceForm = withCardStateProvider(
rowGap: t.space.$4,
})}
>
{totalDueNow.amount > 0 ? (
{isSchedulePayment ? (
<Select
elementId='paymentSource'
options={options}
Expand Down Expand Up @@ -399,17 +431,8 @@ const ExistingPaymentSourceForm = withCardStateProvider(
width: '100%',
}}
isLoading={card.isLoading}
>
<Text
localizationKey={
totalDueNow.amount > 0
? localizationKeys('commerce.pay', {
amount: `${totalDueNow.currencySymbol}${totalDueNow.amountFormatted}`,
})
: localizationKeys('commerce.subscribe')
}
/>
</Button>
localizationKey={submitLabel}
/>
</Form>
);
},
Expand Down
Loading
Loading