Autosuggest with Radio Buttons
Code
<p>
<button>Focusable element before</button>
</p>
<form>
<div data-adg-autosuggest="">
<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>
<p>
<button>Focusable element after</button>
</p>
@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-autosuggest-options] {
position: absolute;
z-index: 1;
background-color: lightyellow;
border: 1px solid;
padding: 5px 0;
}
[data-adg-autosuggest-option] {
display: block;
}
[data-adg-autosuggest-option]:hover,
[data-adg-autosuggest-option-selected] {
cursor: pointer;
background-color: #000;
color: lightyellow;
}
[data-adg-autosuggest-alerts] p {
margin: 0;
}
[data-adg-autosuggest-alerts] kbd::before {
content: "«";
}
[data-adg-autosuggest-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() {
// 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?
var AdgAutocomplete;
AdgAutocomplete = (function() {
var config, uniqueIdCount;
class AdgAutocomplete {
constructor(el, options = {}) {
var jsonOptions, key, val;
this.$el = $(el);
this.config = config;
for (key in options) {
val = options[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('');
this.attachEvents();
}
// 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-autosuggest";
}
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;
}
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;
}
};
uniqueIdCount = 1;
config = {
debugMessage: false,
hiddenCssClass: 'adg-visually-hidden',
optionsContainer: 'fieldset',
optionsContainerLabel: 'legend',
alertsContainerId: 'alerts',
numberInTotalText: '[number] options in total',
numberFilteredText: '[number] of [total] options for [filter]'
};
return AdgAutocomplete;
}).call(this);
$(document).ready(function() {
return $('[data-adg-autosuggest]').each(function() {
return new AdgAutocomplete(this);
});
});
}).call(this);
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"","sourceRoot":"","sources":["<anonymous>"],"names":[],"mappings":"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","sourcesContent":["# 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 AdgAutocomplete\n  uniqueIdCount = 1\n  \n  config =\n    debugMessage:   false\n    hiddenCssClass: 'adg-visually-hidden'\n    \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  constructor: (el, options = {}) ->\n    @$el = $(el)\n\n    @config = config\n    for key, val of options\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  # 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-autosuggest\"\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  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-autosuggest]').each ->\n    new AdgAutocomplete @"]}
//# sourceURL=coffeescript