This repository contains a polyfill for Apple Pay JS for use for testing on devices that do not natively support Apple Pay JS.
Apple Pay JS is a way of accepting Apple Pay in websites using Safari in either iOS 10 (and later) and macOS for users who have a TouchID compatible device.
This polyfill provides a way to make ApplePaySession
available for testing your implementation in browsers that would otherwise not provide support for Apple Pay JS, such as in Chrome on Windows.
The polyfill supports the Apple Pay JS API for versions 1 to 10. The Payment Request API is not supported.
To create a minimal working integration using the polyfill, first install the JavaScript using bower:
$ bower install applepayjs-polyfill
Then reference it in your HTML (don't forget to remove it in production environments):
<script src="/lib/applepayjs-polyfill/ApplePaySession.js"></script>
Next, configure the callbacks on the ApplePaySessionPolyfill
object to return the values you want:
// Set the merchant identifier to drive ApplePaySession.canMakePaymentsWithActiveCard()
ApplePaySessionPolyfill.setMerchantIdentifier("My_Merchant_Identifier");
// Optionally change the validation URL for merchant validation (the default is the URL for the Apple Pay Sandbox)
// ApplePaySessionPolyfill.setValidationURL("https://someurl.somedomain.com")
// Re-declare the function to create a PaymentContact for billing
// See https://developer.apple.com/documentation/apple_pay_on_the_web/applepaypaymentcontact
// "phoneticGivenName" and "phoneticFamilyName" fields are available starting in API version 3.
ApplePaySessionPolyfill.createBillingContact = function (session) {
return {
phoneNumber: '(408) 555-5555',
emailAddress: 'ravipatel@example.com',
givenName: 'Ravi',
familyName: 'Patel',
phoneticGivenName: 'Ravi',
phoneticFamilyName: 'Patel',
addressLines: [
'Address Line 1',
'Address Line 2'
],
subLocality: '',
locality: 'Cupertino',
postalCode: '95014-2083',
subAdministrativeArea: '',
administrativeArea: 'CA',
country: 'United States"',
countryCode: 'US'
};
};
// Re-declare the function to create a PaymentContact for shipping
// You can copy the same interface from "createBillingContact";
ApplePaySessionPolyfill.createShippingContact = function (session) {
return ApplePaySessionPolyfill.createBillingContact();
};
// Re-declare the function to create a PaymentToken for an authorized payment
// See https://developer.apple.com/documentation/apple_pay_on_the_web/applepaypaymenttoken
// and https://developer.apple.com/documentation/apple_pay_on_the_web/applepaypaymentmethod
ApplePaySessionPolyfill.createPaymentToken = function (session) {
return {
paymentMethod: {
displayName: 'Visa 1233',
network: 'visa',
type: 'credit',
paymentPass: {
primaryAccountIdentifier: 'AAAAAAAAAAAA',
primaryAccountNumberSuffix: '1233',
deviceAccountIdentifier: '999999999999999',
deviceAccountNumberSuffix: '9999',
activationState: 'activated'
},
billingContact: ApplePaySessionPolyfill.createBillingContact()
},
transactionIdentifier: 'AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA',
paymentData: {
data: "aAa9A9aA9aA9aAa9+9aAAA999a9aAA/AaAAA9A9AaaaAa9+AAAaAaaa9AaaA99AaAA9Aa9aAA+aAaA9AaAAAA==",
signature: "AAAAAAaAAAa9AAAAAaAAAAAAAAAaAaAAAaaaaaaAAAAAAaAAAAAAAaaaaaaA9a9AAaAAAAAAAAAA9aAA==",
header: {
publicKeyHash: "aaaaAA99aaA9aaa99A9Aaa9Aa9aaaa9A9a9AAAaaAaA=",
ephemeralPublicKey: "AAaaAaAAAaAAaa9AAAAAAaAAaa9AAAaAAaAAAa9/99AaAaAaaAaaAaA9AaAaaaAa+aAaa==",
transactionId: "a9a9a9a99aa99999999a99a9aaa999aaa9aa99aa99a9a99aaa99999aaa9aa99a"
},
version: "FAKE_v1"
}
};
};
Now you should be able to test your implementation of ApplePaySession
in an HTML page in a browser that does not already provide the object in window
. For example:
if ("ApplePaySession" in window && ApplePaySession.canMakePayments() === true) {
var button = document.getElementsByClassName("apple-pay-button")[0];
button.addEventListener("click", function (e) {
e.preventDefault();
var paymentRequest = {
countryCode: "GB",
currencyCode: "GBP",
merchantCapabilities: ["supports3DS"],
supportedNetworks: ["amex", "masterCard", "visa"],
total: {
label: "Just Eat",
amount: "1.00"
}
};
var session = new ApplePaySession(10, paymentRequest);
session.onvalidatemerchant = function (event) {
/* Merchant validation implementation */
};
session.onpaymentauthorized = function (event) {
/* Payment authorization implementation */
};
session.begin();
});
}
If you need to test displaying the "Set Up Apple Pay" button, use the setUserSetupStatus(bool)
function, as shown below, to specify that the user has not yet set up Apple Pay on the device.
ApplePaySessionPolyfill.setUserSetupStatus(false);
By default this value is set to true
so that Apple Pay is available in the polyfill.
If you need to test compatibility with devices that do not support Apple Pay set up, then delete the function from ApplePaySession
before your implementation code is loaded:
delete ApplePaySession.openPaymentSetup;
The Apple Pay documentation states:
The
onpaymentauthorized
function must complete the payment and respond by callingcompletePayment
before the 30 second timeout, after which a message appears stating that the payment could not be completed.
You can change this timeout with setAuthorizationTimeout(milliseconds)
.
ApplePaySessionPolyfill.setAuthorizationTimeout(1000);
By default this value is set to 30000 milliseconds (30 seconds) in the polyfill like in a real Apple Pay session.
Any feedback or issues can be added to the issues for this project in GitHub.
This project is licensed under the Apache 2.0 license.