From 976b9b57bb0156767e6fd88b3adf746674a9f971 Mon Sep 17 00:00:00 2001 From: David Bolter Date: Thu, 13 Nov 2008 15:51:09 +0000 Subject: [PATCH] =?UTF-8?q?Added=20accordion=20keyboard=20and=20ARIA=20sup?= =?UTF-8?q?port,=20and=20tests.=20Mozilla=20needs=20this=20for=20website.?= =?UTF-8?q?=20(partial=20review:=20Scott=20Gonz=C3=A1lez,=20fix=20for=2035?= =?UTF-8?q?53)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- tests/accordion.js | 18 +++++++++++++ ui/ui.accordion.js | 66 ++++++++++++++++++++++++++++++++++++++++++++-- 2 files changed, 82 insertions(+), 2 deletions(-) diff --git a/tests/accordion.js b/tests/accordion.js index ff3719b41..6ede9bc06 100644 --- a/tests/accordion.js +++ b/tests/accordion.js @@ -129,4 +129,22 @@ test("accordionchange event, open closed and close again", function() { .accordion("activate", 0); }); +test("accessibility", function () { + expect(9); + var ac = $('#list1').accordion().accordion("activate", 1); + var headers = $(".ui-accordion-header"); + + equals( headers.eq(1).attr("tabindex"), "0", "active header should have tabindex=0"); + equals( headers.eq(0).attr("tabindex"), "-1", "inactive header should have tabindex=-1"); + equals( ac.attr("role"), "tablist", "main role"); + equals( headers.attr("role"), "tab", "tab roles"); + equals( headers.next().attr("role"), "tabpanel", "tabpanel roles"); + equals( headers.eq(1).attr("aria-expanded"), "true", "active tab has aria-expanded"); + equals( headers.eq(0).attr("aria-expanded"), "false", "inactive tab has aria-expanded"); + ac.accordion("activate", 0); + equals( headers.eq(0).attr("aria-expanded"), "true", "newly active tab has aria-expanded"); + equals( headers.eq(1).attr("aria-expanded"), "false", "newly inactive tab has aria-expanded"); +}); + + })(jQuery); diff --git a/ui/ui.accordion.js b/ui/ui.accordion.js index 902a5654e..2831475ae 100644 --- a/ui/ui.accordion.js +++ b/ui/ui.accordion.js @@ -41,7 +41,7 @@ $.widget("ui.accordion", { this.element.addClass("ui-accordion"); $('').insertBefore(options.headers); $('').appendTo(options.headers); - options.headers.addClass("ui-accordion-header").attr("tabindex", "0"); + options.headers.addClass("ui-accordion-header"); } var maxHeight; @@ -60,23 +60,83 @@ $.widget("ui.accordion", { maxHeight = Math.max(maxHeight, $(this).outerHeight()); }).height(maxHeight); } + + this.element.attr('role','tablist'); + + var self=this; + options.headers + .attr('role','tab') + .bind('keydown', function(e) { return self._keydown(e); }) + .next() + .attr('role','tabpanel'); options.headers .not(options.active || "") + .attr('aria-expanded','false') + .attr("tabIndex", "-1") .next() .hide(); - options.active.parent().andSelf().addClass(options.selectedClass); + + // make sure at least one header is in the tab order + if (!options.active.length) { + options.headers.eq(0).attr('tabIndex','0'); + } else { + options.active + .attr('aria-expanded','true') + .attr("tabIndex", "0") + .parent().andSelf().addClass(options.selectedClass); + } + + // only need links in taborder for Safari + if (!$.browser.safari) + options.headers.find('a').attr('tabIndex','-1'); if (options.event) { this.element.bind((options.event) + ".accordion", clickHandler); } }, + + _keydown: function(e) { + if (this.options.disabled || e.altKey || e.ctrlKey) + return; + + var keyCode = $.keyCode; + + var length = this.options.headers.length; + var currentIndex = this.options.headers.index(e.target); + var toFocus = false; + + if (e.keyCode == keyCode.RIGHT || e.keyCode == keyCode.DOWN){ + + toFocus = this.options.headers[(currentIndex + 1) % length]; + + } else if (e.keyCode == keyCode.LEFT || e.keyCode == keyCode.UP) { + + toFocus = this.options.headers[(currentIndex - 1 + length) % length]; + + } else if (e.keyCode == keyCode.SPACE || e.keyCode == keyCode.ENTER) { + + return clickHandler.call(this.element[0], { target: e.target }); + + } + + if (toFocus) { + $(e.target).attr('tabIndex','-1'); + $(toFocus).attr('tabIndex','0'); + toFocus.focus(); + return false; + } + + return true; + }, + activate: function(index) { // call clickHandler with custom event clickHandler.call(this.element[0], { target: findActive( this.options.headers, index )[0] }); }, + destroy: function() { this.options.headers.parent().andSelf().removeClass(this.options.selectedClass); this.options.headers.prev(".ui-accordion-left").remove(); @@ -189,6 +249,8 @@ function toggle(toShow, toHide, data, clickedActive, down) { } complete(true); } + toHide.prev().attr('aria-expanded','false').attr("tabIndex", "-1"); + toShow.prev().attr('aria-expanded','true').attr("tabIndex", "0").focus();; } function clickHandler(event) {