diff --git a/lib/block-supports/layout.php b/lib/block-supports/layout.php index 15a7a131f8a9b0..b61311362893a6 100644 --- a/lib/block-supports/layout.php +++ b/lib/block-supports/layout.php @@ -291,6 +291,23 @@ function gutenberg_get_layout_style( $selector, $layout, $has_block_gap_support return ''; } +/** + * Gets classname from last tag in a string of HTML. + * + * @param string $html markup to be processed. + * @return string String of inner wrapper classnames. + */ +function gutenberg_get_classnames_from_last_tag( $html ) { + $tags = new WP_HTML_Tag_Processor( $html ); + $last_classnames = ''; + + while ( $tags->next_tag() ) { + $last_classnames = $tags->get_attribute( 'class' ); + } + + return $last_classnames; +} + /** * Renders the layout config to the block wrapper. * @@ -320,7 +337,6 @@ function gutenberg_render_layout_support_flag( $block_content, $block ) { $class_names = array(); $layout_definitions = _wp_array_get( $global_layout_settings, array( 'definitions' ), array() ); - $block_classname = wp_get_block_default_classname( $block['blockName'] ); $container_class = wp_unique_id( 'wp-container-' ); $layout_classname = ''; @@ -397,7 +413,7 @@ function gutenberg_render_layout_support_flag( $block_content, $block ) { $should_skip_gap_serialization = gutenberg_should_skip_block_supports_serialization( $block_type, 'spacing', 'blockGap' ); $style = gutenberg_get_layout_style( - ".$block_classname.$container_class", + ".$container_class.$container_class", $used_layout, $has_block_gap_support, $gap_value, @@ -412,16 +428,22 @@ function gutenberg_render_layout_support_flag( $block_content, $block ) { } } - /* - * This assumes the hook only applies to blocks with a single wrapper. - * A limitation of this hook is that nested inner blocks wrappers are not yet supported. - */ - $content = preg_replace( - '/' . preg_quote( 'class="', '/' ) . '/', - 'class="' . esc_attr( implode( ' ', $class_names ) ) . ' ', - $block_content, - 1 - ); + /** + * The first chunk of innerContent contains the block markup up until the inner blocks start. + * We want to target the opening tag of the inner blocks wrapper, which is the last tag in that chunk. + */ + $inner_content_classnames = isset( $block['innerContent'][0] ) && 'string' === gettype( $block['innerContent'][0] ) ? gutenberg_get_classnames_from_last_tag( $block['innerContent'][0] ) : ''; + + $content = new WP_HTML_Tag_Processor( $block_content ); + if ( $inner_content_classnames ) { + $content->next_tag( array( 'class_name' => $inner_content_classnames ) ); + foreach ( $class_names as $class_name ) { + $content->add_class( $class_name ); + } + } else { + $content->next_tag(); + $content->add_class( implode( ' ', $class_names ) ); + } return $content; } @@ -433,6 +455,7 @@ function gutenberg_render_layout_support_flag( $block_content, $block ) { 'register_attribute' => 'gutenberg_register_layout_support', ) ); + if ( function_exists( 'wp_render_layout_support_flag' ) ) { remove_filter( 'render_block', 'wp_render_layout_support_flag' ); } diff --git a/packages/block-editor/src/components/block-edit/index.js b/packages/block-editor/src/components/block-edit/index.js index 9b2875b3018a30..747e25b820847d 100644 --- a/packages/block-editor/src/components/block-edit/index.js +++ b/packages/block-editor/src/components/block-edit/index.js @@ -20,11 +20,12 @@ import { BlockEditContextProvider, useBlockEditContext } from './context'; export { useBlockEditContext }; export default function BlockEdit( props ) { - const { name, isSelected, clientId } = props; + const { name, isSelected, clientId, __unstableLayoutClassNames } = props; const context = { name, isSelected, clientId, + __unstableLayoutClassNames, }; return ( ); diff --git a/packages/block-editor/src/components/inner-blocks/index.js b/packages/block-editor/src/components/inner-blocks/index.js index 4105d7291957ed..8afc6482eb33fa 100644 --- a/packages/block-editor/src/components/inner-blocks/index.js +++ b/packages/block-editor/src/components/inner-blocks/index.js @@ -150,7 +150,9 @@ const ForwardedInnerBlocks = forwardRef( ( props, ref ) => { * @see https://github.com/WordPress/gutenberg/blob/HEAD/packages/block-editor/src/components/inner-blocks/README.md */ export function useInnerBlocksProps( props = {}, options = {} ) { - const { clientId } = useBlockEditContext(); + const { __unstableDisableLayoutClassNames } = options; + const { clientId, __unstableLayoutClassNames: layoutClassNames = '' } = + useBlockEditContext(); const isSmallScreen = useViewportMatch( 'medium', '<' ); const { __experimentalCaptureToolbars, hasOverlay } = useSelect( ( select ) => { @@ -200,12 +202,14 @@ export function useInnerBlocksProps( props = {}, options = {} ) { innerBlocksProps.value && innerBlocksProps.onChange ? ControlledInnerBlocks : UncontrolledInnerBlocks; + return { ...props, ref, className: classnames( props.className, 'block-editor-block-list__layout', + __unstableDisableLayoutClassNames ? '' : layoutClassNames, { 'has-overlay': hasOverlay, } diff --git a/packages/block-editor/src/hooks/layout.js b/packages/block-editor/src/hooks/layout.js index 73d655c57c2acb..eeea8f3df2c8ce 100644 --- a/packages/block-editor/src/hooks/layout.js +++ b/packages/block-editor/src/hooks/layout.js @@ -9,11 +9,7 @@ import { kebabCase } from 'lodash'; */ import { createHigherOrderComponent, useInstanceId } from '@wordpress/compose'; import { addFilter } from '@wordpress/hooks'; -import { - getBlockDefaultClassName, - getBlockSupport, - hasBlockSupport, -} from '@wordpress/blocks'; +import { getBlockSupport, hasBlockSupport } from '@wordpress/blocks'; import { useSelect } from '@wordpress/data'; import { Button, @@ -366,9 +362,8 @@ export const withLayoutStyles = createHigherOrderComponent( const layoutClasses = hasLayoutBlockSupport ? useLayoutClasses( block ) : null; - const selector = `.${ getBlockDefaultClassName( - name - ) }.wp-container-${ id }`; + // Higher specificity to override defaults from theme.json. + const selector = `.wp-container-${ id }.wp-container-${ id }`; const blockGapSupport = useSetting( 'spacing.blockGap' ); const hasBlockGapSupport = blockGapSupport !== null; @@ -413,7 +408,10 @@ export const withLayoutStyles = createHigherOrderComponent( />, element ) } - + ); } diff --git a/packages/block-library/src/gallery/gallery.js b/packages/block-library/src/gallery/gallery.js index 9a551b6e0d9f78..e6176cc8a7256c 100644 --- a/packages/block-library/src/gallery/gallery.js +++ b/packages/block-library/src/gallery/gallery.js @@ -26,6 +26,7 @@ export const Gallery = ( props ) => { mediaPlaceholder, insertBlocksAfter, blockProps, + __unstableLayoutClassNames: layoutClassNames, } = props; const { align, columns, caption, imageCrop } = attributes; @@ -42,6 +43,7 @@ export const Gallery = ( props ) => { { ...innerBlocksProps } className={ classnames( blockProps.className, + layoutClassNames, 'blocks-gallery-grid', { [ `align${ align }` ]: align, diff --git a/packages/block-library/src/group/edit.js b/packages/block-library/src/group/edit.js index 9d43c4339ba1f5..efe0d0ede9ebc8 100644 --- a/packages/block-library/src/group/edit.js +++ b/packages/block-library/src/group/edit.js @@ -35,7 +35,12 @@ const htmlElementMessages = { ), }; -function GroupEdit( { attributes, setAttributes, clientId } ) { +function GroupEdit( { + attributes, + setAttributes, + clientId, + __unstableLayoutClassNames: layoutClassNames, +} ) { const { hasInnerBlocks, themeSupportsLayout } = useSelect( ( select ) => { const { getBlock, getSettings } = select( blockEditorStore ); @@ -55,7 +60,9 @@ function GroupEdit( { attributes, setAttributes, clientId } ) { const { type = 'default' } = usedLayout; const layoutSupportEnabled = themeSupportsLayout || type === 'flex'; - const blockProps = useBlockProps(); + const blockProps = useBlockProps( { + className: ! layoutSupportEnabled ? layoutClassNames : null, + } ); const innerBlocksProps = useInnerBlocksProps( layoutSupportEnabled @@ -67,6 +74,7 @@ function GroupEdit( { attributes, setAttributes, clientId } ) { ? undefined : InnerBlocks.ButtonBlockAppender, __experimentalLayout: layoutSupportEnabled ? usedLayout : undefined, + __unstableDisableLayoutClassNames: ! layoutSupportEnabled, } ); diff --git a/packages/block-library/src/post-content/edit.js b/packages/block-library/src/post-content/edit.js index 1949553b2540f6..e190101aa05155 100644 --- a/packages/block-library/src/post-content/edit.js +++ b/packages/block-library/src/post-content/edit.js @@ -84,8 +84,8 @@ function Content( props ) { ); } -function Placeholder() { - const blockProps = useBlockProps(); +function Placeholder( { layoutClassNames } ) { + const blockProps = useBlockProps( { className: layoutClassNames } ); return (

@@ -118,7 +118,11 @@ function RecursionError() { ); } -export default function PostContentEdit( { context, attributes } ) { +export default function PostContentEdit( { + context, + attributes, + __unstableLayoutClassNames: layoutClassNames, +} ) { const { postId: contextPostId, postType: contextPostType } = context; const { layout = {} } = attributes; const hasAlreadyRendered = useHasRecursion( contextPostId ); @@ -132,7 +136,7 @@ export default function PostContentEdit( { context, attributes } ) { { contextPostId && contextPostType ? ( ) : ( - + ) } );