diff --git a/lib/compat/wordpress-5.9/block-template-utils.php b/lib/compat/wordpress-5.9/block-template-utils.php new file mode 100644 index 00000000000000..66f65cec82eeb2 --- /dev/null +++ b/lib/compat/wordpress-5.9/block-template-utils.php @@ -0,0 +1,828 @@ + WP_TEMPLATE_PART_AREA_UNCATEGORIZED, + 'label' => __( 'General', 'gutenberg' ), + 'description' => __( + 'General templates often perform a specific role like displaying post content, and are not tied to any particular area.', + 'gutenberg' + ), + 'icon' => 'layout', + 'area_tag' => 'div', + ), + array( + 'area' => WP_TEMPLATE_PART_AREA_HEADER, + 'label' => __( 'Header', 'gutenberg' ), + 'description' => __( + 'The Header template defines a page area that typically contains a title, logo, and main navigation.', + 'gutenberg' + ), + 'icon' => 'header', + 'area_tag' => 'header', + ), + array( + 'area' => WP_TEMPLATE_PART_AREA_FOOTER, + 'label' => __( 'Footer', 'gutenberg' ), + 'description' => __( + 'The Footer template defines a page area that typically contains site credits, social links, or any other combination of blocks.', + 'gutenberg' + ), + 'icon' => 'footer', + 'area_tag' => 'footer', + ), + ); + + /** + * Filters the list of allowed template part area values. + * + * @param array $default_areas An array of supported area objects. + */ + return apply_filters( 'default_wp_template_part_areas', $default_area_definitions ); + } +} + +if ( ! function_exists( 'get_default_block_template_types' ) ) { + /** + * Returns a filtered list of default template types, containing their + * localized titles and descriptions. + * + * @return array The default template types. + */ + function get_default_block_template_types() { + $default_template_types = array( + 'index' => array( + 'title' => _x( 'Index', 'Template name', 'gutenberg' ), + 'description' => __( 'The default template used when no other template is available. This is a required template in WordPress.', 'gutenberg' ), + ), + 'home' => array( + 'title' => _x( 'Home', 'Template name', 'gutenberg' ), + 'description' => __( 'Template used for the main page that displays blog posts. This is the front page by default in WordPress. If a static front page is set, this is the template used for the page that contains the latest blog posts.', 'gutenberg' ), + ), + 'front-page' => array( + 'title' => _x( 'Front Page', 'Template name', 'gutenberg' ), + 'description' => __( 'Template used to render the front page of the site, whether it displays blog posts or a static page. The front page template takes precedence over the "Home" template.', 'gutenberg' ), + ), + 'singular' => array( + 'title' => _x( 'Singular', 'Template name', 'gutenberg' ), + 'description' => __( 'Template used for displaying single views of the content. This template is a fallback for the Single, Post, and Page templates, which take precedence when they exist.', 'gutenberg' ), + ), + 'single' => array( + 'title' => _x( 'Single Post', 'Template name', 'gutenberg' ), + 'description' => __( 'Template used to display a single blog post.', 'gutenberg' ), + ), + 'page' => array( + 'title' => _x( 'Page', 'Template name', 'gutenberg' ), + 'description' => __( 'Template used to display individual pages.', 'gutenberg' ), + ), + 'archive' => array( + 'title' => _x( 'Archive', 'Template name', 'gutenberg' ), + 'description' => __( 'The archive template displays multiple entries at once. It is used as a fallback for the Category, Author, and Date templates, which take precedence when they are available.', 'gutenberg' ), + ), + 'author' => array( + 'title' => _x( 'Author', 'Template name', 'gutenberg' ), + 'description' => __( 'Archive template used to display a list of posts from a single author.', 'gutenberg' ), + ), + 'category' => array( + 'title' => _x( 'Category', 'Template name', 'gutenberg' ), + 'description' => __( 'Archive template used to display a list of posts from the same category.', 'gutenberg' ), + ), + 'taxonomy' => array( + 'title' => _x( 'Taxonomy', 'Template name', 'gutenberg' ), + 'description' => __( 'Archive template used to display a list of posts from the same taxonomy.', 'gutenberg' ), + ), + 'date' => array( + 'title' => _x( 'Date', 'Template name', 'gutenberg' ), + 'description' => __( 'Archive template used to display a list of posts from a specific date.', 'gutenberg' ), + ), + 'tag' => array( + 'title' => _x( 'Tag', 'Template name', 'gutenberg' ), + 'description' => __( 'Archive template used to display a list of posts with a given tag.', 'gutenberg' ), + ), + 'attachment' => array( + 'title' => __( 'Media', 'gutenberg' ), + 'description' => __( 'Template used to display individual media items or attachments.', 'gutenberg' ), + ), + 'search' => array( + 'title' => _x( 'Search', 'Template name', 'gutenberg' ), + 'description' => __( 'Template used to display search results.', 'gutenberg' ), + ), + 'privacy-policy' => array( + 'title' => __( 'Privacy Policy', 'gutenberg' ), + 'description' => '', + ), + '404' => array( + 'title' => _x( '404', 'Template name', 'gutenberg' ), + 'description' => __( 'Template shown when no content is found.', 'gutenberg' ), + ), + ); + + /** + * Filters the list of template types. + * + * @param array $default_template_types An array of template types, formatted as [ slug => [ title, description ] ]. + * + * @since 5.x.x + */ + return apply_filters( 'default_template_types', $default_template_types ); + } +} + +if ( ! function_exists( '_filter_block_template_part_area' ) ) { + /** + * Checks whether the input 'area' is a supported value. + * Returns the input if supported, otherwise returns the 'uncategorized' value. + * + * @param string $type Template part area name. + * + * @return string Input if supported, else the uncategorized value. + */ + function _filter_block_template_part_area( $type ) { + $allowed_areas = array_map( + function ( $item ) { + return $item['area']; + }, + get_allowed_block_template_part_areas() + ); + if ( in_array( $type, $allowed_areas, true ) ) { + return $type; + } + + /* translators: %1$s: Template area type, %2$s: the uncategorized template area value. */ + $warning_message = sprintf( __( '"%1$s" is not a supported wp_template_part area value and has been added as "%2$s".', 'gutenberg' ), $type, WP_TEMPLATE_PART_AREA_UNCATEGORIZED ); + trigger_error( $warning_message, E_USER_NOTICE ); + return WP_TEMPLATE_PART_AREA_UNCATEGORIZED; + } +} + +if ( ! function_exists( '_get_block_templates_paths' ) ) { + /** + * Finds all nested template part file paths in a theme's directory. + * + * @access private + * + * @param string $base_directory The theme's file path. + * @return array $path_list A list of paths to all template part files. + */ + function _get_block_templates_paths( $base_directory ) { + $path_list = array(); + if ( file_exists( $base_directory ) ) { + $nested_files = new RecursiveIteratorIterator( new RecursiveDirectoryIterator( $base_directory ) ); + $nested_html_files = new RegexIterator( $nested_files, '/^.+\.html$/i', RecursiveRegexIterator::GET_MATCH ); + foreach ( $nested_html_files as $path => $file ) { + $path_list[] = $path; + } + } + return $path_list; + } +} + +if ( ! function_exists( '_get_block_template_file' ) ) { + /** + * Retrieves the template file from the theme for a given slug. + * + * @access private + * @internal + * + * @param string $template_type wp_template or wp_template_part. + * @param string $slug template slug. + * + * @return array|null Template. + */ + function _get_block_template_file( $template_type, $slug ) { + $template_base_paths = array( + 'wp_template' => 'block-templates', + 'wp_template_part' => 'block-template-parts', + ); + $themes = array( + get_stylesheet() => get_stylesheet_directory(), + get_template() => get_template_directory(), + ); + foreach ( $themes as $theme_slug => $theme_dir ) { + $file_path = $theme_dir . '/' . $template_base_paths[ $template_type ] . '/' . $slug . '.html'; + if ( file_exists( $file_path ) ) { + $new_template_item = array( + 'slug' => $slug, + 'path' => $file_path, + 'theme' => $theme_slug, + 'type' => $template_type, + ); + + if ( 'wp_template_part' === $template_type ) { + return _add_block_template_part_area_info( $new_template_item ); + } + + if ( 'wp_template' === $template_type ) { + return _add_block_template_info( $new_template_item ); + } + + return $new_template_item; + } + } + + return null; + } +} + +if ( ! function_exists( '_get_block_templates_files' ) ) { + /** + * Retrieves the template files from the theme. + * + * @access private + * @internal + * + * @param string $template_type wp_template or wp_template_part. + * + * @return array Template. + */ + function _get_block_templates_files( $template_type ) { + $template_base_paths = array( + 'wp_template' => 'block-templates', + 'wp_template_part' => 'block-template-parts', + ); + $themes = array( + get_stylesheet() => get_stylesheet_directory(), + get_template() => get_template_directory(), + ); + + $template_files = array(); + foreach ( $themes as $theme_slug => $theme_dir ) { + $theme_template_files = _get_block_templates_paths( $theme_dir . '/' . $template_base_paths[ $template_type ] ); + foreach ( $theme_template_files as $template_file ) { + $template_base_path = $template_base_paths[ $template_type ]; + $template_slug = substr( + $template_file, + // Starting position of slug. + strpos( $template_file, $template_base_path . DIRECTORY_SEPARATOR ) + 1 + strlen( $template_base_path ), + // Subtract ending '.html'. + -5 + ); + $new_template_item = array( + 'slug' => $template_slug, + 'path' => $template_file, + 'theme' => $theme_slug, + 'type' => $template_type, + ); + + if ( 'wp_template_part' === $template_type ) { + $template_files[] = _add_block_template_part_area_info( $new_template_item ); + } + + if ( 'wp_template' === $template_type ) { + $template_files[] = _add_block_template_info( $new_template_item ); + } + } + } + + return $template_files; + } +} + +if ( ! function_exists( '_add_block_template_info' ) ) { + /** + * Attempts to add custom template information to the template item. + * + * @param array $template_item Template to add information to (requires 'slug' field). + * @return array Template + */ + function _add_block_template_info( $template_item ) { + if ( ! WP_Theme_JSON_Resolver_Gutenberg::theme_has_support() ) { + return $template_item; + } + + $theme_data = WP_Theme_JSON_Resolver_Gutenberg::get_theme_data()->get_custom_templates(); + if ( isset( $theme_data[ $template_item['slug'] ] ) ) { + $template_item['title'] = $theme_data[ $template_item['slug'] ]['title']; + $template_item['postTypes'] = $theme_data[ $template_item['slug'] ]['postTypes']; + } + + return $template_item; + } +} + +if ( ! function_exists( '_add_block_template_part_area_info' ) ) { + /** + * Attempts to add the template part's area information to the input template. + * + * @param array $template_info Template to add information to (requires 'type' and 'slug' fields). + * + * @return array Template. + */ + function _add_block_template_part_area_info( $template_info ) { + if ( WP_Theme_JSON_Resolver_Gutenberg::theme_has_support() ) { + $theme_data = WP_Theme_JSON_Resolver_Gutenberg::get_theme_data()->get_template_parts(); + } + + if ( isset( $theme_data[ $template_info['slug'] ]['area'] ) ) { + $template_info['title'] = $theme_data[ $template_info['slug'] ]['title']; + $template_info['area'] = _filter_block_template_part_area( $theme_data[ $template_info['slug'] ]['area'] ); + } else { + $template_info['area'] = WP_TEMPLATE_PART_AREA_UNCATEGORIZED; + } + + return $template_info; + } +} + +if ( ! function_exists( '_flatten_blocks' ) ) { + /** + * Returns an array containing the references of + * the passed blocks and their inner blocks. + * + * @param array $blocks array of blocks. + * + * @return array block references to the passed blocks and their inner blocks. + */ + function _flatten_blocks( &$blocks ) { + $all_blocks = array(); + $queue = array(); + foreach ( $blocks as &$block ) { + $queue[] = &$block; + } + + while ( count( $queue ) > 0 ) { + $block = &$queue[0]; + array_shift( $queue ); + $all_blocks[] = &$block; + + if ( ! empty( $block['innerBlocks'] ) ) { + foreach ( $block['innerBlocks'] as &$inner_block ) { + $queue[] = &$inner_block; + } + } + } + + return $all_blocks; + } +} + +if ( ! function_exists( '_inject_theme_attribute_in_block_template_content' ) ) { + /** + * Parses wp_template content and injects the current theme's + * stylesheet as a theme attribute into each wp_template_part + * + * @param string $template_content serialized wp_template content. + * + * @return string Updated wp_template content. + */ + function _inject_theme_attribute_in_block_template_content( $template_content ) { + $has_updated_content = false; + $new_content = ''; + $template_blocks = parse_blocks( $template_content ); + + $blocks = _flatten_blocks( $template_blocks ); + foreach ( $blocks as &$block ) { + if ( + 'core/template-part' === $block['blockName'] && + ! isset( $block['attrs']['theme'] ) + ) { + $block['attrs']['theme'] = wp_get_theme()->get_stylesheet(); + $has_updated_content = true; + } + } + + if ( $has_updated_content ) { + foreach ( $template_blocks as &$block ) { + $new_content .= serialize_block( $block ); + } + + return $new_content; + } + + return $template_content; + } +} + +if ( ! function_exists( '_build_block_template_result_from_file' ) ) { + /** + * Build a unified template object based on a theme file. + * + * @param array $template_file Theme file. + * @param array $template_type wp_template or wp_template_part. + * + * @return WP_Block_Template Template. + */ + function _build_block_template_result_from_file( $template_file, $template_type ) { + $default_template_types = get_default_block_template_types(); + $template_content = file_get_contents( $template_file['path'] ); + $theme = wp_get_theme()->get_stylesheet(); + + $template = new WP_Block_Template(); + $template->id = $theme . '//' . $template_file['slug']; + $template->theme = $theme; + $template->content = _inject_theme_attribute_in_block_template_content( $template_content ); + $template->slug = $template_file['slug']; + $template->source = 'theme'; + $template->type = $template_type; + $template->title = ! empty( $template_file['title'] ) ? $template_file['title'] : $template_file['slug']; + $template->status = 'publish'; + $template->has_theme_file = true; + $template->is_custom = true; + + if ( 'wp_template' === $template_type && isset( $default_template_types[ $template_file['slug'] ] ) ) { + $template->description = $default_template_types[ $template_file['slug'] ]['description']; + $template->title = $default_template_types[ $template_file['slug'] ]['title']; + $template->is_custom = false; + } + + if ( 'wp_template' === $template_type && isset( $template_file['postTypes'] ) ) { + $template->post_types = $template_file['postTypes']; + } + + if ( 'wp_template_part' === $template_type && isset( $template_file['area'] ) ) { + $template->area = $template_file['area']; + } + + return $template; + } +} + +if ( ! function_exists( '_build_block_template_result_from_post' ) ) { + /** + * Build a unified template object based a post Object. + * + * @param WP_Post $post Template post. + * + * @return WP_Block_Template|WP_Error Template. + */ + function _build_block_template_result_from_post( $post ) { + $default_template_types = get_default_block_template_types(); + $terms = get_the_terms( $post, 'wp_theme' ); + + if ( is_wp_error( $terms ) ) { + return $terms; + } + + if ( ! $terms ) { + return new WP_Error( 'template_missing_theme', __( 'No theme is defined for this template.', 'gutenberg' ) ); + } + + $theme = $terms[0]->name; + $has_theme_file = wp_get_theme()->get_stylesheet() === $theme && + null !== _get_block_template_file( $post->post_type, $post->post_name ); + + $template = new WP_Block_Template(); + $template->wp_id = $post->ID; + $template->id = $theme . '//' . $post->post_name; + $template->theme = $theme; + $template->content = $post->post_content; + $template->slug = $post->post_name; + $template->source = 'custom'; + $template->type = $post->post_type; + $template->description = $post->post_excerpt; + $template->title = $post->post_title; + $template->status = $post->post_status; + $template->has_theme_file = $has_theme_file; + $template->is_custom = true; + + if ( 'wp_template' === $post->post_type && isset( $default_template_types[ $template->slug ] ) ) { + $template->is_custom = false; + } + + if ( 'wp_template_part' === $post->post_type ) { + $type_terms = get_the_terms( $post, 'wp_template_part_area' ); + if ( ! is_wp_error( $type_terms ) && false !== $type_terms ) { + $template->area = $type_terms[0]->name; + } + } + + return $template; + } +} + + +/** + * Retrieves a list of unified template objects based on a query. + * + * @param array $query { + * Optional. Arguments to retrieve templates. + * + * @type array $slug__in List of slugs to include. + * @type int $wp_id Post ID of customized template. + * @type string $area A 'wp_template_part_area' taxonomy value to filter by (for wp_template_part template type only). + * @type string $post_type Post type to get the templates for. + * } + * @param array $template_type wp_template or wp_template_part. + * + * @return array Templates. + */ +function gutenberg_get_block_templates( $query = array(), $template_type = 'wp_template' ) { + /** + * Filters the block templates array before the query takes place. + * + * Return a non-null value to bypass the WordPress queries. + * + * @since 10.8 + * + * @param WP_Block_Template[]|null $block_templates Return an array of block templates to short-circuit the default query, + * or null to allow WP to run it's normal queries. + * @param array $query { + * Optional. Arguments to retrieve templates. + * + * @type array $slug__in List of slugs to include. + * @type int $wp_id Post ID of customized template. + * @type string $post_type Post type to get the templates for. + * } + * @param array $template_type wp_template or wp_template_part. + */ + $templates = apply_filters( 'pre_get_block_templates', null, $query, $template_type ); + if ( ! is_null( $templates ) ) { + return $templates; + } + + $post_type = isset( $query['post_type'] ) ? $query['post_type'] : ''; + $wp_query_args = array( + 'post_status' => array( 'auto-draft', 'draft', 'publish' ), + 'post_type' => $template_type, + 'posts_per_page' => -1, + 'no_found_rows' => true, + 'tax_query' => array( + array( + 'taxonomy' => 'wp_theme', + 'field' => 'name', + 'terms' => wp_get_theme()->get_stylesheet(), + ), + ), + ); + + if ( 'wp_template_part' === $template_type && isset( $query['area'] ) ) { + $wp_query_args['tax_query'][] = array( + 'taxonomy' => 'wp_template_part_area', + 'field' => 'name', + 'terms' => $query['area'], + ); + $wp_query_args['tax_query']['relation'] = 'AND'; + } + + if ( isset( $query['slug__in'] ) ) { + $wp_query_args['post_name__in'] = $query['slug__in']; + } + + // This is only needed for the regular templates/template parts CPT listing and editor. + if ( isset( $query['wp_id'] ) ) { + $wp_query_args['p'] = $query['wp_id']; + } else { + $wp_query_args['post_status'] = 'publish'; + } + + $template_query = new WP_Query( $wp_query_args ); + $query_result = array(); + foreach ( $template_query->posts as $post ) { + $template = _build_block_template_result_from_post( $post ); + + if ( is_wp_error( $template ) ) { + continue; + } + + if ( $post_type && ! $template->is_custom ) { + continue; + } + + $query_result[] = $template; + } + + if ( ! isset( $query['wp_id'] ) ) { + $template_files = _get_block_templates_files( $template_type ); + foreach ( $template_files as $template_file ) { + $template = _build_block_template_result_from_file( $template_file, $template_type ); + + if ( $post_type && ! $template->is_custom ) { + continue; + } + + if ( $post_type && + isset( $template->post_types ) && + ! in_array( $post_type, $template->post_types, true ) + ) { + continue; + } + + $is_not_custom = false === array_search( + wp_get_theme()->get_stylesheet() . '//' . $template_file['slug'], + array_column( $query_result, 'id' ), + true + ); + $fits_slug_query = + ! isset( $query['slug__in'] ) || in_array( $template_file['slug'], $query['slug__in'], true ); + $fits_area_query = + ! isset( $query['area'] ) || $template_file['area'] === $query['area']; + $should_include = $is_not_custom && $fits_slug_query && $fits_area_query; + if ( $should_include ) { + $query_result[] = $template; + } + } + } + /** + * Filters the array of queried block templates array after they've been fetched. + * + * @since 10.8 + * + * @param WP_Block_Template[] $query_result Array of found block templates. + * @param array $query { + * Optional. Arguments to retrieve templates. + * + * @type array $slug__in List of slugs to include. + * @type int $wp_id Post ID of customized template. + * } + * @param array $template_type wp_template or wp_template_part. + */ + return apply_filters( 'get_block_templates', $query_result, $query, $template_type ); +} + +/** + * Retrieves a single unified template object using its id. + * + * @param string $id Template unique identifier (example: theme_slug//template_slug). + * @param array $template_type wp_template or wp_template_part. + * + * @return WP_Block_Template|null Template. + */ +function gutenberg_get_block_template( $id, $template_type = 'wp_template' ) { + /** + * Filters the block templates array before the query takes place. + * + * Return a non-null value to bypass the WordPress queries. + * + * @since 10.8 + * + * @param WP_Block_Template|null $block_template Return block template object to short-circuit the default query, + * or null to allow WP to run it's normal queries. + * @param string $id Template unique identifier (example: theme_slug//template_slug). + * @param array $template_type wp_template or wp_template_part. + */ + $block_template = apply_filters( 'pre_get_block_template', null, $id, $template_type ); + if ( ! is_null( $block_template ) ) { + return $block_template; + } + + $parts = explode( '//', $id, 2 ); + if ( count( $parts ) < 2 ) { + return null; + } + list( $theme, $slug ) = $parts; + $wp_query_args = array( + 'post_name__in' => array( $slug ), + 'post_type' => $template_type, + 'post_status' => array( 'auto-draft', 'draft', 'publish', 'trash' ), + 'posts_per_page' => 1, + 'no_found_rows' => true, + 'tax_query' => array( + array( + 'taxonomy' => 'wp_theme', + 'field' => 'name', + 'terms' => $theme, + ), + ), + ); + $template_query = new WP_Query( $wp_query_args ); + $posts = $template_query->posts; + + if ( count( $posts ) > 0 ) { + $template = _build_block_template_result_from_post( $posts[0] ); + + if ( ! is_wp_error( $template ) ) { + return $template; + } + } + + $block_template = get_block_file_template( $id, $template_type ); + + /** + * Filters the array of queried block templates array after they've been fetched. + * + * @since 10.8 + * + * @param WP_Block_Template $block_template The found block template. + * @param string $id Template unique identifier (example: theme_slug//template_slug). + * @param array $template_type wp_template or wp_template_part. + */ + return apply_filters( 'get_block_template', $block_template, $id, $template_type ); +} + +if ( ! function_exists( 'get_block_file_template' ) ) { + /** + * Retrieves a single unified template object using its id. + * Retrieves the file template. + * + * @param string $id Template unique identifier (example: theme_slug//template_slug). + * @param array $template_type wp_template or wp_template_part. + * + * @return WP_Block_Template|null File template. + */ + function get_block_file_template( $id, $template_type = 'wp_template' ) { + /** + * Filters the block templates array before the query takes place. + * + * Return a non-null value to bypass the WordPress queries. + * + * @since 10.8 + * + * @param WP_Block_Template|null $block_template Return block template object to short-circuit the default query, + * or null to allow WP to run it's normal queries. + * @param string $id Template unique identifier (example: theme_slug//template_slug). + * @param array $template_type wp_template or wp_template_part. + */ + $block_template = apply_filters( 'pre_get_block_file_template', null, $id, $template_type ); + if ( ! is_null( $block_template ) ) { + return $block_template; + } + + $parts = explode( '//', $id, 2 ); + if ( count( $parts ) < 2 ) { + /** This filter is documented at the end of this function */ + return apply_filters( 'get_block_file_template', null, $id, $template_type ); + } + list( $theme, $slug ) = $parts; + + if ( wp_get_theme()->get_stylesheet() !== $theme ) { + /** This filter is documented at the end of this function */ + return apply_filters( 'get_block_file_template', null, $id, $template_type ); + } + + $template_file = _get_block_template_file( $template_type, $slug ); + if ( null === $template_file ) { + /** This filter is documented at the end of this function */ + return apply_filters( 'get_block_file_template', null, $id, $template_type ); + } + + $block_template = _build_block_template_result_from_file( $template_file, $template_type ); + + /** + * Filters the array of queried block templates array after they've been fetched. + * + * @since 10.8 + * + * @param null|WP_Block_Template $block_template The found block template. + * @param string $id Template unique identifier (example: theme_slug//template_slug). + * @param array $template_type wp_template or wp_template_part. + */ + return apply_filters( 'get_block_file_template', $block_template, $id, $template_type ); + } +} + +if ( ! function_exists( 'block_template_part' ) ) { + /** + * Print a template-part. + * + * @param string $part The template-part to print. Use "header" or "footer". + * + * @return void + */ + function block_template_part( $part ) { + $template_part = gutenberg_get_block_template( get_stylesheet() . '//' . $part, 'wp_template_part' ); + if ( ! $template_part || empty( $template_part->content ) ) { + return; + } + echo do_blocks( $template_part->content ); + } +} + +if ( ! function_exists( 'block_header_area' ) ) { + /** + * Print the header template-part. + * + * @return void + */ + function block_header_area() { + block_template_part( 'header' ); + } +} + +if ( ! function_exists( 'block_footer_area' ) ) { + /** + * Print the footer template-part. + * + * @return void + */ + function block_footer_area() { + block_template_part( 'footer' ); + } +} diff --git a/lib/full-site-editing/class-gutenberg-rest-templates-controller.php b/lib/compat/wordpress-5.9/class-gutenberg-rest-templates-controller.php similarity index 97% rename from lib/full-site-editing/class-gutenberg-rest-templates-controller.php rename to lib/compat/wordpress-5.9/class-gutenberg-rest-templates-controller.php index dafb15509bd012..eb7f047a8e98df 100644 --- a/lib/full-site-editing/class-gutenberg-rest-templates-controller.php +++ b/lib/compat/wordpress-5.9/class-gutenberg-rest-templates-controller.php @@ -173,7 +173,7 @@ public function get_item_permissions_check( $request ) { */ public function get_item( $request ) { if ( isset( $request['source'] ) && 'theme' === $request['source'] ) { - $template = gutenberg_get_block_file_template( $request['id'], $this->post_type ); + $template = get_block_file_template( $request['id'], $this->post_type ); } else { $template = gutenberg_get_block_template( $request['id'], $this->post_type ); } @@ -209,7 +209,7 @@ public function update_item( $request ) { if ( isset( $request['source'] ) && 'theme' === $request['source'] ) { wp_delete_post( $template->wp_id, true ); - return $this->prepare_item_for_response( gutenberg_get_block_file_template( $request['id'], $this->post_type ), $request ); + return $this->prepare_item_for_response( get_block_file_template( $request['id'], $this->post_type ), $request ); } $changes = $this->prepare_item_for_database( $request ); @@ -377,9 +377,9 @@ protected function prepare_item_for_database( $request ) { if ( 'wp_template_part' === $this->post_type ) { if ( isset( $request['area'] ) ) { - $changes->tax_input['wp_template_part_area'] = gutenberg_filter_template_part_area( $request['area'] ); + $changes->tax_input['wp_template_part_area'] = _filter_block_template_part_area( $request['area'] ); } elseif ( null !== $template && 'custom' !== $template->source && $template->area ) { - $changes->tax_input['wp_template_part_area'] = gutenberg_filter_template_part_area( $template->area ); + $changes->tax_input['wp_template_part_area'] = _filter_block_template_part_area( $template->area ); } elseif ( ! $template->area ) { $changes->tax_input['wp_template_part_area'] = WP_TEMPLATE_PART_AREA_UNCATEGORIZED; } diff --git a/lib/full-site-editing/block-templates.php b/lib/full-site-editing/block-templates.php index cc1621f6713978..a3db2bd0402503 100644 --- a/lib/full-site-editing/block-templates.php +++ b/lib/full-site-editing/block-templates.php @@ -6,592 +6,6 @@ * @subpackage REST_API */ -/** - * Finds all nested template part file paths in a theme's directory. - * - * @access private - * - * @param string $base_directory The theme's file path. - * @return array $path_list A list of paths to all template part files. - */ -function _gutenberg_get_template_paths( $base_directory ) { - $path_list = array(); - if ( file_exists( $base_directory ) ) { - $nested_files = new RecursiveIteratorIterator( new RecursiveDirectoryIterator( $base_directory ) ); - $nested_html_files = new RegexIterator( $nested_files, '/^.+\.html$/i', RecursiveRegexIterator::GET_MATCH ); - foreach ( $nested_html_files as $path => $file ) { - $path_list[] = $path; - } - } - return $path_list; -} - -/** - * Retrieves the template file from the theme for a given slug. - * - * @access private - * @internal - * - * @param string $template_type wp_template or wp_template_part. - * @param string $slug template slug. - * - * @return array|null Template. - */ -function _gutenberg_get_template_file( $template_type, $slug ) { - $template_base_paths = array( - 'wp_template' => 'block-templates', - 'wp_template_part' => 'block-template-parts', - ); - $themes = array( - get_stylesheet() => get_stylesheet_directory(), - get_template() => get_template_directory(), - ); - foreach ( $themes as $theme_slug => $theme_dir ) { - $file_path = $theme_dir . '/' . $template_base_paths[ $template_type ] . '/' . $slug . '.html'; - if ( file_exists( $file_path ) ) { - $new_template_item = array( - 'slug' => $slug, - 'path' => $file_path, - 'theme' => $theme_slug, - 'type' => $template_type, - ); - - if ( 'wp_template_part' === $template_type ) { - return _gutenberg_add_template_part_area_info( $new_template_item ); - } - - if ( 'wp_template' === $template_type ) { - return _gutenberg_add_template_info( $new_template_item ); - } - - return $new_template_item; - } - } - - return null; -} - -/** - * Retrieves the template files from the theme. - * - * @access private - * @internal - * - * @param string $template_type wp_template or wp_template_part. - * - * @return array Template. - */ -function _gutenberg_get_template_files( $template_type ) { - $template_base_paths = array( - 'wp_template' => 'block-templates', - 'wp_template_part' => 'block-template-parts', - ); - $themes = array( - get_stylesheet() => get_stylesheet_directory(), - get_template() => get_template_directory(), - ); - - $template_files = array(); - foreach ( $themes as $theme_slug => $theme_dir ) { - $theme_template_files = _gutenberg_get_template_paths( $theme_dir . '/' . $template_base_paths[ $template_type ] ); - foreach ( $theme_template_files as $template_file ) { - $template_base_path = $template_base_paths[ $template_type ]; - $template_slug = substr( - $template_file, - // Starting position of slug. - strpos( $template_file, $template_base_path . DIRECTORY_SEPARATOR ) + 1 + strlen( $template_base_path ), - // Subtract ending '.html'. - -5 - ); - $new_template_item = array( - 'slug' => $template_slug, - 'path' => $template_file, - 'theme' => $theme_slug, - 'type' => $template_type, - ); - - if ( 'wp_template_part' === $template_type ) { - $template_files[] = _gutenberg_add_template_part_area_info( $new_template_item ); - } - - if ( 'wp_template' === $template_type ) { - $template_files[] = _gutenberg_add_template_info( $new_template_item ); - } - } - } - - return $template_files; -} - -/** - * Attempts to add custom template information to the template item. - * - * @param array $template_item Template to add information to (requires 'slug' field). - * @return array Template - */ -function _gutenberg_add_template_info( $template_item ) { - if ( ! WP_Theme_JSON_Resolver_Gutenberg::theme_has_support() ) { - return $template_item; - } - - $theme_data = WP_Theme_JSON_Resolver_Gutenberg::get_theme_data()->get_custom_templates(); - if ( isset( $theme_data[ $template_item['slug'] ] ) ) { - $template_item['title'] = $theme_data[ $template_item['slug'] ]['title']; - $template_item['postTypes'] = $theme_data[ $template_item['slug'] ]['postTypes']; - } - - return $template_item; -} - -/** - * Attempts to add the template part's area information to the input template. - * - * @param array $template_info Template to add information to (requires 'type' and 'slug' fields). - * - * @return array Template. - */ -function _gutenberg_add_template_part_area_info( $template_info ) { - if ( WP_Theme_JSON_Resolver_Gutenberg::theme_has_support() ) { - $theme_data = WP_Theme_JSON_Resolver_Gutenberg::get_theme_data()->get_template_parts(); - } - - if ( isset( $theme_data[ $template_info['slug'] ]['area'] ) ) { - $template_info['title'] = $theme_data[ $template_info['slug'] ]['title']; - $template_info['area'] = gutenberg_filter_template_part_area( $theme_data[ $template_info['slug'] ]['area'] ); - } else { - $template_info['area'] = WP_TEMPLATE_PART_AREA_UNCATEGORIZED; - } - - return $template_info; -} - -/** - * Returns an array containing the references of - * the passed blocks and their inner blocks. - * - * @param array $blocks array of blocks. - * - * @return array block references to the passed blocks and their inner blocks. - */ -function _gutenberg_flatten_blocks( &$blocks ) { - $all_blocks = array(); - $queue = array(); - foreach ( $blocks as &$block ) { - $queue[] = &$block; - } - - while ( count( $queue ) > 0 ) { - $block = &$queue[0]; - array_shift( $queue ); - $all_blocks[] = &$block; - - if ( ! empty( $block['innerBlocks'] ) ) { - foreach ( $block['innerBlocks'] as &$inner_block ) { - $queue[] = &$inner_block; - } - } - } - - return $all_blocks; -} - -if ( ! function_exists( '_inject_theme_attribute_in_block_template_content' ) ) { - /** - * Parses wp_template content and injects the current theme's - * stylesheet as a theme attribute into each wp_template_part - * - * @param string $template_content serialized wp_template content. - * - * @return string Updated wp_template content. - */ - function _inject_theme_attribute_in_block_template_content( $template_content ) { - $has_updated_content = false; - $new_content = ''; - $template_blocks = parse_blocks( $template_content ); - - $blocks = _gutenberg_flatten_blocks( $template_blocks ); - foreach ( $blocks as &$block ) { - if ( - 'core/template-part' === $block['blockName'] && - ! isset( $block['attrs']['theme'] ) - ) { - $block['attrs']['theme'] = wp_get_theme()->get_stylesheet(); - $has_updated_content = true; - } - } - - if ( $has_updated_content ) { - foreach ( $template_blocks as &$block ) { - $new_content .= serialize_block( $block ); - } - - return $new_content; - } - - return $template_content; - } -} -/** - * Build a unified template object based on a theme file. - * - * @param array $template_file Theme file. - * @param array $template_type wp_template or wp_template_part. - * - * @return WP_Block_Template Template. - */ -function _gutenberg_build_template_result_from_file( $template_file, $template_type ) { - $default_template_types = gutenberg_get_default_template_types(); - $template_content = file_get_contents( $template_file['path'] ); - $theme = wp_get_theme()->get_stylesheet(); - - $template = new WP_Block_Template(); - $template->id = $theme . '//' . $template_file['slug']; - $template->theme = $theme; - $template->content = _inject_theme_attribute_in_block_template_content( $template_content ); - $template->slug = $template_file['slug']; - $template->source = 'theme'; - $template->type = $template_type; - $template->title = ! empty( $template_file['title'] ) ? $template_file['title'] : $template_file['slug']; - $template->status = 'publish'; - $template->has_theme_file = true; - $template->is_custom = true; - - if ( 'wp_template' === $template_type && isset( $default_template_types[ $template_file['slug'] ] ) ) { - $template->description = $default_template_types[ $template_file['slug'] ]['description']; - $template->title = $default_template_types[ $template_file['slug'] ]['title']; - $template->is_custom = false; - } - - if ( 'wp_template' === $template_type && isset( $template_file['postTypes'] ) ) { - $template->post_types = $template_file['postTypes']; - } - - if ( 'wp_template_part' === $template_type && isset( $template_file['area'] ) ) { - $template->area = $template_file['area']; - } - - return $template; -} - -/** - * Build a unified template object based a post Object. - * - * @param WP_Post $post Template post. - * - * @return WP_Block_Template|WP_Error Template. - */ -function _gutenberg_build_template_result_from_post( $post ) { - $default_template_types = gutenberg_get_default_template_types(); - $terms = get_the_terms( $post, 'wp_theme' ); - - if ( is_wp_error( $terms ) ) { - return $terms; - } - - if ( ! $terms ) { - return new WP_Error( 'template_missing_theme', __( 'No theme is defined for this template.', 'gutenberg' ) ); - } - - $theme = $terms[0]->name; - $has_theme_file = wp_get_theme()->get_stylesheet() === $theme && - null !== _gutenberg_get_template_file( $post->post_type, $post->post_name ); - - $template = new WP_Block_Template(); - $template->wp_id = $post->ID; - $template->id = $theme . '//' . $post->post_name; - $template->theme = $theme; - $template->content = $post->post_content; - $template->slug = $post->post_name; - $template->source = 'custom'; - $template->type = $post->post_type; - $template->description = $post->post_excerpt; - $template->title = $post->post_title; - $template->status = $post->post_status; - $template->has_theme_file = $has_theme_file; - $template->is_custom = true; - - if ( 'wp_template' === $post->post_type && isset( $default_template_types[ $template->slug ] ) ) { - $template->is_custom = false; - } - - if ( 'wp_template_part' === $post->post_type ) { - $type_terms = get_the_terms( $post, 'wp_template_part_area' ); - if ( ! is_wp_error( $type_terms ) && false !== $type_terms ) { - $template->area = $type_terms[0]->name; - } - } - - return $template; -} - -/** - * Retrieves a list of unified template objects based on a query. - * - * @param array $query { - * Optional. Arguments to retrieve templates. - * - * @type array $slug__in List of slugs to include. - * @type int $wp_id Post ID of customized template. - * @type string $area A 'wp_template_part_area' taxonomy value to filter by (for wp_template_part template type only). - * @type string $post_type Post type to get the templates for. - * } - * @param array $template_type wp_template or wp_template_part. - * - * @return array Templates. - */ -function gutenberg_get_block_templates( $query = array(), $template_type = 'wp_template' ) { - /** - * Filters the block templates array before the query takes place. - * - * Return a non-null value to bypass the WordPress queries. - * - * @since 10.8 - * - * @param WP_Block_Template[]|null $block_templates Return an array of block templates to short-circuit the default query, - * or null to allow WP to run it's normal queries. - * @param array $query { - * Optional. Arguments to retrieve templates. - * - * @type array $slug__in List of slugs to include. - * @type int $wp_id Post ID of customized template. - * @type string $post_type Post type to get the templates for. - * } - * @param array $template_type wp_template or wp_template_part. - */ - $templates = apply_filters( 'pre_get_block_templates', null, $query, $template_type ); - if ( ! is_null( $templates ) ) { - return $templates; - } - - $post_type = isset( $query['post_type'] ) ? $query['post_type'] : ''; - $wp_query_args = array( - 'post_status' => array( 'auto-draft', 'draft', 'publish' ), - 'post_type' => $template_type, - 'posts_per_page' => -1, - 'no_found_rows' => true, - 'tax_query' => array( - array( - 'taxonomy' => 'wp_theme', - 'field' => 'name', - 'terms' => wp_get_theme()->get_stylesheet(), - ), - ), - ); - - if ( 'wp_template_part' === $template_type && isset( $query['area'] ) ) { - $wp_query_args['tax_query'][] = array( - 'taxonomy' => 'wp_template_part_area', - 'field' => 'name', - 'terms' => $query['area'], - ); - $wp_query_args['tax_query']['relation'] = 'AND'; - } - - if ( isset( $query['slug__in'] ) ) { - $wp_query_args['post_name__in'] = $query['slug__in']; - } - - // This is only needed for the regular templates/template parts CPT listing and editor. - if ( isset( $query['wp_id'] ) ) { - $wp_query_args['p'] = $query['wp_id']; - } else { - $wp_query_args['post_status'] = 'publish'; - } - - $template_query = new WP_Query( $wp_query_args ); - $query_result = array(); - foreach ( $template_query->posts as $post ) { - $template = _gutenberg_build_template_result_from_post( $post ); - - if ( is_wp_error( $template ) ) { - continue; - } - - if ( $post_type && ! $template->is_custom ) { - continue; - } - - $query_result[] = $template; - } - - if ( ! isset( $query['wp_id'] ) ) { - $template_files = _gutenberg_get_template_files( $template_type ); - foreach ( $template_files as $template_file ) { - $template = _gutenberg_build_template_result_from_file( $template_file, $template_type ); - - if ( $post_type && ! $template->is_custom ) { - continue; - } - - if ( $post_type && - isset( $template->post_types ) && - ! in_array( $post_type, $template->post_types, true ) - ) { - continue; - } - - $is_not_custom = false === array_search( - wp_get_theme()->get_stylesheet() . '//' . $template_file['slug'], - array_column( $query_result, 'id' ), - true - ); - $fits_slug_query = - ! isset( $query['slug__in'] ) || in_array( $template_file['slug'], $query['slug__in'], true ); - $fits_area_query = - ! isset( $query['area'] ) || $template_file['area'] === $query['area']; - $should_include = $is_not_custom && $fits_slug_query && $fits_area_query; - if ( $should_include ) { - $query_result[] = $template; - } - } - } - - /** - * Filters the array of queried block templates array after they've been fetched. - * - * @since 10.8 - * - * @param WP_Block_Template[] $query_result Array of found block templates. - * @param array $query { - * Optional. Arguments to retrieve templates. - * - * @type array $slug__in List of slugs to include. - * @type int $wp_id Post ID of customized template. - * } - * @param array $template_type wp_template or wp_template_part. - */ - return apply_filters( 'get_block_templates', $query_result, $query, $template_type ); -} - -/** - * Retrieves a single unified template object using its id. - * - * @param string $id Template unique identifier (example: theme_slug//template_slug). - * @param array $template_type wp_template or wp_template_part. - * - * @return WP_Block_Template|null Template. - */ -function gutenberg_get_block_template( $id, $template_type = 'wp_template' ) { - /** - * Filters the block templates array before the query takes place. - * - * Return a non-null value to bypass the WordPress queries. - * - * @since 10.8 - * - * @param WP_Block_Template|null $block_template Return block template object to short-circuit the default query, - * or null to allow WP to run it's normal queries. - * @param string $id Template unique identifier (example: theme_slug//template_slug). - * @param array $template_type wp_template or wp_template_part. - */ - $block_template = apply_filters( 'pre_get_block_template', null, $id, $template_type ); - if ( ! is_null( $block_template ) ) { - return $block_template; - } - - $parts = explode( '//', $id, 2 ); - if ( count( $parts ) < 2 ) { - return null; - } - list( $theme, $slug ) = $parts; - $wp_query_args = array( - 'post_name__in' => array( $slug ), - 'post_type' => $template_type, - 'post_status' => array( 'auto-draft', 'draft', 'publish', 'trash' ), - 'posts_per_page' => 1, - 'no_found_rows' => true, - 'tax_query' => array( - array( - 'taxonomy' => 'wp_theme', - 'field' => 'name', - 'terms' => $theme, - ), - ), - ); - $template_query = new WP_Query( $wp_query_args ); - $posts = $template_query->posts; - - if ( count( $posts ) > 0 ) { - $template = _gutenberg_build_template_result_from_post( $posts[0] ); - - if ( ! is_wp_error( $template ) ) { - return $template; - } - } - - $block_template = gutenberg_get_block_file_template( $id, $template_type ); - - /** - * Filters the array of queried block templates array after they've been fetched. - * - * @since 10.8 - * - * @param WP_Block_Template $block_template The found block template. - * @param string $id Template unique identifier (example: theme_slug//template_slug). - * @param array $template_type wp_template or wp_template_part. - */ - return apply_filters( 'get_block_template', $block_template, $id, $template_type ); -} - -/** - * Retrieves a single unified template object using its id. - * Retrieves the file template. - * - * @param string $id Template unique identifier (example: theme_slug//template_slug). - * @param array $template_type wp_template or wp_template_part. - * - * @return WP_Block_Template|null File template. - */ -function gutenberg_get_block_file_template( $id, $template_type = 'wp_template' ) { - /** - * Filters the block templates array before the query takes place. - * - * Return a non-null value to bypass the WordPress queries. - * - * @since 10.8 - * - * @param WP_Block_Template|null $block_template Return block template object to short-circuit the default query, - * or null to allow WP to run it's normal queries. - * @param string $id Template unique identifier (example: theme_slug//template_slug). - * @param array $template_type wp_template or wp_template_part. - */ - $block_template = apply_filters( 'pre_get_block_file_template', null, $id, $template_type ); - if ( ! is_null( $block_template ) ) { - return $block_template; - } - - $parts = explode( '//', $id, 2 ); - if ( count( $parts ) < 2 ) { - /** This filter is documented at the end of this function */ - return apply_filters( 'get_block_file_template', null, $id, $template_type ); - } - list( $theme, $slug ) = $parts; - - if ( wp_get_theme()->get_stylesheet() !== $theme ) { - /** This filter is documented at the end of this function */ - return apply_filters( 'get_block_file_template', null, $id, $template_type ); - } - - $template_file = _gutenberg_get_template_file( $template_type, $slug ); - if ( null === $template_file ) { - /** This filter is documented at the end of this function */ - return apply_filters( 'get_block_file_template', null, $id, $template_type ); - } - - $block_template = _gutenberg_build_template_result_from_file( $template_file, $template_type ); - - /** - * Filters the array of queried block templates array after they've been fetched. - * - * @since 10.8 - * - * @param null|WP_Block_Template $block_template The found block template. - * @param string $id Template unique identifier (example: theme_slug//template_slug). - * @param array $template_type wp_template or wp_template_part. - */ - return apply_filters( 'get_block_file_template', $block_template, $id, $template_type ); -} - /** * Generates a unique slug for templates or template parts. * diff --git a/lib/full-site-editing/default-template-types.php b/lib/full-site-editing/default-template-types.php index 1a9c2b023dbe8e..daaf3c5fb9e5ba 100644 --- a/lib/full-site-editing/default-template-types.php +++ b/lib/full-site-editing/default-template-types.php @@ -5,90 +5,6 @@ * @package gutenberg */ -/** - * Returns a filtered list of default template types, containing their - * localized titles and descriptions. - * - * @return array The default template types. - */ -function gutenberg_get_default_template_types() { - $default_template_types = array( - 'index' => array( - 'title' => _x( 'Index', 'Template name', 'gutenberg' ), - 'description' => __( 'The default template used when no other template is available. This is a required template in WordPress.', 'gutenberg' ), - ), - 'home' => array( - 'title' => _x( 'Home', 'Template name', 'gutenberg' ), - 'description' => __( 'Template used for the main page that displays blog posts. This is the front page by default in WordPress. If a static front page is set, this is the template used for the page that contains the latest blog posts.', 'gutenberg' ), - ), - 'front-page' => array( - 'title' => _x( 'Front Page', 'Template name', 'gutenberg' ), - 'description' => __( 'Template used to render the front page of the site, whether it displays blog posts or a static page. The front page template takes precedence over the "Home" template.', 'gutenberg' ), - ), - 'singular' => array( - 'title' => _x( 'Singular', 'Template name', 'gutenberg' ), - 'description' => __( 'Template used for displaying single views of the content. This template is a fallback for the Single, Post, and Page templates, which take precedence when they exist.', 'gutenberg' ), - ), - 'single' => array( - 'title' => _x( 'Single Post', 'Template name', 'gutenberg' ), - 'description' => __( 'Template used to display a single blog post.', 'gutenberg' ), - ), - 'page' => array( - 'title' => _x( 'Page', 'Template name', 'gutenberg' ), - 'description' => __( 'Template used to display individual pages.', 'gutenberg' ), - ), - 'archive' => array( - 'title' => _x( 'Archive', 'Template name', 'gutenberg' ), - 'description' => __( 'The archive template displays multiple entries at once. It is used as a fallback for the Category, Author, and Date templates, which take precedence when they are available.', 'gutenberg' ), - ), - 'author' => array( - 'title' => _x( 'Author', 'Template name', 'gutenberg' ), - 'description' => __( 'Archive template used to display a list of posts from a single author.', 'gutenberg' ), - ), - 'category' => array( - 'title' => _x( 'Category', 'Template name', 'gutenberg' ), - 'description' => __( 'Archive template used to display a list of posts from the same category.', 'gutenberg' ), - ), - 'taxonomy' => array( - 'title' => _x( 'Taxonomy', 'Template name', 'gutenberg' ), - 'description' => __( 'Archive template used to display a list of posts from the same taxonomy.', 'gutenberg' ), - ), - 'date' => array( - 'title' => _x( 'Date', 'Template name', 'gutenberg' ), - 'description' => __( 'Archive template used to display a list of posts from a specific date.', 'gutenberg' ), - ), - 'tag' => array( - 'title' => _x( 'Tag', 'Template name', 'gutenberg' ), - 'description' => __( 'Archive template used to display a list of posts with a given tag.', 'gutenberg' ), - ), - 'attachment' => array( - 'title' => __( 'Media', 'gutenberg' ), - 'description' => __( 'Template used to display individual media items or attachments.', 'gutenberg' ), - ), - 'search' => array( - 'title' => _x( 'Search', 'Template name', 'gutenberg' ), - 'description' => __( 'Template used to display search results.', 'gutenberg' ), - ), - 'privacy-policy' => array( - 'title' => __( 'Privacy Policy', 'gutenberg' ), - 'description' => '', - ), - '404' => array( - 'title' => _x( '404', 'Template name', 'gutenberg' ), - 'description' => __( 'Template shown when no content is found.', 'gutenberg' ), - ), - ); - - /** - * Filters the list of template types. - * - * @param array $default_template_types An array of template types, formatted as [ slug => [ title, description ] ]. - * - * @since 5.x.x - */ - return apply_filters( 'default_template_types', $default_template_types ); -} - /** * Converts the default template types array from associative to indexed, * to be used in JS, where numeric keys (e.g. '404') always appear before alphabetical @@ -104,7 +20,7 @@ function gutenberg_get_default_template_types() { */ function gutenberg_get_indexed_default_template_types() { $indexed_template_types = array(); - $default_template_types = gutenberg_get_default_template_types(); + $default_template_types = get_default_block_template_types(); foreach ( $default_template_types as $slug => $template_type ) { $template_type['slug'] = (string) $slug; $indexed_template_types[] = $template_type; @@ -120,5 +36,5 @@ function gutenberg_get_indexed_default_template_types() { * @return string[] List of all overrideable default template type slugs. */ function gutenberg_get_template_type_slugs() { - return array_keys( gutenberg_get_default_template_types() ); + return array_keys( get_default_block_template_types() ); } diff --git a/lib/full-site-editing/edit-site-export.php b/lib/full-site-editing/edit-site-export.php index 93ca0127a41fef..6a29c235434951 100644 --- a/lib/full-site-editing/edit-site-export.php +++ b/lib/full-site-editing/edit-site-export.php @@ -18,7 +18,7 @@ function _remove_theme_attribute_from_content( $template_content ) { $new_content = ''; $template_blocks = parse_blocks( $template_content ); - $blocks = _gutenberg_flatten_blocks( $template_blocks ); + $blocks = _flatten_blocks( $template_blocks ); foreach ( $blocks as $key => $block ) { if ( 'core/template-part' === $block['blockName'] && isset( $block['attrs']['theme'] ) ) { unset( $blocks[ $key ]['attrs']['theme'] ); diff --git a/lib/full-site-editing/template-parts.php b/lib/full-site-editing/template-parts.php index f7b3e075088a05..3b3dd6fa16e4df 100644 --- a/lib/full-site-editing/template-parts.php +++ b/lib/full-site-editing/template-parts.php @@ -89,20 +89,6 @@ function gutenberg_register_wp_template_part_area_taxonomy() { } add_action( 'init', 'gutenberg_register_wp_template_part_area_taxonomy' ); -// Define constants for supported wp_template_part_area taxonomy. -if ( ! defined( 'WP_TEMPLATE_PART_AREA_HEADER' ) ) { - define( 'WP_TEMPLATE_PART_AREA_HEADER', 'header' ); -} -if ( ! defined( 'WP_TEMPLATE_PART_AREA_FOOTER' ) ) { - define( 'WP_TEMPLATE_PART_AREA_FOOTER', 'footer' ); -} -if ( ! defined( 'WP_TEMPLATE_PART_AREA_SIDEBAR' ) ) { - define( 'WP_TEMPLATE_PART_AREA_SIDEBAR', 'sidebar' ); -} -if ( ! defined( 'WP_TEMPLATE_PART_AREA_UNCATEGORIZED' ) ) { - define( 'WP_TEMPLATE_PART_AREA_UNCATEGORIZED', 'uncategorized' ); -} - /** * Fixes the label of the 'wp_template_part' admin menu entry. */ @@ -140,7 +126,12 @@ function gutenberg_fix_template_part_admin_menu_entry() { * * @param int $post_id Post ID. */ -function set_unique_slug_on_create_template_part( $post_id ) { +function gutenberg_set_unique_slug_on_create_template_part( $post_id ) { + // This is the core function with the same functionality. + if ( function_exists( 'wp_set_unique_slug_on_create_template_part' ) ) { + return; + } + $post = get_post( $post_id ); if ( 'auto-draft' !== $post->post_status ) { return; @@ -160,111 +151,5 @@ function set_unique_slug_on_create_template_part( $post_id ) { wp_set_post_terms( $post_id, wp_get_theme()->get_stylesheet(), 'wp_theme' ); } } -add_action( 'save_post_wp_template_part', 'set_unique_slug_on_create_template_part' ); -if ( ! function_exists( 'get_allowed_block_template_part_areas' ) ) { - /** - * Returns a filtered list of allowed area values for template parts. - * - * @return array The supported template part area values. - */ - function get_allowed_block_template_part_areas() { - $default_area_definitions = array( - array( - 'area' => WP_TEMPLATE_PART_AREA_UNCATEGORIZED, - 'label' => __( 'General', 'gutenberg' ), - 'description' => __( - 'General templates often perform a specific role like displaying post content, and are not tied to any particular area.', - 'gutenberg' - ), - 'icon' => 'layout', - 'area_tag' => 'div', - ), - array( - 'area' => WP_TEMPLATE_PART_AREA_HEADER, - 'label' => __( 'Header', 'gutenberg' ), - 'description' => __( - 'The Header template defines a page area that typically contains a title, logo, and main navigation.', - 'gutenberg' - ), - 'icon' => 'header', - 'area_tag' => 'header', - ), - array( - 'area' => WP_TEMPLATE_PART_AREA_FOOTER, - 'label' => __( 'Footer', 'gutenberg' ), - 'description' => __( - 'The Footer template defines a page area that typically contains site credits, social links, or any other combination of blocks.', - 'gutenberg' - ), - 'icon' => 'footer', - 'area_tag' => 'footer', - ), - ); - - /** - * Filters the list of allowed template part area values. - * - * @param array $default_areas An array of supported area objects. - */ - return apply_filters( 'default_wp_template_part_areas', $default_area_definitions ); - } -} - -/** - * Checks whether the input 'area' is a supported value. - * Returns the input if supported, otherwise returns the 'uncategorized' value. - * - * @param string $type Template part area name. - * - * @return string Input if supported, else the uncategorized value. - */ -function gutenberg_filter_template_part_area( $type ) { - $allowed_areas = array_map( - function ( $item ) { - return $item['area']; - }, - get_allowed_block_template_part_areas() - ); - if ( in_array( $type, $allowed_areas, true ) ) { - return $type; - } - - /* translators: %1$s: Template area type, %2$s: the uncategorized template area value. */ - $warning_message = sprintf( __( '"%1$s" is not a supported wp_template_part area value and has been added as "%2$s".', 'gutenberg' ), $type, WP_TEMPLATE_PART_AREA_UNCATEGORIZED ); - trigger_error( $warning_message, E_USER_NOTICE ); - return WP_TEMPLATE_PART_AREA_UNCATEGORIZED; -} - -/** - * Print a template-part. - * - * @param string $part The template-part to print. Use "header" or "footer". - * - * @return void - */ -function gutenberg_block_template_part( $part ) { - $template_part = gutenberg_get_block_template( get_stylesheet() . '//' . $part, 'wp_template_part' ); - if ( ! $template_part || empty( $template_part->content ) ) { - return; - } - echo do_blocks( $template_part->content ); -} - -/** - * Print the header template-part. - * - * @return void - */ -function gutenberg_block_header_area() { - gutenberg_block_template_part( 'header' ); -} - -/** - * Print the footer template-part. - * - * @return void - */ -function gutenberg_block_footer_area() { - gutenberg_block_template_part( 'footer' ); -} +add_action( 'save_post_wp_template_part', 'gutenberg_set_unique_slug_on_create_template_part' ); diff --git a/lib/load.php b/lib/load.php index 3d07c8832a7ed1..e234d062b981ed 100644 --- a/lib/load.php +++ b/lib/load.php @@ -56,7 +56,7 @@ function gutenberg_is_experiment_enabled( $name ) { if ( ! class_exists( 'WP_Rest_Customizer_Nonces' ) ) { require_once __DIR__ . '/class-wp-rest-customizer-nonces.php'; } - require_once __DIR__ . '/full-site-editing/class-gutenberg-rest-templates-controller.php'; + require_once __DIR__ . '/compat/wordpress-5.9/class-gutenberg-rest-templates-controller.php'; if ( ! class_exists( 'WP_REST_Block_Editor_Settings_Controller' ) ) { require_once dirname( __FILE__ ) . '/class-wp-rest-block-editor-settings-controller.php'; } @@ -83,6 +83,7 @@ function gutenberg_is_experiment_enabled( $name ) { require __DIR__ . '/compat.php'; require __DIR__ . '/compat/wordpress-5.8/index.php'; require __DIR__ . '/compat/wordpress-5.8.1/index.php'; +require __DIR__ . '/compat/wordpress-5.9/block-template-utils.php'; require __DIR__ . '/compat/wordpress-5.9/default-editor-styles.php'; require __DIR__ . '/compat/wordpress-5.9/get-global-styles-and-settings.php'; require __DIR__ . '/utils.php'; diff --git a/phpunit/class-block-templates-test.php b/phpunit/class-block-templates-test.php index 280f18c20d134b..7d0cfa3b3bc6c2 100644 --- a/phpunit/class-block-templates-test.php +++ b/phpunit/class-block-templates-test.php @@ -79,8 +79,8 @@ public static function wpTearDownAfterClass() { wp_delete_post( self::$post->ID ); } - function test_gutenberg_build_template_result_from_file() { - $template = _gutenberg_build_template_result_from_file( + function test_build_block_template_result_from_file() { + $template = _build_block_template_result_from_file( array( 'slug' => 'single', 'path' => __DIR__ . '/fixtures/template.html', @@ -98,7 +98,7 @@ function test_gutenberg_build_template_result_from_file() { $this->assertEquals( 'wp_template', $template->type ); // Test template parts. - $template_part = _gutenberg_build_template_result_from_file( + $template_part = _build_block_template_result_from_file( array( 'slug' => 'header', 'path' => __DIR__ . '/fixtures/template.html', @@ -117,8 +117,8 @@ function test_gutenberg_build_template_result_from_file() { $this->assertEquals( WP_TEMPLATE_PART_AREA_HEADER, $template_part->area ); } - function test_gutenberg_build_template_result_from_post() { - $template = _gutenberg_build_template_result_from_post( + function test_build_block_template_result_from_post() { + $template = _build_block_template_result_from_post( self::$post, 'wp_template' ); @@ -134,7 +134,7 @@ function test_gutenberg_build_template_result_from_post() { $this->assertEquals( 'wp_template', $template->type ); // Test template parts. - $template_part = _gutenberg_build_template_result_from_post( + $template_part = _build_block_template_result_from_post( self::$template_part_post, 'wp_template_part' ); @@ -194,7 +194,7 @@ function test_inject_theme_attribute_in_block_template_content() { /** * Should retrieve the template from the theme files. */ - function test_gutenberg_get_block_template_from_file() { + function test_get_block_template_from_file() { $id = get_stylesheet() . '//' . 'index'; $template = gutenberg_get_block_template( $id, 'wp_template' ); $this->assertEquals( $id, $template->id ); @@ -219,7 +219,7 @@ function test_gutenberg_get_block_template_from_file() { /** * Should retrieve the template from the CPT. */ - function test_gutenberg_get_block_template_from_post() { + function test_get_block_template_from_post() { $id = get_stylesheet() . '//' . 'my_template'; $template = gutenberg_get_block_template( $id, 'wp_template' ); $this->assertEquals( $id, $template->id ); @@ -287,22 +287,22 @@ function( $template ) { /** * Should flatten nested blocks */ - function test_gutenberg_flatten_blocks() { + function test_flatten_blocks() { $content_template_part_inside_group = ''; $blocks = parse_blocks( $content_template_part_inside_group ); - $actual = _gutenberg_flatten_blocks( $blocks ); + $actual = _flatten_blocks( $blocks ); $expected = array( $blocks[0], $blocks[0]['innerBlocks'][0] ); $this->assertEquals( $expected, $actual ); $content_template_part_inside_group_inside_group = ''; $blocks = parse_blocks( $content_template_part_inside_group_inside_group ); - $actual = _gutenberg_flatten_blocks( $blocks ); + $actual = _flatten_blocks( $blocks ); $expected = array( $blocks[0], $blocks[0]['innerBlocks'][0], $blocks[0]['innerBlocks'][0]['innerBlocks'][0] ); $this->assertEquals( $expected, $actual ); $content_without_inner_blocks = ''; $blocks = parse_blocks( $content_without_inner_blocks ); - $actual = _gutenberg_flatten_blocks( $blocks ); + $actual = _flatten_blocks( $blocks ); $expected = array( $blocks[0] ); $this->assertEquals( $expected, $actual ); }