forked from mirrors/gecko-dev
Differential Revision: https://phabricator.services.mozilla.com/D7473 --HG-- extra : moz-landing-system : lando
417 lines
15 KiB
JavaScript
417 lines
15 KiB
JavaScript
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
import PaymentStateSubscriberMixin from "../mixins/PaymentStateSubscriberMixin.js";
|
|
import paymentRequest from "../paymentRequest.js";
|
|
|
|
import "../components/currency-amount.js";
|
|
import "../components/payment-request-page.js";
|
|
import "../components/accepted-cards.js";
|
|
import "./address-picker.js";
|
|
import "./address-form.js";
|
|
import "./basic-card-form.js";
|
|
import "./completion-error-page.js";
|
|
import "./order-details.js";
|
|
import "./payment-method-picker.js";
|
|
import "./shipping-option-picker.js";
|
|
|
|
/* import-globals-from ../unprivileged-fallbacks.js */
|
|
|
|
/**
|
|
* <payment-dialog></payment-dialog>
|
|
*/
|
|
|
|
export default class PaymentDialog extends PaymentStateSubscriberMixin(HTMLElement) {
|
|
constructor() {
|
|
super();
|
|
this._template = document.getElementById("payment-dialog-template");
|
|
this._cachedState = {};
|
|
}
|
|
|
|
connectedCallback() {
|
|
let contents = document.importNode(this._template.content, true);
|
|
this._hostNameEl = contents.querySelector("#host-name");
|
|
|
|
this._cancelButton = contents.querySelector("#cancel");
|
|
this._cancelButton.addEventListener("click", this.cancelRequest);
|
|
|
|
this._payButton = contents.querySelector("#pay");
|
|
this._payButton.addEventListener("click", this);
|
|
|
|
this._viewAllButton = contents.querySelector("#view-all");
|
|
this._viewAllButton.addEventListener("click", this);
|
|
|
|
this._mainContainer = contents.getElementById("main-container");
|
|
this._orderDetailsOverlay = contents.querySelector("#order-details-overlay");
|
|
|
|
this._shippingAddressPicker = contents.querySelector("address-picker.shipping-related");
|
|
this._shippingOptionPicker = contents.querySelector("shipping-option-picker");
|
|
this._shippingRelatedEls = contents.querySelectorAll(".shipping-related");
|
|
this._payerRelatedEls = contents.querySelectorAll(".payer-related");
|
|
this._payerAddressPicker = contents.querySelector("address-picker.payer-related");
|
|
this._paymentMethodPicker = contents.querySelector("payment-method-picker");
|
|
this._acceptedCardsList = contents.querySelector("accepted-cards");
|
|
this._manageText = contents.querySelector(".manage-text");
|
|
this._manageText.addEventListener("click", this);
|
|
|
|
this._header = contents.querySelector("header");
|
|
|
|
this._errorText = contents.querySelector("header > .page-error");
|
|
|
|
this._disabledOverlay = contents.getElementById("disabled-overlay");
|
|
|
|
this.appendChild(contents);
|
|
|
|
super.connectedCallback();
|
|
}
|
|
|
|
disconnectedCallback() {
|
|
this._cancelButton.removeEventListener("click", this.cancelRequest);
|
|
this._payButton.removeEventListener("click", this.pay);
|
|
this._viewAllButton.removeEventListener("click", this);
|
|
super.disconnectedCallback();
|
|
}
|
|
|
|
handleEvent(event) {
|
|
if (event.type == "click") {
|
|
switch (event.currentTarget) {
|
|
case this._viewAllButton:
|
|
let orderDetailsShowing = !this.requestStore.getState().orderDetailsShowing;
|
|
this.requestStore.setState({ orderDetailsShowing });
|
|
break;
|
|
case this._payButton:
|
|
this.pay();
|
|
break;
|
|
case this._manageText:
|
|
if (event.target instanceof HTMLAnchorElement) {
|
|
this.openPreferences(event);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
openPreferences(event) {
|
|
paymentRequest.openPreferences();
|
|
event.preventDefault();
|
|
}
|
|
|
|
cancelRequest() {
|
|
paymentRequest.cancel();
|
|
}
|
|
|
|
pay() {
|
|
let {
|
|
selectedPayerAddress,
|
|
selectedPaymentCard,
|
|
selectedPaymentCardSecurityCode,
|
|
} = this.requestStore.getState();
|
|
|
|
paymentRequest.pay({
|
|
selectedPayerAddressGUID: selectedPayerAddress,
|
|
selectedPaymentCardGUID: selectedPaymentCard,
|
|
selectedPaymentCardSecurityCode,
|
|
});
|
|
}
|
|
|
|
changeShippingAddress(shippingAddressGUID) {
|
|
paymentRequest.changeShippingAddress({
|
|
shippingAddressGUID,
|
|
});
|
|
}
|
|
|
|
changeShippingOption(optionID) {
|
|
paymentRequest.changeShippingOption({
|
|
optionID,
|
|
});
|
|
}
|
|
|
|
_isPayerRequested(paymentOptions) {
|
|
return paymentOptions.requestPayerName ||
|
|
paymentOptions.requestPayerEmail ||
|
|
paymentOptions.requestPayerPhone;
|
|
}
|
|
|
|
_getAdditionalDisplayItems(state) {
|
|
let methodId = state.selectedPaymentCard;
|
|
let modifier = paymentRequest.getModifierForPaymentMethod(state, methodId);
|
|
if (modifier && modifier.additionalDisplayItems) {
|
|
return modifier.additionalDisplayItems;
|
|
}
|
|
return [];
|
|
}
|
|
|
|
_updateCompleteStatus(state) {
|
|
let {completeStatus} = state.request;
|
|
switch (completeStatus) {
|
|
case "fail":
|
|
case "timeout":
|
|
case "unknown":
|
|
state.page = {
|
|
id: `completion-${completeStatus}-error`,
|
|
};
|
|
state.changesPrevented = false;
|
|
break;
|
|
case "": {
|
|
// When we get a DOM update for an updateWith() or retry() the completeStatus
|
|
// is "" when we need to show non-final screens. Don't set the page as we
|
|
// may be on a form instead of payment-summary
|
|
state.changesPrevented = false;
|
|
break;
|
|
}
|
|
}
|
|
return state;
|
|
}
|
|
|
|
/**
|
|
* Set some state from the privileged parent process.
|
|
* Other elements that need to set state should use their own `this.requestStore.setState`
|
|
* method provided by the `PaymentStateSubscriberMixin`.
|
|
*
|
|
* @param {object} state - See `PaymentsStore.setState`
|
|
*/
|
|
setStateFromParent(state) {
|
|
let oldAddresses = paymentRequest.getAddresses(this.requestStore.getState());
|
|
if (state.request) {
|
|
state = this._updateCompleteStatus(state);
|
|
}
|
|
this.requestStore.setState(state);
|
|
|
|
// Check if any foreign-key constraints were invalidated.
|
|
state = this.requestStore.getState();
|
|
let {
|
|
selectedPayerAddress,
|
|
selectedPaymentCard,
|
|
selectedShippingAddress,
|
|
selectedShippingOption,
|
|
} = state;
|
|
let addresses = paymentRequest.getAddresses(state);
|
|
let shippingOptions = state.request.paymentDetails.shippingOptions;
|
|
let shippingAddress = selectedShippingAddress && addresses[selectedShippingAddress];
|
|
let oldShippingAddress = selectedShippingAddress &&
|
|
oldAddresses[selectedShippingAddress];
|
|
|
|
// Ensure `selectedShippingAddress` never refers to a deleted address.
|
|
// We also compare address timestamps to notify about changes
|
|
// made outside the payments UI.
|
|
if (shippingAddress) {
|
|
// invalidate the cached value if the address was modified
|
|
if (oldShippingAddress &&
|
|
shippingAddress.guid == oldShippingAddress.guid &&
|
|
shippingAddress.timeLastModified != oldShippingAddress.timeLastModified) {
|
|
delete this._cachedState.selectedShippingAddress;
|
|
}
|
|
} else if (selectedShippingAddress !== null) {
|
|
// null out the `selectedShippingAddress` property if it is undefined,
|
|
// or if the address it pointed to was removed from storage.
|
|
log.debug("resetting invalid/deleted shipping address");
|
|
this.requestStore.setState({
|
|
selectedShippingAddress: null,
|
|
});
|
|
}
|
|
|
|
// Ensure `selectedPaymentCard` never refers to a deleted payment card and refers
|
|
// to a payment card if one exists.
|
|
let basicCards = paymentRequest.getBasicCards(state);
|
|
if (!basicCards[selectedPaymentCard]) {
|
|
// Determining the initial selection is tracked in bug 1455789
|
|
this.requestStore.setState({
|
|
selectedPaymentCard: Object.keys(basicCards)[0] || null,
|
|
selectedPaymentCardSecurityCode: null,
|
|
});
|
|
}
|
|
|
|
// Ensure `selectedShippingOption` never refers to a deleted shipping option and
|
|
// matches the merchant's selected option if the user hasn't made a choice.
|
|
if (shippingOptions && (!selectedShippingOption ||
|
|
!shippingOptions.find(option => option.id == selectedShippingOption))) {
|
|
this._cachedState.selectedShippingOption = selectedShippingOption;
|
|
this.requestStore.setState({
|
|
// Use the DOM's computed selected shipping option:
|
|
selectedShippingOption: state.request.shippingOption,
|
|
});
|
|
}
|
|
|
|
// Ensure `selectedPayerAddress` never refers to a deleted address and refers
|
|
// to an address if one exists.
|
|
if (!addresses[selectedPayerAddress]) {
|
|
this.requestStore.setState({
|
|
selectedPayerAddress: Object.keys(addresses)[0] || null,
|
|
});
|
|
}
|
|
}
|
|
|
|
_renderPayButton(state) {
|
|
let completeStatus = state.request.completeStatus;
|
|
switch (completeStatus) {
|
|
case "processing":
|
|
case "success":
|
|
case "unknown": {
|
|
this._payButton.disabled = true;
|
|
this._payButton.textContent = this._payButton.dataset[completeStatus + "Label"];
|
|
break;
|
|
}
|
|
case "": {
|
|
// initial/default state
|
|
this._payButton.textContent = this._payButton.dataset.label;
|
|
const INVALID_CLASS_NAME = "invalid-selected-option";
|
|
this._payButton.disabled =
|
|
(state.request.paymentOptions.requestShipping &&
|
|
(!this._shippingAddressPicker.selectedOption ||
|
|
this._shippingAddressPicker.classList.contains(INVALID_CLASS_NAME) ||
|
|
!this._shippingOptionPicker.selectedOption)) ||
|
|
(this._isPayerRequested(state.request.paymentOptions) &&
|
|
(!this._payerAddressPicker.selectedOption ||
|
|
this._payerAddressPicker.classList.contains(INVALID_CLASS_NAME))) ||
|
|
!this._paymentMethodPicker.securityCodeInput.isValid ||
|
|
!this._paymentMethodPicker.selectedOption ||
|
|
this._paymentMethodPicker.classList.contains(INVALID_CLASS_NAME) ||
|
|
state.changesPrevented;
|
|
break;
|
|
}
|
|
case "fail":
|
|
case "timeout": {
|
|
// pay button is hidden in fail/timeout states.
|
|
this._payButton.textContent = this._payButton.dataset.label;
|
|
this._payButton.disabled = true;
|
|
break;
|
|
}
|
|
default: {
|
|
throw new Error(`Invalid completeStatus: ${completeStatus}`);
|
|
}
|
|
}
|
|
}
|
|
|
|
_renderPayerFields(state) {
|
|
let paymentOptions = state.request.paymentOptions;
|
|
let payerRequested = this._isPayerRequested(paymentOptions);
|
|
for (let element of this._payerRelatedEls) {
|
|
element.hidden = !payerRequested;
|
|
}
|
|
|
|
if (payerRequested) {
|
|
let fieldNames = new Set(); // default: ["name", "tel", "email"]
|
|
if (paymentOptions.requestPayerName) {
|
|
fieldNames.add("name");
|
|
}
|
|
if (paymentOptions.requestPayerEmail) {
|
|
fieldNames.add("email");
|
|
}
|
|
if (paymentOptions.requestPayerPhone) {
|
|
fieldNames.add("tel");
|
|
}
|
|
this._payerAddressPicker.setAttribute("address-fields", [...fieldNames].join(" "));
|
|
// For the payer picker we want to have a line break after the name field (#1)
|
|
// if all three fields are requested.
|
|
if (fieldNames.size == 3) {
|
|
this._payerAddressPicker.setAttribute("break-after-nth-field", 1);
|
|
} else {
|
|
this._payerAddressPicker.removeAttribute("break-after-nth-field");
|
|
}
|
|
} else {
|
|
this._payerAddressPicker.removeAttribute("address-fields");
|
|
}
|
|
this._payerAddressPicker.dataset.addAddressTitle = this.dataset.payerTitleAdd;
|
|
this._payerAddressPicker.dataset.editAddressTitle = this.dataset.payerTitleEdit;
|
|
}
|
|
|
|
stateChangeCallback(state) {
|
|
super.stateChangeCallback(state);
|
|
|
|
// Don't dispatch change events for initial selectedShipping* changes at initialization
|
|
// if requestShipping is false.
|
|
if (state.request.paymentOptions.requestShipping) {
|
|
if (state.selectedShippingAddress != this._cachedState.selectedShippingAddress) {
|
|
this.changeShippingAddress(state.selectedShippingAddress);
|
|
}
|
|
|
|
if (state.selectedShippingOption != this._cachedState.selectedShippingOption) {
|
|
this.changeShippingOption(state.selectedShippingOption);
|
|
}
|
|
}
|
|
|
|
this._cachedState.selectedShippingAddress = state.selectedShippingAddress;
|
|
this._cachedState.selectedShippingOption = state.selectedShippingOption;
|
|
}
|
|
|
|
render(state) {
|
|
let request = state.request;
|
|
let paymentDetails = request.paymentDetails;
|
|
this._hostNameEl.textContent = request.topLevelPrincipal.URI.displayHost;
|
|
|
|
let displayItems = request.paymentDetails.displayItems || [];
|
|
let additionalItems = this._getAdditionalDisplayItems(state);
|
|
this._viewAllButton.hidden = !displayItems.length && !additionalItems.length;
|
|
|
|
let shippingType = state.request.paymentOptions.shippingType || "shipping";
|
|
this._shippingAddressPicker.dataset.addAddressTitle =
|
|
this.dataset[shippingType + "AddressTitleAdd"];
|
|
this._shippingAddressPicker.dataset.editAddressTitle =
|
|
this.dataset[shippingType + "AddressTitleEdit"];
|
|
let addressPickerLabel = this._shippingAddressPicker.dataset[shippingType + "AddressLabel"];
|
|
this._shippingAddressPicker.setAttribute("label", addressPickerLabel);
|
|
let optionPickerLabel = this._shippingOptionPicker.dataset[shippingType + "OptionsLabel"];
|
|
this._shippingOptionPicker.setAttribute("label", optionPickerLabel);
|
|
|
|
let totalItem = paymentRequest.getTotalItem(state);
|
|
let totalAmountEl = this.querySelector("#total > currency-amount");
|
|
totalAmountEl.value = totalItem.amount.value;
|
|
totalAmountEl.currency = totalItem.amount.currency;
|
|
|
|
// Show the total header on the address and basic card pages only during
|
|
// on-boarding(FTU) and on the payment summary page.
|
|
this._header.hidden = !state.page.onboardingWizard && state.page.id != "payment-summary";
|
|
|
|
this._orderDetailsOverlay.hidden = !state.orderDetailsShowing;
|
|
let genericError = "";
|
|
if (this._shippingAddressPicker.selectedOption &&
|
|
(!request.paymentDetails.shippingOptions ||
|
|
!request.paymentDetails.shippingOptions.length)) {
|
|
genericError = this._errorText.dataset[shippingType + "GenericError"];
|
|
}
|
|
this._errorText.textContent = paymentDetails.error || genericError;
|
|
|
|
let paymentOptions = request.paymentOptions;
|
|
for (let element of this._shippingRelatedEls) {
|
|
element.hidden = !paymentOptions.requestShipping;
|
|
}
|
|
|
|
this._renderPayerFields(state);
|
|
|
|
let isMac = /mac/i.test(navigator.platform);
|
|
for (let manageTextEl of this._manageText.children) {
|
|
manageTextEl.hidden = manageTextEl.dataset.os == "mac" ? !isMac : isMac;
|
|
let link = manageTextEl.querySelector("a");
|
|
// The href is only set to be exposed to accessibility tools so users know what will open.
|
|
// The actual opening happens from the click event listener.
|
|
link.href = "about:preferences#privacy-address-autofill";
|
|
}
|
|
|
|
this._renderPayButton(state);
|
|
|
|
for (let page of this._mainContainer.querySelectorAll(":scope > .page")) {
|
|
page.hidden = state.page.id != page.id;
|
|
}
|
|
|
|
if (state.changesPrevented) {
|
|
this.setAttribute("changes-prevented", "");
|
|
} else {
|
|
this.removeAttribute("changes-prevented");
|
|
}
|
|
this.setAttribute("complete-status", request.completeStatus);
|
|
this._disabledOverlay.hidden = !state.changesPrevented;
|
|
}
|
|
|
|
static maybeCreateFieldErrorElement(container) {
|
|
let span = container.querySelector(".error-text");
|
|
if (!span) {
|
|
span = document.createElement("span");
|
|
span.className = "error-text";
|
|
container.appendChild(span);
|
|
}
|
|
return span;
|
|
}
|
|
}
|
|
|
|
customElements.define("payment-dialog", PaymentDialog);
|