Copyright © 2017 W3C® (MIT, ERCIM, Keio, Beihang). W3C liability, trademark and permissive document license rules apply.
The Payment Request API provides a standard way to initiate payment requests from Web pages and applications. User agents implementing that API prompt the user to select a way to handle the payment request, after which the user agent returns a payment response to the originating site. This specification defines capabilities that enable Web applications to handle payment requests.
This section describes the status of this document at the time of its publication. Other documents may supersede this document. A list of current W3C publications and the latest revision of this technical report can be found in the W3C technical reports index at https://www.w3.org/TR/.
The Web Payments Working Group maintains a list of all bug reports that the group has not yet addressed. This draft highlights some of the pending issues that are still to be discussed in the working group. No decision has been taken on the outcome of these issues including whether they are valid. Pull requests with proposed specification text for outstanding issues are strongly encouraged.
This document was published by the Web Payments Working Group as a Working Draft. This document is intended to become a W3C Recommendation.
Publication as a Working Draft does not imply endorsement by the W3C Membership. This is a draft document and may be updated, replaced or obsoleted by other documents at any time. It is inappropriate to cite this document as other than work in progress.
This document was produced by a group operating under the 5 February 2004 W3C Patent Policy. W3C maintains a public list of any patent disclosures made in connection with the deliverables of the group; that page also includes instructions for disclosing a patent. An individual who has actual knowledge of a patent which the individual believes contains Essential Claim(s) must disclose the information in accordance with section 6 of the W3C Patent Policy.
This document is governed by the 1 March 2017 W3C Process Document.
This section is non-normative.
The Web Payments Working Group seeks to streamline payments on the Web to help reduce "shopping cart abandonment" and make it easier to deploy new payment methods on the Web. It has published the Payment Request API [payment-request] as a standard way to initiate payment requests from E-Commerce Web sites and applications.
A payment app is a Web application that can handle payment requests on behalf of the user. This specification defines a number of new Web platform features to handle payment requests:
PaymentRequestEvent
). A
payment handler is an event handler for the
PaymentRequestEvent
.
PaymentInstrument
s.
This specification does not address how software built with operating-system specific mechanisms (e.g., "native mobile apps") handle payment requests.
The term "payment app" may be useful as a shorthand for "Web app that can handle payments with Payment Request API."
As well as sections marked as non-normative, all authoring guidelines, diagrams, examples, and notes in this specification are non-normative. Everything else in this specification is normative.
The key words MAY, MUST, NOT REQUIRED, SHOULD, and SHOULD NOT are to be interpreted as described in [RFC2119].
This specification defines one class of products:
A user agent MUST behave as described in this specification to be considered conformant. In this specification, user agent means a Web browser or other interactive user agent as defined in [HTML5].
User agents MAY implement algorithms given in this specification in any way desired, so long as the end result is indistinguishable from the result that would be obtained by the specification's algorithms.
A conforming Payment Handler API user agent MUST also be a conforming implementation of the IDL fragments of this specification, as described in the “Web IDL” specification. [ WEBIDL-LS]
This specification relies on several other underlying specifications.
TypeError
, and JSON.stringify
are defined by [ECMA-262-2015].
The term JSON-serialize applied to a given object means to run the algorithm specified by the original value of the JSON.stringify
function on the supplied object, passing the supplied object as the sole argument, and return the resulting string. This can throw an exception.
DOMException and the following DOMException types from [WEBIDL-LS] are used:
ServiceWorkerRegistration
,
ServiceWorkerGlobalScope
,
handle
functional event, extend lifetime
promises, and scope URL are defined in [
SERVICE-WORKERS].
In this document we envision the following flow:
instruments
and wallets
.
capabilities
play a role in matching computations.
instruments
and
wallets
supported by the payment handler.
instruments
of the candidate payment handlers. The user agent displays and groups these choices according to information (labels and icons) provided at registration or otherwise available from the Web app.
PaymentRequestEvent
(cf. the user interaction
task source) in the service worker whose PaymentManager the instrument was registered with. The PaymentRequestEvent
includes some information from the PaymentRequest (defined in [
payment-request]) as well as additional information (e.g., origin and selected instrument).
An origin may implement a payment app with more than one service worker and therefore multiple payment handlers may be registered per origin. The handler that is invoked is determined by the selection made by the user of a payment instrument. The service worker which stored the payment instrument with its PaymentManager is the one that will be invoked.
This section is non-normative.
The logic of a payment handler is driven by the payment methods that it supports. Some payment methods, such as basic-card expect little to no processing by the payment handler which simply returns payment card details in the response. It is then the job of the payee website to process the payment using the returned data as input.
In contrast, some payment methods, such as a crypto-currency payments or bank originated credit transfers, require that the payment handler initiate processing of the payment. In such cases the payment handler will return a payment reference, endpoint URL or some other data that the payee website can use to determine the outcome of the payment (as opposed to processing the payment itself).
Handling a payment request may include numerous interactions: with the user through a new window or other APIs (such as [ WebCryptoAPI]) or with other services and origins through web requests or other means.
This specification does not address these activities that occur between the payment handler accepting the PaymentRequestEvent
and the payment handler returning a response. All of these activities which may be required to configure the payment handler and handle the payment request, are left to the implementation of the payment handler, including:
Thus, an origin will rely on many other Web technologies defined elsewhere for lifecycle management, security, user authentication, user interaction, and so on.
This section is non-normative.
This section is non-normative.
This specification does not address how third-party mobile payment apps interact (through proprietary mechanisms) with user agents, or how user agents themselves provide simple payment app functionality.
ServiceWorkerRegistration
interface
This specification extends the ServiceWorkerRegistration
interface with the addition of a paymentManager
attribute.
partial interface ServiceWorkerRegistration
{
readonly attribute PaymentManager paymentManager
;
};
[SecureContext]
interface PaymentManager {
[SameObject]
readonly attribute PaymentInstruments
instruments
;
[SameObject]
readonly attribute PaymentWallets
wallets
;
[Exposed=Window]
static Promise<PermissionState> requestPermission();
};
The PaymentManager is used by payment apps to manage their associated wallets, instruments and supported payment methods.
instruments
attribute
This attribute allows manipulation of payment instruments associated with a service worker (and therefore its payment handler). To be a candidate payment handler, a handler must have at least one registered payment instrument to present to the user. That instrument needs to match the payment methods and required capabilities specified by the payment request.
wallets
attribute
This attribute is used to group payment instruments (e.g., to group together "business account" payment instruments separately from "personal account" payment instruments). Developers can provide a name and icon for such a group. The use of this grouping mechanism by payment handlers is completely optional. If payment handlers use this grouping mechanism, then matching payment instruments that do not appear in any groups should still be presented to users by the user agent for selection.
Should the API support providing grouping information ("wallets") to the user agent? What should requirements be on user agents to display or provide for user interaction with this information?
requestPermission()
method
The user agent is NOT REQUIRED to prompt the user to grant permission to the origin for each new supported payment method.
When called, this method executes the following steps:
PaymentInstruments
interface
interface PaymentInstruments
{
Promise<boolean> delete(DOMString instrumentKey);
Promise<PaymentInstrument
> get(DOMString instrumentKey);
Promise<sequence<DOMString>> keys();
Promise<boolean> has(DOMString instrumentKey);
Promise<void> set(DOMString instrumentKey,
PaymentInstrument
details);
Promise<void> clear();
};
The PaymentInstruments
interface represents a collection of payment instruments, each uniquely identified by an
instrumentKey. The instrumentKey identifier will be passed to the payment handler to indicate the
PaymentInstrument
selected by the user.
delete()
method
When called, this method executes the following steps:
PaymentInstrument
with a matching instrumentKey, remove it from the collection and resolve p with true.
get()
method
When called, this method executes the following steps:
PaymentInstrument
with a matching instrumentKey, resolve p with that PaymentInstrument
.
NotFoundError
".
keys()
method
When called, this method executes the following steps:
PaymentInstrument
s contained in the collection, in original insertion order.
has()
method
When called, this method executes the following steps:
PaymentInstrument
with a matching instrumentKey, resolve p with
true.
set()
method
When called, this method executes the following steps:
NotAllowedError
.
icons
member of
details is present, then:
icons
member of details is present, then for each
icon in details.icons
:
PaymentInstrument
with a matching instrumentKey, replace it with the
PaymentInstrument
in details.
PaymentInstrument
in
details as a new member of the collection and associate it with the key instrumentKey.
clear()
method
When called, this method executes the following steps:
PaymentInstrument
s from the collection and resolve p.
It be useful to have a clear()
method for interfaces that store things.
PaymentInstrument
dictionary
dictionary PaymentInstrument
{
required DOMString name
;
sequence<ImageObject
> icons
;
sequence<DOMString> enabledMethods
;
object capabilities
;
};
name
member
name
member is a string that represents the label for this PaymentInstrument
as it is usually displayed to the user.
icons
member
icons
member is an array of image objects that can serve as iconic representations of the payment instrument when presented to the user for selection.
enabledMethods
member
enabledMethods
member is a list of one or more
payment method identifiers of the payment methods supported by this instrument.
capabilities
member
capabilities
member is a list of payment-method-specific capabilities that this payment handler is capable of supporting for this instrument. For example, for the basic-card payment method, this object will consist of an object with two fields: one for supportedNetworks, and another for
supportedTypes.
PaymentWallets
interface
interface PaymentWallets
{
Promise<boolean> delete(DOMString walletKey);
Promise<PaymentWallet
> get(DOMString walletKey);
Promise<sequence<DOMString>> keys();
Promise<boolean> has(DOMString walletKey);
Promise<void> set(DOMString walletKey,
PaymentWallet
details);
Promise<void> clear();
};
Wallets are collections of payment instruments.
Where it appears, the walletKey argument is a unique identifier for the wallet.
delete()
method
When called, this method executes the following steps:
PaymentWallet
with a matching walletKey, remove it from the collection and resolve p with true.
get()
method
When called, this method executes the following steps:
PaymentWallet
with a matching walletKey, resolve p with that
PaymentWallet
.
NotFoundError
".
keys()
method
When called, this method executes the following steps:
PaymentWallet
s contained in the collection, in original insertion order.
has()
method
When called, this method executes the following steps:
PaymentWallet
with a matching walletKey, resolve p with
true.
set()
method
When called, this method executes the following steps:
NotAllowedError
.
icons
member of
details is present, then:
icons
member of
details is present, then for each icon in
details.icons
:
PaymentWallet
with a matching walletKey, replace it with the
PaymentWallet
in details.
PaymentWallet
in
details as a new member of the collection and associate it with the key walletKey.
clear()
method
When called, this method executes the following steps:
PaymentWallets
from the collection and resolve p.
PaymentWallet
dictionary
dictionary PaymentWallet
{
required DOMString name
;
sequence<ImageObject
> icons
;
required sequence<DOMString> instrumentKeys
;
};
name
member
name
member is a string that represents the label for this wallet as it is usually displayed to the user.
icons
member
icons
member is an array of image objects that can serve as iconic representations of the wallet when presented to the user for selection.
instrumentKeys
member
instrumentKeys
member is a list of one or more
instrumentKeys from PaymentManager.instruments
, indicating which PaymentInstrument
objects are associated with this Wallet, and should be displayed as being "contained in" the wallet. While it is not generally good practice, there is no restriction that prevents a PaymentInstrument
from appearing in more than one Wallet.
ImageObject
dictionary
src
member
src
member is used to specify the ImageObject
's source. It is a URL from which the user agent can fetch the image’s data.
sizes
member
sizes
member is used to specify the ImageObject
's sizes. It follows the spec of sizes member in HTML link element, which is a string consisting of an unordered
set of unique space-separated tokens which are ASCII case-insensitive that represents the dimensions of an image. Each keyword is either an
ASCII
case-insensitive match for the string "any", or a value that consists of two valid non-negative integers that do not have a leading U+0030 DIGIT ZERO (0) character and that are separated by a single U+0078 LATIN SMALL LETTER X or U+0058 LATIN CAPITAL LETTER X character. The keywords represent icon sizes in raw pixels (as opposed to CSS pixels). When multiple image objects are available, a user agent MAY use the value to decide which icon is most suitable for a display context (and ignore any that are inappropriate). The parsing steps for the sizes
member MUST follow the parsing steps for
HTML link element sizes attribute.
type
member
type
member is used to specify the ImageObject
's MIME type. It is a hint as to the media type of the image. The purpose of this member is to allow a user agent to ignore images of media types it does not support.
When this algorithm with inputImages parameter is invoked, the user agent must run the following steps:
ImageObject
.
type
is not a valid MIME type or the value of type is not a supported media format, then return an empty Sequence of ImageObject
.
sizes
is not a valid value, then return an empty Sequence of ImageObject
.
src
with the
context object's
relevant settings
object's API base URL.
ImageObject
.
ImageObject
.
src
to
url.
According to the step 2.3, it is also possible to use the relative url for image.src
. The following examples illustrate how relative URL resolution works in different execution contexts.
<-- In this example, code is located in https://www.example.com/bobpay/index.html -->
<script>
const instrumentKey = "c8126178-3bba-4d09-8f00-0771bcfd3b11";
const { registration } = await navigator.serviceWorker.register("/register/sw.js");
await registration.paymentManager.paymentInstruments.set({
instrumentKey,
{
name: "My Bob Pay Account: john@example.com",
enabledMethods: ["https://bobpay.com/"],
icons: [{
src: "icon/lowres.webp",
sizes: "48x48",
type: "image/webp"
}]
});
const { storedInstrument } =
await registration.paymentManager.paymentInstruments.get(instrumentKey);
// storedInstrument.icons[0].src == "https://www.example.com/bobpay/icon/lowres.webp";
</script>
// In this example, code is located in https://www.example.com/register/sw.js
const instrumentKey = "c8126178-3bba-4d09-8f00-0771bcfd3b11";
await self.registration.paymentManager.paymentInstruments.set({
instrumentKey,
{
name: "My Bob Pay Account: john@example.com",
enabledMethods: ["https://bobpay.com/"],
icons: [{
src: "../bobpay/icon/lowres.webp",
sizes: "48x48",
type: "image/webp"
}]
});
const { storedInstrument } =
await registration.paymentManager.paymentInstruments.get(instrumentKey);
// storedInstrument.icons[0].src == "https://www.example.com/bobpay/icon/lowres.webp";
This section is non-normative.
The following example shows how to register a payment handler:
button.addEventListener("click", async() => {
const permission =
await navigator.permissions.query({ name: "paymenthandler" });
switch (permission) {
case "denied":
return;
case "prompt":
const result = await registration.paymentManager.requestPermission();
if (result !== "granted") {
return;
}
break;
}
const { registration } =
await navigator.serviceWorker.register('/sw.js');
if (!registration.paymentManager) {
return; // not supported, so bail out.
}
// Excellent, we got it! Let's now set up the user's cards.
await addInstruments(registration);
}, { once: true });
function addInstruments(registration) {
const instrumentPromises = [
registration.paymentManager.instruments.set(
"dc2de27a-ca5e-4fbd-883e-b6ded6c69d4f",
{
name: "Visa ending ****4756",
enabledMethods: ["basic-card"],
capabilities: {
supportedNetworks: ['visa'],
supportedTypes: ['credit']
}
}),
registration.paymentManager.instruments.set(
"c8126178-3bba-4d09-8f00-0771bcfd3b11",
{
name: "My Bob Pay Account: john@example.com",
enabledMethods: ["https://bobpay.com/"]
}),
registration.paymentManager.instruments.set(
"new-card",
{
name: "Add new credit/debit card to ExampleApp",
enabledMethods: ["basic-card"],
capabilities: {
supportedNetworks:
['visa','mastercard','amex','discover'],
supportedTypes: ['credit','debit','prepaid']
}
}),
];
return Promise.all(instrumentPromises).then(() => {
registration.paymentManager.wallets.set(
"12a1b7e5-16c0-4c09-a312-9b191d08517b",
{
name: "Acme Bank Personal Accounts",
icons: [
{ src: "icon/lowres.webp",
sizes: "48x48",
type: "image/webp"
},
{ src: "icon/lowres",
sizes: "48x48"
}
],
instrumentKeys: [
"dc2de27a-ca5e-4fbd-883e-b6ded6c69d4f",
"c8126178-3bba-4d09-8f00-0771bcfd3b11",
"new-card"
]
});
});
};
After applying the matching algorithm defined in Payment Request API, the user agent displays a list of instruments from matching payment apps for the user to make a selection. This specification includes a limited number of display requirements; most user experience details are left to implementers.
The second bullet above may be deleted based on PR API issue 481.
The following are examples of payment handler ordering:
The Working Group has discussed two types of merchant preferences related to payment apps: (1) highlighting merchant-preferred payment apps already registered by the user and (2) recommending payment apps not yet registered by the user. The current draft of the specification does not address either point, and the Working Group is seeking feedback on the importance of these use cases. Note that for the second capability, merchants can recommend payment apps through other mechanisms such as links from their web sites.
The user agent MUST enable the user to select any displayed instrument.
In issue 98 there has been push-back to always requiring display of instruments (e.g., on a mobile devices). User agents can incrementally show instruments. Or user agents can return an empty instrumentKey and it becomes the payment app's responsibility to display instruments to the user.
At times, the same origin may wish to group instruments with greater flexibility and granularity than merely "by origin." These use cases include:
A Wallet is a grouping of instruments for display purposes.
To enable developers to build payment apps in a variety of ways, we decouple the registration (and subsequent display) of instruments from how payment handlers respond to a PaymentRequestEvent
. However, the user agent is responsible for communicating the user's selection in the event.
Users agents may wish to enable the user to select individual displayed Instruments. The payment handler would receive information about the selected Instrument and could take action, potentially eliminating an extra click (first open the payment app then select the Instrument).
Again related to issue 98: Should we require that, if displayed, individual instruments must be selectable? Or should we allow flexibility that instruments may be displayed, but selecting any one invokes all registered payment handlers? One idea that has been suggested: the user agent (e.g., on a mobile device) could first display the app-level icon/logo. Upon selection, the user agent could display the Instruments in a submenu.
Once the user has selected an Instrument, the user agent fires a
PaymentRequestEvent
and uses the subsequent
PaymentHandlerResponse
to create a PaymentReponse for [
payment-request].
Payment Request API supports delegation of responsibility to manage an abort to a payment app. There is a proposal to add a paymentRequestAborted event to the Payment Handler interface. The event will have a respondWith method that takes a boolean parameter indicating if the paymentRequest has been successfully aborted.
ServiceWorkerGlobalScope
This specification extends the ServiceWorkerGlobalScope
interface.
partial interface ServiceWorkerGlobalScope
{
attribute EventHandler onpaymentrequest
;
};
onpaymentrequest
attribute
The onpaymentrequest
attribute is an event handler whose corresponding event handler event type is
paymentrequest.
PaymentRequestEvent
The PaymentRequestEvent
represents a received
PaymentRequest.
[Constructor(DOMString type, PaymentRequestEventInit
eventInitDict),
Exposed=ServiceWorker]
interface PaymentRequestEvent
: ExtendableEvent {
readonly attribute USVString topLevelOrigin
;
readonly attribute USVString paymentRequestOrigin
;
readonly attribute DOMString paymentRequestId
;
readonly attribute FrozenArray<PaymentMethodData> methodData
;
readonly attribute object total
;
readonly attribute FrozenArray<PaymentDetailsModifier> modifiers
;
readonly attribute DOMString instrumentKey
;
Promise<WindowClient> openWindow
(USVString url);
void respondWith
(Promise<PaymentHandlerResponse
> handlerResponse);
};
topLevelOrigin
attribute
This attribute is a string that indicates the origin of the top level payee web page. The string MUST be formatted according to the "Unicode Serialization of an Origin" algorithm defined in section 6.1 of [RFC6454].
paymentRequestOrigin
attribute
This attribute is a string that indicates the origin where a PaymentRequest was initialized. When a PaymentRequest is initialized in the
topLevelOrigin
, the attributes have the same value, otherwise the attributes have different values. For example, when a
PaymentRequest is initialized within an iframe from an origin other than topLevelOrigin
, the value of this attribute is the origin of the iframe. The string MUST be formatted according to the "Unicode
Serialization of an Origin" algorithm defined in section 6.1 of [
RFC6454].
paymentRequestId
attribute
When getting, the paymentRequestId
attribute returns the [[details]].
id from the PaymentRequest that corresponds to this PaymentRequestEvent
.
methodData
attribute
This attribute contains PaymentMethodData dictionaries containing the payment method identifiers for the payment methods that the web site accepts and any associated payment method specific data. It is populated from the PaymentRequest using the MethodData Population Algorithm defined below.
total
attribute
This attribute indicates the total amount being requested for payment. It is of type PaymentItem dictionary as defined in [
payment-request], and initialized with a structured clone of the total
field of the PaymentDetailsInit provided when the corresponding PaymentRequest object was instantiated.
modifiers
attribute
This sequence of PaymentDetailsModifier dictionaries contains modifiers for particular payment method identifiers (e.g., if the payment amount or currency type varies based on a per-payment-method basis). It is populated from the PaymentRequest using the Modifiers Population Algorithm defined below.
instrumentKey
attribute
This attribute indicates the PaymentInstrument
selected by the user. It corresponds to the instrumentKey
provided to the
PaymentManager.instruments interface during registration.
openWindow
() method
This method is used by the payment handler to show a window to the user. When called, it runs the open window algorithm.
respondWith
() method
This method is used by the payment handler to provide a
PaymentHandlerResponse
when the payment successfully completes.
Should payment apps receive user data stored in the user agent upon explicit consent from the user? The payment app could request permission either at installation or when the payment app is first invoked.
PaymentRequestEventInit
dictionary
dictionary PaymentRequestEventInit
: ExtendableEventInit {
USVString topLevelOrigin
;
USVString paymentRequestOrigin
;
DOMString paymentRequestId
;
sequence<PaymentMethodData> methodData
;
PaymentItem total
;
sequence<PaymentDetailsModifier> modifiers
;
DOMString instrumentKey
;
};
The topLevelOrigin
, paymentRequestOrigin
,
paymentRequestId
, methodData
,
total
, modifiers
, and
instrumentKey
members share their definitions with those defined for PaymentRequestEvent
To initialize the value of the methodData
, the user agent
MUST perform the following steps or their equivalent:
PaymentInstrument
instrument in the
payment handler's PaymentManager.instruments
, add all entries in instrument.enabledMethods
to
registeredMethods.
methodData
to dataList.
To initialize the value of the modifiers
, the user agent
MUST perform the following steps or their equivalent:
PaymentInstrument
instrument in the
payment handler's PaymentManager.instruments
, add all entries in instrument.enabledMethods
to
registeredMethods.
modifiers
in the corresponding payment request, perform the following steps:
total
to a structured
clone of inModifier.total
.
modifiers
to modifierList.
Instances of PaymentRequestEvent
are created with the internal slots in the following table:
Internal Slot | Default Value | Description (non-normative) |
---|---|---|
[[windowClient]] | null | The currently active WindowClient. This is set if a payment handler is currently showing a window to the user. Otherwise, it is null. |
[[fetchedImage]] | undefined | This value is a result of fetching image object or a fallback image provided by the user agent. |
Upon receiving a PaymentRequest by way of PaymentRequest.show() and subsequent user selection of a payment instrument, the user agent MUST run the following steps:
ServiceWorkerRegistration
corresponding to the
PaymentInstrument
selected by the user.
InvalidStateError
" and terminate these steps.
ServiceWorkerRegistration
of registration and
callbackSteps set to the following steps:
PaymentRequestEvent
interface, with the event type
paymentrequest, which does not bubble, cannot be canceled, and has no default action.
topLevelOrigin
attribute of e to the origin of the top level payee web page.
paymentRequestOrigin
attribute of e to the origin of the context where PaymentRequest was initialized.
paymentRequestId
attribute of e to the [[details]].id from the
PaymentRequest.
methodData
and modifiers
attributes of
e by executing the MethodData Population Algorithm and Modifiers Population Algorithm respectively.
total
attribute of e to a structured clone of the total field on the PaymentDetailsInit from the corresponding PaymentRequest.
PaymentInstrument
.
PaymentHandlerResponse
, reject the Promise that was created by PaymentRequest.show() with a
DOMException whose value "OperationError
".
An invoked payment handler may or may not need to display information about itself or request user input. Some examples of potential payment handler displays include:
A payment handler that requires visual display and user interaction, may call openWindow() to display a page to the user.
Since user agents know that this method is connected to the
PaymentRequestEvent
, they SHOULD render the window in a way that is consistent with the flow and not confusing to the user. The resulting window client is bound to the tab/window that initiated the
PaymentRequest. A single payment handler SHOULD NOT be allowed to open more than one client window using this method.
This algorithm resembles the Open Window Algorithm in the Service Workers specification.
Should we refer to the Service Workers specification instead of copying their steps?
PaymentRequestEvent
.
PaymentRequestEvent
.
InvalidStateError
".
about:blank
, return a
Promise rejected with a TypeError
.
SecurityError
.
InvalidAccessError
.
InvalidStateError
" and abort these steps.
SecurityError
".
PaymentRequestEvent
This section is non-normative.
This example shows how to write a service worker that listens to the
PaymentRequestEvent
. When a PaymentRequestEvent
is received, the service worker opens a window to interact with the user.
self.addEventListener('paymentrequest', function(e) {
e.respondWith(new Promise(function(resolve, reject) {
self.addEventListener('message', listener = function(e) {
self.removeEventListener('message', listener);
if (e.data.hasOwnProperty('name')) {
reject(e.data);
} else {
resolve(e.data);
}
});
e.openWindow("https://www.example.com/bobpay/pay")
.then(function(windowClient) {
windowClient.postMessage(e.data);
})
.catch(function(err) {
reject(err);
});
}));
});
The Web Payments Working Group plans to revisit these two examples.
Using the simple scheme described above, a trivial HTML page that is loaded into the payment handler window to implement the basic card scheme might look like the following:
<form id="form">
<table>
<tr><th>Cardholder Name:</th><td><input name="cardholderName"></td></tr>
<tr><th>Card Number:</th><td><input name="cardNumber"></td></tr>
<tr><th>Expiration Month:</th><td><input name="expiryMonth"></td></tr>
<tr><th>Expiration Year:</th><td><input name="expiryYear"></td></tr>
<tr><th>Security Code:</th><td><input name="cardSecurityCode"></td></tr>
<tr><th></th><td><input type="submit" value="Pay"></td></tr>
</table>
</form>
<script>
window.addEventListener("message", function(e) {
var form = document.getElementById("form");
/* Note: message sent from payment app is available in e.data */
form.onsubmit = function() {
/* See https://w3c.github.io/webpayments-methods-card/#basiccardresponse */
var basicCardResponse = {};
[ "cardholderName", "cardNumber","expiryMonth","expiryYear","cardSecurityCode"]
.forEach(function(field) {
basicCardResponse[field] = form.elements[field].value;
});
/* See https://w3c.github.io/payment-handler/#paymenthandlerresponse-dictionary */
var paymentAppResponse = {
methodName: "basic-card",
details: details
};
e.source.postMessage(paymentAppResponse);
window.close();
}
});
</script>
PaymentHandlerResponse
dictionary
dictionary PaymentHandlerResponse
{
DOMString methodName
;
object details
;
};
methodName
attribute
The payment method identifier for the payment method that the user selected to fulfil the transaction.
details
attribute
A JSON-serializable object that provides a payment method specific message used by the merchant to process the transaction and determine successful fund transfer.
The user agent receives a successful response from the payment handler through resolution of the Promise provided to the
respondWith() function of the corresponding PaymentRequestEvent
dictionary. The application is expected to resolve the Promise with a PaymentHandlerResponse
instance containing the payment response. In case of user cancellation or error, the application may signal failure by rejecting the Promise.
If the Promise is rejected, the user agent MUST run the payment app failure algorithm. The exact details of this algorithm are left to implementers. Acceptable behaviors include, but are not limited to:
If the Promise is successfully resolved, the user agent MUST run the user accepts the payment request algorithm as defined in [ payment-request], replacing steps 6 and 7 with these steps or their equivalent:
PaymentHandlerResponse
instance used to resolve the
PaymentRequestEvent
.respondWith() Promise.
methodName
is not present or not set to one of the values from
PaymentRequestEvent
.methodData
, run the payment
app failure algorithm and terminate these steps.
methodName
and assign it to
response.methodName
.
The following example shows how to respond to a payment request:
paymentRequestEvent.respondWith(new Promise(function(accept,reject) {
/* ... processing may occur here ... */
accept({
methodName: "basic-card",
details: {
cardHolderName: "John Smith",
cardNumber: "1232343451234",
expiryMonth: "12",
expiryYear : "2020",
cardSecurityCode: "123"
}
});
});
[payment-request] defines an ID that parties in the ecosystem (including payment app providers and payees) may use for reconciliation after network or other failures.
The Web Payments Working Group is also discussing Payment App authenticity; see the (draft) Payment Method Manifest.
partial interfaceServiceWorkerRegistration
{ readonly attribute PaymentManagerpaymentManager
; }; [SecureContext] interface PaymentManager { [SameObject] readonly attributePaymentInstruments
instruments
; [SameObject] readonly attributePaymentWallets
wallets
; [Exposed=Window] static Promise<PermissionState> requestPermission(); }; interfacePaymentInstruments
{ Promise<boolean> delete(DOMString instrumentKey); Promise<PaymentInstrument
> get(DOMString instrumentKey); Promise<sequence<DOMString>> keys(); Promise<boolean> has(DOMString instrumentKey); Promise<void> set(DOMString instrumentKey,PaymentInstrument
details); Promise<void> clear(); }; dictionaryPaymentInstrument
{ required DOMStringname
; sequence<ImageObject
>icons
; sequence<DOMString>enabledMethods
; objectcapabilities
; }; interfacePaymentWallets
{ Promise<boolean> delete(DOMString walletKey); Promise<PaymentWallet
> get(DOMString walletKey); Promise<sequence<DOMString>> keys(); Promise<boolean> has(DOMString walletKey); Promise<void> set(DOMString walletKey,PaymentWallet
details); Promise<void> clear(); }; dictionaryPaymentWallet
{ required DOMStringname
; sequence<ImageObject
>icons
; required sequence<DOMString>instrumentKeys
; }; dictionaryImageObject
{ required USVStringsrc
; DOMStringsizes
; DOMStringtype
; }; partial interfaceServiceWorkerGlobalScope
{ attribute EventHandleronpaymentrequest
; }; [Constructor(DOMString type,PaymentRequestEventInit
eventInitDict), Exposed=ServiceWorker] interfacePaymentRequestEvent
: ExtendableEvent { readonly attribute USVStringtopLevelOrigin
; readonly attribute USVStringpaymentRequestOrigin
; readonly attribute DOMStringpaymentRequestId
; readonly attribute FrozenArray<PaymentMethodData>methodData
; readonly attribute objecttotal
; readonly attribute FrozenArray<PaymentDetailsModifier>modifiers
; readonly attribute DOMStringinstrumentKey
; Promise<WindowClient>openWindow
(USVString url); voidrespondWith
(Promise<PaymentHandlerResponse
> handlerResponse); }; dictionaryPaymentRequestEventInit
: ExtendableEventInit { USVStringtopLevelOrigin
; USVStringpaymentRequestOrigin
; DOMStringpaymentRequestId
; sequence<PaymentMethodData>methodData
; PaymentItemtotal
; sequence<PaymentDetailsModifier>modifiers
; DOMStringinstrumentKey
; }; dictionaryPaymentHandlerResponse
{ DOMStringmethodName
; objectdetails
; };