diff --git a/.vscode/settings.json b/.vscode/settings.json index 8a1e359..5ad0e97 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -9,5 +9,6 @@ "editor.formatOnSave": true, "editor.codeActionsOnSave": { "source.fixAll.eslint": true - } + }, + "angular.enable-strict-mode-prompt": false } diff --git a/README.md b/README.md index 4582552..70d4dc6 100644 --- a/README.md +++ b/README.md @@ -9,6 +9,41 @@ See Google Pay in action: This repository contains Google Pay button implementations for compatible with popular website frameworks even easier. +## Web + +This is a bare bones, plain vanilla JavaScript implementation of the Google Pay button. These examples are designed +to launch into a Project IDX Workspace, ready to run. + +- [Example code basic](./examples/html/gpay-web-101/) + + + + Open in IDX + + + +- [Example code advanced](./examples/html/gpay-web-201/) + + + + Open in IDX + + + ## Web component [![npm version](https://badge.fury.io/js/%40google-pay%2Fbutton-element.svg)][npm-element] diff --git a/examples/html/gpay-web-101/README.md b/examples/html/gpay-web-101/README.md new file mode 100644 index 0000000..f5d9c84 --- /dev/null +++ b/examples/html/gpay-web-101/README.md @@ -0,0 +1,58 @@ +# Google Pay for Web + + + + + + + Open in IDX + + + +## About + +This project is a minimum viable integration of Google Pay for Web using HTML +and JavaScript. For a more complete integration, refer to [`gpay-web-201/`][17] +project template. + +## Learning + +To learn about the code in this project template, follow to the +[Google Pay API for Web 101: Basic][10] codelab. Learning paths help you +get the most of your integration by taking you through a guided developer +journey from start to finish. + +## Create a merchant account + +While a merchant ID isn't required for testing your integration in a `TEST` +enviornment, you will need one when you deploy to a `PRODUCTION` environment. +Register with the [Google Pay & Wallet Console][15] to receive your merchant ID. +It's quick and easy! Get yours now. + +## Community + +- Join the conversation in the [#payments channel on Discord][12] +- Follow [@GooglePayDevs on X][13] for announcements about more content like this +- Subscribe to the [Google for Developers][14] YouTube channel + +## Support + +- Question about this template? Ask in the [discussions][16] section of the + Google Pay button repo. +- For assistance with your implementation, create a support ticket from the + [Google Pay & Wallet Console][15]. + +[10]: https://codelabs.developers.google.com/codelabs/gpay-web-101 +[12]: https://goo.gle/payments-dev-community +[13]: https://x.com/GooglePayDevs +[14]: https://goo.gle/developers +[15]: https://goo.gle/3Cg8KxJ +[16]: https://github.com/google-pay/google-pay-button/discussions +[17]: https://github.com/google-pay/google-pay-button/tree/main/examples/html/gpay-web-201/ diff --git a/examples/html/gpay-web-101/dev.nix b/examples/html/gpay-web-101/dev.nix new file mode 100644 index 0000000..007800b --- /dev/null +++ b/examples/html/gpay-web-101/dev.nix @@ -0,0 +1,50 @@ +/* + Copyright 2024 Google LLC + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + https://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. + */ + +# To learn more about how to use Nix to configure your environment +# see: https://developers.google.com/idx/guides/customize-idx-env +{ pkgs, ... }: { + # Which nixpkgs channel to use. + channel = "stable-24.05"; # or "unstable" + + # Use https://search.nixos.org/packages to find packages + packages = [ + pkgs.python311 + ]; + + # Sets environment variables in the workspace + env = {}; + idx = { + workspace = { + onCreate = { + default.openFiles = [ + "README.md" + "index.html" + ]; + }; + }; + + previews = { + enable = true; + previews = { + web = { + command = ["python3" "-m" "http.server" "$PORT" "--bind" "0.0.0.0"]; + manager = "web"; + }; + }; + }; + }; +} diff --git a/examples/html/gpay-web-101/dev/index.html b/examples/html/gpay-web-101/dev/index.html new file mode 100644 index 0000000..98dfde9 --- /dev/null +++ b/examples/html/gpay-web-101/dev/index.html @@ -0,0 +1,32 @@ + + + + + + + + Google Pay API for Web + + + +
+

