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

Components: Speak Notice message #15745

Merged
merged 16 commits into from
Feb 7, 2020
Merged

Components: Speak Notice message #15745

merged 16 commits into from
Feb 7, 2020

Conversation

aduth
Copy link
Member

@aduth aduth commented May 20, 2019

Fixes #9442
Closes #15708
Related: #15594

This pull request seeks to move spoken message handling for notices from the notices store into the Notice component.

Implementation notes:

Update (2019-12-10): The commentary in this section has since received feedback at #15745 (comment) , with subsequent revisions summarized at #15745 (comment)

I could use some accessibility feedback here. Per #9442, there's a recommendation to use the role attribute for notices affected by withNotices. It's unclear whether this is appropriate for all notices. The current implementation here preserves existing behavior so far as using wp.a11y.speak, but notices (both existing and from withNotices) will default to a politeness level corresponding to the status of the message (error messages as 'assertive', success messages as 'polite'). From the reading of the relevant roles, it doesn't seem like this must be a guaranteed 1-to-1 mapping (i.e. a success notice could be a time-sensitive, assertive message?), but is implemented instead as a sensible default. Notices can be configured to override this default by assigning role="polite" or role="assertive". The role is not in-fact assigned to the rendered element, but its usage with speak is assumed to be a suitable equivalent behavior.

Feedback needed:

  • Is the default behavior sensible?
  • Is the actual behavior for role in not assigning the DOM attribute acceptable?
    • In my testing, assigning it to the element would cause a duplicate spoken message, if speak is also used.
  • Would it be better to always use role and never wp.a11y.speak?
  • Should role be exposed at all, or instead based entirely from status? Or always 'assertive'?

Testing instructions:

Verify there are no regressions in the spoken messages of notices (example: published post "Post published!" notice).

Verify that notices mounted by withNotices (or ad hoc use of the Notice component) are spoken.

@aduth aduth added [Focus] Accessibility (a11y) Changes that impact accessibility and need corresponding review (e.g. markup changes). [Feature] UI Components Impacts or related to the UI component system [Package] Components /packages/components Needs Accessibility Feedback Need input from accessibility [Package] Notices /packages/notices labels May 20, 2019
@afercia
Copy link
Contributor

afercia commented May 25, 2019

@aduth thanks for working on this, I've finally found some time to have a look at it.

speak() vs. alert and status roles

Per #9442, there's a recommendation to use the role attribute for notices affected by withNotices

That was a suggestion: given the notices don't use speak(), a role alert or status would have worked. However, if notices are going to use speak(), they shouldn't use a role. As you pointed out, using both would cause a duplicate spoken message.

The alert and status roles are just specialized types of live regions. The speak() live regions implement the same behavior by the use of live region attributes. Quoting from the spec:

role=alert
A type of live region with important, and usually time-sensitive, information.
Elements with the role alert have an implicit aria-live value of assertive, and an implicit aria-atomic value of true.

role=status
A type of live region whose content is advisory information for the user but is not important enough to justify an alert
Elements with the role status have an implicit aria-live value of polite and an implicit aria-atomic value of true.

One difference is that the speak() live regions use explicit attributes:

aria-live="polite / assertive" aria-atomic="true" aria-relevant="additions text"

Basically, never use both speak() and alert/status roles together.

Which one should be used?

Theoretically, we should leverage existing standards and prefer role alert/status. However, there's one big caveat.

Some screen readers are able to announce elements with role alert / status that are "rendered on the fly" and added to the DOM. Not all screen readers are able to do that though. And that's not the recommended way to use these roles.

For maximum compatibility with the various browser/screen reader combos, it is expected that the elements with role alert / status are present in the DOM on page load. Typically, this is not the way the Gutenberg Notices work. They're added to the DOM. Instead, the speak() live regions are rendered when the DOM is ready and screen readers behave better: they're able to understand they have to "monitor" for changes within those elements.

References:

Using ARIA role=alert or Live Regions to Identify Errors
https://www.w3.org/WAI/WCAG21/Techniques/aria/ARIA19

The error container must be present in the DOM on page load for the error message to be spoken by most screen readers

Example:
https://www.w3.org/WAI/WCAG21/working-examples/aria-alert-identify-errors/

Using role=status to present status messages
https://www.w3.org/WAI/WCAG21/Techniques/aria/ARIA22

