diff --git a/packages/extensions/venia-sample-payments-cashondelivery/README.md b/packages/extensions/venia-sample-payments-cashondelivery/README.md new file mode 100644 index 0000000000..b0aa937037 --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/README.md @@ -0,0 +1,10 @@ +# venia-sample-payments-cashondelivery + +This package provides the `Cash On Delivery` payment method for Venia. + +## Installation + +To install this extension, add it as a `devDependency` to your app. +project: + +`yarn add -D @magento/venia-sample-payments-cashondelivery` diff --git a/packages/extensions/venia-sample-payments-cashondelivery/intercept.js b/packages/extensions/venia-sample-payments-cashondelivery/intercept.js new file mode 100644 index 0000000000..1042660671 --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/intercept.js @@ -0,0 +1,41 @@ +module.exports = targets => { + const { specialFeatures } = targets.of('@magento/pwa-buildpack'); + specialFeatures.tap(flags => { + /** + * Wee need to activate esModules, cssModules and GQL Queries to allow build pack to load our extension + * {@link https://magento.github.io/pwa-studio/pwa-buildpack/reference/configure-webpack/#special-flags}. + */ + flags[targets.name] = { + esModules: true, + cssModules: true, + graphqlQueries: true + }; + }); + + const { + checkoutPagePaymentTypes, + editablePaymentTypes, + summaryPagePaymentTypes + } = targets.of('@magento/venia-ui'); + checkoutPagePaymentTypes.tap(payments => + payments.add({ + paymentCode: 'cashondelivery', + importPath: + '@magento/venia-sample-payments-cashondelivery/src/components/cashondelivery.js' + }) + ); + editablePaymentTypes.tap(editablePaymentTypes => { + editablePaymentTypes.add({ + paymentCode: 'cashondelivery', + importPath: + '@magento/venia-sample-payments-cashondelivery/src/components/editcod.js' + }); + }); + summaryPagePaymentTypes.tap(paymentSummaries => + paymentSummaries.add({ + paymentCode: 'cashondelivery', + importPath: + '@magento/venia-sample-payments-cashondelivery/src/components/summarycod.js' + }) + ); +}; diff --git a/packages/extensions/venia-sample-payments-cashondelivery/package.json b/packages/extensions/venia-sample-payments-cashondelivery/package.json new file mode 100644 index 0000000000..eb3b7f2d06 --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/package.json @@ -0,0 +1,27 @@ +{ + "name": "@magento/venia-sample-payments-cashondelivery", + "version": "0.0.1", + "publishConfig": { + "access": "public" + }, + "description": "Provides demo Cash On Delivery Payment Method for PWA Studio.", + "main": "./intercept.js", + "scripts": { + "clean": " " + }, + "repository": "github:magento/pwa-studio", + "license": "(OSL-3.0 OR AFL-3.0)", + "peerDependencies": { + "@magento/peregrine": "~14.4.1", + "@magento/pwa-buildpack": "~11.5.3", + "@magento/venia-ui": "~11.5.0", + "react": "~17.0.1", + "react-intl": "~5.20.0", + "react-router-dom": "~5.2.0" + }, + "pwa-studio": { + "targets": { + "intercept": "./intercept" + } + } +} diff --git a/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/__snapshots__/cashondelivery.spec.js.snap b/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/__snapshots__/cashondelivery.spec.js.snap new file mode 100644 index 0000000000..4849002e4c --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/__snapshots__/cashondelivery.spec.js.snap @@ -0,0 +1,12 @@ +// Jest Snapshot v1, https://goo.gl/fbAQLP + +exports[`Should render CashOnDelivery component correctly`] = ` +
+ +
+`; diff --git a/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/__snapshots__/editcod.spec.js.snap b/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/__snapshots__/editcod.spec.js.snap new file mode 100644 index 0000000000..83573fbdbc --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/__snapshots__/editcod.spec.js.snap @@ -0,0 +1,15 @@ +// Jest Snapshot v1, https://goo.gl/fbAQLP + +exports[`renders correctly 1`] = ` +
+ +
+`; diff --git a/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/__snapshots__/summarycod.spec.js.snap b/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/__snapshots__/summarycod.spec.js.snap new file mode 100644 index 0000000000..26fcf254fd --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/__snapshots__/summarycod.spec.js.snap @@ -0,0 +1,67 @@ +// Jest Snapshot v1, https://goo.gl/fbAQLP + +exports[`renders correctly 1`] = ` +
+
+
+
+
+ + + +
+ + +
+
+ +
+ +
+
+`; diff --git a/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/cashondelivery.spec.js b/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/cashondelivery.spec.js new file mode 100644 index 0000000000..b5054f9207 --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/cashondelivery.spec.js @@ -0,0 +1,35 @@ +import React from 'react'; +import { createTestInstance } from '@magento/peregrine'; +import { useCashondelivery } from '../../talons/useCashondelivery'; + +import CashOnDelivery from '../cashondelivery'; + +jest.mock('@magento/venia-ui/lib/classify'); +jest.mock('../../talons/useCashondelivery', () => { + return { + useCashondelivery: jest.fn().mockReturnValue({ + onBillingAddressChangedError: jest.fn(), + onBillingAddressChangedSuccess: jest.fn() + }) + }; +}); + +jest.mock( + '@magento/venia-ui/lib/components/CheckoutPage/BillingAddress', + () => props => +); + +const useCashondeliveryReturnValue = { + onBillingAddressChangedError: jest.fn(), + onBillingAddressChangedSuccess: jest.fn() +}; + +test('Should render CashOnDelivery component correctly', () => { + useCashondelivery.mockReturnValueOnce({ + ...useCashondeliveryReturnValue + }); + + const tree = createTestInstance(); + + expect(tree.toJSON()).toMatchSnapshot(); +}); diff --git a/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/editcod.spec.js b/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/editcod.spec.js new file mode 100644 index 0000000000..0e02cbe5f9 --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/editcod.spec.js @@ -0,0 +1,30 @@ +import React from 'react'; +import { createTestInstance } from '@magento/peregrine'; + +import EditCod from '../editcod'; + +jest.mock('@magento/venia-ui/lib/classify'); +jest.mock('../cashondelivery', () => props => ( + +)); + +const mocks = { + onPaymentReady: jest.fn().mockName('onPaymentReady'), + onPaymentSuccess: jest.fn().mockName('onPaymentSuccess'), + onPaymentError: jest.fn().mockName('onPaymentError'), + resetShouldSubmit: jest.fn().mockName('resetShouldSubmit') +}; + +test('renders correctly', () => { + // Arrange. + const props = { + ...mocks, + shouldSubmit: false + }; + + // Act. + const tree = createTestInstance(); + + // Assert. + expect(tree.toJSON()).toMatchSnapshot(); +}); diff --git a/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/summarycod.spec.js b/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/summarycod.spec.js new file mode 100644 index 0000000000..8cdf105c3d --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/src/components/__tests__/summarycod.spec.js @@ -0,0 +1,30 @@ +import React from 'react'; +import { createTestInstance } from '@magento/peregrine'; + +import SummaryCod from '../summarycod'; + +jest.mock('@magento/venia-ui/lib/classify'); +jest.mock('react-intl', () => ({ + FormattedMessage: props => ( +
+ ) +})); +jest.mock('@magento/venia-ui/lib/components/LinkButton', () => props => ( + +)); +jest.mock('@magento/venia-ui/lib/components/Icon', () => props => ( + +)); + +test('renders correctly', () => { + // Arrange. + const props = { + onEdit: jest.fn().mockName('onEdit') + }; + + // Act. + const tree = createTestInstance(); + + // Assert. + expect(tree.toJSON()).toMatchSnapshot(); +}); diff --git a/packages/extensions/venia-sample-payments-cashondelivery/src/components/cashondelivery.js b/packages/extensions/venia-sample-payments-cashondelivery/src/components/cashondelivery.js new file mode 100644 index 0000000000..7f892d9b2e --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/src/components/cashondelivery.js @@ -0,0 +1,46 @@ +import React from 'react'; +import { useStyle } from '@magento/venia-ui/lib/classify'; +import { shape, string, bool, func } from 'prop-types'; +import BillingAddress from '@magento/venia-ui/lib/components/CheckoutPage/BillingAddress'; +import { useCashondelivery } from '../talons/useCashondelivery'; +import defaultClasses from './cashondelivery.module.css'; + +/** + * The CashOnDelivery component renders all information to handle cashondelivery payment. + * + * @param {Boolean} props.shouldSubmit boolean value which represents if a payment nonce request has been submitted + * @param {Function} props.onPaymentSuccess callback to invoke when the a payment nonce has been generated + * @param {Function} props.onPaymentReady callback to invoke when the component is ready + * @param {Function} props.onPaymentError callback to invoke when component throws an error + * @param {Function} props.resetShouldSubmit callback to reset the shouldSubmit flag + */ +const CashOnDelivery = props => { + const classes = useStyle(defaultClasses, props.classes); + + const { + onBillingAddressChangedError, + onBillingAddressChangedSuccess + } = useCashondelivery(props); + + return ( +
+ +
+ ); +}; + +CashOnDelivery.propTypes = { + classes: shape({ root: string }), + shouldSubmit: bool.isRequired, + onPaymentSuccess: func, + onPaymentReady: func, + onPaymentError: func, + resetShouldSubmit: func.isRequired +}; + +export default CashOnDelivery; diff --git a/packages/extensions/venia-sample-payments-cashondelivery/src/components/cashondelivery.module.css b/packages/extensions/venia-sample-payments-cashondelivery/src/components/cashondelivery.module.css new file mode 100644 index 0000000000..7fa00f61cd --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/src/components/cashondelivery.module.css @@ -0,0 +1,3 @@ +.cod_root { + padding-top: 1.125rem; +} diff --git a/packages/extensions/venia-sample-payments-cashondelivery/src/components/editcod.js b/packages/extensions/venia-sample-payments-cashondelivery/src/components/editcod.js new file mode 100644 index 0000000000..4c2a7ec7f7 --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/src/components/editcod.js @@ -0,0 +1,47 @@ +import React from 'react'; +import { shape, string, bool, func } from 'prop-types'; + +import { useStyle } from '@magento/venia-ui/lib/classify'; + +import CashOnDelivery from './cashondelivery'; +import defaultClasses from './cashondelivery.module.css'; + +/** + * The edit view for the Cod payment method. + */ +const EditCod = props => { + const { + onPaymentReady, + onPaymentSuccess, + onPaymentError, + resetShouldSubmit, + shouldSubmit + } = props; + + const classes = useStyle(defaultClasses, props.classes); + + return ( +
+ +
+ ); +}; + +export default EditCod; + +EditCod.propTypes = { + classes: shape({ + root: string + }), + onPaymentReady: func.isRequired, + onPaymentSuccess: func.isRequired, + onPaymentError: func.isRequired, + resetShouldSubmit: func.isRequired, + shouldSubmit: bool +}; diff --git a/packages/extensions/venia-sample-payments-cashondelivery/src/components/summary.module.css b/packages/extensions/venia-sample-payments-cashondelivery/src/components/summary.module.css new file mode 100644 index 0000000000..179532469f --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/src/components/summary.module.css @@ -0,0 +1,42 @@ +.root { + display: grid; + gap: 1.125rem; + padding: 2rem; +} + +.heading_container { + display: grid; + grid-auto-flow: column; + grid-template-columns: 1fr; +} + +.heading { + font-weight: 600; +} + +.edit_button { + color: rgb(var(--venia-brand-color-1-700)); + padding: 1rem; + margin: -1rem; +} + +.edit_icon { + stroke: rgb(var(--venia-brand-color-1-700)); +} + +.edit_text { +} + +.cod_details_container { + display: grid; + gap: 0.5rem; +} + +.payment_type { +} + +@media screen(-lg) { + .edit_text { + display: none; + } +} diff --git a/packages/extensions/venia-sample-payments-cashondelivery/src/components/summarycod.js b/packages/extensions/venia-sample-payments-cashondelivery/src/components/summarycod.js new file mode 100644 index 0000000000..346514048d --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/src/components/summarycod.js @@ -0,0 +1,72 @@ +import React from 'react'; +import { func, shape, string } from 'prop-types'; +import { FormattedMessage } from 'react-intl'; +import { Edit2 as EditIcon } from 'react-feather'; +import { useStyle } from '@magento/venia-ui/lib/classify'; +import Icon from '@magento/venia-ui/lib/components/Icon'; +import LinkButton from '@magento/venia-ui/lib/components/LinkButton'; + +import defaultClasses from './summary.module.css'; + +/** + * The SummaryCod component of the Cash On Delivery payment method extension. + */ +const SummaryCod = props => { + const { onEdit } = props; + + const classes = useStyle(defaultClasses, props.classes); + + return ( +
+
+
+ +
+ + + + + + +
+
+ + + +
+
+ ); +}; + +export default SummaryCod; + +SummaryCod.propTypes = { + classes: shape({ + root: string, + cod_details_container: string, + edit_button: string, + edit_icon: string, + edit_text: string, + heading_container: string, + heading: string, + payment_type: string + }), + onEdit: func +}; diff --git a/packages/extensions/venia-sample-payments-cashondelivery/src/talons/cashondelivery.gql.js b/packages/extensions/venia-sample-payments-cashondelivery/src/talons/cashondelivery.gql.js new file mode 100644 index 0000000000..b21152799d --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/src/talons/cashondelivery.gql.js @@ -0,0 +1,24 @@ +import { gql } from '@apollo/client'; + +export const SET_COD_PAYMENT_METHOD_ON_CART = gql` + mutation setPaymentMethodOnCart($cartId: String!) { + setPaymentMethodOnCart( + input: { + cart_id: $cartId + payment_method: { code: "cashondelivery" } + } + ) { + cart { + id + selected_payment_method { + code + title + } + } + } + } +`; + +export default { + setCodPaymentMethodOnCartMutation: SET_COD_PAYMENT_METHOD_ON_CART +}; diff --git a/packages/extensions/venia-sample-payments-cashondelivery/src/talons/useCashondelivery.js b/packages/extensions/venia-sample-payments-cashondelivery/src/talons/useCashondelivery.js new file mode 100644 index 0000000000..0e16ac68f4 --- /dev/null +++ b/packages/extensions/venia-sample-payments-cashondelivery/src/talons/useCashondelivery.js @@ -0,0 +1,80 @@ +import { useCallback, useEffect } from 'react'; +import { useMutation } from '@apollo/client'; +import mergeOperations from '@magento/peregrine/lib/util/shallowMerge'; +import { useCartContext } from '@magento/peregrine/lib/context/cart'; + +import DEFAULT_OPERATIONS from './cashondelivery.gql'; + +/** + * Talon to handle cod payment. + * + * @param {Boolean} props.shouldSubmit boolean value which represents if a payment nonce request has been submitted + * @param {Function} props.onPaymentSuccess callback to invoke when the a payment nonce has been generated + * @param {Function} props.onPaymentError callback to invoke when component throws an error + * @param {Function} props.resetShouldSubmit callback to reset the shouldSubmit flag + * @param {DocumentNode} props.operations.setCodPaymentMethodOnCartMutation mutation to set cashondelivery as payment + * + * @returns { + * onBillingAddressChangedError: Function, + * onBillingAddressChangedSuccess: Function + * } + */ +export const useCashondelivery = props => { + const operations = mergeOperations(DEFAULT_OPERATIONS, props.operations); + + const { setCodPaymentMethodOnCartMutation } = operations; + + const [{ cartId }] = useCartContext(); + + const { resetShouldSubmit, onPaymentSuccess, onPaymentError } = props; + + const [ + updatePaymentMethod, + { + error: paymentMethodMutationError, + called: paymentMethodMutationCalled, + loading: paymentMethodMutationLoading + } + ] = useMutation(setCodPaymentMethodOnCartMutation); + + /** + * This function will be called if cant not set address. + */ + const onBillingAddressChangedError = useCallback(() => { + resetShouldSubmit(); + }, [resetShouldSubmit]); + + /** + * This function will be called if address was successfully set. + */ + const onBillingAddressChangedSuccess = useCallback(() => { + updatePaymentMethod({ + variables: { cartId } + }); + }, [updatePaymentMethod, cartId]); + + useEffect(() => { + const paymentMethodMutationCompleted = + paymentMethodMutationCalled && !paymentMethodMutationLoading; + + if (paymentMethodMutationCompleted && !paymentMethodMutationError) { + onPaymentSuccess(); + } + + if (paymentMethodMutationCompleted && paymentMethodMutationError) { + onPaymentError(); + } + }, [ + paymentMethodMutationError, + paymentMethodMutationLoading, + paymentMethodMutationCalled, + onPaymentSuccess, + onPaymentError, + resetShouldSubmit + ]); + + return { + onBillingAddressChangedError, + onBillingAddressChangedSuccess + }; +};