diff --git a/packages/block-editor/CHANGELOG.md b/packages/block-editor/CHANGELOG.md index d681f0f85e350a..a675dba2adc6f6 100644 --- a/packages/block-editor/CHANGELOG.md +++ b/packages/block-editor/CHANGELOG.md @@ -1,5 +1,9 @@ ## Master +### Breaking Changes + +- The default `URLInput` `autoFocus` prop value has changed from `true` to `false`. If you relied on the auto-focus behavior of the input, you must explicitly assign an `autoFocus={ true }` prop. Note that (in accordance with the purpose of this change) it is usually inadvisable to auto-focus an input. Refer to the [component README](https://github.com/WordPress/gutenberg/blob/master/packages/block-editor/src/components/url-input/README.md) for more information. + ## 3.7.0 (2020-02-10) ### New Features diff --git a/packages/block-editor/src/components/link-control/index.js b/packages/block-editor/src/components/link-control/index.js index 2e7234f7809e31..256df5d5555071 100644 --- a/packages/block-editor/src/components/link-control/index.js +++ b/packages/block-editor/src/components/link-control/index.js @@ -179,7 +179,7 @@ function LinkControl( { ); const [ isResolvingLink, setIsResolvingLink ] = useState( false ); const [ errorMessage, setErrorMessage ] = useState( null ); - const isEndingEditWithFocus = useRef( false ); + const isTogglingEditingWithFocusIntent = useRef( false ); const { fetchSearchSuggestions } = useSelect( ( select ) => { const { getSettings } = select( 'core/block-editor' ); @@ -209,7 +209,7 @@ function LinkControl( { // guaranteed. The link input may continue to be shown if the next value // is still unassigned after calling `onChange`. const hadFocusLoss = - isEndingEditWithFocus.current && + isTogglingEditingWithFocusIntent.current && wrapperNode.current && ! wrapperNode.current.contains( document.activeElement ); @@ -223,7 +223,7 @@ function LinkControl( { nextFocusTarget.focus(); } - isEndingEditWithFocus.current = false; + isTogglingEditingWithFocusIntent.current = false; }, [ isEditingLink ] ); /** @@ -241,6 +241,40 @@ function LinkControl( { }; }, [] ); + /** + * Sets editing state and marks that focus may need to be restored after + * the next render, using an optional explicit `setFocus` argument. Normally + * it should not be necessary to call this function, and instead rely on the + * behavior of `setIsEditingLinkWithRetainedFocus`. However, in cases where + * editing state is toggled in response to button clicks, this function + * normalizes browser inconsistencies where focus may not be assigned, + * considering any button click as effectively having focus. + * + * @see https://developer.mozilla.org/en-US/docs/Web/HTML/Element/button#Clicking_and_focus + * + * @param {boolean} nextIsEditingLink State toggle value to set. + * @param {boolean} [setFocus=true] Whether focus should be restored. + */ + function setIsEditingLinkWithFocus( nextIsEditingLink, setFocus = true ) { + isTogglingEditingWithFocusIntent.current = setFocus; + + setIsEditingLink( nextIsEditingLink ); + } + + /** + * Sets editing state and marks that focus may need to be restored after + * the next render, if focus was within the wrapper while toggled. + * + * @param {boolean} nextIsEditingLink State toggle value to set. + */ + function setIsEditingLinkWithRetainedFocus( nextIsEditingLink ) { + const setFocus = + !! wrapperNode.current && + wrapperNode.current.contains( document.activeElement ); + + setIsEditingLinkWithFocus( nextIsEditingLink, setFocus ); + } + /** * onChange LinkControlSearchInput event handler * @@ -321,18 +355,6 @@ function LinkControl( { } ); }; - /** - * Cancels editing state and marks that focus may need to be restored after - * the next render, if focus was within the wrapper when editing finished. - */ - function stopEditing() { - isEndingEditWithFocus.current = - !! wrapperNode.current && - wrapperNode.current.contains( document.activeElement ); - - setIsEditingLink( false ); - } - /** * Determines whether a given value could be a URL. Note this does not * guarantee the value is a URL only that it looks like it might be one. For @@ -386,9 +408,9 @@ function LinkControl( { // Only set link if request is resolved, otherwise enable edit mode. if ( newSuggestion ) { onChange( newSuggestion ); - stopEditing(); + setIsEditingLinkWithRetainedFocus( false ); } else { - setIsEditingLink( true ); + setIsEditingLinkWithRetainedFocus( true ); } } catch ( error ) { if ( error && error.isCanceled ) { @@ -402,12 +424,12 @@ function LinkControl( { ) ); setIsResolvingLink( false ); - setIsEditingLink( true ); + setIsEditingLinkWithRetainedFocus( true ); } }; const handleSelectSuggestion = ( suggestion, _value = {} ) => { - setIsEditingLink( false ); + setIsEditingLinkWithRetainedFocus( false ); onChange( { ..._value, ...suggestion } ); }; @@ -514,7 +536,7 @@ function LinkControl( { suggestion={ suggestion } index={ index } onClick={ () => { - stopEditing(); + setIsEditingLinkWithFocus( false ); onChange( { ...value, ...suggestion } ); } } isSelected={ index === selectedSuggestion } @@ -555,7 +577,7 @@ function LinkControl( { await handleOnCreate( inputValue ); } else { handleSelectSuggestion( suggestion, value ); - stopEditing(); + setIsEditingLinkWithRetainedFocus( false ); } } } renderSuggestions={ @@ -595,7 +617,7 @@ function LinkControl( {