Check that the container destined to hold the status message has a role attribute with a value of status before the status message occurs.

Example:
https://www.w3.org/WAI/WCAG21/working-examples/aria-role-status-searchresults/

Considering all this, I'd recommend to use speak().

Code related considerations

The role is not in-fact assigned to the rendered element

Looking at the code, I was initially confused by this. It's just about naming. With this implementation, we're not setting ARIA roles, so I'd suggest to rename the prop and the getDefaultRole() function.

Seems to me what is needed here is to just determine the politeness level. Considering also that developers are not required to know what a role is, I'd suggest to:

  • rename the role prop to politeness
  • rename getDefaultRole() to getDefaultPoliteness() and make it return the politeness value

Is the default behavior sensible?

Looks good to me, as long as there's a way to override the default politeness level. As you pointed out, there's no 1-to-1 mapping between the notices type and the politeness level. For example, there may be cases where a success notice would need to be communicated with an assertive politeness level.

I'd also suggest to well document this and make very clear the difference is:

  • assertive: for important, and usually time-sensitive, information that will interrupt anything else the screen reader is announcing in that moment
  • polite: for advisory information that shouldn't interrupt what the screen reader is announcing in that moment (the "speech queue") or interrupt the current task

Worth also noting that these values are suggestion and assistive technologies may override them based on internal heuristics. For more details: https://www.w3.org/TR/wai-aria-1.1/#aria-live

Is the actual behavior for role in not assigning the DOM attribute acceptable?

Yes. I'd just rename things as commented above.

In my testing, assigning it to the element would cause a duplicate spoken message, if speak is also used.
Would it be better to always use role and never wp.a11y.speak?

See above.

Should role be exposed at all, or instead based entirely from status? Or always 'assertive'?

Not sure I fully understand, I guess the previous considerations cover this point?

Add a way to pass a custom message

While testing the ContrastChecker notice, I realized that the current implementation uses the notice content (children) as the message. This may not be always appropriate.

Note: the ContrastChecker still uses its own useEffect() / speak() which I guess should be removed.

In some cases, we're intentionally using a shorter, simpler message. For example in the ContrastChecker the visible messages are longer, e.g.:

This color combination may be hard for people to read. Try using a darker background color and/or a brighter text color.

While we've opted to use a horter, simpler, message for speak():

This color combination may be hard for people to read.

This is because the longer messages would add a lot of noise and in this cases a shorter message is preferable.

While the use of children as message seems a sensible default, seems to me we have to take into account the need for custom messages. I guess a message prop would work? If there's consensus, this part should be well explained in the documentation.

Note on current documentation:

Not sure this part is accurate:

Do use a Notice when you want to communicate a message of medium importance.

Referring to "medium importance" could be a bit misleading

Question on NoticeList

I'm not sure I understand if all this will work with the NoticeList component, but that's very likely because of my poor coding skills 🙂

@afercia afercia removed the Needs Accessibility Feedback Need input from accessibility label May 25, 2019
@youknowriad
Copy link
Contributor

Should we update this to include the snackbar component?

@youknowriad
Copy link
Contributor

Would be good to refresh and land this PR :)

@aduth aduth force-pushed the update/notice-speak branch from 3e61a21 to 8ca6db1 Compare December 10, 2019 20:39
@aduth aduth requested review from ellatrix and mkaz as code owners December 10, 2019 20:39
@aduth
Copy link
Member Author

aduth commented Dec 10, 2019

Apologies for my delay in revisiting this pull request, and a belated "Thank you!" to @afercia for your extended feedback.

I've had a chance to refresh this today. Specifically, this includes:

  • Rename the new Notice role prop to politeness, expecting one of the valid aria-live attribute values. This includes extended documentation on what's expected with these values (based on feedback that this be provided as a suggestion, and detailed overview of what's expected from each of 'polite' and 'assertive').
  • Include a new Notice prop spokenMessage to have full control over the message spoken in the notice.
  • Use this new spokenMessage prop in the implementation of ContrastChecker for its custom (shortened) message.
  • Implement the same speak behavior in the Snackbar component, including the same politeness, spokenMessage props from Notice. The default politeness for Snackbar is fixed to a value of 'polite', consistent with the prior behavior and in accordance with its documented usage for low-priority, non-interruptive messages. This can be overridden using the politeness prop.

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.

