Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add ability to prevent editing blocks using useBlockEditingMode() #50643

Merged
merged 17 commits into from
May 24, 2023

Conversation

noisysocks
Copy link
Member

@noisysocks noisysocks commented May 16, 2023

What, why, how

Alternative to #50082.
Required to implement #50857.
Part of #44461.

This PR adds a new API to block-editor that lets one restrict the editing UI of a block and its children. It's based off of the suggestion by @youknowriad in #50082 (comment).

A new hook is added called useBlockEditingMode( clientId, mode ):

function PostTitleEdit( { clientId } ) {
	useBlockEditingMode( clientId, 'disabled' );
	return (
		...
	);
}

A block can call this in its edit function (or anywhere, really) with mode set to one of the three options:

  • 'disabled' - This prevents editing the block entirely. The block cannot be selected or interacted with. It uses the same pointer-events: disabled mechanism that is used when templateLock is set to 'contentOnly'.
  • 'contentOnly' - This allows editing the block but removes most UI shown in the block toolbar so that only content related buttons (e.g. formatting) are shown. The block cannot be moved, deleted, etc. It's similar to the treatment received by a content block when templateLock is set to 'contentOnly'.
  • 'default' - Default. Block is fully editable, moveable, removable, etc.

The block editing mode is inherited from the parent all the way up to rootBlockEditingMode which is specified in the editor settings. This lets one disable the entire editor except for certain blocks which is what is needed to implement #49980.

The same hook can be used to return the current editing mode:

const blockEditingMode = useBlockEditingMode( clientId );

This hook is a convenience wrapper for the getBlockEditingMode( clientId ) selector and setBlockEditingMode( clientId, mode ) action. Both exist in the block-editor store.

Alternatives considered

See #50082 for my first approach at this.

The original idea suggested by @youknowriad in #50082 (comment) was to wrap blocks in a component that uses context to disable its children e.g. <DisableBlocksProvider>. This is not sufficient for our needs though as we need to be able to customise the UI in BlockTools and BlockToolbar which sits outside of the BlockListBlock component tree. We therefore need to be able to look up a block's editing mode by its client ID. Because the editing mode is inherited from the block's parent, this map needs to exist within the block editor store where we have access to the block tree.

Testing Instructions

  1. Apply the patch below by copying it and running pbpaste | git apply.
diff --git a/packages/edit-site/src/components/block-editor/index.js b/packages/edit-site/src/components/block-editor/index.js
index cc5e7c8d92..f97599a9b5 100644
--- a/packages/edit-site/src/components/block-editor/index.js
+++ b/packages/edit-site/src/components/block-editor/index.js
@@ -39,7 +39,9 @@ import EditorCanvas from './editor-canvas';
 import { unlock } from '../../private-apis';
 import EditorCanvasContainer from '../editor-canvas-container';
 
