-
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
Accessibility improvements for Block Inserter #37357
Accessibility improvements for Block Inserter #37357
Conversation
…o be able to focus second element.
packages/edit-post/src/components/secondary-sidebar/inserter-sidebar.js
Outdated
Show resolved
Hide resolved
…and visually hidden button.
packages/edit-post/src/components/secondary-sidebar/inserter-sidebar.js
Outdated
Show resolved
Hide resolved
@@ -30,6 +31,7 @@ export default function InserterSidebar() { | |||
const isMobileViewport = useViewportMatch( 'medium', '<' ); | |||
const [ inserterDialogRef, inserterDialogProps ] = useDialog( { | |||
onClose: () => setIsInserterOpened( false ), | |||
focusOnMount: 'secondElement', |
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.
Is this still needed? It feels very specific to have a "secondElement" option in focusOnMount. The reasoning is that the developer might change what's inside a dialog without touching this call at all (two separate concerns), so basically "secondElement" is meaningless.
In other words, if we really want to target a specific element, we should probably use a dedicate useEffect(() => { ref.focus() } )
instead of trying to include this logic into a reusable hook.
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.
I made this specifically for use in Core because I think it has other good uses. It also had a check to ensure the element existed before focus was attempted. Probably could have done a better job to work on a fallback.
However, I changed to the useEffect approach but this was tricky because I had to figure out a way to only fire useEffect on initial open. It seems useEffect fires every time the component mounts and focus kept happening when I tried to tab out of the search box.
Do you have any suggestions on if this is good enough? Should I try to access edit post store from menu.js directly or is passing a new property fine?
Thanks.
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.
Do you have any suggestions on if this is good enough? Should I try to access edit post store from menu.js directly or is passing a new property fine?
Mmm interesting challenge, I guess neither of these are ideal to me. The Menu shouldn't be aware of an outside isOpen
because it's not something that is always rendered in a toggable sidebar and block-editor shouldn't have any dependency on edit-post (cycle dependency).
We should try to find an alternative.
Can I ask why we want to focus the "second element" in the first place and not the "first element" like all dialogs?
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.
Can I ask why we want to focus the "second element" in the first place and not the "first element" like all dialogs?
Because in the case of the Block Inserter, it is better to focus the search input first as most users will likely search for their block instead of looking through the list. I know that's how I personally use it. If we were to focus the first element, it would focus the Close button which would create extra key strokes to interact with the search field and the Block Inserter content.
What if I added my changes back to the useFocusOnMount hook with additional checks to make sure the element passed in is valid or default to the first? I understand most devs probably won't use this, but there's a need for it in the Core Gutenberg editor. As you said, passing the state probably isn't a good idea as the Inserter is not always used in a sidebar. Although, I did have one question about that. If the Inserter wasn't used in a sidebar, wouldn't that search field get focus by default since it uses useFocusOnMount hook or would it simply focus the first element on the page? I would think it would focus the first element in the Inserter since the useDialog hook. This would also be super bad UX if the Inserter was just on the page, not in a sidebar, and focus got placed there first. Although, maybe it wouldn't work this way because the Inserter doesn't use those hooks, the sidebar in the editor does.
Personally, I think passing that extra parameter is fine that way we don't need access to core/edit-post store and the variable is set to false, so it's not like it is going to effect other code. Users can still use the useFocusOnMount hook as I've only disabled it in the sidebar managed by Gutenberg. Nothing really changes for the Inserter component itself.
Sorry, this was long winded. Thanks.
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.
The focus target for opening a dialog is variable depending on usage. For simple or informational dialogs (e.g., with only one or two focal points), placing focus at the beginning will be best (and ideally that should be the close button). For complex dialogs, however, the usage of the dialog really needs to be taken into consideration. Focusing on the task that needs to be performed using a dialog makes sense. It's still good to place focus as early in the modal as possible, but making it easier to complete the task is a good thing.
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.
@youknowriad Sadly, your solution did not work. 😞 When I tried to have useEffect in InserterSidebar with a ref on the parent div, it never worked even when passing isOpen as an argument. Whenever I used console.log, useEffect was firing too early for the focus in the search field. I tried to delay useEffect with a timeout before the focus, but never could get that right.
I started reading about children functional components receiving refs and decided to use forwardRef all the way to the search component. This seems highly inefficient though. 😞
I could use the useEffect inside the search component but it would still require some type of state tracking or else it would never know when the inserter is open. I wish there was a way to unify the elements so it wasn't all separate.
What are your thoughts about the forwardRef approach? It seems bad practice, but it does work and to my knowledge, it could almost be seen as an improvement since now anyone could have control over the search part of the Inserter.
If not this way, can you think of any other ways? I was thinking maybe this could be refactored in to one component, but that kind of defeats the purpose of React. BTW, when I mentioned console.log above, what I meant was in useEffect, console.log fired before my console.log of the Inserter when it was fully loaded. I tried a timeout of 1000 and it wasn't enough.
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 giving this a try. I appreciate the efforts.
The forwardRef approach is good practice when the forwarded ref represents the actual component, in the case of the inserter library, I'd expect the forwarded ref to be the inserter container, not just a specific element of that component (the input).
I'll take a look at the PR later today and see if I can find something that would work well.
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.
I've pushed a fix to your branch that I think works. Let me know what you think. I also noticed that you had inverted the condition about when to show the close button (should be hidden on desktop).
Last thing, this InserterSidebar
component is something that is also replicated in three other screens (site editor, widgets screen and navigation screen) so three other packages. I think we might want to make the same changes there as well.
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.
@youknowriad Can you check it out now? I also fixed a bug where the Inserter would not close in Widgets view. I adjusted the code in useCallback just a bit, seems to work now.
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.
BTW, this is the same approach I took except I didn't add the "input" selector. 😕 One tiny selector away and it would have worked for me. Thanks for the catch on this and ensuring I didn't create unnecessary ref forwarding to get this done.
const [ inserterDialogRef, inserterDialogProps ] = useDialog( { | ||
onClose: () => setIsInserterOpened( false ), | ||
focusOnMount: null, |
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.
Had to set this to null to prevent the useFocusOnMount hook from firing. Now that useEffect is being used, it would cause duplicate focus events.
…ry/accessibility-fixes-for-inserter
const inserterContentRef = useRef(); | ||
useEffect( () => { | ||
inserterContentRef.current | ||
.querySelector( '.block-editor-inserter__search input' ) |
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.
Could we use pass a searchInputRef
to <Library>
and all the way down to the actual <input>
element here? So that we can omit the query selector, which feels like an implementation detail.
Or we can use useImperativeHandle
to allow calling focusSearchInput()
on the <Library>
ref.
If all of the above can't be done, then I'd prefer using data-*
attribute over class names for explicitness.
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.
I think @youknowriad wanted to avoid doing anything with the block-editor/inserter component because it could be used as a single element. E.g. keep all the implementation in the sidebar files and out of what could be a reusable stand-alone component.
I actually used forwardRef() all the way to the Search component a few commits back.
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.
Yep I noticed that, but I think my approach is different because it's not using forwardRef
to bind the ref to the inserter instance, but passing a different prop searchInputRef
down to the search input element. useImperativeHandle
can also work with forwardRef
so that the ref is referencing the inserter instance but also providing focusSearchInput()
functionality as well.
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.
@kevin940726 I think a ref with an added property is a sane approach indeed. I'm fine either way but we should be mindful of added complexity.
@@ -25,7 +25,7 @@ export default function InserterSidebar() { | |||
const { setIsInserterOpened } = useDispatch( editWidgetsStore ); | |||
|
|||
const closeInserter = useCallback( () => { | |||
return () => setIsInserterOpened( false ); | |||
return setIsInserterOpened( false ); |
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.
That is indeed a good fix.
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.
This is looking good to me.
@youknowriad @kevin940726 Do you think this is good as is or would you like me to work on a different approach with passing the ref to Search component? Also happy to add a:
If it would help with correct targeting for focus. Thanks. |
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.
LGTM 👍
Preferably, I'd love to have a
|
@kevin940726 I will work on one over the weekend and see if I can pass the ref. I still am of the opinion that it's a lot of work for little reward, but maybe there's just something I don't see. Usually I'd be all for it but I think it's important to think of use context here. When I started this PR, I had no idea the library component could be used stand-alone so having a ref attached to it wouldn't actually help that much. Let me give it a test over the weekend though and see what happens. |
Description
This PR adds a Close button to the Block Inserter that displays all the time for screen readers only. This PR also makes an adjustment to useDialog() call passing focusOnMount: null to prevent duplicate focus on Inserter open.
Bug fix: This PR will fix a bug I noticed on the Widgets screen where Close button on Inserter wasn't working at all. Small tweak to useCallback() call fixed this up.
How has this been tested?
Screenshots
Types of changes
Bug fix
Checklist:
*.native.js
files for terms that need renaming or removal).