This is looking good to me 👍

I'd appreciate another a11y sanity check.

@youknowriad
Copy link
Contributor

Can we rebase and merge this @aduth ?

@afercia
Copy link
Contributor

afercia commented Jan 29, 2020

From an a11y perspective looks good to me, thanks! Defaults are sensible, ease of use is also appreciated.

Aside: Testing the ContrastChecker notice I... noticed the contrast check appears to have a lot of issues at the point it's basically unreliable. Is there an issue for that already?

@aduth
Copy link
Member Author

aduth commented Feb 4, 2020

Thanks @youknowriad and @afercia . I'm hoping to rebase and get this merged sometime today.

Aside: Testing the ContrastChecker notice I... noticed the contrast check appears to have a lot of issues at the point it's basically unreliable. Is there an issue for that already?

I'm not personally aware of one.

Comment on lines 18 to 30
useEffect( () => {
speak( __( 'This color combination may be hard for people to read.' ) );
}, [ backgroundColor, textColor ] );
Copy link
Member Author

Choose a reason for hiding this comment

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

One thing that I am seeing in rebasing this: Previously, we would speak this message whenever the background or text color would change, which I assume would be useful as an indication of "the new selection is still hard to read". With the changes proposed here, the message would only be spoken when the notice is first shown.

I'm not sure it's a common use-case that Notice needs to support, but at least for how we're using it here, we might want to either:

  • Revert the changes to this specific component (it was intended as a refactor anyways) and consider it fine if a component wants more granular control around when speak is called.
  • Or: Change the implementation such that we force Notice to remount, or "flip" the prop value of spokenMessage ('This color combination ... ' -> undefined -> 'This color combination ... ').

The first of these is simplest, at least in an effort to avoid blocking the rest of the pull request.

Copy link
Member Author

Choose a reason for hiding this comment

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

For posterity's sake, this is the commit I am going to drop from the rebase, in case we want to cherry-pick or refer to it in the future: 4b237af

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 also added an inline code comment in the rebased 86a1790 to clarify this requirement, in case it catches the attention of some future maintainer.

Copy link
Member Author

Choose a reason for hiding this comment

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

Added test case in 8ce0dae. And glad I did, because while technically this is a refactoring, since the new default behavior of Notice would speak its rendered children, the spokenMessage must be unset when trying to override it in favor of a custom implementation. This is fixed and tested in 737bfef.

@aduth aduth force-pushed the update/notice-speak branch from ddbe697 to 86a1790 Compare February 4, 2020 20:59
@aduth
Copy link
Member Author

aduth commented Feb 4, 2020

In auditing other notices, I observe that we will want to be more conscious of what we render within Notice. I think it reinforces the decision to reject anything other than a string in notices state, and how we should probably want to (continue to) avoid allowing arbitrary HTML within notices.

Consider:

<Notice
className="editor-template-validation-notice"
isDismissible={ false }
status="warning"
>
<p>
{ __(
'The content of your post doesn’t match the template assigned to your post type.'
) }
</p>
<div>
<Button isSecondary onClick={ props.resetTemplateValidity }>
{ __( 'Keep it as is' ) }
</Button>
<Button onClick={ confirmSynchronization } isPrimary>
{ __( 'Reset the template' ) }
</Button>
</div>
</Notice>

My expectation is that without revisions, the spoken message of this notice would be:

The content of your post doesn’t match the template assigned to your post type.Keep it as isReset the template

...since the implementation would use the HTML-stripped generated output of the Notice.

I don't think this is ideal, and I think it can be improved by leveraging the actions prop of Notice, freeing up the children to be only the text of the message. Perhaps as a future enhancement we could consider additional ways to improve messaging by inspecting the actions (e.g. "There are X actions available..."), if it makes sense to.

@aduth
Copy link
Member Author

aduth commented Feb 4, 2020

The changes in 7688318 update the template validation notice to render only its plaintext message as children, so the spoken message should be more coherent:

The content of your post doesn’t match the template assigned to your post type.

The Notice actions prop worked well, with the exception that the "Reset the template" action was the primary action of the two. Support for controlling the isPrimary prop of the rendered button was added in 20e1d28.

It is expected there is a visual change in the rendered notice, though one which I think brings it into consistency with other notices.

Before After
Before After

