Overview
Headless Universal Checkout is currently in beta for Web, iOS, Android and React Native. The following payment methods are unsupported:
ADYEN_IDEAL
, ADYEN_DOTPAY
, ADYEN_BLIK
, XFERS_PAYNOW
, RAPYD_PROMPTPAY
, RAPYD_FAST
,
PRIMER_TEST_KLARNA
, PRIMER_TEST_PAYPAL
, PRIMER_TEST_SOFORT
Web currently only supports PAYMENT_CARD
.
Where there is a need for more customization and control over the checkout experience, a headless version of Primer’s Universal Checkout is available. You can use Headless Universal Checkout with your own UI, giving you more flexibility and allowing you to move faster when making design changes, while still having Universal Checkout capture sensitive PCI card data or other form data.
As with all versions of Universal Checkout, any payment method can be added and configured through the Primer Dashboard, meaning Primer handles the logic of when to display each method.
Primer Headless Universal Checkout works in a simple way:
- 1Get a
clientToken
from your server - 2Start Primer Headless Universal Checkout with the client token
- 3Primer Headless Universal Checkout will then return the
available payment methods
for the session initiated. Those payment methods that have been configured in the Dashboard and whose conditions match the current client session will be returned. - 4Show the list of available payment methods.
- 5When the user selects a payment method, show its UI to enable the user to enter their credentials. Depending on the payment method, you will have to either ask the SDK to render it, or build the UI yourself.
- 6Primer Headless Universal Checkout will then create a payment for you and manage its lifecycle. You will receive a confirmation of payment with a callback to indicate the checkout flow has completed.
Headless Universal Checkout is currently available in beta on Android. This documentation is only relevant for v2.5.0 and upward. The documentation for version v2.4.0 is available here.
Prerequisites
Step 1: Prepare the Headless Universal Checkout Listener
Prepare the PrimerHeadlessUniversalCheckoutListener
that will handle callbacks that happen during the lifecycle of the payment methods and the payment.
Import the PrimerHeadlessUniversalCheckout
SDK and set its listener as shown in the following example:
1234567891011121314151617181920
class CheckoutActivity : AppCompatActivity() { private val listener = object : PrimerHeadlessUniversalCheckoutListener { override fun onAvailablePaymentMethodsLoaded(paymentMethods: List<PrimerHeadlessUniversalCheckoutPaymentMethod>) { // Primer will return available payment methods based on configuration // use it to enable users to select a payment method } override fun onCheckoutCompleted(checkoutData: PrimerCheckoutData) { // Primer checkout has been completed // checkoutData contains the payment that has been created // show an order confirmation screen, fulfil the order... } override fun onFailed(error: PrimerError, checkoutData: PrimerCheckoutData?) { // your custom method to show an error view } }}
Step 2: Start Headless Universal Checkout
Once you have a client token, you can initialize Primer’s Headless Checkout by calling PrimerHeadlessUniversalCheckout.current.start(...)
123456789101112131415161718192021222324
class CheckoutActivity : AppCompatActivity() { // other code goes here private val listener = object : PrimerHeadlessUniversalCheckoutListener { // ... } private fun setupObservers() { viewModel.clientToken.observe(this) { clientToken -> // Start the headless checkout session upon reception of the client token startHeadlessUniversalCheckout(clientToken) } } private fun startHeadlessUniversalCheckout(clientToken: String) { // Start with default settings PrimerHeadlessUniversalCheckout.current.start( this, clientToken, PrimerSettings(), listener ) }}
Once Primer’s Headless Universal Checkout is configured, its listener functions will notify you about Headless Universal Checkout events.
Additionally, you can pass the listener as an argument in the start function, or set it with
PrimerHeadlessUniversalCheckout.current.setListener(listener)
Check the SDK API here to customize your SDK settings.
Based on the values in the Client Session (e.g. currencyCode
or countryCode
) you can display different payment methods in the checkout.
Look at the “Checkout” tab on the Primer Dashboard to configure the rules for payment method display.
Step 3: Show available payment methods
When the checkout is done initialising, the onAvailablePaymentMethodsLoaded
method in the listener will be invoked.
Use this event to show the list of payment methods to the user. You can add your buttons for them, or use Primer SDK’s buttons.
12345678910111213
private val listener = object : PrimerHeadlessUniversalCheckoutListener { override fun onAvailablePaymentMethodsLoaded( paymentMethods: List<PrimerHeadlessUniversalCheckoutPaymentMethod> ) { paymentMethods.forEach { setupPaymentMethod(it.paymentMethodType) } }} // your custom method to render list of payment method buttons UI.private fun setupPaymentMethod(paymentMethodType: String) { // implement payment method UI}
You can construct your own payment method, or use the function makeView
to create a payment method button.
123456789101112131415161718192021
private val listener = object : PrimerHeadlessUniversalCheckoutListener { override fun onAvailablePaymentMethodsLoaded( paymentMethods: List<PrimerHeadlessUniversalCheckoutPaymentMethod> ) { paymentMethods.forEach { setupPaymentMethod(it.paymentMethodType) } }} // your custom method to render the list of payment method buttonsprivate fun setupPaymentMethod(paymentMethodType: String) { val containerView = findViewById<ViewGroup>(R.id.container) val view = headlessUniversalCheckout.makeView(paymentMethodType) view?.apply { containerView.addView(this) setOnClickListener { // TODO } }}
Step 4: Handle payment method selection
When the user selects a payment method, you should show the payment method to the customer, either by asking the SDK to show it, or by building the UI.
Headless Universal Checkout currently handles payment methods in three ways:
- With the function
showPaymentMethod
that renders a view all handled by Primer. This is the case today for redirect-based payment methods and Google Pay. - With a Primer card manager that provides you with the list of data to capture so that you can build your on UI. This is the case today for card payments. Primer card manager provides you already built native components that handle validation and formatting of the input data.
- With a “Raw Data Manager” that provides you with the list of data to capture so that you can build your own UI. This is the case today for card payments.
123456789101112131415161718192021
// your custom method to render the list of payment method buttonsprivate fun setupPaymentMethod(paymentMethodType: String) { val containerView = findViewById<ViewGroup>(R.id.container) val view = headlessUniversalCheckout.makeView(paymentMethodType) view?.apply { containerView.addView(this) setOnClickListener { // for card, render your own UI, see more details below. if (paymentMethodType == "PAYMENT_CARD") { // createForm() } else { // for other available payment methods headlessUniversalCheckout.showPaymentMethod( this.context, paymentMethodType ) } } }}
Step 4.1: Handle showPaymentMethod
Use the function showPaymentMethod
to display payment method that must be handled by Primer.
1234
PrimerHeadlessUniversalCheckout.current.showPaymentMethod( this.context, paymentMethodType)
The user can now interact with Universal Checkout, and the SDK will create the payment. The payment’s data will be returned on onCheckoutCompleted(checkoutData)
configured in Step 1.
Step 4.2: Show card form
Headless Universal Checkout securely captures payment method data while fully embedded in your app. By communicating directly with Primer's PCI-L1 tokenization service, Universal Checkout transforms sensitive customer data into a secure uniform string called a payment method token. Universal Checkout automatically uses this token to create a payment.
If the user wants to pay with a card, you can display a card form for paying by card.
Primer will provide you with the required inputs for a given payment method. Use this to create your inputs.
Step 4.2.1: Primer Card Manager
With the card manager, you can show your fully customized UI and still use all of the features that make Primer great.
Render input elements
First, use the PrimerCardManager
object and render the card form input elements like this:
1234567891011121314151617181920
class CheckoutActivity : AppCompatActivity() { private val cardManager by lazy { PrimerCardManager.newInstance() } private fun createForm() { val elementTypes = cardManager.getRequiredInputElementTypes().orEmpty() val inputElements = elementTypes.map { type -> PrimerEditTextFactory.createFromType(this, type).apply { setPrimerInputElementListener(inputElementListener) } } val cardFormView = findViewById<ViewGroup>(R.id.cardForm) inputElements.forEach { cardFormView.addView(it) } cardManager.setInputElements(inputElements) }}
Subscribe to Validation Listener
Then, listen to the card form validation state and enable your form’s submit button accordingly:
1234567891011
// val submitButton = findViewById<Button>(R.id.pay_with_card) private val cardManagerListener = object : PrimerCardManagerListener { override fun onCardValidationChanged(isCardFormValid: Boolean) { submitButton.isEnabled = isCardFormValid }} private fun setupListeners() { cardManager.setCardManagerListener(cardManagerListener)}
Handle Input Updates (optional)
When displaying a card form, you may want to trigger UI changes based on the updates and/or validation status of an active input field. To achieve this please implement the following listener interface and pass it to the input field:
12345678910111213141516
val inputListener = object : PrimerInputElementListener { override fun inputElementValueChanged(inputElement: PrimerInputElement) { /* TODO */ } override fun inputElementValueIsValid(inputElement: PrimerInputElement, isValid: Boolean) { /* TODO */ } override fun inputElementDidDetectCardType(type: CardType.Type) { /* TODO */ }} cardNumberField.setListener(inputLister)
Configure submit button
Finally, configure the on click listener of the submit button:
1
submitButton.setOnClickListener { cardManager.tokenize() }
When the function tokenize
is called, the SDK will attempt to create a payment. The payment’s data will be returned on onCheckoutCompleted(checkoutData)
configured in Step 1.
Step 4.2.2: Primer Raw Data Manager
With the raw (card) manager, you can use your fully customized UI and still use all of the features that make Primer great.
At any point, you can validate your data by calling setRawData
and by listening to onValidationChanged
.
Primer will return errors specific to each required input.
When the data is valid, you can send it to Primer for further processing by calling submit
.
Configure the PrimerHeadlessUniversalCheckoutRawDataManager and subscribe to callbacks
First, configure the PrimerHeadlessUniversalCheckoutRawDataManager
, and optionally set the onValidationChanged
and the onMetadataChanged
callbacks.
123456789101112131415161718192021222324252627
class CheckoutActivity : AppCompatActivity() { // val submitButton = findViewById<Button>(R.id.pay_with_card) private val cardManager by lazy { PrimerHeadlessUniversalCheckoutRawDataManager.newInstance("PAYMENT_CARD") } private val rawDataManagerListener: PrimerHeadlessUniversalCheckoutRawDataManagerListener = object : PrimerHeadlessUniversalCheckoutRawDataManagerListener { override fun onValidationChanged( isValid: Boolean, errors: List<PrimerInputValidationError> ) { // modify your UI submitButton.isEnabled = isValid } override fun onMetadataChanged(metadata: PrimerPaymentMethodMetadata) { // show card icon } }} private fun setupListeners() { cardManager.setManagerListener(rawDataManagerListener)}
Listen to the card data validation state and enable your form’s submit button accordingly.
Render Input Elements
Use the PrimerHeadlessUniversalCheckoutRawDataManager
object and render the card form input elements like this:
1234567891011121314151617181920212223242526272829303132
class CheckoutActivity : AppCompatActivity() { private fun createForm() { val requiredInputElementTypes = cardManager.getRequiredInputElementTypes() // simple example of creating inputs based on type val inputElements = requiredInputElementTypes.map { type -> TextInputLayout(this).apply { tag = type hint = type.name addView(TextInputEditText(context).apply { id = View.generateViewId() doAfterTextChanged { cardManager.setRawData(getCardData()) } layoutParams = LinearLayout.LayoutParams( RelativeLayout.LayoutParams.MATCH_PARENT, RelativeLayout.LayoutParams.WRAP_CONTENT, ) }) } } val cardFormView = findViewById<ViewGroup>(R.id.cardForm) inputElements.forEach { cardFormView.addView(it) } } private fun getInputTypeValue(inputElementType: PrimerInputElementType) = findViewById<ViewGroup>(R.id.cardForm).findViewWithTag<TextInputLayout>( inputElementType ).editText?.text?.trim().toString()}
getRequiredInputElementTypes
will return a list of PrimerInputElementType
.
e.g. for PAYMENT_CARD
Possible values are CARD_NUMBER
, EXPIRY_DATE
, CVV
, CARDHOLDER_NAME
.
Primer Headless Universal Checkout requires input data to be sent for further processing.
For PAYMENT_CARD
, you should pass PrimerRawCardData
:
1234567
data class PrimerRawCardData( val cardNumber: String, // ex. 4242424242424242 val expiryMonth: String, // ex. 01, 1, 12... val expiryYear: String, // ex. 2023 val cvv: String, // ex. 244 val cardholderName: String? = null,)
Submit input data changes
In order to validate input data, and do further processing, submit input data changes:
1
cardManager.setRawData(getCardData())
All changes will be propagated to PrimerHeadlessUniversalCheckoutRawDataManagerListener
Configure submit button
Finally, configure the on click listener of the submit button
1
submitButton.setOnClickListener { cardManager.submit() }
When the function submit
is called, the SDK will attempt to create a payment. The payment’s data will be returned on onCheckoutCompleted(checkoutData)
configured in Step 1.
Step 5: Handle Listener Callbacks (Optional)
Handle onPreparationStarted
Callback (Optional)
This function will notify you that the tokenization preparation has started. At this point the SDK has not yet started to communicate with Primer's backend. It may be useful in case you want to show a loader or notify user about progress in any other way.
123456
private val listener = object : PrimerHeadlessUniversalCheckoutListener { override fun onPreparationStarted(paymentMethodType: String) { // e.g. show a loading screen. }}
Handle onTokenizationStarted
Callback (Optional)
This function will notify you that the tokenization API call has been fired. It may be useful in case you want to show a loader or notify user about progress in any other way.
123456
private val listener = object : PrimerHeadlessUniversalCheckoutListener { override fun onTokenizationStarted(paymentMethodType: String) { // e.g. show a loading screen. }}
Handle onPaymentMethodShowed
Callback (Optional)
This function will notify you that the payment method you requested to show has been presented.
123456
private val listener = object : PrimerHeadlessUniversalCheckoutListener { override fun onPaymentMethodShowed(paymentMethodType: String) { // e.g. hide a loading screen. }}
Handle errors
Use the onFailed
method to handle any errors emitted by the SDK during the checkout flow.
1234567
private val listener = object : PrimerHeadlessUniversalCheckoutListener { override fun onFailed(error: PrimerError, checkoutData: PrimerCheckoutData?) { // your custom method to show an error view showErrorView(error) }}
Prepare 3DS
- In order to make the SDK lightweight, 3DS requires the addition of the Primer 3DS library. Check out our guide on how to add the Primer 3DS library to your Android app.
- When the user pays by card, the Workflow will decide whether or not a 3DS challenge is required. If so, Headless Universal Checkout will automatically render the 3DS challenge.
- To improve 3DS success rates, it is recommended to pass the following elements in the Client Session:
Field | Description |
---|---|
The customer's email address | |
The customer's billing address |
Learn more about how to configure 3DS!
Did It Work?
If everything went well, you should be able to see the payment you just created on your Dashboard under the Payments menu.