/** * External dependencies */ import classnames from 'classnames'; import { __ } from '@wordpress/i18n'; import PropTypes from 'prop-types'; import QuantitySelector from '@woocommerce/base-components/quantity-selector'; import ProductPrice from '@woocommerce/base-components/product-price'; import ProductName from '@woocommerce/base-components/product-name'; import { useStoreCartItemQuantity } from '@woocommerce/base-hooks'; import { ProductBackorderBadge, ProductImage, ProductLowStockBadge, ProductMetadata, ProductSaleBadge, } from '@woocommerce/base-components/cart-checkout'; import { getCurrencyFromPriceResponse } from '@woocommerce/price-format'; import { __experimentalApplyCheckoutFilter, mustBeString, mustContain, } from '@woocommerce/blocks-checkout'; import Dinero from 'dinero.js'; import { DISPLAY_CART_PRICES_INCLUDING_TAX } from '@woocommerce/block-settings'; import { useCallback, useMemo } from '@wordpress/element'; /** * @typedef {import('@woocommerce/type-defs/cart').CartItem} CartItem */ /** * Convert a Dinero object with precision to store currency minor unit. * * @param {Dinero} priceObject Price object to convert. * @param {Object} currency Currency data. * @return {number} Amount with new minor unit precision. */ const getAmountFromRawPrice = ( priceObject, currency ) => { return priceObject.convertPrecision( currency.minorUnit ).getAmount(); }; /** * Cart line item table row component. * * @param {Object} props * @param {CartItem|Object} props.lineItem */ const CartLineItemRow = ( { lineItem = {} } ) => { const { name: initialName = '', catalog_visibility: catalogVisibility = '', short_description: shortDescription = '', description: fullDescription = '', low_stock_remaining: lowStockRemaining = null, show_backorder_badge: showBackorderBadge = false, quantity_limit: quantityLimit = 99, permalink = '', images = [], variation = [], item_data: itemData = [], prices = { currency_code: 'USD', currency_minor_unit: 2, currency_symbol: '$', currency_prefix: '$', currency_suffix: '', currency_decimal_separator: '.', currency_thousand_separator: ',', price: '0', regular_price: '0', sale_price: '0', price_range: null, raw_prices: { precision: 6, price: '0', regular_price: '0', sale_price: '0', }, }, totals = { currency_code: 'USD', currency_minor_unit: 2, currency_symbol: '$', currency_prefix: '$', currency_suffix: '', currency_decimal_separator: '.', currency_thousand_separator: ',', line_subtotal: '0', line_subtotal_tax: '0', }, extensions, } = lineItem; const { quantity, changeQuantity, removeItem, isPendingDelete, } = useStoreCartItemQuantity( lineItem ); const productPriceValidation = useCallback( ( value ) => mustBeString( value ) && mustContain( value, '' ), [] ); const arg = useMemo( () => ( { context: 'cart', cartItem: lineItem, } ), [ lineItem ] ); const priceCurrency = getCurrencyFromPriceResponse( prices ); const name = __experimentalApplyCheckoutFilter( { filterName: 'itemName', defaultValue: initialName, extensions, arg, validation: mustBeString, } ); const regularAmountSingle = Dinero( { amount: parseInt( prices.raw_prices.regular_price, 10 ), precision: parseInt( prices.raw_prices.precision, 10 ), } ); const purchaseAmountSingle = Dinero( { amount: parseInt( prices.raw_prices.price, 10 ), precision: parseInt( prices.raw_prices.precision, 10 ), } ); const saleAmountSingle = regularAmountSingle.subtract( purchaseAmountSingle ); const saleAmount = saleAmountSingle.multiply( quantity ); const totalsCurrency = getCurrencyFromPriceResponse( totals ); let lineSubtotal = parseInt( totals.line_subtotal, 10 ); if ( DISPLAY_CART_PRICES_INCLUDING_TAX ) { lineSubtotal += parseInt( totals.line_subtotal_tax, 10 ); } const subtotalPrice = Dinero( { amount: lineSubtotal, precision: totalsCurrency.minorUnit, } ); const firstImage = images.length ? images[ 0 ] : {}; const isProductHiddenFromCatalog = catalogVisibility === 'hidden' || catalogVisibility === 'search'; // Allow extensions to filter how the price is displayed. Ie: prepending or appending some values. const productPriceFormat = __experimentalApplyCheckoutFilter( { filterName: 'cartItemPrice', defaultValue: '', extensions, arg, validation: productPriceValidation, } ); const subtotalPriceFormat = __experimentalApplyCheckoutFilter( { filterName: 'subtotalPriceFormat', defaultValue: '', extensions, arg, validation: productPriceValidation, } ); const saleBadgePriceFormat = __experimentalApplyCheckoutFilter( { filterName: 'saleBadgePriceFormat', defaultValue: '', extensions, arg, validation: productPriceValidation, } ); return ( { /* If the image has no alt text, this link is unnecessary and can be hidden. */ } { /* We don't need to make it focusable, because product name has the same link. */ } { isProductHiddenFromCatalog ? ( ) : ( ) } { showBackorderBadge ? ( ) : ( !! lowStockRemaining && ( ) ) }
{ quantity > 1 && ( ) }
); }; CartLineItemRow.propTypes = { lineItem: PropTypes.object, }; export default CartLineItemRow;