-
Notifications
You must be signed in to change notification settings - Fork 4.3k
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
Block Bindings: Fix focus behavior when pressing Enter on connected block #58958
Changes from 11 commits
c7049b7
db9cf94
13469ca
5920feb
67be306
3778cc5
e8f52a6
63dc74f
a0517a9
e6f3bb7
df1730c
5578d8c
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -38,6 +38,7 @@ export function useEnter( props ) { | |
disableLineBreaks, | ||
onSplitAtEnd, | ||
onSplitAtDoubleLineEnd, | ||
shouldDisableEditing, | ||
} = propsRef.current; | ||
|
||
event.preventDefault(); | ||
|
@@ -67,7 +68,13 @@ export function useEnter( props ) { | |
|
||
const { text, start, end } = _value; | ||
|
||
if ( event.shiftKey ) { | ||
if ( shouldDisableEditing ) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can you explain what that branch of the code is doing? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I believe this explanation about how Another option here could be to change From my testing, it seems that when the rich text is not editable, it acts as if the cursor was in the initial character. This causes that when you click on enter, it doesn't work as expected. For that reason, I believe we could assume that when There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I see. I was totally lost, so that made it clear. I think simulating the case that the cursor is at the end should do the trick for most of the blocks 💯 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Just changed that part in this commit. |
||
// Simulate the cursor is at the end of the rich text. | ||
_value.start = _value.text?.length; | ||
_value.end = _value.text?.length; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We shouldn't have to do this stuff if we're just disabling rich text |
||
} | ||
|
||
if ( event.shiftKey && ! shouldDisableEditing ) { | ||
if ( ! disableLineBreaks ) { | ||
onChange( insert( _value, '\n' ) ); | ||
} | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,42 @@ | ||
/** | ||
* WordPress dependencies | ||
*/ | ||
import { useSelect } from '@wordpress/data'; | ||
import { getBlockType, store as blocksStore } from '@wordpress/blocks'; | ||
|
||
/** | ||
* Internal dependencies | ||
*/ | ||
import { BLOCK_BINDINGS_ALLOWED_BLOCKS } from '../../hooks/use-bindings-attributes'; | ||
import { useBlockEditContext } from '../block-edit'; | ||
import { store as blockEditorStore } from '../../store'; | ||
import { unlock } from '../../lock-unlock'; | ||
|
||
export function useShouldDisableEditing() { | ||
const { clientId, name: blockName } = useBlockEditContext(); | ||
|
||
const { getBlockAttributes } = useSelect( blockEditorStore ); | ||
const { getBlockBindingsSource } = unlock( useSelect( blocksStore ) ); | ||
const blockBindings = getBlockAttributes( clientId )?.metadata?.bindings; | ||
|
||
if ( blockBindings && blockName in BLOCK_BINDINGS_ALLOWED_BLOCKS ) { | ||
const blockTypeAttributes = getBlockType( blockName ).attributes; | ||
|
||
for ( const [ attribute, args ] of Object.entries( blockBindings ) ) { | ||
if ( blockTypeAttributes?.[ attribute ]?.source !== 'rich-text' ) { | ||
break; | ||
} | ||
|
||
// If the source is not defined, or if its value of `lockAttributesEditing` is `true`, disable it. | ||
const blockBindingsSource = getBlockBindingsSource( args.source ); | ||
if ( | ||
! blockBindingsSource || | ||
blockBindingsSource.lockAttributesEditing | ||
) { | ||
return true; | ||
} | ||
} | ||
} | ||
|
||
return false; | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -293,12 +293,23 @@ function ButtonEdit( props ) { | |
...spacingProps.style, | ||
...shadowProps.style, | ||
} } | ||
onSplit={ ( value ) => | ||
createBlock( 'core/button', { | ||
...attributes, | ||
text: value, | ||
} ) | ||
} | ||
onSplit={ ( value, isOriginal ) => { | ||
let newAttributes; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I would be also curious to better understand the changes applied here. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Okay, I'll try to do my best from what I understood from my testings 😄 When you use
With the previous logic, we were passing the attributes even if it wasn't the original block and its value it's empty. This happens when you click enter at the end of the block, with the cursor in the last character. You only want to preserve some value and the attributes when you actually split the text. Actually, I just realized that we are preserving the metadata in the new block, which I believe it is not correct. For example, if I have a button and rename it to "Custom button" with text "My button", I put the cursor in the middle and I click Enter, the new block has the "Custom button" name as well. It should be easy to fix just checking if it is the original block or not. I can work on that. Anyway, I believe this last part is not related to this PR because it happens with not bound blocks using rich text: paragraph, heading, button. It could be fixed changing the conditional to something like this: if ( isOriginal ) {
newAttributes = {
...attributes,
text: value,
};
} else if ( value ) {
newAttributes = {
...attributes,
metadata: undefined,
text: value,
};
} But we can work on that in a follow-up if we really want to get rid of the metadata There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Would it work this way? newAttributes = {
...attributes,
metadata: isOriginal ? attributes.metadata : undefined,
text: value,
}; There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I guess 🙂 Much better! There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This seems unrelated to the main problem? Maybe split this out in a separate PR? |
||
if ( isOriginal || value ) { | ||
newAttributes = { | ||
...attributes, | ||
text: value, | ||
}; | ||
} | ||
const block = createBlock( | ||
'core/button', | ||
newAttributes | ||
); | ||
if ( isOriginal ) { | ||
block.clientId = clientId; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why is this necessary? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I assumed it is needed because when we split a paragraph or a button we are actually creating two new blocks, and I guess that createBlock generates a new |
||
} | ||
return block; | ||
} } | ||
onReplace={ onReplace } | ||
onMerge={ mergeBlocks } | ||
identifier="text" | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Back then, it was discussed to include the logic moved to the
useShouldDisableEditing
hook into the initial selector: link. It should improve performance and ensure the hooks are not called more than necessary. Is there any reason to move it outside?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So I've done some additional testing, and it looks like setting
shouldDisableEditing
in the selector results in inconsistent behavior. You're also right in this comment saying that the blur isn't necessary to get the focus to work. In a previous iteration, blurring the target allowed focus to be set to the new block, though apparently this wasn't addressing the core issue.The real problem, I'm realizing, is that because the
shouldDisableEditing
value from the selector is inconsistent, it conflicts with the expected behavior when settingtabIndex
, preventing focus from being set to the new block.Here's a video going over it in more detail:
enter-focus-comparison.mp4
I've pushed another commit removing the blur logic. We can also move hook logic back to the Rich Text
index.js
— it'll work as long as the logic remains outside of the selector. Personally, I think having the hook separate is cleaner and easier to reason with, but I'm happy to take either approach.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for the explanation. I guess there is an issue when the block is selected, but I am not able to identify the why.
I don't have a strong opinion whatever you prefer.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
See my comment with the potential refactoring for the new hook that would address most of the discussed performance implications: #58958 (comment).