From 707887ebc42bcc5199a25b272e27e01a08296e95 Mon Sep 17 00:00:00 2001 From: ValeriyYustunyk Date: Tue, 14 Dec 2021 10:51:12 -0500 Subject: [PATCH] BB-20361: Make Search Autocomplete drop-down accessible by keyboard arrows (#31255) --- .../scss/components/search-autocomplete.scss | 19 +- .../variables/search-autocomplete-config.scss | 10 +- .../js/app/views/search-autocomplete-view.js | 187 ++++++++++++++++-- .../public/templates/search-autocomplete.html | 57 +++--- .../product_search_autocomplete.feature | 54 ++++- .../ProductBundle/Tests/Behat/behat.yml | 3 + 6 files changed, 273 insertions(+), 57 deletions(-) diff --git a/src/Oro/Bundle/ProductBundle/Resources/public/blank/scss/components/search-autocomplete.scss b/src/Oro/Bundle/ProductBundle/Resources/public/blank/scss/components/search-autocomplete.scss index d7bae3de27..5a6032f1df 100644 --- a/src/Oro/Bundle/ProductBundle/Resources/public/blank/scss/components/search-autocomplete.scss +++ b/src/Oro/Bundle/ProductBundle/Resources/public/blank/scss/components/search-autocomplete.scss @@ -2,8 +2,6 @@ .search-autocomplete { position: $search-autocomplete-position; - display: $search-autocomplete-display; - width: $search-autocomplete-width; z-index: $search-autocomplete-z-index; &__content { @@ -15,16 +13,21 @@ } &__item { + margin: $search-autocomplete-item-offset; + padding: $search-autocomplete-item-inner-offset; border-bottom: $search-autocomplete-item-border-bottom; + + &:last-child { + border-bottom-width: 0; + } + + &[aria-selected='true'] { + box-shadow: $search-autocomplete-selected-box-shadow; + } } &__highlight { background: $search-autocomplete-highlight-background; - text-decoration: $search-autocomplete-highlight-text-decoration; - } - - &__footer { - padding: $search-autocomplete-footer-inner-offset; } &__submit { @@ -40,8 +43,6 @@ .search-autocomplete-product { text-decoration: $search-autocomplete-product-text-decoration; display: $search-autocomplete-product-display; - margin: $search-autocomplete-product-offset; - padding: $search-autocomplete-product-inner-offset; &:hover { text-decoration: $search-autocomplete-product-hover-text-decoration; diff --git a/src/Oro/Bundle/ProductBundle/Resources/public/blank/scss/variables/search-autocomplete-config.scss b/src/Oro/Bundle/ProductBundle/Resources/public/blank/scss/variables/search-autocomplete-config.scss index 675f8655bf..f6bcbe3fca 100644 --- a/src/Oro/Bundle/ProductBundle/Resources/public/blank/scss/variables/search-autocomplete-config.scss +++ b/src/Oro/Bundle/ProductBundle/Resources/public/blank/scss/variables/search-autocomplete-config.scss @@ -1,9 +1,8 @@ /* @theme: blank; */ $search-autocomplete-position: absolute !default; -$search-autocomplete-display: block !default; -$search-autocomplete-width: 100% !default; $search-autocomplete-z-index: z('dropdown') + 1 !default; +$search-autocomplete-selected-box-shadow: $focus-visible-style !default; $search-autocomplete-content-min-width: 385px !default; $search-autocomplete-content-display: block !default; @@ -12,11 +11,10 @@ $search-autocomplete-content-float: none !default; $search-autocomplete-content-position: static !default; $search-autocomplete-item-border-bottom: 1px solid get-color('additional', 'light') !default; +$search-autocomplete-item-offset: 0 -#{$offset-y-m + $offset-y-s} !default; +$search-autocomplete-item-inner-offset: #{$offset-y-m + $offset-y-s} #{$offset-y-m + $offset-y-s} !default; $search-autocomplete-highlight-background: get-color('ui', 'warning') !default; -$search-autocomplete-highlight-text-decoration: underline !default; - -$search-autocomplete-footer-inner-offset: #{$offset-y-m + $offset-y-s} 0 !default; $search-autocomplete-submit-line-height: $base-line-height !default; $search-autocomplete-submit-border: none !default; @@ -25,8 +23,6 @@ $search-autocomplete-no-found-inner-offset: #{$offset-y-m + $offset-y-s} 0 !defa $search-autocomplete-product-text-decoration: none !default; $search-autocomplete-product-display: flex !default; -$search-autocomplete-product-offset: 0 -#{$offset-y-m + $offset-y-s} !default; -$search-autocomplete-product-inner-offset: #{$offset-y-m + $offset-y-s} #{$offset-y-m + $offset-y-s} !default; $search-autocomplete-product-hover-text-decoration: none !default; $search-autocomplete-product-image-width: 40px !default; diff --git a/src/Oro/Bundle/ProductBundle/Resources/public/js/app/views/search-autocomplete-view.js b/src/Oro/Bundle/ProductBundle/Resources/public/js/app/views/search-autocomplete-view.js index 3954de6439..d20054cf93 100644 --- a/src/Oro/Bundle/ProductBundle/Resources/public/js/app/views/search-autocomplete-view.js +++ b/src/Oro/Bundle/ProductBundle/Resources/public/js/app/views/search-autocomplete-view.js @@ -4,6 +4,7 @@ import _ from 'underscore'; import BaseView from 'oroui/js/app/views/base/view'; import routing from 'routing'; import template from 'tpl-loader!oroproduct/templates/search-autocomplete.html'; +import 'jquery-ui/tabbable'; const SearchAutocompleteView = BaseView.extend({ optionNames: BaseView.prototype.optionNames.concat([ @@ -41,29 +42,62 @@ const SearchAutocompleteView = BaseView.extend({ events: { change: '_onInputChange', keyup: '_onInputChange', - focus: '_onInputRefresh' + focus: '_onInputRefresh', + keydown: '_onKeyDown' }, previousValue: '', + autocompleteItems: '[role="option"]', + /** * @inheritdoc */ constructor: function SearchAutocompleteView(options) { + this.renderSuggestions = _.debounce(this.renderSuggestions.bind(this), this.delay); SearchAutocompleteView.__super__.constructor.call(this, options); }, + preinitialize() { + this.comboboxId = `combobox-${this.cid}`; + }, + /** * @inheritdoc */ initialize(options) { - this.$el.attr('autocomplete', 'off'); + this.$el.attr({ + 'role': 'combobox', + 'autocomplete': 'off', + 'aria-haspopup': true, + 'aria-expanded': false, + 'aria-autocomplete': 'list', + 'aria-controls': this.comboboxId + }); - this.renderSuggestions = _.debounce(this.renderSuggestions.bind(this), this.delay); + SearchAutocompleteView.__super__.initialize.call(this, options); + }, + + /** + * @inheritdoc + */ + delegateEvents: function() { + SearchAutocompleteView.__super__.delegateEvents.call(this); $('body').on(`click${this.eventNamespace()}`, this._onOutsideAction.bind(this)); - SearchAutocompleteView.__super__.initialize.call(this, options); + return this; + }, + + /** + * @inheritdoc + */ + undelegateEvents: function() { + SearchAutocompleteView.__super__.undelegateEvents.call(this); + + $('body').off(this.eventNamespace()); + + return this; }, getInputString() { @@ -75,7 +109,8 @@ const SearchAutocompleteView = BaseView.extend({ */ getTemplateData: function(data = {}) { return Object.assign(data, { - inputString: this.getInputString() + inputString: this.getInputString(), + comboboxId: this.comboboxId }); }, @@ -86,20 +121,109 @@ const SearchAutocompleteView = BaseView.extend({ if (this.disposed) { return; } - this.close(); + this.closeCombobox(); - $('body').off(this.eventNamespace()); + this.$el.attr('aria-expanded', null); SearchAutocompleteView.__super__.dispose.call(this); }, - close() { + closeCombobox() { if (!this.$popup) { return; } this.$popup.remove(); this.$popup = null; + this.$el.attr({ + 'aria-expanded': false, + 'aria-activedescendant': null + }); + this.undoFocusStyle(); + }, + + hideCombobox() { + if (!this.$popup) { + return; + } + + this.$el.attr({ + 'aria-expanded': false, + 'aria-activedescendant': null + }); + this.$popup.hide(); + this.gerSelectedOption().removeAttr('aria-selected'); + this.undoFocusStyle(); + }, + + showCombobox() { + if (!this.$popup) { + return; + } + + this.$popup.show(); + }, + + hasSelectedOption() { + return this.gerSelectedOption().length > 0; + }, + + getAutocompleteItems() { + return this.$el.next().find(this.autocompleteItems); + }, + + gerSelectedOption() { + return this.getAutocompleteItems().filter((i, el) => $(el).attr('aria-selected') === 'true'); + }, + + getNextOption() { + const $options = this.getAutocompleteItems(); + const $activeOption = this.gerSelectedOption(); + + if ( + $activeOption.length === 0 || + ($options.length - 1 === $options.index($activeOption)) + ) { + return $options.first(); + } + + return $options.eq($options.index($activeOption) + 1); + }, + + getPreviousOption() { + const $options = this.getAutocompleteItems(); + const $activeOption = this.gerSelectedOption(); + + if ( + $activeOption.length === 0 || + $options.index($activeOption) === 0 + ) { + return $options.last(); + } + + return $options.eq($options.index($activeOption) - 1); + }, + + /** + * @param {string} direction + */ + goToOption(direction = 'down') { + const $options = this.getAutocompleteItems(); + const $activeOption = direction === 'down' + ? this.getNextOption() + : this.getPreviousOption() + ; + + this.showCombobox(); + $options.attr('aria-selected', false); + $activeOption.attr('aria-selected', true); + this.$el.attr('aria-activedescendant', $activeOption.attr('id')); + }, + + executeSelectedOption() { + if (this.hasSelectedOption()) { + this.gerSelectedOption().find(':first-child')[0].click(); + } }, _getSearchXHR(inputString) { @@ -117,11 +241,19 @@ const SearchAutocompleteView = BaseView.extend({ * @inheritdoc */ render(suggestions) { - this.close(); + this.closeCombobox(); if (this.getInputString().length) { this.$popup = $(this.template(this.getTemplateData(suggestions))); this.$el.after(this.$popup); + this.$el.attr('aria-expanded', true); + + this.getAutocompleteItems().each((i, el) => { + $(el).attr({ + 'id': _.uniqueId('item-'), + 'aria-selected': false + }).find(':tabbable').attr('tabindex', -1); + }); } return this; @@ -148,6 +280,35 @@ const SearchAutocompleteView = BaseView.extend({ ; }, + _onKeyDown(event) { + switch (event.key) { + case 'Tab': + case 'Escape': + this.hideCombobox(); + break; + case 'ArrowUp': + event.preventDefault(); + this.goToOption('up'); + break; + case 'ArrowDown': + event.preventDefault(); + this.goToOption('down'); + break; + case 'Enter': + case ' ': + this.executeSelectedOption(); + break; + default: + break; + } + + this.undoFocusStyle(); + }, + + undoFocusStyle() { + this.$el.toggleClass('undo-focus', this.hasSelectedOption()); + }, + _onInputChange(event) { const inputString = this.getInputString(); if (inputString === this.previousValue) { @@ -156,7 +317,7 @@ const SearchAutocompleteView = BaseView.extend({ this._shouldShowPopup(inputString) ? this.renderSuggestions(inputString) - : this.close(); + : this.closeCombobox(); this.previousValue = inputString; }, @@ -166,16 +327,16 @@ const SearchAutocompleteView = BaseView.extend({ if (!inputString.length && this.searchXHR) { this.searchXHR.abort(); - }; + } this._shouldShowPopup(inputString) ? this.renderSuggestions(inputString) - : this.close(); + : this.closeCombobox(); }, _onOutsideAction(event) { if (!((event.target === this.el) || (this.$popup && $.contains(this.$popup[0], event.target)))) { - this.close(); + this.closeCombobox(); } } }); diff --git a/src/Oro/Bundle/ProductBundle/Resources/public/templates/search-autocomplete.html b/src/Oro/Bundle/ProductBundle/Resources/public/templates/search-autocomplete.html index 82f3626f3c..3202b3832e 100644 --- a/src/Oro/Bundle/ProductBundle/Resources/public/templates/search-autocomplete.html +++ b/src/Oro/Bundle/ProductBundle/Resources/public/templates/search-autocomplete.html @@ -1,7 +1,7 @@ <% function highlightWords(inputString, value) { if (!!inputString && _.isString(inputString)) { - const highlightPattern = '$&'; + const highlightPattern = '$&'; const inputWords = _.escape(inputString).split(' '); const reg = new RegExp(inputWords.join('|'), 'gi'); return _.escape(value).replace(reg, highlightPattern); @@ -14,37 +14,40 @@
+ + <% }) %> +
  • + +
  • - <% } else { %>
    <%- _.__('oro.product.autocomplete.no_found') %>
    <% } %> diff --git a/src/Oro/Bundle/ProductBundle/Tests/Behat/Features/product-search/product_search_autocomplete.feature b/src/Oro/Bundle/ProductBundle/Tests/Behat/Features/product-search/product_search_autocomplete.feature index 16f94baa24..cd991be716 100644 --- a/src/Oro/Bundle/ProductBundle/Tests/Behat/Features/product-search/product_search_autocomplete.feature +++ b/src/Oro/Bundle/ProductBundle/Tests/Behat/Features/product-search/product_search_autocomplete.feature @@ -32,8 +32,60 @@ Feature: Product search autocomplete And I should see an "Search Autocomplete Submit" element And I should see "See All 3 Results" in the "Search Autocomplete Submit" element + Scenario: Check the search autocomplete navigation by ArrowDown key + When I press "ArrowDown" key on "Search Form Field" element + Then I should see "Search Form Field" element focused + And I should see "Search Autocomplete Item Selected" element with text "Product2" inside "Search Autocomplete" element + When I press "ArrowDown" key on "Search Form Field" element + Then I should see "Search Form Field" element focused + And I should see "Search Autocomplete Item Selected" element with text "Product3" inside "Search Autocomplete" element + When I press "ArrowDown" key on "Search Form Field" element + Then I should see "Search Form Field" element focused + And I should see "Search Autocomplete Item Selected" element with text "Product1" inside "Search Autocomplete" element + When I press "ArrowDown" key on "Search Form Field" element + Then I should see "Search Form Field" element focused + And I should see "Search Autocomplete Item Selected" element with text "See All 3 Results" inside "Search Autocomplete" element + When I press "ArrowDown" key on "Search Form Field" element + Then I should see "Search Form Field" element focused + And I should see "Search Autocomplete Item Selected" element with text "Product2" inside "Search Autocomplete" element + When I press "Esc" key on "Search Form Field" element + Then I should not see an "Search Autocomplete" element + When I press "ArrowDown" key on "Search Form Field" element + Then I should see "Search Form Field" element focused + And I should see "Search Autocomplete Item Selected" element with text "Product2" inside "Search Autocomplete" element + When I press "Enter" key on "Search Form Field" element + And I should see "All Products / Product2" + + Scenario: Check the search autocomplete navigation by ArrowUp key + When I type "Product" in "search" + And I should see an "Search Autocomplete" element + And I press "ArrowUp" key on "Search Form Field" element + Then I should see "Search Form Field" element focused + And I should see "Search Autocomplete Item Selected" element with text "See All 3 Results" inside "Search Autocomplete" element + When I press "ArrowUp" key on "Search Form Field" element + Then I should see "Search Form Field" element focused + And I should see "Search Autocomplete Item Selected" element with text "Product1" inside "Search Autocomplete" element + When I press "ArrowUp" key on "Search Form Field" element + Then I should see "Search Form Field" element focused + And I should see "Search Autocomplete Item Selected" element with text "Product3" inside "Search Autocomplete" element + When I press "ArrowUp" key on "Search Form Field" element + Then I should see "Search Form Field" element focused + And I should see "Search Autocomplete Item Selected" element with text "Product2" inside "Search Autocomplete" element + When I press "ArrowUp" key on "Search Form Field" element + Then I should see "Search Form Field" element focused + And I should see "Search Autocomplete Item Selected" element with text "See All 3 Results" inside "Search Autocomplete" element + When I press "Esc" key on "Search Form Field" element + Then I should not see an "Search Autocomplete" element + When I press "ArrowUp" key on "Search Form Field" element + Then I should see "Search Form Field" element focused + And I should see "Search Autocomplete Item Selected" element with text "See All 3 Results" inside "Search Autocomplete" element + When I press "Enter" key on "Search Form Field" element + Then number of records in "Product Frontend Grid" should be 3 + Scenario: Check the search grid after following by the search autocomplete link - When I click "Search Autocomplete Submit" + When I type "Product" in "search" + And I should see an "Search Autocomplete" element + Then I click "Search Autocomplete Submit" And number of records in "Product Frontend Grid" should be 3 And I type "PSKU3" in "search" And I click "Search Autocomplete Product" diff --git a/src/Oro/Bundle/ProductBundle/Tests/Behat/behat.yml b/src/Oro/Bundle/ProductBundle/Tests/Behat/behat.yml index 283ff6d539..04d88cc61d 100644 --- a/src/Oro/Bundle/ProductBundle/Tests/Behat/behat.yml +++ b/src/Oro/Bundle/ProductBundle/Tests/Behat/behat.yml @@ -743,6 +743,9 @@ oro_behat_extension: Search Autocomplete Item: selector: '.search-autocomplete .search-autocomplete__item' + Search Autocomplete Item Selected: + selector: '.search-autocomplete [aria-selected="true"]' + Search Autocomplete Product: selector: '.search-autocomplete .search-autocomplete-product'