Transaction info and errors will be logged to the console.

+ + + + + diff --git a/examples/html/gpay-web-101/dev/main.js b/examples/html/gpay-web-101/dev/main.js new file mode 100644 index 0000000..f2948ad --- /dev/null +++ b/examples/html/gpay-web-101/dev/main.js @@ -0,0 +1,220 @@ +/** + * Copyright 2024 Google LLC + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +//============================================================================= +// Configuration +//============================================================================= + +// The DOM element that the Google Pay button will be rendered into +const GPAY_BUTTON_CONTAINER_ID = 'gpay-container'; + +// Update the `merchantId` and `merchantName` properties with your own values. +// These fields are optional when the environment is `TEST`. +// Get your merchant Id at https://goo.gle/3Cg8KxJ +const merchantInfo = { + merchantId: '12345678901234567890', + merchantName: 'Example Merchant', +}; + +/** + * This is the base configuration for all Google Pay requests. This + * configuration will be cloned, modified, and used for all Google Pay requests. + * + * @see {@link https://developers.google.com/pay/api/web/guides/test-and-deploy/integration-checklist} + * @see {@link https://developers.google.com/pay/api/web/reference/request-objects} + * @see {@link https://developers.google.com/pay/api/web/reference/request-objects#gateway} + * @see {@link https://developers.google.com/pay/api/web/reference/request-objects#MerchantInfo} + */ +const baseGooglePayRequest = { + apiVersion: 2, + apiVersionMinor: 0, + allowedPaymentMethods: [ + { + type: 'CARD', + parameters: { + allowedAuthMethods: ['PAN_ONLY', 'CRYPTOGRAM_3DS'], + allowedCardNetworks: ['AMEX', 'DISCOVER', 'INTERAC', 'JCB', 'MASTERCARD', 'VISA'], + }, + tokenizationSpecification: { + type: 'PAYMENT_GATEWAY', + parameters: { + gateway: 'example', + gatewayMerchantId: 'exampleGatewayMerchantId', + }, + }, + }, + ], + merchantInfo, +}; + +// Prevent accidental edits to the base configuration. Mutations will be +// handled by cloning the config using deepCopy() and modifying the copy. +Object.freeze(baseGooglePayRequest); + +//============================================================================= +// Google payments client singleton +//============================================================================= + +/** + * A variable to store the Google Payments Client instance. + * Initialized to null to indicate it hasn't been created yet. + */ +let paymentsClient = null; + +/** + * Gets an instance of the Google Payments Client. + * + * This function ensures that only one instance of the Google Payments Client + * is created and reused throughout the application. It lazily initializes + * the client if it hasn't been created yet. + * + * @see {@link https://developers.google.com/pay/api/web/reference/client#PaymentsClient} + * @return {google.payments.api.PaymentsClient} Google Payments Client instance. + */ +function getGooglePaymentsClient() { + // Check if the paymentsClient has already been initialized. + if (paymentsClient === null) { + // If not, create a new instance of the Google Payments Client. + paymentsClient = new google.payments.api.PaymentsClient({ + // Set the environment for the client ('TEST' or 'PRODUCTION'). + // `TEST` is default. + environment: 'TEST', + // Add the merchant information (optional) + merchantInfo, + }); + } + + return paymentsClient; +} + +//============================================================================= +// Helpers +//============================================================================= + +/** + * Creates a deep copy of an object. + * + * This function uses JSON serialization and deserialization to create a deep + * copy of the provided object. It's a convenient way to clone objects without + * worrying about shared references. + * + * @param {Object} obj - The object to be copied. + * @returns {Object} A deep copy of the original object. + */ +const deepCopy = obj => JSON.parse(JSON.stringify(obj)); + +/** + * Renders the Google Pay button to the DOM. + * + * This function creates a Google Pay button using the Google Pay API and adds + * it to the container element specified by `GPAY_BUTTON_CONTAINER_ID`. + * When clicked, button triggers the `onGooglePaymentButtonClicked` handler. + * + * @see {@link https://developers.google.com/pay/api/web/reference/client#createButton} + * @returns {void} + */ +function renderGooglePayButton() { + // Create a Google Pay button using the PaymentsClient. + const button = getGooglePaymentsClient().createButton({ + // Set the click handler for the button to the onGooglePaymentButtonClicked + onClick: onGooglePaymentButtonClicked, + }); + // Add the Google Pay button to the container element on the page. + document.getElementById(GPAY_BUTTON_CONTAINER_ID).appendChild(button); +} + +//============================================================================= +// Event Handlers +//============================================================================= + +/** + * Google Pay API loaded handler + * + * This function will be called by the script tag in index.html when the pay.js + * script has finished loading. Once the script is loaded, it will first check + * to see if the consumer is ready to pay with Google Pay. If they are ready, + * the next thing it does is add the Google Pay button to the page. Otherwise, + * it logs an error to the console. + * + * @see {@link https://developers.google.com/pay/api/web/reference/client#isReadyToPay} + * @returns {void} + */ +function onGooglePayLoaded() { + // Create a deep copy of the base Google Pay request object. + // This ensures that any modifications made to the request object + // do not affect the original base request. + const req = deepCopy(baseGooglePayRequest); + + // Get an instance of the Google Payments Client. + getGooglePaymentsClient() + // Check if the user is ready to pay with Google Pay. + .isReadyToPay(req) + // Handle the response from the isReadyToPay() method. + .then(function (res) { + // If the user is ready to pay with Google Pay... + if (res.result) { + // Render the Google Pay button to the page. + renderGooglePayButton(); + } else { + // If the user is not ready to pay with Google Pay, log + // an error to the console. + console.log('Google Pay is not ready for this user.'); + } + }) + // Handle any errors that occur during the process. + .catch(console.error); +} + +/** + * Google Pay button click handler + * + * @see {@link https://developers.google.com/pay/api/web/reference/client#loadPaymentData} + * @see {@link https://developers.google.com/pay/api/web/reference/response-objects#PaymentMethodTokenizationData} + * @see {@link https://developers.google.com/pay/api/web/reference/request-objects#TransactionInfo} + * @returns {void} + */ +function onGooglePaymentButtonClicked() { + // Create a new request data object for this request + const req = { + ...deepCopy(baseGooglePayRequest), + transactionInfo: { + countryCode: 'US', + currencyCode: 'USD', + totalPriceStatus: 'FINAL', + totalPrice: (Math.random() * 999 + 1).toFixed(2), + }, + }; + + // Write the data to console for debugging + console.log('onGooglePaymentButtonClicked', req); + + // Get an instance of the Google Payments Client. + getGooglePaymentsClient() + // Load the payment data in console for the transaction. + .loadPaymentData(req) + // If the payment is successful, process the payment + .then(function (res) { + // show returned data for debugging + console.log(res); + // @todo pass payment token to your gateway to process payment + // @note DO NOT save the payment credentials for future transactions, + // unless they're used for merchant-initiated transactions with user + // consent in place. + paymentToken = res.paymentMethodData.tokenizationData.token; + }) + // If there is an error, log it to the console. + .catch(console.error); +} diff --git a/examples/html/gpay-web-101/idx-template.json b/examples/html/gpay-web-101/idx-template.json new file mode 100644 index 0000000..a9d0405 --- /dev/null +++ b/examples/html/gpay-web-101/idx-template.json @@ -0,0 +1,8 @@ +{ + "name": "Google Pay API for Web", + "description": "Minimum viable web integration of the Google Pay API on Web.", + "categories": ["Web app", "Misc"], + "icon": "https://www.gstatic.com/images/branding/productlogos/google_pay_round/v6/192px.svg", + "publisher": "Google Pay DevRel", + "website": "https://codelabs.developers.google.com/codelabs/gpay-web-101" +} \ No newline at end of file diff --git a/examples/html/gpay-web-101/idx-template.nix b/examples/html/gpay-web-101/idx-template.nix new file mode 100644 index 0000000..76d6c0f --- /dev/null +++ b/examples/html/gpay-web-101/idx-template.nix @@ -0,0 +1,26 @@ +/* + Copyright 2024 Google LLC + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + https://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. + */ + +{ pkgs, ... }: { + bootstrap = '' + mkdir "$out" + mkdir -p "$out/.idx/" + cp -rf ${./dev.nix} "$out/.idx/dev.nix" + cp -f ${./README.md} "$out/README.md" + shopt -s dotglob; cp -r ${./dev}/* "$out" + chmod -R +w "$out" + ''; +} diff --git a/examples/html/gpay-web-201/README.md b/examples/html/gpay-web-201/README.md new file mode 100644 index 0000000..358a1e9 --- /dev/null +++ b/examples/html/gpay-web-201/README.md @@ -0,0 +1,58 @@ +# Google Pay for Web + + + + + + + Open in IDX + + + +## About + +This project is a more complete integration of Google Pay API for Web using +HTML and JavaScript. For a more bare bones integration, refer to +[`gpay-web-101/`][17] project template. + +## Learning + +To learn about the code in this project template, follow to the +[Google Pay API for Web 201: Advance][10] codelab. Learning paths help you +get the most of your integration by taking you through a guided developer +journey from start to finish. + +## Create a merchant account + +While a merchant ID isn't required for testing your integration in a `TEST` +enviornment, you will need one when you deploy to a `PRODUCTION` environment. +Register with the [Google Pay & Wallet Console][15] to receive your merchant ID. +It's quick and easy! Get yours now. + +## Community + +- Join the conversation in the [#payments channel on Discord][12] +- Follow [@GooglePayDevs on X][13] for announcements about more content like this +- Subscribe to the [Google for Developers][14] YouTube channel + +## Support + +- Question about this template? Ask in the [discussions][16] section of the + Google Pay button repo. +- For assistance with your implementation, create a support ticket from the + [Google Pay & Wallet Console][15]. + +[10]: https://codelabs.developers.google.com/codelabs/gpay-web-201 +[12]: https://goo.gle/payments-dev-community +[13]: https://x.com/GooglePayDevs +[14]: https://goo.gle/developers +[15]: https://goo.gle/3Cmv497 +[16]: https://github.com/google-pay/google-pay-button/discussions +[17]: https://github.com/google-pay/google-pay-button/tree/main/examples/html/gpay-web-101/ diff --git a/examples/html/gpay-web-201/dev.nix b/examples/html/gpay-web-201/dev.nix new file mode 100644 index 0000000..007800b --- /dev/null +++ b/examples/html/gpay-web-201/dev.nix @@ -0,0 +1,50 @@ +/* + Copyright 2024 Google LLC + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + https://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. + */ + +# To learn more about how to use Nix to configure your environment +# see: https://developers.google.com/idx/guides/customize-idx-env +{ pkgs, ... }: { + # Which nixpkgs channel to use. + channel = "stable-24.05"; # or "unstable" + + # Use https://search.nixos.org/packages to find packages + packages = [ + pkgs.python311 + ]; + + # Sets environment variables in the workspace + env = {}; + idx = { + workspace = { + onCreate = { + default.openFiles = [ + "README.md" + "index.html" + ]; + }; + }; + + previews = { + enable = true; + previews = { + web = { + command = ["python3" "-m" "http.server" "$PORT" "--bind" "0.0.0.0"]; + manager = "web"; + }; + }; + }; + }; +} diff --git a/examples/html/gpay-web-201/dev/index.html b/examples/html/gpay-web-201/dev/index.html new file mode 100644 index 0000000..0291d2f --- /dev/null +++ b/examples/html/gpay-web-201/dev/index.html @@ -0,0 +1,33 @@ + + + + + + + + + Google Pay API for Web + + + +
+

