From 525d7cb1d5bef71f0e1981bce594d42cc2efe0f5 Mon Sep 17 00:00:00 2001 From: Kai Hao Date: Wed, 9 Nov 2022 11:13:33 +0800 Subject: [PATCH 1/3] Add basic shuffle patterns to the toolbar --- .../block-editor/src/hooks/content-lock-ui.js | 150 +++++++++++++++--- packages/block-editor/src/hooks/utils.js | 86 +++++++++- 2 files changed, 208 insertions(+), 28 deletions(-) diff --git a/packages/block-editor/src/hooks/content-lock-ui.js b/packages/block-editor/src/hooks/content-lock-ui.js index 2de6d762ceadbb..751f4f6344e22b 100644 --- a/packages/block-editor/src/hooks/content-lock-ui.js +++ b/packages/block-editor/src/hooks/content-lock-ui.js @@ -7,12 +7,18 @@ import { useDispatch, useSelect } from '@wordpress/data'; import { addFilter } from '@wordpress/hooks'; import { __ } from '@wordpress/i18n'; import { useEffect, useRef, useCallback } from '@wordpress/element'; +import { store as blocksStore } from '@wordpress/blocks'; /** * Internal dependencies */ import { store as blockEditorStore } from '../store'; import { BlockControls, BlockSettingsMenuControls } from '../components'; +import { + flattenBlocks, + replaceContentsInBlocks, + areBlocksAlike, +} from './utils'; /** * External dependencies */ @@ -41,6 +47,89 @@ function StopEditingAsBlocksOnOutsideSelect( { return null; } +function ShufflePatternsToolbarItem( { clientId } ) { + // TODO: Probably worth to add this to blocks' selectors. + const getFlattenContentBlocks = useSelect( ( select ) => { + const contentBlockNames = select( blocksStore ) + .getBlockTypes() + .filter( + ( blockType ) => + blockType.name !== 'core/list-item' && + Object.values( blockType.attributes ).some( + ( attribute ) => + attribute.__experimentalRole === 'content' + ) + ) + .map( ( blockType ) => blockType.name ); + + return ( blocks ) => + flattenBlocks( blocks ).filter( ( block ) => + contentBlockNames.includes( block.name ) + ); + }, [] ); + const { contentBlocks, patterns } = useSelect( + ( select ) => { + const blocks = + select( blockEditorStore ).getBlocksByClientId( clientId ); + const _contentBlocks = getFlattenContentBlocks( blocks ); + const allPatterns = + select( blockEditorStore ).__experimentalGetAllowedPatterns(); + + return { + contentBlocks: _contentBlocks, + patterns: allPatterns + .filter( ( pattern ) => { + const patternContentBlocks = getFlattenContentBlocks( + pattern.blocks + ); + return ( + patternContentBlocks.length === + _contentBlocks.length && + _contentBlocks.every( + ( block, index ) => + block.name === + patternContentBlocks[ index ].name + ) + ); + } ) + .filter( + ( pattern ) => + ! areBlocksAlike( blocks, pattern.blocks ) + ), + }; + }, + [ clientId, getFlattenContentBlocks ] + ); + const { replaceBlocks } = useDispatch( blockEditorStore ); + + function shuffle() { + // We're not using `Math.random` for instance ids here. + // eslint-disable-next-line no-restricted-syntax + const randomNumber = Math.floor( Math.random() * patterns.length ); + const pattern = patterns[ randomNumber ]; + const replacedPatternBlocks = replaceContentsInBlocks( + pattern.blocks, + contentBlocks + ).map( ( block ) => { + block.attributes.templateLock = 'contentOnly'; + return block; + } ); + replaceBlocks( clientId, replacedPatternBlocks ); + } + + if ( patterns.length === 0 ) { + return null; + } + + return ( + + + { __( 'Shuffle' ) } + + + ); +} + export const withBlockControls = createHigherOrderComponent( ( BlockEdit ) => ( props ) => { const { getBlockListSettings, getSettings } = @@ -124,33 +213,42 @@ export const withBlockControls = createHigherOrderComponent( ) } { ! isEditingAsBlocks && isContentLocked && props.isSelected && ( - - { ( { onClose } ) => ( - { - __unstableMarkNextChangeAsNotPersistent(); - updateBlockAttributes( props.clientId, { - templateLock: undefined, - } ); - updateBlockListSettings( props.clientId, { - ...getBlockListSettings( + <> + + { ( { onClose } ) => ( + { + __unstableMarkNextChangeAsNotPersistent(); + updateBlockAttributes( props.clientId, { + templateLock: undefined, + } ); + updateBlockListSettings( + props.clientId, + { + ...getBlockListSettings( + props.clientId + ), + templateLock: false, + } + ); + focusModeToRevert.current = + getSettings().focusMode; + updateSettings( { focusMode: true } ); + __unstableSetTemporarilyEditingAsBlocks( props.clientId - ), - templateLock: false, - } ); - focusModeToRevert.current = - getSettings().focusMode; - updateSettings( { focusMode: true } ); - __unstableSetTemporarilyEditingAsBlocks( - props.clientId - ); - onClose(); - } } - > - { __( 'Modify' ) } - - ) } - + ); + onClose(); + } } + > + { __( 'Modify' ) } + + ) } + + + + ) } [ + block, + ...flattenBlocks( block.innerBlocks ), + ] ); +} + +export function replaceContentsInBlocks( sourceBlocks, contentBlocks ) { + const [ contentBlock, ...restContentBlocks ] = contentBlocks; + return sourceBlocks.map( ( block ) => { + if ( block.name !== contentBlock.name ) { + return createBlock( + block.name, + block.attributes, + replaceContentsInBlocks( block.innerBlocks, contentBlocks ) + ); + } + + const blockTypeAttributes = getBlockType( block.name ).attributes; + return createBlock( + block.name, + Object.keys( block.attributes ).reduce( + ( attributes, attributeName ) => { + if ( + blockTypeAttributes[ attributeName ] + .__experimentalRole === 'content' + ) { + attributes[ attributeName ] = + contentBlock.attributes[ attributeName ]; + } else { + attributes[ attributeName ] = + block.attributes[ attributeName ]; + } + return attributes; + }, + {} + ), + replaceContentsInBlocks( block.innerBlocks, restContentBlocks ) + ); + } ); +} + +export function areBlocksAlike( sourceBlocks, targetBlocks ) { + if ( sourceBlocks.length !== targetBlocks.length ) { + return false; + } + + for ( let index = 0; index < sourceBlocks.length; index += 1 ) { + const sourceBlock = sourceBlocks[ index ]; + const targetBlock = targetBlocks[ index ]; + + if ( sourceBlock.name !== targetBlock.name ) { + return false; + } + + const blockTypeAttributes = getBlockType( sourceBlock.name ).attributes; + + if ( + Object.keys( blockTypeAttributes ).some( + ( attribute ) => + attribute !== 'templateLock' && + blockTypeAttributes[ attribute ].__experimentalRole !== + 'content' && + ! isEqual( + sourceBlock.attributes[ attribute ], + targetBlock.attributes[ attribute ] + ) + ) + ) { + return false; + } + + if ( + ! areBlocksAlike( sourceBlock.innerBlocks, targetBlock.innerBlocks ) + ) { + return false; + } + } + + return true; +} From b35142a3681eae831a6775f41f0549dced25351f Mon Sep 17 00:00:00 2001 From: Kai Hao Date: Tue, 22 Nov 2022 16:36:02 +0800 Subject: [PATCH 2/3] Filter out 'design' blocks --- .../block-editor/src/hooks/content-lock-ui.js | 46 +++++++++++++++---- 1 file changed, 36 insertions(+), 10 deletions(-) diff --git a/packages/block-editor/src/hooks/content-lock-ui.js b/packages/block-editor/src/hooks/content-lock-ui.js index 751f4f6344e22b..888596dc64e00d 100644 --- a/packages/block-editor/src/hooks/content-lock-ui.js +++ b/packages/block-editor/src/hooks/content-lock-ui.js @@ -7,7 +7,7 @@ import { useDispatch, useSelect } from '@wordpress/data'; import { addFilter } from '@wordpress/hooks'; import { __ } from '@wordpress/i18n'; import { useEffect, useRef, useCallback } from '@wordpress/element'; -import { store as blocksStore } from '@wordpress/blocks'; +import { store as blocksStore, getBlockType } from '@wordpress/blocks'; /** * Internal dependencies @@ -47,6 +47,36 @@ function StopEditingAsBlocksOnOutsideSelect( { return null; } +function filterBlocksForShuffle( blocks ) { + return flattenBlocks( blocks ).filter( ( block ) => { + const blockType = getBlockType( block.name ); + + return blockType.category !== 'design'; + } ); +} + +function compareFilteredBlocks( sourceBlocks, targetBlocks ) { + if ( sourceBlocks.length !== targetBlocks.length ) { + return false; + } + + const targetBlockNames = {}; + for ( const targetBlock of targetBlocks ) { + targetBlockNames[ targetBlock.name ] ??= 0; + targetBlockNames[ targetBlock.name ] += 1; + } + + for ( const sourceBlock of sourceBlocks ) { + if ( ! targetBlockNames[ sourceBlock.name ] ) { + return false; + } + + targetBlockNames[ sourceBlock.name ] -= 1; + } + + return true; +} + function ShufflePatternsToolbarItem( { clientId } ) { // TODO: Probably worth to add this to blocks' selectors. const getFlattenContentBlocks = useSelect( ( select ) => { @@ -71,6 +101,7 @@ function ShufflePatternsToolbarItem( { clientId } ) { ( select ) => { const blocks = select( blockEditorStore ).getBlocksByClientId( clientId ); + const filteredBlocks = filterBlocksForShuffle( blocks ); const _contentBlocks = getFlattenContentBlocks( blocks ); const allPatterns = select( blockEditorStore ).__experimentalGetAllowedPatterns(); @@ -79,17 +110,12 @@ function ShufflePatternsToolbarItem( { clientId } ) { contentBlocks: _contentBlocks, patterns: allPatterns .filter( ( pattern ) => { - const patternContentBlocks = getFlattenContentBlocks( + const filteredPatternBlocks = filterBlocksForShuffle( pattern.blocks ); - return ( - patternContentBlocks.length === - _contentBlocks.length && - _contentBlocks.every( - ( block, index ) => - block.name === - patternContentBlocks[ index ].name - ) + return compareFilteredBlocks( + filteredBlocks, + filteredPatternBlocks ); } ) .filter( From 30845c69f35e54e8cd7c6f24feecd08768a299ea Mon Sep 17 00:00:00 2001 From: Kai Hao Date: Tue, 22 Nov 2022 17:08:09 +0800 Subject: [PATCH 3/3] Use __experimentalLayout --- packages/block-editor/src/hooks/content-lock-ui.js | 10 ++++------ 1 file changed, 4 insertions(+), 6 deletions(-) diff --git a/packages/block-editor/src/hooks/content-lock-ui.js b/packages/block-editor/src/hooks/content-lock-ui.js index 888596dc64e00d..3b5e1fcaca3557 100644 --- a/packages/block-editor/src/hooks/content-lock-ui.js +++ b/packages/block-editor/src/hooks/content-lock-ui.js @@ -7,7 +7,7 @@ import { useDispatch, useSelect } from '@wordpress/data'; import { addFilter } from '@wordpress/hooks'; import { __ } from '@wordpress/i18n'; import { useEffect, useRef, useCallback } from '@wordpress/element'; -import { store as blocksStore, getBlockType } from '@wordpress/blocks'; +import { store as blocksStore, hasBlockSupport } from '@wordpress/blocks'; /** * Internal dependencies @@ -48,11 +48,9 @@ function StopEditingAsBlocksOnOutsideSelect( { } function filterBlocksForShuffle( blocks ) { - return flattenBlocks( blocks ).filter( ( block ) => { - const blockType = getBlockType( block.name ); - - return blockType.category !== 'design'; - } ); + return flattenBlocks( blocks ).filter( + ( block ) => ! hasBlockSupport( block.name, '__experimentalLayout' ) + ); } function compareFilteredBlocks( sourceBlocks, targetBlocks ) {