If you want to test this notice, the easiest way I have found is:

  1. Assuming you are using the default Gutenberg development environment, enable the "Gutenberg Test Plugin, Templates" plugin
  2. You will need to revise the post type registration code to add 'template_lock' => 'all'
  3. Navigate to Books > Add New
  4. Change to Code Editor
  5. Remove some or all of the markup
  6. Deselect the textarea

@aduth aduth force-pushed the update/notice-speak branch from 7688318 to 92d0e3f Compare February 7, 2020 15:41
Previously assumed to read text content from a paragraph. Standard notices do not include paragraph tag. Read instead as the first child Node.TEXT_NODE nodeValue
@aduth aduth requested review from nerrad and ntwb as code owners February 7, 2020 16:30
@aduth aduth merged commit b12aab0 into master Feb 7, 2020
@aduth aduth deleted the update/notice-speak branch February 7, 2020 20:15
@github-actions github-actions bot added this to the Gutenberg 7.5 milestone Feb 7, 2020
getdave added a commit that referenced this pull request Feb 17, 2020
* Adds new create component to search results

* Styling and i18n

* Display approximation of Page slug to be created

* Fix to ensure border above create button is displayed

* Implement prototype async load of created Page

* Add state and UI to represent link being in a resolving/loading state

This is require to accommodate the new async mode for setting a link. For example when a Page is being created async with the API.

* add basic API integration (inline in component)

* actually create pages, not posts

* Add API to control display of the Create functionality on LinkControl

* Adds initial tests for Create button feature

* Add more test use case data

* extract API call to prop function

* swap api fetch with dispatch

* Improve a11y of label

* Adds coverage for creation of Link and update of “Selected” UI

* Adds tests to cover scenarios where Create Page option should NOT be shown

* Fix tests to avoid edge cases

* Fix test to expose bug in implementation

Previously both tests would pass even though when testing in browser (using real API requests) the one with spaces would fail.

This is because the mocked API always results results which ensures that suggestinos are always shown. However, if the API doesn’t return any results then URLInput has no suggestions state and so the suggestions dropdown doesn’t display which causes the Create Page option also not display. In fact it should display.

* Fix bug introduced in tests

* Hide Create option if suggestion is a single Direct (URL) result only.

This is becase we shouldn’t create pages from anything that looks like a Direct Entry URL.

* remove temporary loading state

* Promote Create button to top level suggestion.

Previously the Create button was not part of the proper suggestions state. This caused several issues:

1. The Create option was only shown if there were suggestions (not what we want in certain circumstances)
2. It wasn’t possible to use keyboard to move to the Create button.
3. a11y concerns regarding having Create option outside of the true suggestions.

We now make the searchSuggestions handler always append a special suggestion to the result set for create. This is a reserved suggestion which is only displayed in the UI when the appropriate conditions are met.

Also fixes bug with un-needed call to `setIsEditingLink( false )` in async `onChange` handler causes invalid test failures.

* Reinstate full test conditions

Left some commented out test conditions. These have revealed that the previous commit did not fix the outstanding broken tests.

* Fix bug whereby editing state was no disabled follow successful onChange

* Allow create button to display for initial suggestions

* Improve test code comments

* Tweaks to code layout and formatting

* Generalise prop to enable on the fly creation of entities

* Improve code comments around promoting CREATE option to a first class suggestion

This code is particularly ambiguous. Adding a clear comment here should remove a lot of confusion as to why this particular work around is required. It is far from ideal but without decoupling `LinkContorl` from `URLInput` it is necessary.

* support other post types in create function

* Use generated Ids for `id` prop of faux suggestions rather than ever decrementing static negative numbers

* Remove Create option from initial suggestions

Addresses #18900 (comment)

If we need to reinstate this functionality we can simply revert this commit.

* Fix broken tests due to incorrect faux API response mock format

* Fix test by relaxing type checking on falsey value

* Catch invalid saveEntityRecord response types and throw error

Throwing here causes a rejected Promise to be returned from `createEntity`. This is then caught and handled inside `LinkControl` in order to centralise error handling.

This area will need additional safeguards adding to catch response types which are errors or which don’t confirm to the expected format of `entity`.

* Add Error handling to LinkControl for Creating Entities

If `createEntities` promise is rejected then the erorr is caught and handled and an appropriate error notice is shown to the user.

