Skip to content

DigitalA11Y

Your Accessibility Partner

  • Home
  • ServicesExpand
    • WCAG Audit Services
    • VPAT/ACR Services
    • Accessibility Consulting
    • PDF Remediation
    • Accessibility Trainings
    • Website Remediation
    • Design Audit
  • Free ToolsExpand
    • Accessibility Checker
    • A11Y Cost Calculator
    • A11Y Bookmarklets
    • Color Contrast Extension
    • WCAG Contrast Checker
  • ResourcesExpand
    • A11Y Articles
    • WCAG Primer
    • ARIA Cheatsheet
    • A11Y Tools
    • A11Y Patterns
    • A11Y Cheatsheets
  • Contact
Search
DigitalA11Y
Your Accessibility Partner
Search

Autocomplete with Radio Buttons

Favorite hobby suggestions
Provides auto-suggestions when entering text

Code

  • HTML
  • CSS
  • JavaScript
<button href="#" id="before">Focusable element before</button>
<form>
  <div data-adg-autocomplete="">
    <div class="control">
      <label for="favorite_hobby_filter">Favorite hobby </label><input aria-describedby="favorite_hobby_filter_help" id="favorite_hobby_filter" type="text" />
      <fieldset hidden="">
        <legend>Favorite hobby suggestions</legend>
        <div class="control">
          <input id="favorite_hobby_hiking" name="hobby" type="radio" /><label for="favorite_hobby_hiking">Hiking</label>
        </div>
        <div class="control">
          <input id="favorite_hobby_dancing" name="hobby" type="radio" /><label for="favorite_hobby_dancing">Dancing</label>
        </div>
        <div class="control">
          <input id="favorite_hobby_gardening" name="hobby" type="radio" /><label for="favorite_hobby_gardening">Gardening</label>
        </div>
        <div class="control">
          <input id="favorite_hobby_meditation" name="hobby" type="radio" /><label for="favorite_hobby_meditation">Meditation</label>
        </div>
        <div class="control">
          <input id="favorite_hobby_gaming" name="hobby" type="radio" /><label for="favorite_hobby_gaming">Gaming</label>
        </div>
      </fieldset>
      <div class="description" id="favorite_hobby_filter_help">
        Provides auto-suggestions when entering text
      </div>
    </div>
  </div>
</form>
<button href="#" id="after">Focusable element after</button>
@charset "UTF-8";
.adg-visually-hidden {
  position: absolute;
  white-space: nowrap;
  width: 1px;
  height: 1px;
  overflow: hidden;
  border: 0;
  padding: 0;
  clip: rect(0 0 0 0);
  clip-path: inset(50%);
  margin: -1px;
}

[data-adg-autocomplete-options] {
  position: absolute;
  z-index: 1;
  background-color: #fff;
  border: 1px solid;
  padding: 5px 0;
}

[data-adg-autocomplete-option] {
  display: block;
}

[data-adg-autocomplete-option]:hover,
[data-adg-autocomplete-option-selected] {
  cursor: pointer;
  outline: 1px solid;
}

[data-adg-autocomplete-alerts] p {
  margin: 0;
}
[data-adg-autocomplete-alerts] kbd::before {
  content: "«";
}
[data-adg-autocomplete-alerts] kbd::after {
  content: "»";
}

.control {
  margin: 6px 0;
}

input[type=text] {
  width: 140px;
}

label {
  display: inline-block;
  width: 120px;
  vertical-align: top;
}

.description {
  margin-left: 120px;
}

