/** * External dependencies */ import { createBlock, createBlocksFromInnerBlocksTemplate, type BlockInstance, } from '@wordpress/blocks'; import { isWpVersion } from '@woocommerce/settings'; import { __, sprintf } from '@wordpress/i18n'; /** * Internal dependencies */ import { INNER_BLOCKS_TEMPLATE as productsInnerBlocksTemplate, QUERY_DEFAULT_ATTRIBUTES as productsQueryDefaultAttributes, } from '../product-query/constants'; import { VARIATION_NAME as productsVariationName } from '../product-query/variations/product-query'; import { createArchiveTitleBlock, createRowBlock } from './utils'; import { OnClickCallbackParameter, type InheritedAttributes } from './types'; const createProductsBlock = ( inheritedAttributes: InheritedAttributes ) => createBlock( 'core/query', { ...productsQueryDefaultAttributes, ...inheritedAttributes, namespace: productsVariationName, query: { ...productsQueryDefaultAttributes.query, inherit: true, }, }, createBlocksFromInnerBlocksTemplate( productsInnerBlocksTemplate ) ); const getBlockifiedTemplate = ( inheritedAttributes: InheritedAttributes, withTermDescription = false ) => [ createBlock( 'woocommerce/breadcrumbs', inheritedAttributes ), createArchiveTitleBlock( 'archive-title', inheritedAttributes ), withTermDescription ? createBlock( 'core/term-description', inheritedAttributes ) : null, createBlock( 'woocommerce/store-notices', inheritedAttributes ), createRowBlock( [ createBlock( 'woocommerce/product-results-count' ), createBlock( 'woocommerce/catalog-sorting' ), ], inheritedAttributes ), createProductsBlock( inheritedAttributes ), ].filter( Boolean ) as BlockInstance[]; const getBlockifiedTemplateWithTermDescription = ( inheritedAttributes: InheritedAttributes ) => getBlockifiedTemplate( inheritedAttributes, true ); const isConversionPossible = () => { // Blockification is possible for the WP version 6.1 and above, // which are the versions the Products block supports. return isWpVersion( '6.1', '>=' ); }; const getDescriptionAllowingConversion = ( templateTitle: string ) => sprintf( /* translators: %s is the template title */ __( 'Transform this template into multiple blocks so you can add, remove, reorder, and customize your %s template.', 'woo-gutenberg-products-block' ), templateTitle ); const getDescriptionDisallowingConversion = ( templateTitle: string ) => sprintf( /* translators: %s is the template title */ __( 'This block serves as a placeholder for your %s. It will display the actual product image, title, price in your store. You can move this placeholder around and add more blocks around to customize the template.', 'woo-gutenberg-products-block' ), templateTitle ); const getDescription = ( templateTitle: string, canConvert: boolean ) => { if ( canConvert ) { return getDescriptionAllowingConversion( templateTitle ); } return getDescriptionDisallowingConversion( templateTitle ); }; const getButtonLabel = () => __( 'Transform into blocks', 'woo-gutenberg-products-block' ); const onClickCallback = ( { clientId, attributes, getBlocks, replaceBlock, selectBlock, }: OnClickCallbackParameter ) => { replaceBlock( clientId, getBlockifiedTemplate( attributes ) ); const blocks = getBlocks(); const groupBlock = blocks.find( ( block ) => block.name === 'core/group' && block.innerBlocks.some( ( innerBlock ) => innerBlock.name === 'woocommerce/store-notices' ) ); if ( groupBlock ) { selectBlock( groupBlock.clientId ); } }; const onClickCallbackWithTermDescription = ( { clientId, attributes, getBlocks, replaceBlock, selectBlock, }: OnClickCallbackParameter ) => { replaceBlock( clientId, getBlockifiedTemplate( attributes, true ) ); const blocks = getBlocks(); const groupBlock = blocks.find( ( block ) => block.name === 'core/group' && block.innerBlocks.some( ( innerBlock ) => innerBlock.name === 'woocommerce/store-notices' ) ); if ( groupBlock ) { selectBlock( groupBlock.clientId ); } }; const productCatalogBlockifyConfig = { getButtonLabel, onClickCallback, getBlockifiedTemplate, }; const productTaxonomyBlockifyConfig = { getButtonLabel, onClickCallback: onClickCallbackWithTermDescription, getBlockifiedTemplate: getBlockifiedTemplateWithTermDescription, }; export const blockifiedProductCatalogConfig = { isConversionPossible, getDescription, blockifyConfig: productCatalogBlockifyConfig, }; export const blockifiedProductTaxonomyConfig = { isConversionPossible, getDescription, blockifyConfig: productTaxonomyBlockifyConfig, };