* Fix reset button positioning when error Notice is displayed above search input

* Force repopulation of search results on create button failure.

* Avoid showing Create UI when there is not input value

* Moves error note below search input UI

* Remove forceUpdate

This doesn’t actually work. Removing.

* Refactor out a handle create function

* Extract common handler for selecting suggestion.

* Add test (failing) to cover creation of entities via keyboard

* Resolve rebase errors

* Removes unused keypress handler.

This appears never to be called. The prop is not used directly within `URLInput` or “spread” from the passed props. Testing by adding console.log and running the tests confirms that the code is not being run. Manual testing also.

* Remove redundant onKeyDown prop

Again onKeyDown prop is not used within `URLInput`. Nor is it spread from props. Never called in tests. Not called when testing manually either.

* Improve naming of handler functions to better reflect purpose

* Fix to allow keyboard and form submit to handle entity creation

Previously only mouse click on the “create” suggestion correctly created a new entity. Improved by ensuring the form `onSubmit` handler also correctly triggers the onCreate handler if the suggestion type is the “create” type.

* Tweak to error positioning styling

* Update to ensure created Page entities conform to object schema required by LinkControl

* Include test for loading state in create entity tests

* Fix errors introduced during rebase

* Fix implementation broken due to rebase.

* Add i18n for Loading state

* Ensure correct timing of stopEditing by awaiting entity creation promise

* Fixes render conditionals to be mutually exclusive

Rebasing with master caused conditionals to be normalised into one. Now we have `isResolving` as state we need to ensure the component renders differently with this conditional involved. The clearest means of doing this is to break the render up into exclusive sections based on conditions.

* Fix broken conditional exposed by test failures.

* Fix to stop users without create pages permission seeing the Create entity UI

* Add test to verify Create UI not shown if valid handler not provided to component

* Renamed 'Loading...' to 'Creating Page' to match mockup

* Adds the <Spinner> component to the Creating Page loading state

* Reduced padding on bottom of LinkControl list results and moved error message up to match design specs more closely.

* Only add separator when it needs separating

We only need a separator when there are other suggestions the Create button needs separation from.

* Committed formatted js

* Fixed broken unit test for checking 'Loading' string instead of updated 'Creating' string

* Removed uniqueID from Manual URLs in Navigation Link results from Link Control

Based on discussion about how to handle link results, I've removed the uniqueId as we may not need an ID at all for manual link results. Other possibilities are to include the ID property again and either set it to null or to the URL value

* Remove unrequired eslint disable rule

Addresses #19775 (comment)

* Update packages/block-editor/src/components/link-control/index.js

Co-Authored-By: Andrew Duthie <[email protected]>

* Rename `errorMsg` prop to be unabbreviated for clarity.

* Remove unnecessary cast to Boolean

Co-Authored-By: Andrew Duthie <[email protected]>

* Update to utilise `rendered` title value.

Co-Authored-By: Andrew Duthie <[email protected]>

* Remove redundant prop usage

* Restore commented out test

This should never have been committed with commented out code.

* Update to invert create entity “type” dependency

Previously `LInkControl` was specifiying the type of `page`. This meant it was “aware” of the entity being created. This commit inverts that so that the consuming component is now required to define the type itself (this has been updated on Nav Block).

* Removes outdated `showCreateEntity` prop

* Fix so that CREATE option not displayed if result is a direct entry URL

Also amends tests which were incorrectly asserting.

* Fix typo

* Adds docblock to newly extracted function for clarity

* Corrects comment which implied a specific visual layout and DOM order

Addresses #19775 (comment)

* Fix to use standardised “gray” var reference

Addresses #19775 (comment)

* Correct comment to reference the correct property.

Addresses #19775 (comment)

* Remove partially applied function as not necessary

Addresses

* #19775 (comment)
* #19775 (comment)

* Reinstate descriptive function name

Addresses #19775 (comment)

* Rename `createEntity` prop to `createSuggestion`

Addresses #19775 (comment)

* Attempt update of type definitions

#19775 (comment)

* Avoid need to use entity term where making it more generic can avoid this.

* Remove unused timeout HOC

* Fix misnamed property

* Update Nav Block create handler to use term “Page” to better reflect purpose

* Refactor handleCreatePage to async/await to avoid nesting

