From dd68caaf6dc87425e15c71dc0e76d354ca16e41a Mon Sep 17 00:00:00 2001 From: youknowriad Date: Tue, 25 May 2021 07:36:58 +0000 Subject: [PATCH] Block Editor: Add the layout block support. The layout block allows containers to define the size of their inner blocks and the allowed allignments. It's only enabled for themes with theme.json files. Props nosolosw. See #53175. Built from https://develop.svn.wordpress.org/trunk@50991 git-svn-id: http://core.svn.wordpress.org/trunk@50600 1a063a9b-81f0-0310-95a4-ce76da25c4cd --- wp-admin/edit-form-blocks.php | 1 + wp-includes/block-supports/layout.php | 100 +++++++++++++++++++++++++- wp-includes/version.php | 2 +- 3 files changed, 101 insertions(+), 2 deletions(-) diff --git a/wp-admin/edit-form-blocks.php b/wp-admin/edit-form-blocks.php index c2fa7d0019..848157de96 100644 --- a/wp-admin/edit-form-blocks.php +++ b/wp-admin/edit-form-blocks.php @@ -209,6 +209,7 @@ $editor_settings = array( 'unlockNonce' => wp_create_nonce( 'update-post_' . $post->ID ), 'ajaxUrl' => admin_url( 'admin-ajax.php' ), ), + 'supportsLayout' => WP_Theme_JSON_Resolver::theme_has_support(), '__experimentalBlockPatterns' => WP_Block_Patterns_Registry::get_instance()->get_all_registered(), '__experimentalBlockPatternCategories' => WP_Block_Pattern_Categories_Registry::get_instance()->get_all_registered(), diff --git a/wp-includes/block-supports/layout.php b/wp-includes/block-supports/layout.php index 76240fa15c..a28a51b4a3 100644 --- a/wp-includes/block-supports/layout.php +++ b/wp-includes/block-supports/layout.php @@ -6,6 +6,104 @@ * @since 5.8.0 */ +/** + * Registers the layout block attribute for block types that support it. + * + * @since 5.8.0 + * @access private + * + * @param WP_Block_Type $block_type Block Type. + */ +function wp_register_layout_support( $block_type ) { + $support_layout = block_has_support( $block_type, array( '__experimentalLayout' ), false ); + if ( $support_layout ) { + if ( ! $block_type->attributes ) { + $block_type->attributes = array(); + } + + if ( ! array_key_exists( 'layout', $block_type->attributes ) ) { + $block_type->attributes['layout'] = array( + 'type' => 'object', + ); + } + } +} + +/** + * Renders the layout config to the block wrapper. + * + * @since 5.8.0 + * @access private + * + * @param string $block_content Rendered block content. + * @param array $block Block object. + * @return string Filtered block content. + */ +function wp_render_layout_support_flag( $block_content, $block ) { + $block_type = WP_Block_Type_Registry::get_instance()->get_registered( $block['blockName'] ); + $support_layout = block_has_support( $block_type, array( '__experimentalLayout' ), false ); + if ( ! $support_layout || ! isset( $block['attrs']['layout'] ) ) { + return $block_content; + } + + $used_layout = $block['attrs']['layout']; + if ( isset( $used_layout['inherit'] ) && $used_layout['inherit'] ) { + $tree = WP_Theme_JSON_Resolver::get_merged_data(); + $default_layout = _wp_array_get( $tree->get_settings(), array( 'layout' ) ); + if ( ! $default_layout ) { + return $block_content; + } + $used_layout = $default_layout; + } + + $id = uniqid(); + $content_size = isset( $used_layout['contentSize'] ) ? $used_layout['contentSize'] : null; + $wide_size = isset( $used_layout['wideSize'] ) ? $used_layout['wideSize'] : null; + + $all_max_width_value = $content_size ? $content_size : $wide_size; + $wide_max_width_value = $wide_size ? $wide_size : $content_size; + + // Make sure there is a single CSS rule, and all tags are stripped for security. + $all_max_width_value = safecss_filter_attr( explode( ';', $all_max_width_value )[0] ); + $wide_max_width_value = safecss_filter_attr( explode( ';', $wide_max_width_value )[0] ); + + $style = ''; + if ( $content_size || $wide_size ) { + $style = ".wp-container-$id > * {"; + $style .= 'max-width: ' . esc_html( $all_max_width_value ) . ';'; + $style .= 'margin-left: auto !important;'; + $style .= 'margin-right: auto !important;'; + $style .= '}'; + + $style .= ".wp-container-$id > .alignwide { max-width: " . esc_html( $wide_max_width_value ) . ';}'; + + $style .= ".wp-container-$id .alignfull { max-width: none; }"; + } + + $style .= ".wp-container-$id .alignleft { float: left; margin-right: 2em; }"; + $style .= ".wp-container-$id .alignright { float: right; margin-left: 2em; }"; + + // This assumes the hook only applies to blocks with a single wrapper. + // I think this is a reasonable limitation for that particular hook. + $content = preg_replace( + '/' . preg_quote( 'class="', '/' ) . '/', + 'class="wp-container-' . $id . ' ', + $block_content, + 1 + ); + + return $content . ''; +} + +// Register the block support. +WP_Block_Supports::get_instance()->register( + 'layout', + array( + 'register_attribute' => 'wp_register_layout_support', + ) +); +add_filter( 'render_block', 'wp_render_layout_support_flag', 10, 2 ); + /** * For themes without theme.json file, make sure * to restore the inner div for the group block @@ -22,9 +120,9 @@ function wp_restore_group_inner_container( $block_content, $block ) { $group_with_inner_container_regex = '/(^\s*]*wp-block-group(\s|")[^>]*>)(\s*]*wp-block-group__inner-container(\s|")[^>]*>)((.|\S|\s)*)/'; - // TODO: Add check for theme.json presence. if ( 'core/group' !== $block['blockName'] || + WP_Theme_JSON_Resolver::theme_has_support() || 1 === preg_match( $group_with_inner_container_regex, $block_content ) ) { return $block_content; diff --git a/wp-includes/version.php b/wp-includes/version.php index c386004274..37ef234902 100644 --- a/wp-includes/version.php +++ b/wp-includes/version.php @@ -13,7 +13,7 @@ * * @global string $wp_version */ -$wp_version = '5.8-alpha-50987'; +$wp_version = '5.8-alpha-50991'; /** * Holds the WordPress DB revision, increments when changes are made to the WordPress DB schema.