fieldset {
  margin: -1px 0 0 120px;
}
fieldset .control {
  margin: 0;
}
fieldset label {
  min-width: 144px;
}
(function() {
  this.Adg = {};

  Adg.Base = (function() {
    var config, uniqueIdCount;

    class Base {
      
        // Constructor. Should not be overridden; use @init() instead.

      // - Arg1: The DOM element on which the script should be applied (will be saved as @$el)
      // - Arg2: An optional hash of options which will be merged into the global default config
      constructor(el, options = {}) {
        var key, val;
        this.$el = $(el);
        this.config = config;
        for (key in options) {
          val = options[key];
          this.config[key] = val;
        }
        this.init();
      }

      // Dummy, must be overridden in inheriting classes.
      init() {
        return this.throwMessageAndPrintObjectsToConsole('Classes extending App must implement method init()!');
      }

      // Prints the given message to the console if config['debug'] is true.
      debugMessage(message) {
        if (this.config.debugMessage) {
          return console.log(`Adg debug: ${message}`);
        }
      }

      // Executes the given selector on @$el and returns the element. Makes sure exactly one element exists.
      findOne(selector) {
        var result;
        result = this.$el.find(selector);
        switch (result.length) {
          case 0:
            return this.throwMessageAndPrintObjectsToConsole(`No object found for ${selector}!`, {
              result: result
            });
          case 1:
            return $(result.first());
          default:
            return this.throwMessageAndPrintObjectsToConsole(`More than one object found for ${selector}!`, {
              result: result
            });
        }
      }

      name() {
        return `adg-${this.constructor.name.toLowerCase()}`;
      }

      addAdgDataAttribute($target, name, value = '') {
        return $target.attr(this.adgDataAttributeName(name), value);
      }

      removeAdgDataAttribute($target, name) {
        return $target.removeAttr(this.adgDataAttributeName(name));
      }

      adgDataAttributeName(name = null) {
        var result;
        result = `data-${this.name()}`;
        if (name) {
          result += `-${name}`;
        }
        return result;
      }

      uniqueId(name) {
        return [this.name(), name, uniqueIdCount++].join('-');
      }

      labelOfInput($inputs) {
        return $inputs.map((i, input) => {
          var $input, $label, id;
          $input = $(input);
          id = $input.attr('id');
          $label = this.findOne(`label[for='${id}']`)[0];
          if ($label.length === 0) {
            $label = $input.closest('label');
            if ($label.length === 0) {
              this.throwMessageAndPrintObjectsToConsole("No corresponding input found for input!", {
                input: $input
              });
            }
          }
          return $label;
        });
      }

      show($el) {
        $el.removeAttr('hidden');
        return $el.show();
      }

      // TODO Would be cool to renounce CSS and solely use the hidden attribute. But jQuery's :visible doesn't seem to work with it!?
      // @throwMessageAndPrintObjectsToConsole("Element is still hidden, although hidden attribute was removed! Make sure there's no CSS like display:none or visibility:hidden left on it!", element: $el) if $el.is(':hidden')
      hide($el) {
        $el.attr('hidden', '');
        return $el.hide();
      }

      throwMessageAndPrintObjectsToConsole(message, elements = {}) {
        console.log(elements);
        throw message;
      }

      text(text, options = {}) {
        var key, value;
        text = this.config[`${text}Text`];
        for (key in options) {
          value = options[key];
          text = text.replace(`[${key}]`, value);
        }
        return text;
      }

    };

    uniqueIdCount = 1;

    config = {
      debugMessage: false,
      hiddenCssClass: 'adg-visually-hidden'
    };

    return Base;

  }).call(this);

  Adg.Autocomplete = (function() {
    var config;

    // Tested in JAWS+IE/FF, NVDA+FF

    // Known issues:
    // - JAWS leaves the input when using up/down without entering something (I guess this is due to screen layout and can be considered intended)
    // - Alert not perceivable upon opening options using up/down
    //     - Possible solution 1: always show options count when filter focused?
    //     - Possible solution 2: wait a moment before adding the alert?
    // - VoiceOver/iOS announces radio buttons as disabled?!
    // - iOS doesn't select all text when option was chosen

    // In general: alerts seem to be most robust in all relevant browsers, but aren't polite. Maybe we'll find a better mechanism to serve browsers individually?
    class Autocomplete extends Adg.Base {
      init() {
        var jsonOptions, key, val;
// Merge config into existing one (not nice, see https://stackoverflow.com/questions/47721699/)
        for (key in config) {
          val = config[key];
          this.config[key] = val;
        }
        jsonOptions = this.$el.attr(this.adgDataAttributeName());
        if (jsonOptions) {
          for (key in jsonOptions) {
            val = jsonOptions[key];
            this.config[key] = val;
          }
        }
        this.debugMessage('start');
        this.initFilter();
        this.initOptions();
        this.initAlerts();
        this.applyCheckedOptionToFilter();
        this.announceOptionsNumber('');
        return this.attachEvents();
      }

      initFilter() {
        this.$filter = this.findOne('input[type="text"]');
        this.addAdgDataAttribute(this.$filter, 'filter');
        this.$filter.attr('autocomplete', 'off');
        return this.$filter.attr('aria-expanded', 'false');
      }

      initOptions() {
        this.$optionsContainer = this.findOne(this.config.optionsContainer);
        this.addAdgDataAttribute(this.$optionsContainer, 'options');
        this.$optionsContainerLabel = this.findOne(this.config.optionsContainerLabel);
        this.$optionsContainerLabel.addClass(this.config.hiddenCssClass);
        this.$options = this.$optionsContainer.find('input[type="radio"]');
        this.addAdgDataAttribute(this.labelOfInput(this.$options), 'option');
        return this.$options.addClass(this.config.hiddenCssClass);
      }

      initAlerts() {
        this.$alertsContainer = $(`<div id='${this.uniqueId(this.config.alertsContainerId)}'></div>`);
        this.$optionsContainerLabel.after(this.$alertsContainer);
        this.$filter.attr('aria-describedby', [this.$filter.attr('aria-describedby'), this.$alertsContainer.attr('id')].join(' ').trim());
        return this.addAdgDataAttribute(this.$alertsContainer, 'alerts');
      }

      attachEvents() {
        this.attachClickEventToFilter();
        this.attachChangeEventToFilter();
        this.attachEscapeKeyToFilter();
        this.attachEnterKeyToFilter();
        this.attachTabKeyToFilter();
        this.attachUpDownKeysToFilter();
        this.attachChangeEventToOptions();
        return this.attachClickEventToOptions();
      }

      attachClickEventToFilter() {
        return this.$filter.click(() => {
          this.debugMessage('click filter');
          if (this.$optionsContainer.is(':visible')) {
            return this.hideOptions();
          } else {
            return this.showOptions();
          }
        });
      }

      attachEscapeKeyToFilter() {
        return this.$filter.keydown((e) => {
          if (e.which === 27) {
            if (this.$optionsContainer.is(':visible')) {
              this.applyCheckedOptionToFilterAndResetOptions();
              return e.preventDefault();
            } else if (this.$options.is(':checked')) {
              this.$options.prop('checked', false);
              this.applyCheckedOptionToFilterAndResetOptions();
              return e.preventDefault(); // Needed for automatic testing only
            } else {
              return $('body').append('<p>Esc passed on.</p>');
            }
          }
        });
      }

      attachEnterKeyToFilter() {
        return this.$filter.keydown((e) => {
          if (e.which === 13) {
            this.debugMessage('enter');
            if (this.$optionsContainer.is(':visible')) {
              this.applyCheckedOptionToFilterAndResetOptions();
              return e.preventDefault(); // Needed for automatic testing only
            } else {
              return $('body').append('<p>Enter passed on.</p>');
            }
          }
        });
      }

      attachTabKeyToFilter() {
        return this.$filter.keydown((e) => {
          if (e.which === 9) {
            this.debugMessage('tab');
            if (this.$optionsContainer.is(':visible')) {
              return this.applyCheckedOptionToFilterAndResetOptions();
            }
          }
        });
      }

      attachUpDownKeysToFilter() {
        return this.$filter.keydown((e) => {
          if (e.which === 38 || e.which === 40) {
            if (this.$optionsContainer.is(':visible')) {
              if (e.which === 38) {
                this.moveSelection('up');
              } else {
                this.moveSelection('down');
              }
            } else {
              this.showOptions();
            }
            return e.preventDefault(); // TODO: Test!
          }
        });
      }

      showOptions() {
        this.debugMessage('(show options)');
        this.show(this.$optionsContainer);
        return this.$filter.attr('aria-expanded', 'true');
      }

      hideOptions() {
        this.debugMessage('(hide options)');
        this.hide(this.$optionsContainer);
        return this.$filter.attr('aria-expanded', 'false');
      }

      moveSelection(direction) {
        var $upcomingOption, $visibleOptions, currentIndex, maxIndex, upcomingIndex;
        $visibleOptions = this.$options.filter(':visible');
        maxIndex = $visibleOptions.length - 1;
        currentIndex = $visibleOptions.index($visibleOptions.parent().find(':checked')); // TODO: is parent() good here?!
        upcomingIndex = direction === 'up' ? currentIndex <= 0 ? maxIndex : currentIndex - 1 : currentIndex === maxIndex ? 0 : currentIndex + 1;
        $upcomingOption = $($visibleOptions[upcomingIndex]);
        return $upcomingOption.prop('checked', true).trigger('change');
      }

      attachChangeEventToOptions() {
        return this.$options.change((e) => {
          this.debugMessage('option change');
          this.applyCheckedOptionToFilter();
          return this.$filter.focus().select();
        });
      }

      applyCheckedOptionToFilterAndResetOptions() {
        this.applyCheckedOptionToFilter();
        this.hideOptions();
        return this.filterOptions();
      }

      applyCheckedOptionToFilter() {
        var $checkedOption, $checkedOptionLabel, $previouslyCheckedOptionLabel;
        this.debugMessage('(apply option to filter)');
        $previouslyCheckedOptionLabel = $(`[${this.adgDataAttributeName('option-selected')}]`);
        if ($previouslyCheckedOptionLabel.length === 1) {
          this.removeAdgDataAttribute($previouslyCheckedOptionLabel, 'option-selected');
        }
        $checkedOption = this.$options.filter(':checked');
        if ($checkedOption.length === 1) {
          $checkedOptionLabel = this.labelOfInput($checkedOption);
          this.$filter.val($.trim($checkedOptionLabel.text()));
          return this.addAdgDataAttribute($checkedOptionLabel, 'option-selected');
        } else {
          return this.$filter.val('');
        }
      }

      attachClickEventToOptions() {
        return this.$options.click((e) => {
          this.debugMessage('click option');
          return this.hideOptions();
        });
      }

      attachChangeEventToFilter() {
        return this.$filter.on('input propertychange paste', (e) => {
          this.debugMessage('(filter changed)');
          this.filterOptions(e.target.value);
          return this.showOptions();
        });
      }

      filterOptions(filter = '') {
        var fuzzyFilter, visibleNumber;
        fuzzyFilter = this.fuzzifyFilter(filter);
        visibleNumber = 0;
        this.$options.each((i, el) => {
          var $option, $optionContainer, regex;
          $option = $(el);
          $optionContainer = $option.parent();
          regex = new RegExp(fuzzyFilter, 'i');
          if (regex.test($optionContainer.text())) {
            visibleNumber++;
            return this.show($optionContainer);
          } else {
            return this.hide($optionContainer);
          }
        });
        return this.announceOptionsNumber(filter, visibleNumber);
      }

      announceOptionsNumber(filter = this.$filter.val(), number = this.$options.length) {
        var message;
        this.$alertsContainer.find('p').remove(); // Remove previous alerts (I'm not sure whether this is the best solution, maybe hiding them would be more robust?)
        message = filter === '' ? this.text('numberInTotal', {
          number: number
        }) : this.text('numberFiltered', {
          number: number,
          total: this.$options.length,
          filter: `<kbd>${filter}</kbd>`
        });
        return this.$alertsContainer.append(`<p role='alert'>${message}</p>`);
      }

      fuzzifyFilter(filter) {
        var escapedCharacter, fuzzifiedFilter, i;
        i = 0;
        fuzzifiedFilter = '';
        while (i < filter.length) {
          escapedCharacter = filter.charAt(i).replace(/[\-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g, "\\$&"); // See https://stackoverflow.com/questions/3446170/escape-string-for-use-in-javascript-regex
          fuzzifiedFilter += `${escapedCharacter}.*?`;
          i++;
        }
        return fuzzifiedFilter;
      }

    };

    config = {
      optionsContainer: 'fieldset',
      optionsContainerLabel: 'legend',
      alertsContainerId: 'alerts',
      numberInTotalText: '[number] options in total',
      numberFilteredText: '[number] of [total] options for [filter]'
    };

    return Autocomplete;

  }).call(this);

  $(document).ready(function() {
    return $('[data-adg-autocomplete]').each(function() {
      return new Adg.Autocomplete(this);
    });
  });

}).call(this);

//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"","sourceRoot":"","sources":["<anonymous>"],"names":[],"mappings":"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","sourcesContent":["@Adg = {}\n\nclass Adg.Base\n  uniqueIdCount = 1\n  \n  config =\n    debugMessage:   false\n    hiddenCssClass: 'adg-visually-hidden'\n  \n  # Constructor. Should not be overridden; use @init() instead.\n  #\n  # - Arg1: The DOM element on which the script should be applied (will be saved as @$el)\n  # - Arg2: An optional hash of options which will be merged into the global default config\n  constructor: (el, options = {}) ->\n    @$el = $(el)\n\n    @config = config\n    for key, val of options\n      @config[key] = val\n    \n    @init()\n\n  # Dummy, must be overridden in inheriting classes.\n  init: ->\n    @throwMessageAndPrintObjectsToConsole 'Classes extending App must implement method init()!'\n\n  # Prints the given message to the console if config['debug'] is true.\n  debugMessage: (message) ->\n    console.log \"Adg debug: #{message}\" if @config.debugMessage\n\n  # Executes the given selector on @$el and returns the element. Makes sure exactly one element exists.\n  findOne: (selector) ->\n    result = @$el.find(selector)\n    switch result.length\n      when 0 then @throwMessageAndPrintObjectsToConsole \"No object found for #{selector}!\", result: result\n      when 1 then $(result.first())\n      else @throwMessageAndPrintObjectsToConsole \"More than one object found for #{selector}!\", result: result\n        \n  name: ->\n    \"adg-#{@constructor.name.toLowerCase()}\"\n        \n  addAdgDataAttribute: ($target, name, value = '') ->\n    $target.attr(@adgDataAttributeName(name), value)\n        \n  removeAdgDataAttribute: ($target, name) ->\n    $target.removeAttr(@adgDataAttributeName(name))\n    \n  adgDataAttributeName: (name = null) ->\n    result = \"data-#{@name()}\"\n    result += \"-#{name}\" if name\n    result\n    \n  uniqueId: (name) ->\n    [@name(), name, uniqueIdCount++].join '-'\n    \n  labelOfInput: ($inputs) ->\n    $inputs.map (i, input) =>\n      $input = $(input)\n      \n      id = $input.attr('id')\n      $label = @findOne(\"label[for='#{id}']\")[0]\n\n      if $label.length == 0\n        $label = $input.closest('label')\n        @throwMessageAndPrintObjectsToConsole \"No corresponding input found for input!\", input: $input if $label.length == 0\n\n      $label\n\n  show: ($el) ->\n    $el.removeAttr('hidden')\n    $el.show()\n\n    # TODO Would be cool to renounce CSS and solely use the hidden attribute. But jQuery's :visible doesn't seem to work with it!?\n    # @throwMessageAndPrintObjectsToConsole(\"Element is still hidden, although hidden attribute was removed! Make sure there's no CSS like display:none or visibility:hidden left on it!\", element: $el) if $el.is(':hidden')\n\n  hide: ($el) ->\n    $el.attr('hidden', '')\n    $el.hide()\n    \n  throwMessageAndPrintObjectsToConsole: (message, elements = {}) ->\n    console.log elements\n    throw message\n    \n  text: (text, options = {}) ->\n    text = @config[\"#{text}Text\"]\n    \n    for key, value of options\n      text = text.replace \"[#{key}]\", value\n      \n    text\n\n# Tested in JAWS+IE/FF, NVDA+FF\n#\n# Known issues:\n# - JAWS leaves the input when using up/down without entering something (I guess this is due to screen layout and can be considered intended)\n# - Alert not perceivable upon opening options using up/down\n#     - Possible solution 1: always show options count when filter focused?\n#     - Possible solution 2: wait a moment before adding the alert?\n# - VoiceOver/iOS announces radio buttons as disabled?!\n# - iOS doesn't select all text when option was chosen\n#\n# In general: alerts seem to be most robust in all relevant browsers, but aren't polite. Maybe we'll find a better mechanism to serve browsers individually?\nclass Adg.Autocomplete extends Adg.Base\n  config =\n    optionsContainer:      'fieldset'\n    optionsContainerLabel: 'legend'\n    alertsContainerId:     'alerts'\n    numberInTotalText:     '[number] options in total'\n    numberFilteredText:    '[number] of [total] options for [filter]'\n  \n  init: ->\n    # Merge config into existing one (not nice, see https://stackoverflow.com/questions/47721699/)\n    for key, val of config\n      @config[key] = val\n      \n    jsonOptions = @$el.attr(@adgDataAttributeName())\n    if jsonOptions\n      for key, val of jsonOptions\n        @config[key] = val\n    \n    @debugMessage 'start'\n\n    @initFilter()\n    @initOptions()\n    @initAlerts()\n    \n    @applyCheckedOptionToFilter()\n    @announceOptionsNumber('')\n\n    @attachEvents()\n    \n  initFilter: ->\n    @$filter = @findOne('input[type=\"text\"]')\n    @addAdgDataAttribute(@$filter, 'filter')\n    @$filter.attr('autocomplete', 'off')\n    @$filter.attr('aria-expanded', 'false')\n    \n  initOptions: ->\n    @$optionsContainer = @findOne(@config.optionsContainer)\n    @addAdgDataAttribute(@$optionsContainer, 'options')\n    \n    @$optionsContainerLabel = @findOne(@config.optionsContainerLabel)\n    @$optionsContainerLabel.addClass(@config.hiddenCssClass)\n    \n    @$options = @$optionsContainer.find('input[type=\"radio\"]')\n    @addAdgDataAttribute(@labelOfInput(@$options), 'option')\n    @$options.addClass(@config.hiddenCssClass)\n    \n  initAlerts: ->\n    @$alertsContainer = $(\"<div id='#{@uniqueId(@config.alertsContainerId)}'></div>\")\n    @$optionsContainerLabel.after(@$alertsContainer)\n    @$filter.attr('aria-describedby', [@$filter.attr('aria-describedby'), @$alertsContainer.attr('id')].join(' ').trim())\n    @addAdgDataAttribute(@$alertsContainer, 'alerts')\n  \n  attachEvents: ->\n    @attachClickEventToFilter()\n    @attachChangeEventToFilter()\n    \n    @attachEscapeKeyToFilter()\n    @attachEnterKeyToFilter()\n    @attachTabKeyToFilter()\n    @attachUpDownKeysToFilter()\n    \n    @attachChangeEventToOptions()\n    @attachClickEventToOptions()\n    \n  attachClickEventToFilter: ->\n    @$filter.click =>\n      @debugMessage 'click filter'\n      if @$optionsContainer.is(':visible')\n        @hideOptions()\n      else\n        @showOptions()\n      \n  attachEscapeKeyToFilter: ->\n    @$filter.keydown (e) =>\n      if e.which == 27\n        if @$optionsContainer.is(':visible')\n          @applyCheckedOptionToFilterAndResetOptions()\n          e.preventDefault()\n        else if @$options.is(':checked')\n          @$options.prop('checked', false)\n          @applyCheckedOptionToFilterAndResetOptions()\n          e.preventDefault()\n        else # Needed for automatic testing only\n          $('body').append('<p>Esc passed on.</p>')\n      \n  attachEnterKeyToFilter: ->\n    @$filter.keydown (e) =>\n      if e.which == 13\n        @debugMessage 'enter'\n        if @$optionsContainer.is(':visible')\n          @applyCheckedOptionToFilterAndResetOptions()\n          e.preventDefault()\n        else # Needed for automatic testing only\n          $('body').append('<p>Enter passed on.</p>')\n      \n  attachTabKeyToFilter: ->\n    @$filter.keydown (e) =>\n      if e.which == 9\n        @debugMessage 'tab'\n        if @$optionsContainer.is(':visible')\n          @applyCheckedOptionToFilterAndResetOptions()\n      \n  attachUpDownKeysToFilter: ->\n    @$filter.keydown (e) =>\n      if e.which == 38 || e.which == 40\n        if @$optionsContainer.is(':visible')\n          if e.which == 38\n            @moveSelection('up')\n          else\n            @moveSelection('down')\n        else\n          @showOptions()\n       \n        e.preventDefault() # TODO: Test!\n    \n  showOptions: ->\n    @debugMessage '(show options)'\n    @show(@$optionsContainer)\n    @$filter.attr('aria-expanded', 'true')\n    \n  hideOptions: ->\n    @debugMessage '(hide options)'\n    @hide(@$optionsContainer)\n    @$filter.attr('aria-expanded', 'false')\n    \n  moveSelection: (direction) ->\n    $visibleOptions = @$options.filter(':visible')\n    \n    maxIndex = $visibleOptions.length - 1\n    currentIndex = $visibleOptions.index($visibleOptions.parent().find(':checked')) # TODO: is parent() good here?!\n    \n    upcomingIndex = if direction == 'up'\n                      if currentIndex <= 0\n                        maxIndex\n                      else\n                        currentIndex - 1\n                    else\n                      if currentIndex == maxIndex\n                        0\n                      else\n                        currentIndex + 1\n\n    $upcomingOption = $($visibleOptions[upcomingIndex])\n    $upcomingOption.prop('checked', true).trigger('change')\n    \n  attachChangeEventToOptions: ->\n    @$options.change (e) =>\n      @debugMessage 'option change'\n      @applyCheckedOptionToFilter()\n      @$filter.focus().select()\n\n  applyCheckedOptionToFilterAndResetOptions: ->\n    @applyCheckedOptionToFilter()\n    @hideOptions()\n    @filterOptions()\n      \n  applyCheckedOptionToFilter: ->\n    @debugMessage '(apply option to filter)'\n    \n    $previouslyCheckedOptionLabel = $(\"[#{@adgDataAttributeName('option-selected')}]\")\n    if $previouslyCheckedOptionLabel.length == 1\n      @removeAdgDataAttribute($previouslyCheckedOptionLabel, 'option-selected')\n   \n    $checkedOption = @$options.filter(':checked')\n    if $checkedOption.length == 1\n      $checkedOptionLabel = @labelOfInput($checkedOption)\n      @$filter.val($.trim($checkedOptionLabel.text()))\n      @addAdgDataAttribute($checkedOptionLabel, 'option-selected')\n    else\n      @$filter.val('')\n      \n  attachClickEventToOptions: ->\n    @$options.click (e) =>\n      @debugMessage 'click option'\n      @hideOptions()\n      \n  attachChangeEventToFilter: ->\n    @$filter.on 'input propertychange paste', (e) =>\n      @debugMessage '(filter changed)'\n      @filterOptions(e.target.value)\n      @showOptions()\n      \n  filterOptions: (filter = '') ->\n    fuzzyFilter = @fuzzifyFilter(filter)\n    visibleNumber = 0\n    \n    @$options.each (i, el) =>\n      $option = $(el)\n      $optionContainer = $option.parent()\n\n      regex = new RegExp(fuzzyFilter, 'i')\n      if regex.test($optionContainer.text())\n        visibleNumber++\n        @show($optionContainer)\n      else\n        @hide($optionContainer)\n        \n    @announceOptionsNumber(filter, visibleNumber)\n    \n  announceOptionsNumber: (filter = @$filter.val(), number = @$options.length) ->\n    @$alertsContainer.find('p').remove() # Remove previous alerts (I'm not sure whether this is the best solution, maybe hiding them would be more robust?)\n    \n    message = if filter == ''\n                @text('numberInTotal', number: number)\n              else\n                @text('numberFiltered', number: number, total: @$options.length, filter: \"<kbd>#{filter}</kbd>\")\n      \n    @$alertsContainer.append(\"<p role='alert'>#{message}</p>\")\n        \n  fuzzifyFilter: (filter) ->\n    i = 0\n    fuzzifiedFilter = ''\n    while i < filter.length\n      escapedCharacter = filter.charAt(i).replace(/[\\-\\[\\]\\/\\{\\}\\(\\)\\*\\+\\?\\.\\\\\\^\\$\\|]/g, \"\\\\$&\") # See https://stackoverflow.com/questions/3446170/escape-string-for-use-in-javascript-regex\n      fuzzifiedFilter += \"#{escapedCharacter}.*?\"\n      i++\n      \n    fuzzifiedFilter\n    \n$(document).ready ->\n  $('[data-adg-autocomplete]').each ->\n    new Adg.Autocomplete @"]}
//# sourceURL=coffeescript

Company

  • About
  • Blog
  • Careers
  • Contact

Services

  • Accessibility Audits
  • Accessibility Consulting
  • VPAT/ACR
  • Accessibility Trainings

Compliance

  • WCAG
  • ADA
  • Section 508
  • EN 301 549
  • EAA
  • AODA Compliance Services — Make Your Digital Assets Accessible in Ontario
  • ACA

Resources

  • Accessibility Resources
  • Understanding WCAG
  • WCAG Checklist
  • Understanding WAI-ARIA

Legal

  • Privacy Policy
  • Terms and Conditions
  • Disclaimer
  • Accessibility Statement for digitala11y.com
  • Sitemap

© 2025 DigitalA11Y
All Rights Reserved

Linkedin Twitter Facebook Instagram YouTube

DigitalA11Y
Plot No 108, 3rd Cross Rd, Saipuri Colony,
Hastinapuri Colony, Sainikpuri, Secunderabad -500094
Telangana, India.

Tel:(+91)99082 66680,
E-mail: [email protected]

Scroll to top
  • Home
  • Services
    • WCAG Audit Services
    • VPAT/ACR Services
    • Accessibility Consulting
    • PDF Remediation
    • Accessibility Trainings
    • Website Remediation
    • Design Audit
  • Free Tools
    • Accessibility Checker
    • A11Y Cost Calculator
    • A11Y Bookmarklets
    • Color Contrast Extension
    • WCAG Contrast Checker
  • Resources
    • A11Y Articles
    • WCAG Primer
    • ARIA Cheatsheet
    • A11Y Tools
    • A11Y Patterns
    • A11Y Cheatsheets
  • Contact