Addresses #19775 (comment)

* Set url items from LinkControl to have sanitized url as the id and updated e2e snapshot

Based on conversations around potential issues with not including an id on navigation links, the ID has been added back in as the sanitized url as it should be fairly unique and not actually matter if it's unique.

* Reinstate keyboard handling of suggestion selection bug fix.

Failing to pass the current input value as `url` of the suggestion causes the keyboard handling to break when hitting `ENTER`. Not entirely sure why at this stage.

* Update comment to better reflect need for `title` and `url` props to reflect input text value

* Remove unwanted reference to entity type.

The more generic we keep the errors the simpler this component needs to be.

Addresses #19775 (comment)

* Corrects usage of aria-label and aria-labelled by on the link control search results

Use aria-labelledby to reference the visible label ID
Use aria-label when no visible label

* Attempt to fix Typescript docblock alignment

* Updated hardcoded color values to scss color variables

* add e2e test for creating page from nav

* accomodate for self-closing link popover after selecting link (prevents react error)

* update comment to match expected test result

* Fix to cancel pending promises on unmount to avoid state updates

Addresses #19775 (comment)

* Make arbitary CSS value correspond to standardised var

Addresses #19775 (comment)

* Removes manual aria live in favour of speak and @wordpress/a11y

Addresses #19775 (comment)

* Remove unwanted whitespace.

Co-Authored-By: Andrew Duthie <[email protected]>

* Rename var to lowercase

Addresses #19775 (comment)

* Avoid unwanted arial-label* roles for visually hidden elements

`aria-labelledby` is only suitable when the referenced element is visible.

`aria-label` should not be used to duplicate content already accessible within the element.

* Move cancellable refs to component scope.

Partially addresses #19775 (comment)

* Lightened border color between link control suggestions and create new page button

* Changed Create new Page text to New Page to match design

* Added <mark> to New page title when creating a new page in the link control

* Make page title of new page creation darker

* Adjusted positioning on the create new page button padding and centering

* Added clarifying comment on negative margin CSS

* Correct tests to select by aria-label now aria-labelledby is removed.

* Correct test to look for new Create button wording

Changed due to Design request to change text but test was not updated.

* Fix to use cancelable async handler and call stopEditing

Previously only the keyboard version of handleCreate was cancellable and treated as async. Now the onClick mouse version is also cancellable.

Moreover we call stopEditing() correctly on resolve.

* Refactor select handlers to use async/await

* Move function invocation within try/catch boundary

* Remove test spying on speak HOC as this isn’t possible via `LinkControl` directly.

* Remove superflous visually hidden text in favour of aria-label

Addresses #19775 (comment)

* Fix space vs tabs linting issue

* Correct spelling typos

* Correct prop rename error from rebase

* Fix missing references

* Remove duplicate render of LinkControlSearchInput introduced during rebase

* Removes unnecessary closing of Link UI

Calling setIsLinkOpen in the useEffect was causing setIsLinkOpen to be called twice. The act of moving focus back to the label is enough to trigger setIsLinkOpen to be called by the `Popover` onClose handler. Trying to call it again in the effect caused an error to be thrown regarding setting state on the unmounted component.

* Adds documentation for createSuggestion prop

* Restore Link Settings to be always visable as per 6c591f4

* Update e2e snapshot

* Revert "Update e2e snapshot"

This reverts commit 98f7e9e.

* Wait for rich-text to be focused instead of waiting for link control to disappear in e2e test

We can't rely on the .block-editor-link-control__search-results-wrapper to be hidden, as there's a quick loading state between when that disappears and the link is focused. Waiting for the link to be focused allows this test to be passed while also checking that focus is placed correctly.

* Added a check on the active element to make sure the focused block matches our newly created page title

* Update test to target specific input field for focus state rather than wrapper

Previously the test was checking for focus on a <div>. Amended to target the <input /> element as that is what actually will receive focus.

* Refactor e2e test to be absolutelty 100% sure we’re in the input before typing

Previously the tests were less than strict about whether the focus was in the input element. Improve this across all tests.

* Attempt to fix e2e test via simplifying selection of create button

* Ensure Create button is “in view” by removing other results.

It’s possible that on certain screen sizes there is not enough room to display the Create button without having to scroll the LinkContorl seaerch results panel. This could cause the selection of the button to fail.

