213 lines
6.7 KiB
TypeScript
213 lines
6.7 KiB
TypeScript
/**
|
|
* External dependencies
|
|
*/
|
|
import { store as noticesStore } from '@wordpress/notices';
|
|
import deprecated from '@wordpress/deprecated';
|
|
import type { BillingAddress, ShippingAddress } from '@woocommerce/settings';
|
|
import { isObject, isString, objectHasProp } from '@woocommerce/types';
|
|
|
|
/**
|
|
* Internal dependencies
|
|
*/
|
|
import {
|
|
emitEventWithAbort,
|
|
isErrorResponse,
|
|
isFailResponse,
|
|
isSuccessResponse,
|
|
noticeContexts,
|
|
ObserverResponse,
|
|
} from '../../base/context/event-emit';
|
|
import { EMIT_TYPES } from '../../base/context/providers/cart-checkout/payment-events/event-emit';
|
|
import type { emitProcessingEventType } from './types';
|
|
import { CART_STORE_KEY } from '../cart';
|
|
import {
|
|
isBillingAddress,
|
|
isShippingAddress,
|
|
} from '../../types/type-guards/address';
|
|
import { isObserverResponse } from '../../types/type-guards/observers';
|
|
import { isValidValidationErrorsObject } from '../../types/type-guards/validation';
|
|
|
|
export const __internalSetExpressPaymentError = ( message?: string ) => {
|
|
return ( { registry } ) => {
|
|
const { createErrorNotice, removeNotice } =
|
|
registry.dispatch( noticesStore );
|
|
if ( message ) {
|
|
createErrorNotice( message, {
|
|
id: 'wc-express-payment-error',
|
|
context: noticeContexts.EXPRESS_PAYMENTS,
|
|
} );
|
|
} else {
|
|
removeNotice(
|
|
'wc-express-payment-error',
|
|
noticeContexts.EXPRESS_PAYMENTS
|
|
);
|
|
}
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Emit the payment_processing event
|
|
*/
|
|
export const __internalEmitPaymentProcessingEvent: emitProcessingEventType = (
|
|
currentObserver,
|
|
setValidationErrors
|
|
) => {
|
|
return ( { dispatch, registry } ) => {
|
|
const { createErrorNotice, removeNotice } =
|
|
registry.dispatch( 'core/notices' );
|
|
removeNotice( 'wc-payment-error', noticeContexts.PAYMENTS );
|
|
return emitEventWithAbort(
|
|
currentObserver,
|
|
EMIT_TYPES.PAYMENT_SETUP,
|
|
{}
|
|
).then( ( observerResponses ) => {
|
|
let successResponse: ObserverResponse | undefined,
|
|
errorResponse: ObserverResponse | undefined,
|
|
billingAddress: BillingAddress | undefined,
|
|
shippingAddress: ShippingAddress | undefined;
|
|
observerResponses.forEach( ( response ) => {
|
|
if ( isSuccessResponse( response ) ) {
|
|
// The last observer response always "wins" for success.
|
|
successResponse = response;
|
|
}
|
|
|
|
// We consider both failed and error responses as an error.
|
|
if (
|
|
isErrorResponse( response ) ||
|
|
isFailResponse( response )
|
|
) {
|
|
errorResponse = response;
|
|
}
|
|
// Extensions may return shippingData, shippingAddress, billingData, and billingAddress in the response,
|
|
// so we need to check for all. If we detect either shippingData or billingData we need to show a
|
|
// deprecated warning for it, but also apply the changes to the wc/store/cart store.
|
|
const {
|
|
billingAddress: billingAddressFromResponse,
|
|
|
|
// Deprecated, but keeping it for now, for compatibility with extensions returning it.
|
|
billingData: billingDataFromResponse,
|
|
shippingAddress: shippingAddressFromResponse,
|
|
|
|
// Deprecated, but keeping it for now, for compatibility with extensions returning it.
|
|
shippingData: shippingDataFromResponse,
|
|
} = response?.meta || {};
|
|
|
|
billingAddress = billingAddressFromResponse as BillingAddress;
|
|
shippingAddress =
|
|
shippingAddressFromResponse as ShippingAddress;
|
|
|
|
if ( billingDataFromResponse ) {
|
|
// Set this here so that old extensions still using billingData can set the billingAddress.
|
|
billingAddress = billingDataFromResponse as BillingAddress;
|
|
deprecated(
|
|
'returning billingData from an onPaymentProcessing observer in WooCommerce Blocks',
|
|
{
|
|
version: '9.5.0',
|
|
alternative: 'billingAddress',
|
|
link: 'https://github.com/woocommerce/woocommerce-blocks/pull/6369',
|
|
}
|
|
);
|
|
}
|
|
|
|
if ( shippingDataFromResponse ) {
|
|
// Set this here so that old extensions still using shippingData can set the shippingAddress.
|
|
shippingAddress =
|
|
shippingDataFromResponse as ShippingAddress;
|
|
deprecated(
|
|
'returning shippingData from an onPaymentProcessing observer in WooCommerce Blocks',
|
|
{
|
|
version: '9.5.0',
|
|
alternative: 'shippingAddress',
|
|
link: 'https://github.com/woocommerce/woocommerce-blocks/pull/8163',
|
|
}
|
|
);
|
|
}
|
|
} );
|
|
|
|
const { setBillingAddress, setShippingAddress } =
|
|
registry.dispatch( CART_STORE_KEY );
|
|
|
|
// Observer returned success, we sync the payment method data and billing address.
|
|
if ( isObserverResponse( successResponse ) && ! errorResponse ) {
|
|
const { paymentMethodData } = successResponse?.meta || {};
|
|
|
|
if ( isBillingAddress( billingAddress ) ) {
|
|
setBillingAddress( billingAddress );
|
|
}
|
|
if ( isShippingAddress( shippingAddress ) ) {
|
|
setShippingAddress( shippingAddress );
|
|
}
|
|
|
|
dispatch.__internalSetPaymentMethodData(
|
|
isObject( paymentMethodData ) ? paymentMethodData : {}
|
|
);
|
|
dispatch.__internalSetPaymentReady();
|
|
} else if ( isFailResponse( errorResponse ) ) {
|
|
const { paymentMethodData } = errorResponse?.meta || {};
|
|
|
|
if (
|
|
objectHasProp( errorResponse, 'message' ) &&
|
|
isString( errorResponse.message ) &&
|
|
errorResponse.message.length
|
|
) {
|
|
let context: string = noticeContexts.PAYMENTS;
|
|
if (
|
|
objectHasProp( errorResponse, 'messageContext' ) &&
|
|
isString( errorResponse.messageContext ) &&
|
|
errorResponse.messageContext.length
|
|
) {
|
|
context = errorResponse.messageContext;
|
|
}
|
|
createErrorNotice( errorResponse.message, {
|
|
id: 'wc-payment-error',
|
|
isDismissible: false,
|
|
context,
|
|
} );
|
|
}
|
|
|
|
if ( isBillingAddress( billingAddress ) ) {
|
|
setBillingAddress( billingAddress );
|
|
}
|
|
|
|
dispatch.__internalSetPaymentMethodData(
|
|
isObject( paymentMethodData ) ? paymentMethodData : {}
|
|
);
|
|
dispatch.__internalSetPaymentError();
|
|
} else if ( isErrorResponse( errorResponse ) ) {
|
|
if (
|
|
objectHasProp( errorResponse, 'message' ) &&
|
|
isString( errorResponse.message ) &&
|
|
errorResponse.message.length
|
|
) {
|
|
let context: string = noticeContexts.PAYMENTS;
|
|
if (
|
|
objectHasProp( errorResponse, 'messageContext' ) &&
|
|
isString( errorResponse.messageContext ) &&
|
|
errorResponse.messageContext.length
|
|
) {
|
|
context = errorResponse.messageContext;
|
|
}
|
|
createErrorNotice( errorResponse.message, {
|
|
id: 'wc-payment-error',
|
|
isDismissible: false,
|
|
context,
|
|
} );
|
|
}
|
|
|
|
dispatch.__internalSetPaymentError();
|
|
|
|
if (
|
|
isValidValidationErrorsObject(
|
|
errorResponse.validationErrors
|
|
)
|
|
) {
|
|
setValidationErrors( errorResponse.validationErrors );
|
|
}
|
|
} else {
|
|
// Otherwise there are no payment methods doing anything so just assume payment method is ready.
|
|
dispatch.__internalSetPaymentReady();
|
|
}
|
|
} );
|
|
};
|
|
};
|