Transaction info and errors will be logged to the console.

+ + + + + diff --git a/examples/html/gpay-web-201/dev/main.js b/examples/html/gpay-web-201/dev/main.js new file mode 100644 index 0000000..8fe4a13 --- /dev/null +++ b/examples/html/gpay-web-201/dev/main.js @@ -0,0 +1,340 @@ +/** + * Copyright 2024 Google LLC + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +//============================================================================= +// Configuration +//============================================================================= + +// The DOM element that the Google Pay button will be rendered into +const GPAY_BUTTON_CONTAINER_ID = 'gpay-container'; + +// Update the `merchantId` and `merchantName` properties with your own values. +// These fields are optional when the environment is `TEST`. +// Get your merchant Id at https://goo.gle/3Cmv497 +const merchantInfo = { + merchantId: '12345678901234567890', + merchantName: 'Example Merchant', +}; + +/** + * This is the base configuration for all Google Pay requests. This + * configuration will be cloned, modified, and used for all Google Pay requests. + * + * @see {@link https://developers.google.com/pay/api/web/guides/test-and-deploy/integration-checklist} + * @see {@link https://developers.google.com/pay/api/web/reference/request-objects} + * @see {@link https://developers.google.com/pay/api/web/reference/request-objects#gateway} + * @see {@link https://developers.google.com/pay/api/web/reference/request-objects#MerchantInfo} + */ +const baseGooglePayRequest = { + apiVersion: 2, + apiVersionMinor: 0, + allowedPaymentMethods: [ + { + type: 'CARD', + parameters: { + allowedAuthMethods: ['PAN_ONLY', 'CRYPTOGRAM_3DS'], + allowedCardNetworks: ['AMEX', 'DISCOVER', 'INTERAC', 'JCB', 'MASTERCARD', 'VISA'], + }, + tokenizationSpecification: { + type: 'PAYMENT_GATEWAY', + parameters: { + gateway: 'example', + gatewayMerchantId: 'exampleGatewayMerchantId', + }, + }, + }, + ], + merchantInfo, +}; + +// Prevent accidental edits to the base configuration. Mutations will be +// handled by cloning the config using deepCopy() and modifying the copy. +Object.freeze(baseGooglePayRequest); + +//============================================================================= +// Google payments client singleton +//============================================================================= + +/** + * A variable to store the Google Payments Client instance. + * Initialized to null to indicate it hasn't been created yet. + */ +let paymentsClient = null; + +/** + * Gets an instance of the Google Payments Client. + * + * This function ensures that only one instance of the Google Payments Client + * is created and reused throughout the application. It lazily initializes + * the client if it hasn't been created yet. + * + * @see {@link https://developers.google.com/pay/api/web/reference/client#PaymentsClient} + * @return {google.payments.api.PaymentsClient} Google Payments Client instance. + */ +function getGooglePaymentsClient() { + // Check if the paymentsClient has already been initialized. + if (paymentsClient === null) { + // If not, create a new instance of the Google Payments Client. + paymentsClient = new google.payments.api.PaymentsClient({ + // Set the environment for the client ('TEST' or 'PRODUCTION'). + // `TEST` is default. + environment: 'TEST', + // Add the merchant information (optional) + merchantInfo, + paymentDataCallbacks: { + onPaymentAuthorized: onPaymentAuthorized, + onPaymentDataChanged: onPaymentDataChanged, + }, + }); + } + + return paymentsClient; +} + +//============================================================================= +// Helpers +//============================================================================= + +/** + * Creates a deep copy of an object. + * + * This function uses JSON serialization and deserialization to create a deep + * copy of the provided object. It's a convenient way to clone objects without + * worrying about shared references. + * + * @param {Object} obj - The object to be copied. + * @returns {Object} A deep copy of the original object. + */ +const deepCopy = obj => JSON.parse(JSON.stringify(obj)); + +/** + * Renders the Google Pay button to the DOM. + * + * This function creates a Google Pay button using the Google Pay API and adds + * it to the container element specified by `GPAY_BUTTON_CONTAINER_ID`. + * When clicked, button triggers the `onGooglePaymentButtonClicked` handler. + * + * @see {@link https://developers.google.com/pay/api/web/reference/client#createButton} + * @returns {void} + */ +function renderGooglePayButton() { + // Create a Google Pay button using the PaymentsClient. + const button = getGooglePaymentsClient().createButton({ + // Set the click handler for the button to the onGooglePaymentButtonClicked + onClick: onGooglePaymentButtonClicked, + // Set the allowed payment methods for the button. + allowedPaymentMethods: baseGooglePayRequest.allowedPaymentMethods, + }); + // Add the Google Pay button to the container element on the page. + document.getElementById(GPAY_BUTTON_CONTAINER_ID).appendChild(button); +} + +/** + * Prefetches Google Pay payment data to improve the payment flow performance. + * + * This function creates a payment data request object and uses the Google Pay client + * to prefetch payment data. This can help speed up the payment process when the user + * actually initiates a transaction. The prefetched data is cached and can be used + * for subsequent payment requests. + * + * @see {@link https://developers.google.com/pay/api/web/reference/request-objects#TransactionInfo} + * @returns {void} + */ +function prefetchGooglePaymentData() { + // Create a deep copy of the base Google Pay request object. + // This ensures that the original request object is not modified. + const req = deepCopy(baseGooglePayRequest); + + // Set the transactionInfo property on the request object. + // This is required for prefetching, even though the values + // are not used for caching. + req.transactionInfo = { + totalPriceStatus: 'NOT_CURRENTLY_KNOWN', + currencyCode: 'USD', + }; + + // Get an instance of the Google Pay client and use it to + // prefetch the payment data. + getGooglePaymentsClient().prefetchPaymentData(req); +} + +//============================================================================= +// Event Handlers +//============================================================================= + +/** + * Google Pay API loaded handler + * + * This function will be called by the script tag in index.html when the pay.js + * script has finished loading. Once the script is loaded, it will first check + * to see if the consumer is ready to pay with Google Pay. If they are ready, + * the next thing it does is add the Google Pay button to the page. Otherwise, + * it logs an error to the console. + * + * @see {@link https://developers.google.com/pay/api/web/reference/client#isReadyToPay} + * @returns {void} + */ +function onGooglePayLoaded() { + // Create a deep copy of the base Google Pay request object. + // This ensures that any modifications made to the request object + // do not affect the original base request. + const req = deepCopy(baseGooglePayRequest); + + // Get an instance of the Google Payments Client. + getGooglePaymentsClient() + // Check if the user is ready to pay with Google Pay. + .isReadyToPay(req) + // Handle the response from the isReadyToPay() method. + .then(function (res) { + // If the user is ready to pay with Google Pay... + if (res.result) { + // Render the Google Pay button to the page. + renderGooglePayButton(); + // Prefetch the payment data to improve performance. + prefetchGooglePaymentData(); + } else { + // If the user is not ready to pay with Google Pay, log + // an error to the console. + console.log('Google Pay is not ready for this user.'); + } + }) + // Handle any errors that occur during the process. + .catch(console.error); +} + +/** + * Google Pay button click handler + * + * @see {@link https://developers.google.com/pay/api/web/reference/client#loadPaymentData} + * @see {@link https://developers.google.com/pay/api/web/reference/response-objects#PaymentMethodTokenizationData} + * @see {@link https://developers.google.com/pay/api/web/reference/request-objects#TransactionInfo} + * @returns {void} + */ +function onGooglePaymentButtonClicked() { + // Create a new request data object for this request + const req = { + ...deepCopy(baseGooglePayRequest), + transactionInfo: { + countryCode: 'US', + currencyCode: 'USD', + totalPriceStatus: 'FINAL', + totalPrice: (Math.random() * 999 + 1).toFixed(2), + }, + callbackIntents: ['PAYMENT_AUTHORIZATION', 'SHIPPING_ADDRESS', 'SHIPPING_OPTION', 'OFFER'], + shippingAddressRequired: true, + shippingOptionRequired: true, + shippingOptionParameters: { + defaultSelectedOptionId: 'shipping-001', + shippingOptions: [ + { + id: 'shipping-001', + label: '$0.00: Free shipping', + description: 'Free Shipping delivered in 5 business days.', + }, + { + id: 'shipping-002', + label: '$1.99: Standard shipping', + description: 'Standard shipping delivered in 3 business days.', + }, + { + id: 'shipping-003', + label: '$1000: Express shipping', + description: 'Express shipping delivered in 1 business day.', + }, + ], + }, + }; + + // Write the data to console for debugging + console.log('onGooglePaymentButtonClicked', req); + + // Get an instance of the Google Payments Client. + getGooglePaymentsClient() + // Load the payment data in console for the transaction. + .loadPaymentData(req) + // If the payment is successful, process the payment + .then(function (res) { + // show returned data for debugging + console.log(res); + // @todo pass payment token to your gateway to process payment + // @note DO NOT save the payment credentials for future transactions, + // unless they're used for merchant-initiated transactions with user + // consent in place. + paymentToken = res.paymentMethodData.tokenizationData.token; + }) + // If there is an error, log it to the console. + .catch(console.error); +} + +function onPaymentAuthorized(paymentData) { + return new Promise(function (resolve, reject) { + // Write the data to console for debugging + console.log('onPaymentAuthorized', paymentData); + + // Do something here to pass token to your gateway + + // To simulate the payment processing, there is a 70% chance of success + const paymentAuthorizationResult = + Math.random() > 0.3 + ? { transactionState: 'SUCCESS' } + : { + transactionState: 'ERROR', + error: { + intent: 'PAYMENT_AUTHORIZATION', + message: 'Insufficient funds', + reason: 'PAYMENT_DATA_INVALID', + }, + }; + + resolve(paymentAuthorizationResult); + }); +} + +function onPaymentDataChanged(intermediatePaymentData) { + return new Promise(function (resolve, reject) { + let paymentDataRequestUpdate = {}; + + // Write the data to console for debugging + console.log('onPaymentDataChanged', intermediatePaymentData); + + switch (intermediatePaymentData.callbackTrigger) { + case 'INITIALIZE': + // Handle initialize + break; + case 'SHIPPING_ADDRESS': + // Read intermediatePaymentData.transactionInfo + // Read intermediatePaymentData.shippingAddress + // Update paymentDataRequestUpdate.newTransactionInfo + break; + case 'SHIPPING_OPTION': + // Read intermediatePaymentData.transactionInfo + // Read intermediatePaymentData.shippingOptionData + // Update paymentDataRequestUpdate.newTransactionInfo + // Update paymentDataRequestUpdate.newShippingOptionParameters + break; + case 'OFFER': + // Read intermediatePaymentData.offerData + // Read intermediatePaymentData.transactionInfo + // Update paymentDataRequestUpdate.newTransactionInfo + // Update paymentDataRequestUpdate.newOfferInfo + break; + default: + // Update paymentDataRequestUpdate.error + } + + resolve(paymentDataRequestUpdate); + }); +} diff --git a/examples/html/gpay-web-201/idx-template.json b/examples/html/gpay-web-201/idx-template.json new file mode 100644 index 0000000..92713b9 --- /dev/null +++ b/examples/html/gpay-web-201/idx-template.json @@ -0,0 +1,8 @@ +{ + "name": "Google Pay API for Web complete", + "description": "A more complete integration of the Google Pay API on Web.", + "categories": ["Web app", "Misc"], + "icon": "https://www.gstatic.com/images/branding/productlogos/google_pay_round/v6/192px.svg", + "publisher": "Google Pay DevRel", + "website": "https://codelabs.developers.google.com/codelabs/gpay-web-201" +} diff --git a/examples/html/gpay-web-201/idx-template.nix b/examples/html/gpay-web-201/idx-template.nix new file mode 100644 index 0000000..4aed0bd --- /dev/null +++ b/examples/html/gpay-web-201/idx-template.nix @@ -0,0 +1,26 @@ +/* + Copyright 2024 Google LLC + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + https://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. + */ + +{ pkgs, ... }: { + bootstrap = '' + mkdir "$out" + mkdir -p "$out/.idx/" + cp -rf ${./dev.nix} "$out/.idx/dev.nix" + cp -f ${./README.md} "$out/README.md" + shopt -s dotglob; cp -r ${./dev}/* "$out" + chmod -R +w "$out" + ''; +}