Testing if this fixes the broken e2e tests or at least makes them more resilient.

* Provide more context on why we mock search and create API requests in e2e test

* Updated routes on mock responses in e2e navigation tests and snapshots to hopefully make e2e tests easier to debug

* Added description and documentation to updateActiveNavigatinoLink

* More explicity select which suggestion we want in updateActiveNavigationLink

* Await input focus before selecting the label on updateActiveNavigationLink

* Fixed xpath selector in updateActiveNavigationLink and made the search page term more unique

* Remove duplicate error notice

* Remove superflous prop

Introduced via rebase (again).

* Simply listbox labelling for a11y

As per this thread in WPOrg Slack (https://wordpress.slack.com/archives/C02RP4X03/p1581500184181100) it’s better to have a HTML based label over aria labels under all circumstances. Therefore despite what it says on MDN docs for listbox (https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/Roles/listbox_role) we simplify to only use a HTML label and refer to that with aria-labelledby.

* Remove manual speak of Notice as it’s now built in to the component

Addresses #15745

* Remove requirement to pass a Promise for `createSuggestion` prop

Addresses #19775 (comment)

* Handle cancellable handler props using refs

Addresses #19775 (comment). Previously we were treating a functional component as though it would live forever when in fact it could easily be unmounted and all internal var references wiped.

Using refs solves this as they persiste between renders.

* Fix Promise flow so that stopEditing is not called on createSuggestion error and error message is shown

There were several problems here.

* Originally the handleCreate method did `return` with a undefined value when the error was cancelled. This caused incorrect logic flows.
* The `stopEditing` method was being called even if the Promise flow was handling an error state. This caused the “Currently selected” UI to show with an `undefined` value.
* The `errorMessage` state was being reset ever time the input “changed”. This meant the error set in the Promise chain was being reset before it could be displayed.

Fixing all the above issues has resolved the errors.

* Fix rebase regressions.

* Remove cleanForSlug

This was used to try and “clean” the id primarily to make it valid as a React key. However we’ve discovered that any string is potentially valid. Moreover cleanForSlug might end up making two urls that are otherwise distinct become identical via stripping out of various parts of the full URL.

See #19775 (comment)

* Removes `id` prop from “Create” result and uses static string for React key prop.

See #19775 (comment)

* Fix e2e test snapshot to account for using full URL as the suggestion prop without cleanForSlug

* Remove explicit “politeness” and use default settings

See #19775 (review)

* Update docs to distinguish suggestion from value.

* Adds comment for “Create” constant

Addresses #19775 (comment)

* Adds type def for suggestion and reformats type defs

Addresses #19775 (comment)

* Fix to not render create button if there is no search term.

This was a throw back to a previous state of the `LinkControl` component whereby we wanted to render a create option to allow the user to create blank pages. This was removed as a requirement but the component was not fixed to account for that.

Addresses: #19775 (comment)

* Reinstate new line to separate imports from code body.

Addresses #19775 (comment)

* Removes unecessary response checking on advice from @aduth

See

* #19775 (comment)
* #19775 (comment)

* Remove stray unwanted comments from tests

Addresses #19775 (comment)

* Update novmenclature to confirm to suggestion as opposed to entity

* Remove redundant portion of conditional

Addresses #19775 (comment)

* Improves method name and supporting comments and doc block.

Addresses #19775 (comment)

* Refactor search results label to utilise VisuallyHidden component

Addresses #19775 (comment)

* Consolidate createSuggestion handling logic within single method

Previously the code to handle the async flow including error handling was duplicated across two handler props. Consolidating helps DRY things and avoid accidental errors being introduced.

Addresses #19775 (comment)

* Add punctuation to comments

Addresses #19775 (comment)

* Update error handling to include the message prop of the Error object if provided.

Addresses #19775 (comment)

* Fix broken comment

Co-authored-by: Marek Hrabe <[email protected]>
Co-authored-by: Jerry Jones <[email protected]>
Co-authored-by: Konstantin Obenland <[email protected]>
Co-authored-by: Andrew Duthie <[email protected]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
[Feature] UI Components Impacts or related to the UI component system [Focus] Accessibility (a11y) Changes that impact accessibility and need corresponding review (e.g. markup changes). [Package] Components /packages/components [Package] Notices /packages/notices
Projects
None yet
3 participants