Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
29 commits
Select commit Hold shift + click to select a range
340d3bf
feat(clerk-js,shared): Add parsing for SBB fields (#7785)
dstaley Feb 18, 2026
a3f8ab3
Merge branch 'main' into feat/seat-based-billing
dstaley Feb 18, 2026
f2e39f4
Merge branch 'main' into feat/seat-based-billing
dstaley Mar 6, 2026
0e036d2
feat(clerk-js,localizations,shared,ui): Render seat costs in PricingT…
dstaley Mar 10, 2026
c0052f1
feat(clerk-js,localizations,msw,shared,ui): Add display for member li…
dstaley Mar 12, 2026
383499a
Merge branch 'main' into feat/seat-based-billing
dstaley Mar 12, 2026
192be6e
fix(ui): Assert non-null fee
dstaley Mar 12, 2026
2fe3197
Merge branch 'main' into feat/seat-based-billing
dstaley Mar 16, 2026
d5b41a1
feat(clerk-js,ui): Show SBB totals in Checkout (#8005)
dstaley Mar 17, 2026
403efb1
feat(ui): Disable invite button when over maxAllowedMemberships (#8089)
dstaley Mar 17, 2026
0104ebd
feat(ui): Render seat limits on subscription items (#8069)
dstaley Mar 17, 2026
174a6f0
feat(ui): Render seat limits in SubscriptionDetails (#8115)
dstaley Mar 19, 2026
7f6e3a4
Merge branch 'main' into feat/seat-based-billing
dstaley Mar 19, 2026
b98d919
fix(clerk-js): Bump bundle limit
dstaley Mar 19, 2026
255bdfc
fix(ui): rm console.log
dstaley Mar 19, 2026
e65db01
text(ui): Update for new DOM structure
dstaley Mar 19, 2026
5d0c49a
test(ui): Remove SBB rendering tests
dstaley Mar 19, 2026
5adbece
fix(ui): Bump bundle limit
dstaley Mar 19, 2026
dc73582
Merge branch 'main' into feat/seat-based-billing
dstaley Mar 24, 2026
02ce6eb
Merge branch 'main' into feat/seat-based-billing
dstaley Mar 25, 2026
a1d50a5
chore(repo): Update changelog
dstaley Mar 25, 2026
d9234e5
fix(localizations,ui): Disable switch to plan button when over seat l…
dstaley Mar 26, 2026
74984c2
Merge branch 'main' into feat/seat-based-billing
dstaley Mar 26, 2026
0278220
fix(ui): Render Card background in PricingTable when plan has seat fe…
dstaley Mar 26, 2026
2390e62
Merge branch 'main' into feat/seat-based-billing
jacekradko Mar 27, 2026
73c9f3d
fix(ui): Guard against invalid API unit price responses (#8176)
dstaley Mar 27, 2026
6d3adfd
fix(ui): Update useMemo deps
dstaley Mar 31, 2026
8dccb02
fix(ui): Use membersCount and pendingInvitationsCount
dstaley Mar 31, 2026
0604eb3
Merge branch 'main' into feat/seat-based-billing
dstaley Mar 31, 2026
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
8 changes: 8 additions & 0 deletions .changeset/cute-ideas-appear.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
---
'@clerk/localizations': minor
'@clerk/clerk-js': minor
'@clerk/shared': minor
'@clerk/ui': minor
---

Add support for seat-based billing plans in Clerk Billing.
2 changes: 1 addition & 1 deletion packages/clerk-js/bundlewatch.config.json
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
{
"files": [
{ "path": "./dist/clerk.js", "maxSize": "543KB" },
{ "path": "./dist/clerk.browser.js", "maxSize": "67KB" },
{ "path": "./dist/clerk.browser.js", "maxSize": "68KB" },
{ "path": "./dist/clerk.legacy.browser.js", "maxSize": "110KB" },
{ "path": "./dist/clerk.no-rhc.js", "maxSize": "309KB" },
{ "path": "./dist/clerk.native.js", "maxSize": "68KB" },
Expand Down
341 changes: 341 additions & 0 deletions packages/clerk-js/sandbox/scenarios/checkout-seats.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,341 @@
import {
clerkHandlers,
http,
HttpResponse,
EnvironmentService,
SessionService,
setClerkState,
type MockScenario,
UserService,
} from '@clerk/msw';

export function CheckoutSeats(): MockScenario {
const user = UserService.create();
const session = SessionService.create(user);

setClerkState({
environment: EnvironmentService.MULTI_SESSION,
session,
user,
});

const subscriptionHandler = http.get('https://*.clerk.accounts.dev/v1/me/billing/subscription', () => {
return HttpResponse.json({
response: {
data: {},
},
});
});

const paymentMethodsHandler = http.get('https://*.clerk.accounts.dev/v1/me/billing/payment_methods', () => {
return HttpResponse.json({
response: {
data: {},
},
});
});

const checkoutAccountCreditHandler = http.post('https://*.clerk.accounts.dev/v1/me/billing/checkouts', () => {
return HttpResponse.json({
response: {
object: 'commerce_checkout',
id: 'string',
plan: {
object: 'commerce_plan',
id: 'string',
name: 'Pro',
fee: {
amount: 0,
amount_formatted: '25.00',
currency: 'string',
currency_symbol: '$',
},
annual_monthly_fee: {
amount: 0,
amount_formatted: 'string',
currency: 'string',
currency_symbol: 'string',
},
annual_fee: {
amount: 0,
amount_formatted: 'string',
currency: 'string',
currency_symbol: 'string',
},
description: null,
is_default: true,
is_recurring: true,
publicly_visible: true,
has_base_fee: true,
for_payer_type: 'string',
slug: 'string',
avatar_url: null,
free_trial_enabled: true,
free_trial_days: null,
features: [
{
object: 'feature',
id: 'string',
name: 'string',
description: null,
slug: 'string',
avatar_url: null,
},
],
},
plan_period: 'month',
payer: {
object: 'commerce_payer',
id: 'string',
instance_id: 'string',
user_id: null,
first_name: null,
last_name: null,
email: null,
organization_id: null,
organization_name: null,
image_url: 'https://example.com',
created_at: 1,
updated_at: 1,
},
payment_method: {
object: 'commerce_payment_method',
id: 'string',
payer_id: 'string',
payment_type: 'card',
is_default: true,
gateway: 'string',
gateway_external_id: 'string',
gateway_external_account_id: null,
last4: null,
status: 'active',
wallet_type: null,
card_type: null,
expiry_year: null,
expiry_month: null,
created_at: 1,
updated_at: 1,
is_removable: true,
},
external_gateway_id: 'string',
status: 'needs_confirmation',
totals: {
subtotal: {
amount: 4500,
amount_formatted: '45.00',
currency: 'string',
currency_symbol: '$',
},
tax_total: {
amount: 1,
amount_formatted: 'string',
currency: 'string',
currency_symbol: 'string',
},
grand_total: {
amount: 1,
amount_formatted: 'string',
currency: 'string',
currency_symbol: 'string',
},
total_due_after_free_trial: {
amount: 1,
amount_formatted: 'string',
currency: 'string',
currency_symbol: 'string',
},
total_due_now: {
amount: 4500,
amount_formatted: '45.00',
currency: 'string',
currency_symbol: '$',
},
past_due: null,
credit: {
amount: 1,
amount_formatted: '5.00',
currency: 'string',
currency_symbol: '$',
},
per_unit_totals: [
{
name: 'seats',
block_size: 1,
tiers: [
{
quantity: 10,
fee_per_block: {
amount: 0,
amount_formatted: '0.00',
currency: 'USD',
currency_symbol: '$',
},
total: {
amount: 0,
amount_formatted: '0.00',
currency: 'USD',
currency_symbol: '$',
},
},
{
quantity: 2,
fee_per_block: {
amount: 1000,
amount_formatted: '10.00',
currency: 'USD',
currency_symbol: '$',
},
total: {
amount: 2000,
amount_formatted: '20.00',
currency: 'USD',
currency_symbol: '$',
},
},
],
},
],
},
subscription_item: {
object: 'commerce_subscription_item',
id: 'string',
instance_id: 'string',
status: 'active',
credit: {
amount: {
amount: 1,
amount_formatted: 'string',
currency: 'string',
currency_symbol: 'string',
},
cycle_days_remaining: 1,
cycle_days_total: 1,
cycle_remaining_percent: 1,
},
plan_id: 'string',
price_id: 'string',
plan: {
object: 'commerce_plan',
id: 'string',
name: 'string',
fee: {
amount: 0,
amount_formatted: 'string',
currency: 'string',
currency_symbol: 'string',
},
annual_monthly_fee: {
amount: 0,
amount_formatted: 'string',
currency: 'string',
currency_symbol: 'string',
},
annual_fee: {
amount: 0,
amount_formatted: 'string',
currency: 'string',
currency_symbol: 'string',
},
description: null,
is_default: true,
is_recurring: true,
publicly_visible: true,
has_base_fee: true,
for_payer_type: 'string',
slug: 'string',
avatar_url: null,
free_trial_enabled: true,
free_trial_days: null,
features: [
{
object: 'feature',
id: 'string',
name: 'string',
description: null,
slug: 'string',
avatar_url: null,
},
],
},
plan_period: 'month',
payment_method_id: 'string',
payment_method: {
object: 'commerce_payment_method',
id: 'string',
payer_id: 'string',
payment_type: 'card',
is_default: true,
gateway: 'string',
gateway_external_id: 'string',
gateway_external_account_id: null,
last4: null,
status: 'active',
wallet_type: null,
card_type: null,
expiry_year: null,
expiry_month: null,
created_at: 1,
updated_at: 1,
is_removable: true,
},
lifetime_paid: {
amount: 0,
amount_formatted: 'string',
currency: 'string',
currency_symbol: 'string',
},
amount: {
amount: 0,
amount_formatted: 'string',
currency: 'string',
currency_symbol: 'string',
},
next_payment: {
amount: {
amount: 0,
amount_formatted: 'string',
currency: 'string',
currency_symbol: 'string',
},
date: 1,
},
payer_id: 'string',
payer: {
object: 'commerce_payer',
id: 'string',
instance_id: 'string',
user_id: null,
first_name: null,
last_name: null,
email: null,
organization_id: null,
organization_name: null,
image_url: 'https://example.com',
created_at: 1,
updated_at: 1,
},
is_free_trial: true,
period_start: 1,
period_end: null,
proration_date: 'string',
canceled_at: null,
past_due_at: null,
ended_at: null,
created_at: 1,
updated_at: 1,
},
plan_period_start: 1,
is_immediate_plan_change: true,
free_trial_ends_at: 1,
needs_payment_method: true,
},
});
});

return {
description: 'Checkout with seats',
handlers: [checkoutAccountCreditHandler, subscriptionHandler, paymentMethodsHandler, ...clerkHandlers],
initialState: { session, user },
name: 'checkout-seats',
};
}
3 changes: 3 additions & 0 deletions packages/clerk-js/sandbox/scenarios/index.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,6 @@
export { UserButtonSignedIn } from './user-button-signed-in';
export { CheckoutAccountCredit } from './checkout-account-credit';
export { CheckoutSeats } from './checkout-seats';
export { OrgProfileSeatLimit } from './org-profile-seat-limit';
export { PricingTableSBB } from './pricing-table-sbb';
export { AnnualOnlyPlans } from './annual-only-plans';
Loading
Loading