From 846e11fddb85070bc72a39b09fee3571f34e5f71 Mon Sep 17 00:00:00 2001 From: Mike Jolley Date: Thu, 25 May 2023 14:46:54 +0100 Subject: [PATCH 1/6] Load frontend styles in editor iframe --- src/BlockTypes/ClassicTemplate.php | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/src/BlockTypes/ClassicTemplate.php b/src/BlockTypes/ClassicTemplate.php index 090f15b8e8d..e61e84771fa 100644 --- a/src/BlockTypes/ClassicTemplate.php +++ b/src/BlockTypes/ClassicTemplate.php @@ -36,7 +36,23 @@ protected function initialize() { parent::initialize(); add_filter( 'render_block', array( $this, 'add_alignment_class_to_wrapper' ), 10, 2 ); add_filter( 'woocommerce_product_query_meta_query', array( $this, 'filter_products_by_stock' ) ); + add_action( 'enqueue_block_assets', array( $this, 'enqueue_block_assets' ) ); + } + /** + * Enqueue assets used for rendering the block in editor context. + * + * This is needed if a block is not yet within the post content--`render` and `enqueue_assets` may not have ran. + */ + public function enqueue_block_assets() { + // Ensures frontend styles for blocks exist in the site editor iframe. + if ( class_exists( 'WC_Frontend_Scripts' ) && is_admin() ) { + $frontend_scripts = new \WC_Frontend_Scripts(); + $styles = $frontend_scripts::get_styles(); + foreach ( $styles as $handle => $style ) { + wp_enqueue_style( $handle, $style['src'], $style['deps'], $style['version'], $style['media'] ); + } + } } /** From ff33d1b45deae8c525cdc81b5297115d4fab9df6 Mon Sep 17 00:00:00 2001 From: Mike Jolley Date: Thu, 25 May 2023 14:50:34 +0100 Subject: [PATCH 2/6] Update placeholder to include skeleton and updated icons --- assets/js/blocks/classic-template/index.tsx | 242 +++++++++++--------- assets/js/blocks/classic-template/types.ts | 1 + 2 files changed, 130 insertions(+), 113 deletions(-) diff --git a/assets/js/blocks/classic-template/index.tsx b/assets/js/blocks/classic-template/index.tsx index b1434cad669..c7d32062912 100644 --- a/assets/js/blocks/classic-template/index.tsx +++ b/assets/js/blocks/classic-template/index.tsx @@ -25,6 +25,7 @@ import { useDispatch, subscribe, useSelect, select } from '@wordpress/data'; import { useEffect, useState } from '@wordpress/element'; import { store as noticesStore } from '@wordpress/notices'; import { useEntityRecord } from '@wordpress/core-data'; +import { woo } from '@woocommerce/icons'; /** * Internal dependencies @@ -128,8 +129,19 @@ const Edit = ( { getButtonLabel, onClickCallback, getBlockifiedTemplate, + getSkeleton, } = conversionConfig[ templateType ]; + const skeleton = getSkeleton ? ( + getSkeleton() + ) : ( + { + ); + const canConvert = isConversionPossible(); const placeholderDescription = getDescription( templateTitle, canConvert ); const [ isPopoverOpen, setIsPopoverOpen ] = useState( false ); @@ -138,123 +150,127 @@ const Edit = ( {
-
-
- - - { __( - 'Classic Product Template', - 'woo-gutenberg-products-block' - ) } - -
-

{ placeholderDescription }

- { canConvert && ( -
-
+
+
+ + WooCommerce + + + { __( + 'Classic Template Placeholder', + 'woo-gutenberg-products-block' + ) } + +
+

+

+ { __( + 'You cannot edit the content of this block. However, you can move it and place other blocks around it.', + 'woo-gutenberg-products-block' + ) } +

+ { canConvert && ( +
+ -
- ) } -
- { + +
+ + ) } + +
+ ) }
diff --git a/assets/js/blocks/classic-template/types.ts b/assets/js/blocks/classic-template/types.ts index f9274706792..fec50b538ac 100644 --- a/assets/js/blocks/classic-template/types.ts +++ b/assets/js/blocks/classic-template/types.ts @@ -31,4 +31,5 @@ export type BlockifiedTemplateConfig = { getDescription: ( templateTitle: string, canConvert: boolean ) => string; getButtonLabel: () => string; onClickCallback: ( params: OnClickCallbackParameter ) => void; + getSkeleton?: ( () => JSX.Element ) | undefined; }; From dce98025efe611d3f8d30e67a275abe9465a4dc3 Mon Sep 17 00:00:00 2001 From: Mike Jolley Date: Thu, 25 May 2023 15:23:39 +0100 Subject: [PATCH 3/6] Update classic template configs --- .../classic-template/archive-product.ts | 20 +- assets/js/blocks/classic-template/editor.scss | 85 ++++--- assets/js/blocks/classic-template/index.tsx | 224 +++++++++--------- .../blocks/classic-template/order-received.ts | 55 ----- .../classic-template/order-received.tsx | 120 ++++++++++ .../product-search-results.ts | 8 +- .../blocks/classic-template/single-product.ts | 8 +- assets/js/blocks/classic-template/types.ts | 16 +- 8 files changed, 318 insertions(+), 218 deletions(-) delete mode 100644 assets/js/blocks/classic-template/order-received.ts create mode 100644 assets/js/blocks/classic-template/order-received.tsx diff --git a/assets/js/blocks/classic-template/archive-product.ts b/assets/js/blocks/classic-template/archive-product.ts index dd93f0b10e0..1a5f391f65d 100644 --- a/assets/js/blocks/classic-template/archive-product.ts +++ b/assets/js/blocks/classic-template/archive-product.ts @@ -147,18 +147,26 @@ const onClickCallbackWithTermDescription = ( { } }; -export const blockifiedProductCatalogConfig = { +const productCatalogBlockifyConfig = { + getButtonLabel, + onClickCallback, getBlockifiedTemplate, +}; + +const productTaxonomyBlockifyConfig = { + getButtonLabel, + onClickCallback: onClickCallbackWithTermDescription, + getBlockifiedTemplate: getBlockifiedTemplateWithTermDescription, +}; + +export const blockifiedProductCatalogConfig = { isConversionPossible, getDescription, - getButtonLabel, - onClickCallback, + blockifyConfig: productCatalogBlockifyConfig, }; export const blockifiedProductTaxonomyConfig = { - getBlockifiedTemplate: getBlockifiedTemplateWithTermDescription, - onClickCallback: onClickCallbackWithTermDescription, isConversionPossible, getDescription, - getButtonLabel, + blockifyConfig: productTaxonomyBlockifyConfig, }; diff --git a/assets/js/blocks/classic-template/editor.scss b/assets/js/blocks/classic-template/editor.scss index c2997f22d81..9728432b163 100644 --- a/assets/js/blocks/classic-template/editor.scss +++ b/assets/js/blocks/classic-template/editor.scss @@ -3,55 +3,66 @@ margin-right: auto; } +.wp-block-woocommerce-classic-template__placeholder-warning { + border-left: 5px solid #2181d2; + padding-left: em(40px); +} +.wp-block-woocommerce-classic-template__placeholder .components-placeholder__fieldset { + display: grid; + grid-template-columns: 1fr; +} +.wp-block-woocommerce-classic-template__placeholder-wireframe, +.wp-block-woocommerce-classic-template__placeholder-copy { + grid-row-start: 1; + grid-column-start: 1; +} .wp-block-woocommerce-classic-template__placeholder-copy { + border: 2px solid $gray-900; + background-color: #fff; + padding: $gap-large $gap-larger; + border-radius: 3px; display: flex; flex-direction: column; max-width: 900px; width: 400px; margin: auto; -} + visibility: hidden; + z-index: 10; -.wp-block-woocommerce-classic-template__placeholder-warning { - border-left: 5px solid #2181d2; - padding-left: em(40px); -} + .wp-block-woocommerce-classic-template__placeholder-copy__icon-container { + margin: 0 0 $gap; -.wp-block-woocommerce-classic-template__placeholder-wireframe { - height: 250px; - background: #e5e5e5; - display: flex; - flex-wrap: wrap; - gap: $gap-large; - margin: auto; + span { + @include font-size(larger); + display: block; + } + .woo-icon { + color: #{$studio-woocommerce-purple}; + @include font-size(large); - @media only screen and (min-width: 768px) { - height: auto; - background: transparent; + svg { + vertical-align: middle; + } + } } -} - -.wp-block-woocommerce-classic-template__placeholder .wp-block-woocommerce-classic-template__placeholder-image { - display: none; - width: 400px; - height: auto; - - @media only screen and (min-width: 768px) { - display: block; + p { + margin: 0 0 $gap; + } + .wp-block-woocommerce-classic-template__placeholder-migration-button-container { + justify-content: center; + margin: $gap 0; } } +.wp-block-woocommerce-classic-template__placeholder-wireframe { + filter: blur(3px); + color: $gray-200 !important; + opacity: 0.5; -.wp-block-woocommerce-classic-template__placeholder-migration-button-container { - justify-content: center; - align-items: center; - margin: 0 auto; -} - -.wp-block-woocommerce-classic-template__placeholder-copy__icon-container { - display: flex; - align-items: center; - gap: $gap-small; - - span { - @include font-size(larger); + // Image based placeholders should fill horizontal width. + > img { + width: 100%; } } +.wp-block-woocommerce-classic-template__placeholder-copy { + visibility: visible; +} diff --git a/assets/js/blocks/classic-template/index.tsx b/assets/js/blocks/classic-template/index.tsx index c7d32062912..dfe086f78ee 100644 --- a/assets/js/blocks/classic-template/index.tsx +++ b/assets/js/blocks/classic-template/index.tsx @@ -57,7 +57,6 @@ const blockifiedFallbackConfig = { isConversionPossible: () => false, getBlockifiedTemplate: () => [], getDescription: () => '', - getButtonLabel: () => '', onClickCallback: () => void 0, }; @@ -79,17 +78,121 @@ const pickBlockClientIds = ( blocks: Array< BlockInstance > ) => return [ ...acc, block.clientId ]; }, [] ); +const ConvertTemplate = ( { blockifyConfig, clientId, attributes } ) => { + const { getButtonLabel, onClickCallback, getBlockifiedTemplate } = + blockifyConfig; + + const [ isPopoverOpen, setIsPopoverOpen ] = useState( false ); + const { replaceBlock, selectBlock, replaceBlocks } = + useDispatch( blockEditorStore ); + + const { getBlocks } = useSelect( ( sel ) => { + return { + getBlocks: sel( blockEditorStore ).getBlocks, + }; + }, [] ); + + const { createInfoNotice } = useDispatch( noticesStore ); + + return ( +
+ +
+ ); +}; + const Edit = ( { clientId, attributes, setAttributes, }: BlockEditProps< Attributes > ) => { - const { replaceBlock, selectBlock, replaceBlocks } = - useDispatch( blockEditorStore ); - - const { getBlocks, editedPostId } = useSelect( ( sel ) => { + const blockProps = useBlockProps(); + const { editedPostId } = useSelect( ( sel ) => { return { - getBlocks: sel( blockEditorStore ).getBlocks, editedPostId: sel( 'core/edit-site' ).getEditedPostId(), }; }, [] ); @@ -102,9 +205,6 @@ const Edit = ( { }; } >( 'postType', 'wp_template', editedPostId ); - const { createInfoNotice } = useDispatch( noticesStore ); - - const blockProps = useBlockProps(); const templateDetails = getTemplateDetailsBySlug( attributes.template, TEMPLATES @@ -126,10 +226,8 @@ const Edit = ( { const { isConversionPossible, getDescription, - getButtonLabel, - onClickCallback, - getBlockifiedTemplate, getSkeleton, + blockifyConfig, } = conversionConfig[ templateType ]; const skeleton = getSkeleton ? ( @@ -144,7 +242,6 @@ const Edit = ( { const canConvert = isConversionPossible(); const placeholderDescription = getDescription( templateTitle, canConvert ); - const [ isPopoverOpen, setIsPopoverOpen ] = useState( false ); return (
@@ -175,101 +272,12 @@ const Edit = ( { 'woo-gutenberg-products-block' ) }

- { canConvert && ( -
- -
+ { canConvert && blockifyConfig && ( + ) }
diff --git a/assets/js/blocks/classic-template/order-received.ts b/assets/js/blocks/classic-template/order-received.ts deleted file mode 100644 index 9df2ebf7a49..00000000000 --- a/assets/js/blocks/classic-template/order-received.ts +++ /dev/null @@ -1,55 +0,0 @@ -/** - * External dependencies - */ - -import { __, sprintf } from '@wordpress/i18n'; - -/** - * Internal dependencies - */ - -const getBlockifiedTemplate = () => []; - -const isConversionPossible = () => { - // Blockification is possible for the WP version 6.1 and above, - // which are the versions the Products block supports. - return false; -}; - -const getDescriptionAllowingConversion = ( templateTitle: string ) => - sprintf( - /* translators: %s is the template title */ - __( - "This block serves as a placeholder for your %s. We recommend upgrading to the Order Received block for more features and to edit it visually. Don't worry, you can always revert back.", - '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 thankyou.php template. 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 = () => - __( 'Upgrade to Products block', 'woo-gutenberg-products-block' ); - -export { - getBlockifiedTemplate, - isConversionPossible, - getDescription, - getButtonLabel, -}; diff --git a/assets/js/blocks/classic-template/order-received.tsx b/assets/js/blocks/classic-template/order-received.tsx new file mode 100644 index 00000000000..0b3d1e305a0 --- /dev/null +++ b/assets/js/blocks/classic-template/order-received.tsx @@ -0,0 +1,120 @@ +/** + * External dependencies + */ + +import { __ } from '@wordpress/i18n'; + +const isConversionPossible = () => { + return false; +}; + +const getDescription = () => { + return __( + 'This block represents the thankyou.php classic template. The actual rendered template may appear different from this placeholder.', + 'woo-gutenberg-products-block' + ); +}; + +const getSkeleton = () => { + return ( +
+
+

Order received

+

+ Thank you. Your order has been received. +

+
    +
  • + Order number: 123 +
  • +
  • + Date: May 25, 2023 +
  • +
  • + Email: shopper@woo.com +
  • +
  • + Total: $20 +
  • +
  • + Payment method: Credit Card +
  • +
+ +
+

+ Order details +

+ + + + + + + + + + + + + + + + + + + + + + + + +
+ Product + + Total +
+ Product{ ' ' } + + × 2 + { ' ' } + + $20.00 +
Subtotal:$20.00
Total:$20.00
+
+ +
+
+
+

+ Billing address +

+
+ 123 Main St +
+ New York, NY 10001 +
+ United States (US) +
+
+ +
+

+ Shipping address +

+
+ 123 Main St +
+ New York, NY 10001 +
+ United States (US) +
+
+
+
+
+
+ ); +}; + +export { isConversionPossible, getDescription, getSkeleton }; diff --git a/assets/js/blocks/classic-template/product-search-results.ts b/assets/js/blocks/classic-template/product-search-results.ts index 44587355401..6ea531e22f2 100644 --- a/assets/js/blocks/classic-template/product-search-results.ts +++ b/assets/js/blocks/classic-template/product-search-results.ts @@ -169,10 +169,10 @@ const onClickCallback = ( { const getButtonLabel = () => __( 'Transform into blocks', 'woo-gutenberg-products-block' ); -export { - getBlockifiedTemplate, - isConversionPossible, - getDescription, +const blockifyConfig = { getButtonLabel, onClickCallback, + getBlockifiedTemplate, }; + +export { isConversionPossible, getDescription, blockifyConfig }; diff --git a/assets/js/blocks/classic-template/single-product.ts b/assets/js/blocks/classic-template/single-product.ts index 1dca28bffa3..439e149dde6 100644 --- a/assets/js/blocks/classic-template/single-product.ts +++ b/assets/js/blocks/classic-template/single-product.ts @@ -112,10 +112,10 @@ const onClickCallback = ( { } }; -export { - getBlockifiedTemplate, - isConversionPossible, - getDescription, +const blockifyConfig = { getButtonLabel, onClickCallback, + getBlockifiedTemplate, }; + +export { isConversionPossible, getDescription, blockifyConfig }; diff --git a/assets/js/blocks/classic-template/types.ts b/assets/js/blocks/classic-template/types.ts index fec50b538ac..30995413933 100644 --- a/assets/js/blocks/classic-template/types.ts +++ b/assets/js/blocks/classic-template/types.ts @@ -23,13 +23,21 @@ export type OnClickCallbackParameter = { selectBlock: ( clientId: string ) => void; }; -export type BlockifiedTemplateConfig = { +type ConversionConfig = { + onClickCallback: ( params: OnClickCallbackParameter ) => void; + getButtonLabel: () => string; getBlockifiedTemplate: ( inheritedAttributes: InheritedAttributes ) => BlockInstance[]; - isConversionPossible: () => boolean; +}; + +export type BlockifiedTemplateConfig = { + // Description of the template, shown in the block placeholder. getDescription: ( templateTitle: string, canConvert: boolean ) => string; - getButtonLabel: () => string; - onClickCallback: ( params: OnClickCallbackParameter ) => void; + // Returns the skeleton HTML for the template, or can be left blank to use the default fallback image. getSkeleton?: ( () => JSX.Element ) | undefined; + // Is conversion possible for the template? + isConversionPossible: () => boolean; + // If conversion is possible, returns the config for the template to be blockified. + blockifyConfig?: ConversionConfig | undefined; }; From a138a82156d3b9e5d8b8be398997f057dd4c5781 Mon Sep 17 00:00:00 2001 From: Paulo Arromba <17236129+wavvves@users.noreply.github.com> Date: Fri, 26 May 2023 16:39:38 +0100 Subject: [PATCH 4/6] 1px border for .wp-block-woocommerce-classic-template__placeholder-copy --- assets/js/blocks/classic-template/editor.scss | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/assets/js/blocks/classic-template/editor.scss b/assets/js/blocks/classic-template/editor.scss index 9728432b163..829cd9d0c70 100644 --- a/assets/js/blocks/classic-template/editor.scss +++ b/assets/js/blocks/classic-template/editor.scss @@ -17,7 +17,7 @@ grid-column-start: 1; } .wp-block-woocommerce-classic-template__placeholder-copy { - border: 2px solid $gray-900; + border: 1px solid $gray-900; background-color: #fff; padding: $gap-large $gap-larger; border-radius: 3px; From f5f891e63694f6f22b98271d6e106b184eabcfa1 Mon Sep 17 00:00:00 2001 From: Mike Jolley Date: Tue, 30 May 2023 12:29:53 +0100 Subject: [PATCH 5/6] Show copy on focus --- assets/js/blocks/classic-template/editor.scss | 32 +++++++++++++++---- 1 file changed, 26 insertions(+), 6 deletions(-) diff --git a/assets/js/blocks/classic-template/editor.scss b/assets/js/blocks/classic-template/editor.scss index 829cd9d0c70..232285a91d5 100644 --- a/assets/js/blocks/classic-template/editor.scss +++ b/assets/js/blocks/classic-template/editor.scss @@ -15,6 +15,7 @@ .wp-block-woocommerce-classic-template__placeholder-copy { grid-row-start: 1; grid-column-start: 1; + transition: 0.3s all ease; } .wp-block-woocommerce-classic-template__placeholder-copy { border: 1px solid $gray-900; @@ -26,7 +27,7 @@ max-width: 900px; width: 400px; margin: auto; - visibility: hidden; + opacity: 0; z-index: 10; .wp-block-woocommerce-classic-template__placeholder-copy__icon-container { @@ -54,15 +55,34 @@ } } .wp-block-woocommerce-classic-template__placeholder-wireframe { - filter: blur(3px); - color: $gray-200 !important; - opacity: 0.5; + pointer-events: none; // Image based placeholders should fill horizontal width. > img { width: 100%; } } -.wp-block-woocommerce-classic-template__placeholder-copy { - visibility: visible; + +.wp-block-woocommerce-legacy-template { + .components-placeholder { + box-shadow: none; + padding: 0; + } +} +.wp-block-woocommerce-legacy-template.is-selected { + .wp-block-woocommerce-classic-template__placeholder-wireframe { + filter: blur(3px); + opacity: 0.5; + + * { + color: $gray-200 !important; + border-color: $gray-200 !important; + } + } + .wp-block-woocommerce-classic-template__placeholder-copy { + opacity: 1; + } + .components-placeholder { + box-shadow: inherit; + } } From 7f591f1817ff3835f189e52deec95c7876987e07 Mon Sep 17 00:00:00 2001 From: Mike Jolley Date: Tue, 30 May 2023 13:43:16 +0100 Subject: [PATCH 6/6] Sample data --- .../classic-template/order-received.tsx | 67 ++++++++++++++----- 1 file changed, 50 insertions(+), 17 deletions(-) diff --git a/assets/js/blocks/classic-template/order-received.tsx b/assets/js/blocks/classic-template/order-received.tsx index 0b3d1e305a0..2408f344151 100644 --- a/assets/js/blocks/classic-template/order-received.tsx +++ b/assets/js/blocks/classic-template/order-received.tsx @@ -19,47 +19,62 @@ const getSkeleton = () => { return (
-

Order received

+

+ { __( 'Order received', 'woo-gutenberg-products-block' ) } +

- Thank you. Your order has been received. + { __( + 'Thank you. Your order has been received.', + 'woo-gutenberg-products-block' + ) }

  • - Order number: 123 + { __( 'Order number', 'woo-gutenberg-products-block' ) } + : 123
  • - Date: May 25, 2023 + { __( 'Date', 'woo-gutenberg-products-block' ) }:{ ' ' } + May 25, 2023
  • - Email: shopper@woo.com + { __( 'Email', 'woo-gutenberg-products-block' ) }:{ ' ' } + shopper@woo.com
  • - Total: $20 -
  • -
  • - Payment method: Credit Card + { __( 'Total', 'woo-gutenberg-products-block' ) }:{ ' ' } + $20.00

- Order details + { __( + 'Order details', + 'woo-gutenberg-products-block' + ) }

- + - + @@ -87,7 +114,10 @@ const getSkeleton = () => {

- Billing address + { __( + 'Billing address', + 'woo-gutenberg-products-block' + ) }

123 Main St @@ -100,7 +130,10 @@ const getSkeleton = () => {

- Shipping address + { __( + 'Shipping address', + 'woo-gutenberg-products-block' + ) }

123 Main St
- Product + { __( + 'Product', + 'woo-gutenberg-products-block' + ) } - Total + { __( + 'Total', + 'woo-gutenberg-products-block' + ) }
- Product{ ' ' } + Sample Product{ ' ' } × 2 { ' ' } @@ -72,11 +87,23 @@ const getSkeleton = () => {
Subtotal: + { __( + 'Subtotal', + 'woo-gutenberg-products-block' + ) } + : + $20.00
Total: + { __( + 'Total', + 'woo-gutenberg-products-block' + ) } + : + $20.00