-const { ExperimentalBlockEditorProvider } = unlock( blockEditorPrivateApis );
+const { ExperimentalBlockEditorProvider, useBlockEditingMode } = unlock(
+	blockEditorPrivateApis
+);
 
 const LAYOUT = {
 	type: 'default',
@@ -47,6 +49,11 @@ const LAYOUT = {
 	alignments: [],
 };
 
+function SetRootBlockEditingMode( { mode } ) {
+	useBlockEditingMode( mode );
+	return null;
+}
+
 export default function BlockEditor() {
 	const { setIsInserterOpened } = useDispatch( editSiteStore );
 	const { storedSettings, templateType, canvasMode } = useSelect(
@@ -162,6 +169,7 @@ export default function BlockEditor() {
 			onChange={ onChange }
 			useSubRegistry={ false }
 		>
+			<SetRootBlockEditingMode mode="disabled" />
 			<TemplatePartConverter />
 			<SidebarInspectorFill>
 				<BlockInspector />
diff --git a/packages/edit-site/src/hooks/block-editing-mode.js b/packages/edit-site/src/hooks/block-editing-mode.js
new file mode 100644
index 0000000000..24d63f122f
--- /dev/null
+++ b/packages/edit-site/src/hooks/block-editing-mode.js
@@ -0,0 +1,50 @@
+/**
+ * WordPress dependencies
+ */
+import { createHigherOrderComponent } from '@wordpress/compose';
+import { addFilter } from '@wordpress/hooks';
+import {
+	store as blockEditorStore,
+	privateApis as blockEditorPrivateApis,
+} from '@wordpress/block-editor';
+import { useSelect } from '@wordpress/data';
+
+/**
+ * Internal dependencies
+ */
+import { unlock } from '../private-apis';
+
+const { useBlockEditingMode } = unlock( blockEditorPrivateApis );
+
+export const withBlockEditingMode = createHigherOrderComponent(
+	( BlockEdit ) => ( props ) => {
+		const mode = useSelect( ( select ) => {
+			if (
+				[
+					'core/post-title',
+					'core/post-featured-image',
+					'core/post-content',
+				].includes( props.name )
+			) {
+				return 'contentOnly';
+			}
+			if (
+				select( blockEditorStore ).getBlockParentsByBlockName(
+					props.clientId,
+					'core/post-content'
+				).length
+			) {
+				return 'default';
+			}
+		} );
+		useBlockEditingMode( mode );
+		return <BlockEdit { ...props } />;
+	},
+	'withBlockEditingMode'
+);
+
+addFilter(
+	'editor.BlockEdit',
+	'core/edit-site/block-editing-mode',
+	withBlockEditingMode
+);
diff --git a/packages/edit-site/src/hooks/index.js b/packages/edit-site/src/hooks/index.js
index 513634c55b..de9dc8c3f7 100644
--- a/packages/edit-site/src/hooks/index.js
+++ b/packages/edit-site/src/hooks/index.js
@@ -1,6 +1,7 @@
 /**
  * Internal dependencies
  */
+import './block-editing-mode';
 import './components';
 import './push-changes-to-global-styles';
 import './template-part-edit';
  1. Navigate to Appearance → Editor → Pages and select a page to edit.
  2. All blocks should be locked except for the Post Title, Post Featured Image and Post Content blocks.

To test the existing contentOnly functionality for regressions:

  1. Add a pattern that has a content lock, e.g. this one.
  2. Edit a post, page, or template.
  3. Insert that pattern.

@noisysocks noisysocks added [Type] Enhancement A suggestion for improvement. [Package] Block editor /packages/block-editor Needs Dev Note Requires a developer note for a major WordPress release cycle labels May 16, 2023
@noisysocks noisysocks requested review from youknowriad and talldan May 16, 2023 00:55
@noisysocks
Copy link
Member Author

WDYT of this alternative approach @youknowriad @talldan? If we like it I can add tests and docs.

Comment on lines 236 to 238
'is-editing-disabled':
blockEditingMode === 'disabled' ||
( hasContentLockedParent && ! isContentBlock ),
Copy link
Member Author

@noisysocks noisysocks May 16, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Something I'm not sure about is whether or not to have getEditingMode contain the template lock logic as well so that components only need to call getBlockEditingMode instead of both getBlockEditingMode and getTemplateLock.

It makes the selector logic more complicated, but means that blocks/consumers only have to worry about calling one method and adjusting the UI accordingly.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It makes the selector logic more complicated, but means that blocks/consumers only have to worry about calling one method and adjusting the UI accordingly.

If the API goes public my vote is for this. Maybe it'll be make it easier to maintain too if the conditions are all in one spot?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also to note - if I remember correctly, there's also the potential for getBlockEditingMode to take over the work of (or even replace) __unstableGetContentLockingParent?

The while loops look very similar.

From what I can see __unstableGetContentLockingParent is exclusively used to check if a block is locked by its "inert" parent - the parent's clientId is never used.

In my mind, it would be handy to know whether a block is disabled with one call.

but maybe we should avoid over-thinking/optimization altogether 😄

It's a tricky one. I'm also thinking, is there nothing a few meaningfully-named functions or properties couldn't tell us?

isBlockEditingDisabled(); // blockEditingMode === 'disabled'
isLockedByParent();  // !! __unstableGetContentLockingParent( clientId )
isContentEditingEnabled(); // blockEditingMode === 'contentOnly' || templateLock === 'contentOnly'

See useBlockLock (packages/block-editor/src/components/block-lock/use-block-lock.js), that returns these sorts of things.

I'm not sure if it's related, but I'm wondering also if we need to taken into account any "lock" instructions on the block's attributes. See canEditBlock (packages/block-editor/src/store/selectors.js).

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've put the contentOnly template lock logic into getBlockEditingMode in one of my more recent commits. I think it's definitely nice having only one selector to call, and hopefully the JSDoc comments I help demystify it. What do you think?

Respecting canEditBlock is an interesting idea. I'll investigate. From memory canEditBlock is a bit weird and we only use it in the navigation block.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

OK so you can test what canEditBlock does yourself by selecting a Navigation block, open the settings menu, selecting Lock, then enabling Lock editing.

When you do this you'll see that you can select the Navigation block but that there's an overlay meaning you can't select its children or edit anything. This is different from either 'disabled' which prevents section altogether or 'contentOnly' which allows editing but hides toolbars, etc.

I suppose we could add a fourth mode called e.g. 'overlay' to getBlockEditingMode which encapsulates this behaviour. I'm not really sure there's any benefit to doing this though. getBlockEditingMode is a private API so it's something we can change easily in the future if there's a need.

@noisysocks noisysocks removed the Needs Dev Note Requires a developer note for a major WordPress release cycle label May 16, 2023
@github-actions
Copy link

github-actions bot commented May 16, 2023

Size Change: +914 B (0%)

Total Size: 1.4 MB

Filename Size Change
build/block-editor/content-rtl.css 4.23 kB +111 B (+3%)
build/block-editor/content.css 4.23 kB +111 B (+3%)
build/block-editor/index.min.js 199 kB +377 B (0%)
build/block-editor/style-rtl.css 15.1 kB -47 B (0%)
build/block-editor/style.css 15.1 kB -46 B (0%)
build/block-library/blocks/post-comments-form/style-rtl.css 508 B +7 B (+1%)
build/block-library/blocks/post-comments-form/style.css 508 B +7 B (+1%)
build/block-library/index.min.js 205 kB +92 B (0%)
build/block-library/style-rtl.css 12.9 kB +125 B (+1%)
build/block-library/style.css 12.9 kB +126 B (+1%)
build/components/index.min.js 232 kB -5 B (0%)
build/data/index.min.js 8.68 kB -5 B (0%)
build/edit-site/index.min.js 63.9 kB -4 B (0%)
build/edit-site/style-rtl.css 10.5 kB +32 B (0%)
build/edit-site/style.css 10.5 kB +32 B (0%)
build/editor/index.min.js 45.8 kB +1 B (0%)
ℹ️ View Unchanged
Filename Size
build/a11y/index.min.js 982 B
build/annotations/index.min.js 2.76 kB
build/api-fetch/index.min.js 2.33 kB
build/autop/index.min.js 2.14 kB
build/blob/index.min.js 472 B
build/block-directory/index.min.js 7.18 kB
build/block-directory/style-rtl.css 1.02 kB
build/block-directory/style.css 1.02 kB
build/block-editor/default-editor-styles-rtl.css 381 B
build/block-editor/default-editor-styles.css 381 B
build/block-library/blocks/archives/editor-rtl.css 61 B
build/block-library/blocks/archives/editor.css 60 B
build/block-library/blocks/archives/style-rtl.css 90 B
build/block-library/blocks/archives/style.css 90 B
build/block-library/blocks/audio/editor-rtl.css 150 B
build/block-library/blocks/audio/editor.css 150 B
build/block-library/blocks/audio/style-rtl.css 122 B
build/block-library/blocks/audio/style.css 122 B
build/block-library/blocks/audio/theme-rtl.css 126 B
build/block-library/blocks/audio/theme.css 126 B
build/block-library/blocks/avatar/editor-rtl.css 116 B
build/block-library/blocks/avatar/editor.css 116 B
build/block-library/blocks/avatar/style-rtl.css 91 B
build/block-library/blocks/avatar/style.css 91 B
build/block-library/blocks/block/editor-rtl.css 305 B
build/block-library/blocks/block/editor.css 305 B
build/block-library/blocks/button/editor-rtl.css 584 B
build/block-library/blocks/button/editor.css 582 B
build/block-library/blocks/button/style-rtl.css 624 B
build/block-library/blocks/button/style.css 623 B
build/block-library/blocks/buttons/editor-rtl.css 337 B
build/block-library/blocks/buttons/editor.css 337 B
build/block-library/blocks/buttons/style-rtl.css 332 B
build/block-library/blocks/buttons/style.css 332 B
build/block-library/blocks/calendar/style-rtl.css 239 B
build/block-library/blocks/calendar/style.css 239 B
build/block-library/blocks/categories/editor-rtl.css 113 B
build/block-library/blocks/categories/editor.css 112 B
build/block-library/blocks/categories/style-rtl.css 124 B
build/block-library/blocks/categories/style.css 124 B
build/block-library/blocks/code/editor-rtl.css 53 B
build/block-library/blocks/code/editor.css 53 B
build/block-library/blocks/code/style-rtl.css 121 B
build/block-library/blocks/code/style.css 121 B
build/block-library/blocks/code/theme-rtl.css 124 B
build/block-library/blocks/code/theme.css 124 B
build/block-library/blocks/columns/editor-rtl.css 108 B
build/block-library/blocks/columns/editor.css 108 B
build/block-library/blocks/columns/style-rtl.css 409 B
build/block-library/blocks/columns/style.css 409 B
build/block-library/blocks/comment-author-avatar/editor-rtl.css 125 B
build/block-library/blocks/comment-author-avatar/editor.css 125 B
build/block-library/blocks/comment-content/style-rtl.css 92 B
build/block-library/blocks/comment-content/style.css 92 B
build/block-library/blocks/comment-template/style-rtl.css 199 B
build/block-library/blocks/comment-template/style.css 198 B
build/block-library/blocks/comments-pagination-numbers/editor-rtl.css 123 B
build/block-library/blocks/comments-pagination-numbers/editor.css 121 B
build/block-library/blocks/comments-pagination/editor-rtl.css 222 B
build/block-library/blocks/comments-pagination/editor.css 209 B
build/block-library/blocks/comments-pagination/style-rtl.css 235 B
build/block-library/blocks/comments-pagination/style.css 231 B
build/block-library/blocks/comments-title/editor-rtl.css 75 B
build/block-library/blocks/comments-title/editor.css 75 B
build/block-library/blocks/comments/editor-rtl.css 840 B
build/block-library/blocks/comments/editor.css 839 B
build/block-library/blocks/comments/style-rtl.css 637 B
build/block-library/blocks/comments/style.css 636 B
build/block-library/blocks/cover/editor-rtl.css 647 B
build/block-library/blocks/cover/editor.css 650 B
build/block-library/blocks/cover/style-rtl.css 1.61 kB
build/block-library/blocks/cover/style.css 1.6 kB
build/block-library/blocks/details/editor-rtl.css 65 B
build/block-library/blocks/details/editor.css 65 B
build/block-library/blocks/details/style-rtl.css 159 B
build/block-library/blocks/details/style.css 159 B
build/block-library/blocks/embed/editor-rtl.css 293 B
build/block-library/blocks/embed/editor.css 293 B
build/block-library/blocks/embed/style-rtl.css 410 B
build/block-library/blocks/embed/style.css 410 B
build/block-library/blocks/embed/theme-rtl.css 126 B
build/block-library/blocks/embed/theme.css 126 B
build/block-library/blocks/file/editor-rtl.css 316 B
build/block-library/blocks/file/editor.css 316 B
build/block-library/blocks/file/interactivity.min.js 395 B
build/block-library/blocks/file/style-rtl.css 269 B
build/block-library/blocks/file/style.css 270 B
build/block-library/blocks/file/view.min.js 375 B
build/block-library/blocks/freeform/editor-rtl.css 2.44 kB
build/block-library/blocks/freeform/editor.css 2.44 kB
build/block-library/blocks/freeform/style-rtl.css 246 B
build/block-library/blocks/freeform/style.css 246 B
build/block-library/blocks/gallery/editor-rtl.css 947 B
build/block-library/blocks/gallery/editor.css 952 B
build/block-library/blocks/gallery/style-rtl.css 1.53 kB
build/block-library/blocks/gallery/style.css 1.53 kB
build/block-library/blocks/gallery/theme-rtl.css 108 B
build/block-library/blocks/gallery/theme.css 108 B
build/block-library/blocks/group/editor-rtl.css 654 B
build/block-library/blocks/group/editor.css 654 B
build/block-library/blocks/group/style-rtl.css 57 B
build/block-library/blocks/group/style.css 57 B
build/block-library/blocks/group/theme-rtl.css 78 B
build/block-library/blocks/group/theme.css 78 B
build/block-library/blocks/heading/style-rtl.css 76 B
build/block-library/blocks/heading/style.css 76 B
build/block-library/blocks/html/editor-rtl.css 336 B
build/block-library/blocks/html/editor.css 337 B
build/block-library/blocks/image/editor-rtl.css 834 B
build/block-library/blocks/image/editor.css 833 B
build/block-library/blocks/image/style-rtl.css 652 B
build/block-library/blocks/image/style.css 652 B
build/block-library/blocks/image/theme-rtl.css 126 B
build/block-library/blocks/image/theme.css 126 B
build/block-library/blocks/latest-comments/style-rtl.css 357 B
build/block-library/blocks/latest-comments/style.css 357 B
build/block-library/blocks/latest-posts/editor-rtl.css 213 B
build/block-library/blocks/latest-posts/editor.css 212 B
build/block-library/blocks/latest-posts/style-rtl.css 478 B
build/block-library/blocks/latest-posts/style.css 478 B
build/block-library/blocks/list/style-rtl.css 88 B
build/block-library/blocks/list/style.css 88 B
build/block-library/blocks/media-text/editor-rtl.css 266 B
build/block-library/blocks/media-text/editor.css 263 B
build/block-library/blocks/media-text/style-rtl.css 507 B
build/block-library/blocks/media-text/style.css 505 B
build/block-library/blocks/more/editor-rtl.css 431 B
build/block-library/blocks/more/editor.css 431 B
build/block-library/blocks/navigation-link/editor-rtl.css 712 B
build/block-library/blocks/navigation-link/editor.css 711 B
build/block-library/blocks/navigation-link/style-rtl.css 115 B
build/block-library/blocks/navigation-link/style.css 115 B
build/block-library/blocks/navigation-submenu/editor-rtl.css 296 B
build/block-library/blocks/navigation-submenu/editor.css 295 B
build/block-library/blocks/navigation/editor-rtl.css 2.33 kB
build/block-library/blocks/navigation/editor.css 2.33 kB
build/block-library/blocks/navigation/interactivity.min.js 865 B
build/block-library/blocks/navigation/style-rtl.css 2.21 kB
build/block-library/blocks/navigation/style.css 2.2 kB
build/block-library/blocks/navigation/view-modal.min.js 2.81 kB
build/block-library/blocks/navigation/view.min.js 443 B
build/block-library/blocks/nextpage/editor-rtl.css 395 B
build/block-library/blocks/nextpage/editor.css 395 B
build/block-library/blocks/page-list/editor-rtl.css 401 B
build/block-library/blocks/page-list/editor.css 401 B
build/block-library/blocks/page-list/style-rtl.css 175 B
build/block-library/blocks/page-list/style.css 175 B
build/block-library/blocks/paragraph/editor-rtl.css 174 B
build/block-library/blocks/paragraph/editor.css 174 B
build/block-library/blocks/paragraph/style-rtl.css 279 B
build/block-library/blocks/paragraph/style.css 281 B
build/block-library/blocks/post-author/style-rtl.css 175 B
build/block-library/blocks/post-author/style.css 176 B
build/block-library/blocks/post-comments-form/editor-rtl.css 96 B
build/block-library/blocks/post-comments-form/editor.css 96 B
build/block-library/blocks/post-date/style-rtl.css 61 B
build/block-library/blocks/post-date/style.css 61 B
build/block-library/blocks/post-excerpt/editor-rtl.css 71 B
build/block-library/blocks/post-excerpt/editor.css 71 B
build/block-library/blocks/post-excerpt/style-rtl.css 141 B
build/block-library/blocks/post-excerpt/style.css 141 B
build/block-library/blocks/post-featured-image/editor-rtl.css 588 B
build/block-library/blocks/post-featured-image/editor.css 586 B
build/block-library/blocks/post-featured-image/style-rtl.css 319 B
build/block-library/blocks/post-featured-image/style.css 319 B
build/block-library/blocks/post-navigation-link/style-rtl.css 153 B
build/block-library/blocks/post-navigation-link/style.css 153 B
build/block-library/blocks/post-template/editor-rtl.css 99 B
build/block-library/blocks/post-template/editor.css 98 B
build/block-library/blocks/post-template/style-rtl.css 281 B
build/block-library/blocks/post-template/style.css 281 B
build/block-library/blocks/post-terms/style-rtl.css 96 B
build/block-library/blocks/post-terms/style.css 96 B
build/block-library/blocks/post-time-to-read/style-rtl.css 69 B
build/block-library/blocks/post-time-to-read/style.css 69 B
build/block-library/blocks/post-title/style-rtl.css 100 B
build/block-library/blocks/post-title/style.css 100 B
build/block-library/blocks/preformatted/style-rtl.css 103 B
build/block-library/blocks/preformatted/style.css 103 B
build/block-library/blocks/pullquote/editor-rtl.css 135 B
build/block-library/blocks/pullquote/editor.css 135 B
build/block-library/blocks/pullquote/style-rtl.css 335 B
build/block-library/blocks/pullquote/style.css 335 B
build/block-library/blocks/pullquote/theme-rtl.css 167 B
build/block-library/blocks/pullquote/theme.css 167 B
build/block-library/blocks/query-pagination-numbers/editor-rtl.css 122 B
build/block-library/blocks/query-pagination-numbers/editor.css 121 B
build/block-library/blocks/query-pagination/editor-rtl.css 221 B
build/block-library/blocks/query-pagination/editor.css 211 B
build/block-library/blocks/query-pagination/style-rtl.css 288 B
build/block-library/blocks/query-pagination/style.css 284 B
build/block-library/blocks/query-title/style-rtl.css 63 B
build/block-library/blocks/query-title/style.css 63 B
build/block-library/blocks/query/editor-rtl.css 450 B
build/block-library/blocks/query/editor.css 449 B
build/block-library/blocks/quote/style-rtl.css 222 B
build/block-library/blocks/quote/style.css 222 B
build/block-library/blocks/quote/theme-rtl.css 223 B
build/block-library/blocks/quote/theme.css 226 B
build/block-library/blocks/read-more/style-rtl.css 132 B
build/block-library/blocks/read-more/style.css 132 B
build/block-library/blocks/rss/editor-rtl.css 149 B
build/block-library/blocks/rss/editor.css 149 B
build/block-library/blocks/rss/style-rtl.css 289 B
build/block-library/blocks/rss/style.css 288 B
build/block-library/blocks/search/editor-rtl.css 178 B
build/block-library/blocks/search/editor.css 178 B
build/block-library/blocks/search/style-rtl.css 434 B
build/block-library/blocks/search/style.css 432 B
build/block-library/blocks/search/theme-rtl.css 114 B
build/block-library/blocks/search/theme.css 114 B
build/block-library/blocks/separator/editor-rtl.css 146 B
build/block-library/blocks/separator/editor.css 146 B
build/block-library/blocks/separator/style-rtl.css 234 B
build/block-library/blocks/separator/style.css 234 B
build/block-library/blocks/separator/theme-rtl.css 194 B
build/block-library/blocks/separator/theme.css 194 B
build/block-library/blocks/shortcode/editor-rtl.css 323 B
build/block-library/blocks/shortcode/editor.css 323 B
build/block-library/blocks/site-logo/editor-rtl.css 754 B
build/block-library/blocks/site-logo/editor.css 754 B
build/block-library/blocks/site-logo/style-rtl.css 203 B
build/block-library/blocks/site-logo/style.css 203 B
build/block-library/blocks/site-tagline/editor-rtl.css 86 B
build/block-library/blocks/site-tagline/editor.css 86 B
build/block-library/blocks/site-title/editor-rtl.css 116 B
build/block-library/blocks/site-title/editor.css 116 B
build/block-library/blocks/site-title/style-rtl.css 57 B
build/block-library/blocks/site-title/style.css 57 B
build/block-library/blocks/social-link/editor-rtl.css 184 B
build/block-library/blocks/social-link/editor.css 184 B
build/block-library/blocks/social-links/editor-rtl.css 674 B
build/block-library/blocks/social-links/editor.css 673 B
build/block-library/blocks/social-links/style-rtl.css 1.4 kB
build/block-library/blocks/social-links/style.css 1.39 kB
build/block-library/blocks/spacer/editor-rtl.css 348 B
build/block-library/blocks/spacer/editor.css 348 B
build/block-library/blocks/spacer/style-rtl.css 48 B
build/block-library/blocks/spacer/style.css 48 B
build/block-library/blocks/table/editor-rtl.css 433 B
build/block-library/blocks/table/editor.css 433 B
build/block-library/blocks/table/style-rtl.css 645 B
build/block-library/blocks/table/style.css 644 B
build/block-library/blocks/table/theme-rtl.css 146 B
build/block-library/blocks/table/theme.css 146 B
build/block-library/blocks/tag-cloud/style-rtl.css 251 B
build/block-library/blocks/tag-cloud/style.css 253 B
build/block-library/blocks/template-part/editor-rtl.css 403 B
build/block-library/blocks/template-part/editor.css 403 B
build/block-library/blocks/template-part/theme-rtl.css 101 B
build/block-library/blocks/template-part/theme.css 101 B
build/block-library/blocks/text-columns/editor-rtl.css 95 B
build/block-library/blocks/text-columns/editor.css 95 B
build/block-library/blocks/text-columns/style-rtl.css 166 B
build/block-library/blocks/text-columns/style.css 166 B
build/block-library/blocks/verse/style-rtl.css 99 B
build/block-library/blocks/verse/style.css 99 B
build/block-library/blocks/video/editor-rtl.css 552 B
build/block-library/blocks/video/editor.css 555 B
build/block-library/blocks/video/style-rtl.css 174 B
build/block-library/blocks/video/style.css 174 B
build/block-library/blocks/video/theme-rtl.css 126 B
build/block-library/blocks/video/theme.css 126 B
build/block-library/classic-rtl.css 179 B
build/block-library/classic.css 179 B
build/block-library/common-rtl.css 1.1 kB
build/block-library/common.css 1.1 kB
build/block-library/editor-elements-rtl.css 75 B
build/block-library/editor-elements.css 75 B
build/block-library/editor-rtl.css 12 kB
build/block-library/editor.css 12 kB
build/block-library/elements-rtl.css 54 B
build/block-library/elements.css 54 B
build/block-library/interactivity/runtime.min.js 2.23 kB
build/block-library/interactivity/vendors.min.js 8.15 kB
build/block-library/reset-rtl.css 478 B
build/block-library/reset.css 478 B
build/block-library/theme-rtl.css 686 B
build/block-library/theme.css 691 B
build/block-serialization-default-parser/index.min.js 1.12 kB
build/block-serialization-spec-parser/index.min.js 2.83 kB
build/blocks/index.min.js 50.9 kB
build/commands/index.min.js 15 kB
build/commands/style-rtl.css 827 B
build/commands/style.css 827 B
build/components/style-rtl.css 11.7 kB
build/components/style.css 11.7 kB
build/compose/index.min.js 12.4 kB
build/core-commands/index.min.js 1.8 kB
build/core-data/index.min.js 16.5 kB
build/customize-widgets/index.min.js 12.2 kB
build/customize-widgets/style-rtl.css 1.38 kB
build/customize-widgets/style.css 1.38 kB
build/data-controls/index.min.js 708 B
build/date/index.min.js 40.5 kB
build/deprecated/index.min.js 507 B
build/dom-ready/index.min.js 324 B
build/dom/index.min.js 4.72 kB
build/edit-post/classic-rtl.css 544 B
build/edit-post/classic.css 545 B
build/edit-post/index.min.js 35.2 kB
build/edit-post/style-rtl.css 7.76 kB
build/edit-post/style.css 7.75 kB
build/edit-widgets/index.min.js 17.3 kB
build/edit-widgets/style-rtl.css 4.53 kB
build/edit-widgets/style.css 4.53 kB
build/editor/style-rtl.css 3.54 kB
build/editor/style.css 3.53 kB
build/element/index.min.js 4.89 kB
build/escape-html/index.min.js 537 B
build/format-library/index.min.js 7.77 kB
build/format-library/style-rtl.css 554 B
build/format-library/style.css 553 B
build/hooks/index.min.js 1.64 kB
build/html-entities/index.min.js 448 B
build/i18n/index.min.js 3.73 kB
build/is-shallow-equal/index.min.js 527 B
build/keyboard-shortcuts/index.min.js 1.78 kB
build/keycodes/index.min.js 1.91 kB
build/list-reusable-blocks/index.min.js 2.14 kB
build/list-reusable-blocks/style-rtl.css 836 B
build/list-reusable-blocks/style.css 836 B
build/media-utils/index.min.js 2.97 kB
build/notices/index.min.js 963 B
build/plugins/index.min.js 1.85 kB
build/preferences-persistence/index.min.js 2.22 kB
build/preferences/index.min.js 1.33 kB
build/primitives/index.min.js 944 B
build/priority-queue/index.min.js 1.52 kB
build/private-apis/index.min.js 939 B
build/react-i18n/index.min.js 696 B
build/react-refresh-entry/index.min.js 8.44 kB
build/react-refresh-runtime/index.min.js 7.31 kB
build/redux-routine/index.min.js 2.74 kB
build/reusable-blocks/index.min.js 2.25 kB
build/reusable-blocks/style-rtl.css 243 B
build/reusable-blocks/style.css 243 B
build/rich-text/index.min.js 11 kB
build/router/index.min.js 1.78 kB
build/server-side-render/index.min.js 2.08 kB
build/shortcode/index.min.js 1.42 kB
build/style-engine/index.min.js 1.52 kB
build/token-list/index.min.js 644 B
build/url/index.min.js 3.65 kB
build/vendors/inert-polyfill.min.js 2.48 kB
build/vendors/react-dom.min.js 41.8 kB
build/vendors/react.min.js 4.02 kB
build/viewport/index.min.js 1.08 kB
build/warning/index.min.js 268 B
build/widgets/index.min.js 7.28 kB
build/widgets/style-rtl.css 1.15 kB
build/widgets/style.css 1.16 kB
build/wordcount/index.min.js 1.06 kB

compressed-size-action

@github-actions
Copy link

github-actions bot commented May 16, 2023

Flaky tests detected in 25f3e58.
Some tests passed with failed attempts. The failures may not be related to this commit but are still reported for visibility. See the documentation for more information.

🔍 Workflow run URL: https://github.com/WordPress/gutenberg/actions/runs/5052708088
📝 Reported issues:

Copy link
Contributor

@youknowriad youknowriad left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If this gives us the right solution, in terms of behavior, then I'm sold. I like this API and I can even see ourselves making it public.

Some thoughts/questions:

  • Will this have any impact on performance?
  • I know that we can't use inert/useDisabled for the "contentOnly" mode but maybe we can still consolidate this with __unstableHasActiveBlockOverlayActive (make the latter use the former maybe) and when the mode is "disabled", rely on useDisabled? (It doesn't really have to be here, but curious for your thoughts)
  • I like how this also replaces the "contentOnly" lock behavior (so not really a new mode or something)

@noisysocks
Copy link
Member Author

noisysocks commented May 19, 2023

Thanks @ramonjd for jumping on a call with me to review this PR. Notes from that call:

  • Change useBlockEditingMode( clientId, mode )useBlockEditingMode( mode ). Look up clientId from context. That way can call this hook from top level instead of having a public settings.rootBlockEditingMode.
  • Put the "is template content locked?" logic into getBlockEditingMode selector so that we're not repeating ourselves everywhere. Document that the selector will do this logic. Possibly should be two selectors? Possibly should be done as a follow up?
  • List View and breadcrumbs need work. Disabled blocks shouldn't be shown there.

Will this have any impact on performance?

Will double check but should be okay so long as the getBlockEditingMode selector is memoised. (Need to do this.)

I know that we can't use inert/useDisabled for the "contentOnly" mode but maybe we can still consolidate this with __unstableHasActiveBlockOverlayActive (make the latter use the former maybe) and when the mode is "disabled", rely on useDisabled? (It doesn't really have to be here, but curious for your thoughts)

@ramonjd and I spoke about this. We could put inert on blocks where we know that all of the block's descendants are disabled. Maybe something worth exploring in a follow up.

I like how this also replaces the "contentOnly" lock behavior (so not really a new mode or something)

To be clear: templateLock = 'contentOnly' still exists. Just adding a different mechanism to activate that same functionality.

I'll address all this feedback and hopefully have this ready for review/merge early next week.

Copy link
Member

@ramonjd ramonjd left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Great stuff. I've mainly added notes from last week.

I like the direction (now I know a bit more about the problem, constraints and history, thank you!)

Happy to keep testing as it iterates.

packages/block-editor/src/store/private-selectors.js Outdated Show resolved Hide resolved
Comment on lines 236 to 238
'is-editing-disabled':
blockEditingMode === 'disabled' ||
( hasContentLockedParent && ! isContentBlock ),
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also to note - if I remember correctly, there's also the potential for getBlockEditingMode to take over the work of (or even replace) __unstableGetContentLockingParent?

The while loops look very similar.

From what I can see __unstableGetContentLockingParent is exclusively used to check if a block is locked by its "inert" parent - the parent's clientId is never used.

In my mind, it would be handy to know whether a block is disabled with one call.

but maybe we should avoid over-thinking/optimization altogether 😄

It's a tricky one. I'm also thinking, is there nothing a few meaningfully-named functions or properties couldn't tell us?

isBlockEditingDisabled(); // blockEditingMode === 'disabled'
isLockedByParent();  // !! __unstableGetContentLockingParent( clientId )
isContentEditingEnabled(); // blockEditingMode === 'contentOnly' || templateLock === 'contentOnly'

See useBlockLock (packages/block-editor/src/components/block-lock/use-block-lock.js), that returns these sorts of things.

I'm not sure if it's related, but I'm wondering also if we need to taken into account any "lock" instructions on the block's attributes. See canEditBlock (packages/block-editor/src/store/selectors.js).

packages/block-editor/src/store/private-selectors.js Outdated Show resolved Hide resolved
@noisysocks noisysocks requested a review from nerrad as a code owner May 22, 2023 06:41
@ramonjd
Copy link
Member

ramonjd commented May 23, 2023

I tested content locking in the site and post editor for the following blocks (which activate templateLock in their block.json files)

  • navigation
  • group
  • cover
  • columns
  • column

Child content is locked as I expected with no regressions.

The only change I can see, which is an added bonus, is the addition of locked symbols on the child block items in list view.

Trunk This PR
Screenshot 2023-05-23 at 7 45 03 pm Screenshot 2023-05-23 at 7 37 51 pm

Copy link
Member

@ramonjd ramonjd left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for persisting with this one @noisysocks

@noisysocks noisysocks self-assigned this May 23, 2023
@talldan
Copy link
Contributor

talldan commented May 26, 2023

@noisysocks Random thought, I wonder whether it might be worth consolidating visual/html mode into this API.

At the moment there's a very similar API for that (getBlockMode / toggleBlockMode).

I could imagine default, contentOnly and disabled are all considered visual (with default being the normal visual mode), and an html option exists for html mode.

@bph bph added the Needs Dev Note Requires a developer note for a major WordPress release cycle label May 30, 2023
@bph
Copy link
Contributor

bph commented May 30, 2023

The key thing that makes me happy is that the API is entirely private now so we're not tied down to anything. (@noisysocks

If this gives us the right solution, in terms of behavior, then I'm sold. I like this API and I can even see ourselves making it public. (@youknowriad

Curation / Governance is a big topic right now and this change caught my attention, as there is a need to add extensibility to parts of the site editor to enable more granular governance. It seems this API could be made public and delight a few agency and product developers. Maybe it's too early, hard for me to say, though. It might be worth a discussion, especially if this wouldn't be a hard lift.
cc @mtias

Comment on lines +23 to +36
const removeDisabledBlocks = ( tree ) => {
return tree.flatMap( ( { clientId, innerBlocks, ...rest } ) => {
if ( getBlockEditingMode( clientId ) === 'disabled' ) {
return removeDisabledBlocks( innerBlocks );
}
return [
{
clientId,
innerBlocks: removeDisabledBlocks( innerBlocks ),
...rest,
},
];
} );
};
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This will return a new array reference on each call and cause the whole ListView to be rendered, even if the client IDs tree is the same.

Should we extract this into a new private memoized selector?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I tested something out by adding the following selector to packages/block-editor/src/store/private-selectors.js.

const EMPTY_ARRAY = [];
export const getEnabledBlocks = createSelector(
	( state, blocks = EMPTY_ARRAY ) =>
		blocks.flatMap( ( { clientId, innerBlocks, ...rest } ) => {
			if ( getBlockEditingMode( state, clientId ) === 'disabled' ) {
				return getEnabledBlocks( state, innerBlocks );
			}
			return [
				{
					clientId,
					innerBlocks: getEnabledBlocks( state, innerBlocks ),
					...rest,
				},
			];
		} ),
	( state ) => [ state.blockEditingModes, state.blocks.parents ]
);

Used in packages/block-editor/src/components/list-view/use-list-view-client-ids.js

...
			const {
				getDraggedBlockClientIds,
				getSelectedBlockClientIds,
				__unstableGetClientIdsTree,
				getEnabledBlocks,
			} = unlock( select( blockEditorStore ) );
...
				clientIdsTree: getEnabledBlocks(
					blocks ?? __unstableGetClientIdsTree( rootClientId )
				),
...

And it does prevent entire tree rerenders, which might be helpful for posts with lots of blocks:

Before

2023-05-31.14.46.03.mp4

After

2023-05-31.14.44.52.mp4

I'm just not sure about the dependency array [ state.blockEditingModes, state.blocks.parents ], which I copied from getExplicitBlockEditingMode() since that function is called in getEnabledBlocks() anyway.

Comment on lines +34 to +37
parents: getBlockParents( selectedBlockClientId ).filter(
( parentClientId ) =>
getBlockEditingMode( parentClientId ) !== 'disabled'
),
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same here.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
[Feature] Page Content Focus Ability to toggle between focusing on editing the page and editing the template in the site editor. [Package] Block editor /packages/block-editor [Type] Enhancement A suggestion for improvement.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants