Migrate to the Payment Element with the Checkout Sessions API
Accept many payment methods with a single Element, while also managing taxes, shipping, discounts, currency conversion, and more.
Previously, each payment method (cards, iDEAL, and so on) required a separate Element. By migrating to the Payment Element, you can accept many payment methods with a single Element. You can use additional capabilities by migrating to Checkout Sessions from Payment Intents, which enables your integration to manage subscriptions, discounts, shipping, and currency conversion.
If you’re using the Card Element with PaymentIntents or SetupIntents, and only want to migrate to the Payment Element, see migrate to the Payment Element instead. You can also compare other payment integrations if neither fit your use case.
PaymentIntents and SetupIntents each have their own set of migration guidelines. See the appropriate guide for your integration path, including example code.
If your existing integration uses the Payment Intents API to create and track payments or save card details during a payment, follow the steps below to use the Payment Element with Checkout Sessions.
Enable payment methods
Caution
This integration path doesn’t support BLIK or pre-authorised debits that use the Automated Clearing Settlement System (ACSS).
View your payment methods settings and enable the payment methods you want to support. You need at least one payment method enabled to create a Checkout Session.
By default, Stripe enables cards and other common payment methods that can help you reach more customers, but we recommend turning on additional payment methods that are relevant for your business and customers. See Payment method support for product and payment method support, and our pricing page for fees.
Migrate your PaymentIntent creation callServer-side
Set the SDK to use at least the 2025-03-31.
API version.
Because the Payment Element allows you to accept multiple payment methods, we recommend using dynamic payment methods, which are automatically enabled if you don’t pass payment_
into the Checkout Session. When enabled, Stripe evaluates the currency, payment method restrictions, and other parameters to determine the list of payment methods available for your customers. We prioritise payment methods that increase conversion and are most relevant to the currency and location of the customer.
Update your PaymentIntent creation call to create a Checkout Session instead. In the Checkout Sessions instance, you’ll pass:
line_
: Represents what’s in the orderitems ui_
: Indicates that you’re using embedded componentsmode: custom mode: payment
: Indicates that you’ll accept one-off payments for the Checkout Sessionreturn_
: Represents the URL to redirect your customer back to after they authenticate or cancel their payment on the payment method’s app or site.url
In addition, return the Checkout Session’s client_secret to the client-side to use later.
Each Checkout Session generates a PaymentIntent upon confirmation. If you want to retain any extra parameters from your current integration while creating a PaymentIntent, refer to the options available in payment_intent_data.
Collect customer emailClient-side
Migrating to embedded components requires the additional step of collecting your customer’s email.
Add the Payment ElementClient-side
You can now replace the Card Element and individual payment method Elements with the Payment Element. The Payment Element automatically adjusts to collect input fields based on the payment method and country (for example, full billing address collection for SEPA Direct Debit) so you don’t have to maintain customised input fields any more.
The following example replaces CardElement
with PaymentElement
:
Update the submit handlerClient-side
Instead of using individual confirm methods like stripe.
or stripe.
, use checkout.confirm to collect payment information and submit it to Stripe.
To confirm the Checkout Session, update your submit handler to use checkout.
instead of individual confirm methods.
When called, checkout.
attempts to complete any required actions, such as displaying a 3DS window or redirecting them to a bank authorisation page. When confirmation is complete, customers redirect to the return_
you configured, which normally corresponds to a page on your website that provides the status of the payment.
If you want to keep the same checkout flow for card payments and only redirect for redirect-based payment methods, you can set redirect to if_
.
The following code example replaces stripe.
with checkout.
:
// Create the PaymentIntent and obtain clientSecret const res = await fetch("/create-intent", { method: "POST", headers: {"Content-Type": "application/json"}, }); const {client_secret: clientSecret} = await res.json(); const handleSubmit = async (event) => { event.preventDefault(); if (!stripe) { // Stripe.js hasn't yet loaded. // Make sure to disable form submission until Stripe.js has loaded. return; } setLoading(true); const {error} = await stripe.confirmCardPayment(clientSecret, { payment_method: { card: elements.getElement(CardElement) } }); if (error) { handleError(error); } };
const handleSubmit = async (event) => { event.preventDefault(); if (!stripe) { // Stripe.js hasn't yet loaded. // Make sure to disable form submission until Stripe.js has loaded. return; } setLoading(true); const {error} = await checkout.confirm(); if (error) { handleError(error); } };
Handle post-payment eventsServer-side
Stripe sends a checkout.session.completed event when the payment completes. Use the Dashboard webhook tool or follow the webhook guide to receive these events and run actions, such as sending an order confirmation email to your customer, logging the sale in a database, or starting a shipping workflow.
Listen for these events rather than waiting on a callback from the client. On the client, the customer could close the browser window or quit the app before the callback executes, and malicious clients could manipulate the response. Setting up your integration to listen for asynchronous events is what enables you to accept different types of payment methods with a single integration.
In addition to handling the checkout.
event, we recommend handling these other events when collecting payments with the Payment Element:
Event | Description | Action |
---|---|---|
checkout.session.completed | Sent when a customer successfully completes a payment. | Send the customer an order confirmation and fulfill their order. |
checkout_session.async_payment_succeeded | Sent when payment by a customer using a delayed payment method finally succeeds. | Send the customer an order confirmation and fulfill their order. |
checkout.session.async_payment_failed | Sent when a customer attempts a payment, but the payment fails. | If a payment transitions from async_ , offer the customer another attempt to pay. |
checkout.session.expired | Sent when a customer’s checkout session has expired, which is after 24 hours. | If a payment transitions from expired to payment_ , offer the customer an attempt to reload the checkout page and create a new checkout session. |
Test the integration
- Navigate to your checkout page.
- Fill out the payment details with a payment method from the following table. For card payments:
- Enter any future date for card expiry.
- Enter any 3-digit number for CVC.
- Enter any billing postal code.
- Submit the payment to Stripe.
- Go to the Dashboard and look for the payment on the Transactions page. If your payment succeeded, you’ll see it in that list.
- Click your payment to see more details, like billing information and the list of purchased items. You can use this information to fulfil the order.
See Testing for additional information to test your integration.