diff --git a/packages/block-library/src/block/edit.js b/packages/block-library/src/block/edit.js
index 6e6379571cc9e3..2dd24a5fbd9596 100644
--- a/packages/block-library/src/block/edit.js
+++ b/packages/block-library/src/block/edit.js
@@ -38,7 +38,7 @@ import { name as patternBlockName } from './index';
import { unlock } from '../lock-unlock';
const { useLayoutClasses } = unlock( blockEditorPrivateApis );
-const { PARTIAL_SYNCING_SUPPORTED_BLOCKS } = unlock( patternsPrivateApis );
+const { isOverridableBlock } = unlock( patternsPrivateApis );
const fullAlignments = [ 'full', 'wide', 'left', 'right' ];
@@ -90,21 +90,9 @@ const useInferredLayout = ( blocks, parentLayout ) => {
}, [ blocks, parentLayout ] );
};
-function hasOverridableAttributes( block ) {
- return (
- Object.keys( PARTIAL_SYNCING_SUPPORTED_BLOCKS ).includes(
- block.name
- ) &&
- !! block.attributes.metadata?.bindings &&
- Object.values( block.attributes.metadata.bindings ).some(
- ( binding ) => binding.source === 'core/pattern-overrides'
- )
- );
-}
-
function hasOverridableBlocks( blocks ) {
return blocks.some( ( block ) => {
- if ( hasOverridableAttributes( block ) ) return true;
+ if ( isOverridableBlock( block ) ) return true;
return hasOverridableBlocks( block.innerBlocks );
} );
}
@@ -133,7 +121,7 @@ function applyInitialContentValuesToInnerBlocks(
const metadataName =
legacyIdMap?.[ block.clientId ] ?? block.attributes.metadata?.name;
- if ( ! metadataName || ! hasOverridableAttributes( block ) ) {
+ if ( ! metadataName || ! isOverridableBlock( block ) ) {
return { ...block, innerBlocks };
}
@@ -184,7 +172,7 @@ function getContentValuesFromInnerBlocks( blocks, defaultValues, legacyIdMap ) {
}
const metadataName =
legacyIdMap?.[ block.clientId ] ?? block.attributes.metadata?.name;
- if ( ! metadataName || ! hasOverridableAttributes( block ) ) {
+ if ( ! metadataName || ! isOverridableBlock( block ) ) {
continue;
}
@@ -217,7 +205,7 @@ function setBlockEditMode( setEditMode, blocks, mode ) {
blocks.forEach( ( block ) => {
const editMode =
mode ||
- ( hasOverridableAttributes( block ) ? 'contentOnly' : 'disabled' );
+ ( isOverridableBlock( block ) ? 'contentOnly' : 'disabled' );
setEditMode( block.clientId, editMode );
setBlockEditMode(
diff --git a/packages/edit-post/src/components/sidebar/settings-sidebar/index.js b/packages/edit-post/src/components/sidebar/settings-sidebar/index.js
index 50e103cb8b444c..2264dfaaa56d1e 100644
--- a/packages/edit-post/src/components/sidebar/settings-sidebar/index.js
+++ b/packages/edit-post/src/components/sidebar/settings-sidebar/index.js
@@ -43,6 +43,7 @@ import { unlock } from '../../../lock-unlock';
const { PostCardPanel } = unlock( editorPrivateApis );
const { Tabs } = unlock( componentsPrivateApis );
+const { PatternOverridesPanel } = unlock( editorPrivateApis );
const SIDEBAR_ACTIVE_BY_DEFAULT = Platform.select( {
web: true,
@@ -123,6 +124,7 @@ const SidebarContent = ( {
+
>
) }
diff --git a/packages/edit-site/src/components/sidebar-edit-mode/template-panel/index.js b/packages/edit-site/src/components/sidebar-edit-mode/template-panel/index.js
index c19cf45e4551bb..38e2ad46949061 100644
--- a/packages/edit-site/src/components/sidebar-edit-mode/template-panel/index.js
+++ b/packages/edit-site/src/components/sidebar-edit-mode/template-panel/index.js
@@ -9,8 +9,8 @@ import {
PostExcerptPanel,
PostLastRevisionPanel,
PostTaxonomiesPanel,
- store as editorStore,
privateApis as editorPrivateApis,
+ store as editorStore,
} from '@wordpress/editor';
import { store as coreStore } from '@wordpress/core-data';
import { __ } from '@wordpress/i18n';
@@ -31,7 +31,7 @@ import { TEMPLATE_PART_POST_TYPE } from '../../../utils/constants';
import { unlock } from '../../../lock-unlock';
const { PostCardPanel } = unlock( editorPrivateApis );
-
+const { PatternOverridesPanel } = unlock( editorPrivateApis );
const { useHistory } = unlock( routerPrivateApis );
function TemplatesList( { availableTemplates, onSelect } ) {
@@ -141,6 +141,7 @@ export default function TemplatePanel() {
+
>
);
}
diff --git a/packages/editor/src/components/pattern-overrides-panel/index.js b/packages/editor/src/components/pattern-overrides-panel/index.js
new file mode 100644
index 00000000000000..47cd6282b3fedd
--- /dev/null
+++ b/packages/editor/src/components/pattern-overrides-panel/index.js
@@ -0,0 +1,26 @@
+/**
+ * WordPress dependencies
+ */
+import { useSelect } from '@wordpress/data';
+import { privateApis as patternsPrivateApis } from '@wordpress/patterns';
+
+/**
+ * Internal dependencies
+ */
+import { store as editorStore } from '../../store';
+import { unlock } from '../../lock-unlock';
+
+const { OverridesPanel } = unlock( patternsPrivateApis );
+
+export default function PatternOverridesPanel() {
+ const supportsPatternOverridesPanel = useSelect(
+ ( select ) => select( editorStore ).getCurrentPostType() === 'wp_block',
+ []
+ );
+
+ if ( ! supportsPatternOverridesPanel ) {
+ return null;
+ }
+
+ return ;
+}
diff --git a/packages/editor/src/private-apis.js b/packages/editor/src/private-apis.js
index 10143bdf002627..ea42d6ad5fde5b 100644
--- a/packages/editor/src/private-apis.js
+++ b/packages/editor/src/private-apis.js
@@ -10,6 +10,7 @@ import DocumentTools from './components/document-tools';
import InserterSidebar from './components/inserter-sidebar';
import ListViewSidebar from './components/list-view-sidebar';
import ModeSwitcher from './components/mode-switcher';
+import PatternOverridesPanel from './components/pattern-overrides-panel';
import PluginPostExcerpt from './components/post-excerpt/plugin';
import PostPanelRow from './components/post-panel-row';
import PostViewLink from './components/post-view-link';
@@ -27,6 +28,7 @@ lock( privateApis, {
InserterSidebar,
ListViewSidebar,
ModeSwitcher,
+ PatternOverridesPanel,
PluginPostExcerpt,
PostPanelRow,
PostViewLink,
diff --git a/packages/patterns/src/api/index.js b/packages/patterns/src/api/index.js
new file mode 100644
index 00000000000000..07f3b2321a127c
--- /dev/null
+++ b/packages/patterns/src/api/index.js
@@ -0,0 +1,23 @@
+/**
+ * Internal dependencies
+ */
+import { PARTIAL_SYNCING_SUPPORTED_BLOCKS } from '../constants';
+
+/**
+ * Determines whether a block is overridable.
+ *
+ * @param {WPBlock} block The block to test.
+ *
+ * @return {boolean} `true` if a block is overridable, `false` otherwise.
+ */
+export function isOverridableBlock( block ) {
+ return (
+ Object.keys( PARTIAL_SYNCING_SUPPORTED_BLOCKS ).includes(
+ block.name
+ ) &&
+ !! block.attributes.metadata?.bindings &&
+ Object.values( block.attributes.metadata.bindings ).some(
+ ( binding ) => binding.source === 'core/pattern-overrides'
+ )
+ );
+}
diff --git a/packages/patterns/src/components/overrides-panel.js b/packages/patterns/src/components/overrides-panel.js
new file mode 100644
index 00000000000000..b567b3c372daa2
--- /dev/null
+++ b/packages/patterns/src/components/overrides-panel.js
@@ -0,0 +1,45 @@
+/**
+ * WordPress dependencies
+ */
+import {
+ privateApis as blockEditorPrivateApis,
+ store as blockEditorStore,
+} from '@wordpress/block-editor';
+import { PanelBody } from '@wordpress/components';
+import { useSelect } from '@wordpress/data';
+import { useMemo } from '@wordpress/element';
+import { __ } from '@wordpress/i18n';
+
+/**
+ * Internal dependencies
+ */
+import { isOverridableBlock } from '../api';
+import { unlock } from '../lock-unlock';
+
+const { BlockQuickNavigation } = unlock( blockEditorPrivateApis );
+
+export default function OverridesPanel() {
+ const allClientIds = useSelect(
+ ( select ) => select( blockEditorStore ).getClientIdsWithDescendants(),
+ []
+ );
+ const { getBlock } = useSelect( blockEditorStore );
+ const clientIdsWithOverrides = useMemo(
+ () =>
+ allClientIds.filter( ( clientId ) => {
+ const block = getBlock( clientId );
+ return isOverridableBlock( block );
+ } ),
+ [ allClientIds, getBlock ]
+ );
+
+ if ( ! clientIdsWithOverrides?.length ) {
+ return null;
+ }
+
+ return (
+
+
+
+ );
+}
diff --git a/packages/patterns/src/private-apis.js b/packages/patterns/src/private-apis.js
index 54ad5a4aa47d1b..15ff161305f4a6 100644
--- a/packages/patterns/src/private-apis.js
+++ b/packages/patterns/src/private-apis.js
@@ -2,6 +2,7 @@
* Internal dependencies
*/
import { lock } from './lock-unlock';
+import OverridesPanel from './components/overrides-panel';
import {
default as CreatePatternModal,
CreatePatternModalContents,
@@ -10,6 +11,7 @@ import {
default as DuplicatePatternModal,
useDuplicatePatternProps,
} from './components/duplicate-pattern-modal';
+import { isOverridableBlock } from './api';
import RenamePatternModal from './components/rename-pattern-modal';
import PatternsMenuItems from './components';
import RenamePatternCategoryModal from './components/rename-pattern-category-modal';
@@ -27,9 +29,11 @@ import {
export const privateApis = {};
lock( privateApis, {
+ OverridesPanel,
CreatePatternModal,
CreatePatternModalContents,
DuplicatePatternModal,
+ isOverridableBlock,
useDuplicatePatternProps,
RenamePatternModal,
PatternsMenuItems,