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

Datepicker with Radio Buttons


Available dates
Provides datepicker upon clicking or arrow keys usage

Code

  • HTML
  • CSS
  • JavaScript
<button href="#" id="before">Focusable element before</button>
<form>
  <div data-adg-datepicker="">
    <div class="control"><br>
      <label for="birthday">Birthday </label><input aria-describedby="birthday_help" id="birthday" type="text" />
      <fieldset hidden="">
        <legend>Available dates</legend>
      </fieldset>
      <div class="description" id="birthday_help">
        Provides datepicker upon clicking or arrow keys usage
      </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-datepicker] {
  position: relative;
}

[data-adg-datepicker-options] {
  position: absolute;
  float: left;
  background-color: lightyellow;
  border: 1px solid;
  padding: 5px 0;
}

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

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

[data-adg-datepicker-alerts] p {
  margin: 0;
}
[data-adg-datepicker-alerts] kbd::before {
  content: "«";
}
[data-adg-datepicker-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 {
  width: 100%;
}
(function() {
  var AdgDatepicker;

  AdgDatepicker = (function() {
    var config;

    class AdgDatepicker {
      constructor(el, options = {}) {
        this.$el = $(el);
        this.config = config;
        this.currentDate = this.config["date"];
        this.initInput();
        this.initOptions();
        this.applyCheckedOptionToInput();
      }

      // 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-datepicker";
      }

      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;
      }

      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;
      }

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

      initOptions() {
        this.$optionsContainer = this.findOne('fieldset');
        this.addAdgDataAttribute(this.$optionsContainer, 'options');
        this.$optionsContainerLabel = this.findOne('legend');
        this.$optionsContainerLabel.addClass('adg-visually-hidden');
        this.initDate();
        return this.setSelection(this.currentDate.getDate() - 1, false);
      }

      getFirstMonthDay(date) {
        var m, y;
        y = date.getFullYear();
        m = date.getMonth();
        return new Date(y, m, 1);
      }

      getLastMonthDay(date) {
        var m, y;
        y = date.getFullYear();
        m = date.getMonth();
        return new Date(y, m + 1, 0);
      }

      initDate() {
        var $dateTable, $tr, day, daysOfMonth, firstDay, i, id, j, k, lastDay, len, len1, ref, value, weekday;
        this.$optionsContainer.find("table").remove();
        $dateTable = $(`<table border='1'><caption>${this.config["monthNames"][this.currentDate.getMonth()]} ${this.currentDate.getFullYear()}</caption><thead></thead></table>`);
        ref = this.config["dayNames"];
        for (j = 0, len = ref.length; j < len; j++) {
          weekday = ref[j];
          $dateTable.find("thead").append(`<th>${weekday}</th>`);
        }
        this.$optionsContainer.append($dateTable);
        firstDay = this.getFirstMonthDay(this.currentDate);
        lastDay = this.getLastMonthDay(this.currentDate);
        daysOfMonth = [];
        day = firstDay;
        while (day <= lastDay) {
          daysOfMonth.push(new Date(day));
          day.setDate(day.getDate() + 1);
        }
        
        // Add empty days at beginning
        i = 1;
        firstDay = daysOfMonth[0].getDay();
        while (i < firstDay) {
          daysOfMonth.unshift(null);
          i++;
        }
        
        // Add empty days at end
        i = daysOfMonth[daysOfMonth.length - 1].getDay();
        while (i > 0 && i < 6) {
          daysOfMonth.push(null);
          i++;
        }
        $tr = null;
        for (i = k = 0, len1 = daysOfMonth.length; k < len1; i = ++k) {
          day = daysOfMonth[i];
          if (i % 7 === 0) {
            $tr = $("<tr></tr>");
            $dateTable.append($tr);
          }
          value = day ? (id = `favorite_hobby_${i}`, `<input type='radio' name='hobby' id='${id}' /><label for='${id}'><span class='adg-visually-hidden'>${this.getDayName(day.getDay())}, </span>${day.getDate()}<span class='adg-visually-hidden'> of ${this.config['monthNames'][day.getMonth()]} ${day.getFullYear()}</span></label>`) : "";
          $tr.append(`<td class='control'>${value}</td>`);
        }
        this.$options = this.$optionsContainer.find('input[type="radio"]');
        this.attachOptionsEvents();
        this.addAdgDataAttribute(this.labelOfInput(this.$options), 'option');
        return this.$options.addClass('adg-visually-hidden');
      }

      getDayName(day) {
        if (day === 0) {
          day = 6;
        }
        return this.config['dayNames'][day - 1];
      }

      attachInputEvents() {
        this.attachClickEventToInput();
        this.attachEscapeKeyToInput();
        this.attachEnterKeyToInput();
        this.attachTabKeyToInput();
        return this.attachUpDownKeysToInput();
      }

      attachOptionsEvents() {
        this.attachArrowKeysToOptions();
        this.attachChangeEventToOptions();
        this.attachClickEventToOptionLabels();
        this.attachEnterEventToOptions();
        return this.attachTabEventToOptions();
      }

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

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

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

      attachTabKeyToInput() {
        return this.$input.keydown((e) => {
          if (e.which === 9) {
            if (this.$optionsContainer.is(':visible')) {
              return this.applyCheckedOptionToInputAndResetOptions();
            }
          }
        });
      }

      attachUpDownKeysToInput() {
        return this.$input.keydown((e) => {
          if (e.which === 38 || e.which === 40) {
            this.showOptions();
            return e.preventDefault(); // TODO: Test!
          }
        });
      }

      showOptions() {
        this.show(this.$optionsContainer);
        this.$input.attr('aria-expanded', 'true');
        if (this.$options.filter(':checked').length === 0) {
          this.currentDate = this.config["date"];
          this.initDate();
          this.setSelection(this.currentDate.getDate() - 1);
        }
        return this.$options.filter(':checked').focus();
      }

      hideOptions() {
        this.hide(this.$optionsContainer);
        this.$input.attr('aria-expanded', 'false');
        return this.$input.focus();
      }

      moveSelection(direction) {
        var currentIndex, maxIndex, upcomingIndex;
        maxIndex = this.$options.length - 1;
        currentIndex = this.$options.index(this.$options.parent().find(':checked')); // TODO: is parent() good here?!
        upcomingIndex = direction === 'left' ? currentIndex <= 0 ? (this.currentDate = this.previousMonth(this.currentDate), this.initDate(), -1) : currentIndex - 1 : direction === 'up' ? currentIndex - 7 < 0 ? (this.currentDate = this.previousMonth(this.currentDate), this.initDate(), -1) : currentIndex - 7 : direction === 'right' ? currentIndex === maxIndex ? (this.currentDate = this.nextMonth(this.currentDate), this.initDate(), 0) : currentIndex + 1 : direction === 'down' ? currentIndex + 7 > maxIndex ? (this.currentDate = this.nextMonth(this.currentDate), this.initDate(), 0) : currentIndex + 7 : void 0; // TODO: Calculate index for the current week day // TODO: Calculate index for the current week day
        return this.setSelection(upcomingIndex);
      }

      setSelection(current, change = true) {
        var $currentOption;
        if (current === -1) {
          current = this.$options.length - 1;
        }
        $currentOption = $(this.$options[current]);
        $currentOption.prop('checked', true);
        if (change) {
          $currentOption.trigger('change');
          return $currentOption.focus();
        }
      }

      previousMonth(now) {
        if (now.getMonth() === 0) {
          return new Date(now.getFullYear() - 1, 11, 1);
        } else {
          return new Date(now.getFullYear(), now.getMonth() - 1, 1);
        }
      }

      nextMonth(now) {
        if (now.getMonth() === 11) {
          return new Date(now.getFullYear() + 1, 11, 1);
        } else {
          return new Date(now.getFullYear(), now.getMonth() + 1, 1);
        }
      }

      attachArrowKeysToOptions() {
        return this.$options.keydown((e) => {
          if (e.which === 37 || e.which === 38 || e.which === 39 || e.which === 40) {
            if (e.which === 37) {
              this.moveSelection('left');
            } else if (e.which === 38) {
              this.moveSelection('up');
            } else if (e.which === 39) {
              this.moveSelection('right');
            } else if (e.which === 40) {
              this.moveSelection('down');
            }
            return e.preventDefault(); // TODO: Test!
          }
        });
      }

      attachChangeEventToOptions() {
        return this.$options.change((e) => {
          return this.applyCheckedOptionToInput();
        });
      }

      applyCheckedOptionToInputAndResetOptions() {
        this.applyCheckedOptionToInput();
        return this.hideOptions();
      }

      applyCheckedOptionToInput() {
        var $checkedOption, $checkedOptionLabel, $previouslyCheckedOptionLabel;
        $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.$input.val($.trim($checkedOptionLabel.text()));
          return this.addAdgDataAttribute($checkedOptionLabel, 'option-selected');
        } else {
          return this.$input.val('');
        }
      }

      attachClickEventToOptionLabels() {
        return this.labelOfInput(this.$options).click((e) => {
          return this.hideOptions();
        });
      }

      attachEnterEventToOptions() {
        return this.$options.keydown((e) => {
          if (e.which === 13) {
            this.hideOptions();
            e.preventDefault();
            return e.stopPropagation();
          }
        });
      }

      attachTabEventToOptions() {
        return this.$options.keydown((e) => {
          if (e.which === 9) {
            return this.hideOptions();
          }
        });
      }

    };

    config = {
      date: new Date(),
      dayNames: ["Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday", "Sunday"],
      monthNames: ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"]
    };

    return AdgDatepicker;

  }).call(this);

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

}).call(this);

