diff --git a/assets/js/atomic/blocks/product-elements/stock-indicator/attributes.ts b/assets/js/atomic/blocks/product-elements/stock-indicator/attributes.ts
index d7194c6d347..cf1ec2d22be 100644
--- a/assets/js/atomic/blocks/product-elements/stock-indicator/attributes.ts
+++ b/assets/js/atomic/blocks/product-elements/stock-indicator/attributes.ts
@@ -12,6 +12,10 @@ export const blockAttributes: BlockAttributes = {
type: 'boolean',
default: false,
},
+ isDescendantOfAllProducts: {
+ type: 'boolean',
+ default: false,
+ },
};
export default blockAttributes;
diff --git a/assets/js/atomic/blocks/product-elements/stock-indicator/block.tsx b/assets/js/atomic/blocks/product-elements/stock-indicator/block.tsx
index c933a74fe33..544b020f9db 100644
--- a/assets/js/atomic/blocks/product-elements/stock-indicator/block.tsx
+++ b/assets/js/atomic/blocks/product-elements/stock-indicator/block.tsx
@@ -7,7 +7,11 @@ import {
useInnerBlockLayoutContext,
useProductDataContext,
} from '@woocommerce/shared-context';
-import { useColorProps, useTypographyProps } from '@woocommerce/base-hooks';
+import {
+ useColorProps,
+ useTypographyProps,
+ useSpacingProps,
+} from '@woocommerce/base-hooks';
import { withProductDataContext } from '@woocommerce/shared-hocs';
import type { HTMLAttributes } from 'react';
@@ -17,22 +21,43 @@ import type { HTMLAttributes } from 'react';
import './style.scss';
import type { BlockAttributes } from './types';
-const lowStockText = ( lowStock: number ): string => {
- return sprintf(
- /* translators: %d stock amount (number of items in stock for product) */
- __( '%d left in stock', 'woo-gutenberg-products-block' ),
- lowStock
- );
-};
-
-const stockText = ( inStock: boolean, isBackordered: boolean ): string => {
- if ( isBackordered ) {
+/**
+ * Get stock text based on stock. For example:
+ * - In stock
+ * - Out of stock
+ * - Available on backorder
+ * - 2 left in stock
+ *
+ * @param stockInfo Object containing stock information.
+ * @param stockInfo.isInStock Whether product is in stock.
+ * @param stockInfo.isLowStock Whether product is low in stock.
+ * @param stockInfo.lowStockAmount Number of items left in stock.
+ * @param stockInfo.isOnBackorder Whether product is on backorder.
+ * @return string Stock text.
+ */
+const getTextBasedOnStock = ( {
+ isInStock = false,
+ isLowStock = false,
+ lowStockAmount = null,
+ isOnBackorder = false,
+}: {
+ isInStock?: boolean;
+ isLowStock?: boolean;
+ lowStockAmount?: number | null;
+ isOnBackorder?: boolean;
+} ): string => {
+ if ( isLowStock && lowStockAmount !== null ) {
+ return sprintf(
+ /* translators: %d stock amount (number of items in stock for product) */
+ __( '%d left in stock', 'woo-gutenberg-products-block' ),
+ lowStockAmount
+ );
+ } else if ( isOnBackorder ) {
return __( 'Available on backorder', 'woo-gutenberg-products-block' );
+ } else if ( isInStock ) {
+ return __( 'In stock', 'woo-gutenberg-products-block' );
}
-
- return inStock
- ? __( 'In Stock', 'woo-gutenberg-products-block' )
- : __( 'Out of Stock', 'woo-gutenberg-products-block' );
+ return __( 'Out of stock', 'woo-gutenberg-products-block' );
};
type Props = BlockAttributes & HTMLAttributes< HTMLDivElement >;
@@ -43,8 +68,9 @@ export const Block = ( props: Props ): JSX.Element | null => {
const { product } = useProductDataContext();
const colorProps = useColorProps( props );
const typographyProps = useTypographyProps( props );
+ const spacingProps = useSpacingProps( props );
- if ( ! product.id || ! product.is_purchasable ) {
+ if ( ! product.id ) {
return null;
}
@@ -54,28 +80,39 @@ export const Block = ( props: Props ): JSX.Element | null => {
return (
- { lowStock
- ? lowStockText( lowStock )
- : stockText( inStock, isBackordered ) }
+ { getTextBasedOnStock( {
+ isInStock: inStock,
+ isLowStock: !! lowStock,
+ lowStockAmount: lowStock,
+ isOnBackorder: isBackordered,
+ } ) }
);
};
diff --git a/assets/js/atomic/blocks/product-elements/stock-indicator/edit.tsx b/assets/js/atomic/blocks/product-elements/stock-indicator/edit.tsx
index 162ab4311d8..83efbf324ef 100644
--- a/assets/js/atomic/blocks/product-elements/stock-indicator/edit.tsx
+++ b/assets/js/atomic/blocks/product-elements/stock-indicator/edit.tsx
@@ -24,7 +24,9 @@ const Edit = ( {
setAttributes,
context,
}: BlockEditProps< BlockAttributes > & { context: Context } ): JSX.Element => {
- const blockProps = useBlockProps();
+ const { style, ...blockProps } = useBlockProps( {
+ className: 'wc-block-components-product-stock-indicator',
+ } );
const blockAttrs = {
...attributes,
@@ -38,7 +40,15 @@ const Edit = ( {
);
return (
-
+
diff --git a/assets/js/atomic/blocks/product-elements/stock-indicator/supports.ts b/assets/js/atomic/blocks/product-elements/stock-indicator/supports.ts
index 6059f2454ee..6c6f9120303 100644
--- a/assets/js/atomic/blocks/product-elements/stock-indicator/supports.ts
+++ b/assets/js/atomic/blocks/product-elements/stock-indicator/supports.ts
@@ -2,17 +2,39 @@
* External dependencies
*/
import { isFeaturePluginBuild } from '@woocommerce/block-settings';
+import {
+ // @ts-expect-error We check if this exists before using it.
+ // eslint-disable-next-line @wordpress/no-unsafe-wp-apis
+ __experimentalGetSpacingClassesAndStyles,
+} from '@wordpress/block-editor';
+
+/**
+ * Internal dependencies
+ */
+import sharedConfig from '../shared/config';
export const supports = {
- ...( isFeaturePluginBuild() && {
- color: {
- text: true,
- background: false,
- link: false,
- },
- typography: {
- fontSize: true,
+ ...sharedConfig.supports,
+ color: {
+ text: true,
+ background: true,
+ },
+ typography: {
+ fontSize: true,
+ lineHeight: true,
+ ...( isFeaturePluginBuild() && {
+ __experimentalFontWeight: true,
+ __experimentalFontFamily: true,
+ __experimentalFontStyle: true,
+ __experimentalTextTransform: true,
+ __experimentalTextDecoration: true,
+ __experimentalLetterSpacing: true,
+ } ),
+ },
+ ...( typeof __experimentalGetSpacingClassesAndStyles === 'function' && {
+ spacing: {
+ margin: true,
+ padding: true,
},
- __experimentalSelector: '.wc-block-components-product-stock-indicator',
} ),
};
diff --git a/assets/js/atomic/blocks/product-elements/stock-indicator/types.ts b/assets/js/atomic/blocks/product-elements/stock-indicator/types.ts
index b695a3b5f5d..c68e4b51c4e 100644
--- a/assets/js/atomic/blocks/product-elements/stock-indicator/types.ts
+++ b/assets/js/atomic/blocks/product-elements/stock-indicator/types.ts
@@ -1,4 +1,5 @@
export interface BlockAttributes {
productId: number;
isDescendentOfQueryLoop: boolean;
+ isDescendantOfAllProducts: boolean;
}
diff --git a/src/BlockTypes/ProductStockIndicator.php b/src/BlockTypes/ProductStockIndicator.php
index 51ae70dc3fd..f95ad94951e 100644
--- a/src/BlockTypes/ProductStockIndicator.php
+++ b/src/BlockTypes/ProductStockIndicator.php
@@ -22,28 +22,6 @@ class ProductStockIndicator extends AbstractBlock {
*/
protected $api_version = '2';
- /**
- * Get block supports. Shared with the frontend.
- * IMPORTANT: If you change anything here, make sure to update the JS file too.
- *
- * @return array
- */
- protected function get_block_type_supports() {
- return array(
- 'color' =>
- array(
- 'link' => false,
- 'background' => false,
- 'text' => true,
- ),
- 'typography' =>
- array(
- 'fontSize' => true,
- ),
- '__experimentalSelector' => '.wc-block-components-product-stock-indicator',
- );
- }
-
/**
* Register script and style assets for the block type before it is registered.
*
@@ -125,7 +103,7 @@ protected function render( $attributes, $content, $block ) {
$classnames .= $is_on_backorder ? ' wc-block-components-product-stock-indicator--available-on-backorder ' : '';
$output = '';
- $output .= '