//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"","sourceRoot":"","sources":["<anonymous>"],"names":[],"mappings":"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","sourcesContent":["class AdgDatepicker\n  config =\n    date:       new Date()\n    dayNames:   [\"Monday\", \"Tuesday\", \"Wednesday\", \"Thursday\", \"Friday\", \"Saturday\", \"Sunday\"]\n    monthNames: [\"January\", \"February\", \"March\", \"April\", \"May\", \"June\", \"July\", \"August\", \"September\", \"October\", \"November\", \"December\"]\n  \n  constructor: (el, options = {}) ->\n    @$el = $(el)\n    @config = config\n    \n    @currentDate = @config[\"date\"]\n\n    @initInput()\n    @initOptions()\n\n    @applyCheckedOptionToInput()\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-datepicker\"\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  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  initInput: ->\n    @$input = @findOne('input[type=\"text\"]')\n    @$input.attr('autocomplete', 'off')\n    @$input.attr('aria-expanded', 'false')\n    @attachInputEvents()\n\n  initOptions: ->\n    @$optionsContainer = @findOne('fieldset')\n    @addAdgDataAttribute(@$optionsContainer, 'options')\n\n    @$optionsContainerLabel = @findOne('legend')\n    @$optionsContainerLabel.addClass('adg-visually-hidden')\n    \n    @initDate()\n    @setSelection(@currentDate.getDate() - 1, false)\n    \n  getFirstMonthDay: (date) ->\n    y = date.getFullYear()\n    m = date.getMonth()\n    new Date(y, m, 1)\n    \n  getLastMonthDay: (date) ->\n    y = date.getFullYear()\n    m = date.getMonth()\n    new Date(y, m + 1, 0)\n    \n  initDate: () ->\n    @$optionsContainer.find(\"table\").remove()\n    $dateTable = $(\"<table border='1'><caption>#{@config[\"monthNames\"][@currentDate.getMonth()]} #{@currentDate.getFullYear()}</caption><thead></thead></table>\")\n    for weekday in @config[\"dayNames\"]\n      $dateTable.find(\"thead\").append(\"<th>#{weekday}</th>\")\n      \n    @$optionsContainer.append($dateTable)\n    \n    firstDay = @getFirstMonthDay(@currentDate)\n    lastDay = @getLastMonthDay(@currentDate)\n    \n    daysOfMonth = []\n    day = firstDay\n    while day <= lastDay\n      daysOfMonth.push new Date(day)\n      day.setDate day.getDate() + 1\n      \n    # Add empty days at beginning\n    i = 1\n    firstDay = daysOfMonth[0].getDay()\n    while i < firstDay\n      daysOfMonth.unshift null\n      i++\n      \n    # Add empty days at end\n    i = daysOfMonth[daysOfMonth.length - 1].getDay()\n    while i > 0 && i < 6\n      daysOfMonth.push null\n      i++\n      \n    $tr = null\n    for day, i in daysOfMonth\n      if i % 7 == 0\n        $tr = $(\"<tr></tr>\")\n        $dateTable.append($tr)\n        \n      value = if day\n                id = \"favorite_hobby_#{i}\"\n\n                \"<input type='radio' name='hobby' id='#{id}' /><label for='#{id}'><span class='adg-visually-hidden'>#{@getDayName(day.getDay())}, </span>#{day.getDate()}<span class='adg-visually-hidden'> of #{@config['monthNames'][day.getMonth()]} #{day.getFullYear()}</span></label>\"\n              else\n                \"\"\n      $tr.append(\"<td class='control'>#{value}</td>\")\n      \n    @$options = @$optionsContainer.find('input[type=\"radio\"]')\n    @attachOptionsEvents()\n\n    @addAdgDataAttribute(@labelOfInput(@$options), 'option')\n    @$options.addClass('adg-visually-hidden')\n    \n  getDayName: (day) ->\n    day = 6 if day == 0\n    @config['dayNames'][day - 1]\n\n  attachInputEvents: ->\n    @attachClickEventToInput()\n\n    @attachEscapeKeyToInput()\n    @attachEnterKeyToInput()\n    @attachTabKeyToInput()\n    @attachUpDownKeysToInput()\n\n  attachOptionsEvents: ->\n    @attachArrowKeysToOptions()\n    @attachChangeEventToOptions()\n    @attachClickEventToOptionLabels()\n    @attachEnterEventToOptions()\n    @attachTabEventToOptions()\n\n  attachClickEventToInput: ->\n    @$input.click =>\n      if @$optionsContainer.is(':visible')\n        @hideOptions()\n      else\n        @showOptions()\n\n  attachEscapeKeyToInput: ->\n    @$input.keydown (e) =>\n      if e.which == 27\n        if @$optionsContainer.is(':visible')\n          @applyCheckedOptionToInputAndResetOptions()\n          e.preventDefault()\n        else if @$options.is(':checked')\n          @$options.prop('checked', false)\n          @applyCheckedOptionToInputAndResetOptions()\n          e.preventDefault()\n        else # Needed for automatic testing only\n          $('body').append('<p>Esc passed on.</p>')\n\n  attachEnterKeyToInput: ->\n    @$input.keydown (e) =>\n      if e.which == 13\n        if @$optionsContainer.is(':visible')\n          @applyCheckedOptionToInputAndResetOptions()\n          e.preventDefault()\n        else # Needed for automatic testing only\n          $('body').append('<p>Enter passed on.</p>')\n\n  attachTabKeyToInput: ->\n    @$input.keydown (e) =>\n      if e.which == 9\n        if @$optionsContainer.is(':visible')\n          @applyCheckedOptionToInputAndResetOptions()\n\n  attachUpDownKeysToInput: ->\n    @$input.keydown (e) =>\n      if e.which == 38 || e.which == 40\n        @showOptions()\n        e.preventDefault() # TODO: Test!\n\n  showOptions: ->\n    @show(@$optionsContainer)\n    @$input.attr('aria-expanded', 'true')\n    \n    if @$options.filter(':checked').length == 0\n      @currentDate = @config[\"date\"]\n      @initDate()\n      @setSelection(@currentDate.getDate() - 1)\n\n    @$options.filter(':checked').focus()\n\n  hideOptions: ->\n    @hide(@$optionsContainer)\n    @$input.attr('aria-expanded', 'false')\n    @$input.focus()\n\n  moveSelection: (direction) ->\n    maxIndex = @$options.length - 1\n    currentIndex = @$options.index(@$options.parent().find(':checked')) # TODO: is parent() good here?!\n    \n    upcomingIndex = if direction == 'left'\n                      if currentIndex <= 0\n                        @currentDate = @previousMonth(@currentDate)\n                        @initDate()\n                        -1\n                      else\n                        currentIndex - 1\n                    else if direction == 'up'\n                      if currentIndex - 7 < 0\n                        @currentDate = @previousMonth(@currentDate)\n                        @initDate()\n                        -1 # TODO: Calculate index for the current week day\n                      else\n                        currentIndex - 7\n                    else if direction == 'right'\n                      if currentIndex == maxIndex\n                        @currentDate = @nextMonth(@currentDate)\n                        @initDate()\n                        0\n                      else\n                        currentIndex + 1\n                    else if direction == 'down'\n                      if currentIndex + 7 > maxIndex\n                        @currentDate = @nextMonth(@currentDate)\n                        @initDate()\n                        0 # TODO: Calculate index for the current week day\n                      else\n                        currentIndex + 7\n\n    @setSelection(upcomingIndex)\n                        \n  setSelection: (current, change = true) ->\n    if current == -1\n      current = @$options.length - 1\n      \n    $currentOption = $(@$options[current])\n    $currentOption.prop('checked', true)\n  \n    if change\n      $currentOption.trigger('change')\n      $currentOption.focus()\n    \n  previousMonth: (now) ->\n    if now.getMonth() == 0\n      new Date(now.getFullYear() - 1, 11, 1)\n    else\n      new Date(now.getFullYear(), now.getMonth() - 1, 1)\n    \n  nextMonth: (now) ->\n    if now.getMonth() == 11\n      new Date(now.getFullYear() + 1, 11, 1)\n    else\n      new Date(now.getFullYear(), now.getMonth() + 1, 1)\n\n  attachArrowKeysToOptions: ->\n    @$options.keydown (e) =>\n      if e.which == 37 || e.which == 38 || e.which == 39 || e.which == 40\n        if e.which == 37\n          @moveSelection('left')\n        else if e.which == 38\n          @moveSelection('up')\n        else if e.which == 39\n          @moveSelection('right')\n        else if e.which == 40\n          @moveSelection('down')\n          \n        e.preventDefault() # TODO: Test!\n  \n  attachChangeEventToOptions: ->\n    @$options.change (e) =>\n      @applyCheckedOptionToInput()\n\n  applyCheckedOptionToInputAndResetOptions: ->\n    @applyCheckedOptionToInput()\n    @hideOptions()\n\n  applyCheckedOptionToInput: ->\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      @$input.val($.trim($checkedOptionLabel.text()))\n      @addAdgDataAttribute($checkedOptionLabel, 'option-selected')\n    else\n      @$input.val('')\n\n  attachClickEventToOptionLabels: ->\n    @labelOfInput(@$options).click (e) =>\n      @hideOptions()\n\n  attachEnterEventToOptions: ->\n    @$options.keydown (e) =>\n      if e.which == 13\n        @hideOptions()\n        e.preventDefault()\n        e.stopPropagation()\n\n  attachTabEventToOptions: ->\n    @$options.keydown (e) =>\n      if e.which == 9\n        @hideOptions()\n    \n$(document).ready ->\n  $('[data-adg-datepicker]').each ->\n    new AdgDatepicker @"]}
//# 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