← Back to team overview

launchpad-reviewers team mailing list archive

[Merge] lp:~sinzui/launchpad/overlay-tab into lp:launchpad

 

Curtis Hovey has proposed merging lp:~sinzui/launchpad/overlay-tab into lp:launchpad.

Requested reviews:
  Launchpad code reviewers (launchpad-reviewers)
Related bugs:
  Bug #1019064 in Launchpad itself: "Users cannot use keyboard to navigate the choicesource picker"
  https://bugs.launchpad.net/launchpad/+bug/1019064

For more details, see:
https://code.launchpad.net/~sinzui/launchpad/overlay-tab/+merge/112901

We discovered that we cannot use the keyboard to tab or use arrow keys
to move between the links in the choicesource pickers added the to
+filebug. There is nothing special about the pickers...keyboard
navigation was broken on every form a select menu was replaced with.

--------------------------------------------------------------------

RULES

    Pre-implementation: no one
    * Discover what is missing or what was added that broken tabbing.

    Addendum
    * It is possible to tab out of the so-called modal overview. This
      causes confusion because you can tab to elements underneath the
      overlay.
    * Prevent users from tabbing out of the overlay. The user must
      dismiss it using the escape key or the close button.

QA

    You can see a demo of the broken and fixes cases at
    http://people.canonical.com/~curtis/broken-and-fix-tabs.ogv

    On qastaging:
    * Visit https://bugs.qastaging.launchpad.net/gdp/+bug/932086
    * Tab to the status action, press enter.
    * Used tab to cycle through the links.
    * Use shift+tab to cycle backwards though the links.
    * Press enter on the status you want.
    * Verify the bug's status was changed.
    * Tab to the assignee, press enter.
    * Search for curtis
    * Used tab to cycle through the links.
      Verifying that the hidden links in the picker do cause you to
      tab outside of the overlay.
    * Tab to sinzui's email address and press enter to expand.
    * Tab to the Select link and press enter.
    * Verify sinzui was selected.

LINT

    lib/lp/app/javascript/choiceedit/choiceedit.js
    lib/lp/app/javascript/overlay/overlay.js
    lib/lp/app/javascript/overlay/assets/skins/sam/pretty-overlay-skin.css
    lib/lp/app/javascript/overlay/tests/test_overlay.js

TEST

    ./bin/test -vv -t test_choiceedit -t test_overlay lp.app.tests.test_yui

IMPLEMENTATION

Removed bad code that caused the close button to steal focus.
    lib/lp/app/javascript/choiceedit/choiceedit.js

Fixed the CSS for the close button, which was empty, invalid, and unusable
by screen readers.
    lib/lp/app/javascript/overlay/assets/skins/sam/pretty-overlay-skin.css

Add a handler to watch the tab key. When you tab past the last node, focus
it moved to the first node. If you shift-tab before the first node, focus
is moved to the last node. This is trickier for person and target pickers
because they have hidden nodes that you can expand to see. I wrote a
method to locate and filter the links that you cannot see...those that
the browser did not allocate height to on the page. As you may note from
one of the tests, we cannot check the elements style of computed style
because the parent element may cause it to be hidden.
    lib/lp/app/javascript/overlay/overlay.js
    lib/lp/app/javascript/overlay/tests/test_overlay.js
-- 
https://code.launchpad.net/~sinzui/launchpad/overlay-tab/+merge/112901
Your team Launchpad code reviewers is requested to review the proposed merge of lp:~sinzui/launchpad/overlay-tab into lp:launchpad.
=== modified file 'lib/lp/app/javascript/choiceedit/choiceedit.js'
--- lib/lp/app/javascript/choiceedit/choiceedit.js	2012-06-15 18:07:24 +0000
+++ lib/lp/app/javascript/choiceedit/choiceedit.js	2012-06-30 14:09:23 +0000
@@ -578,12 +578,7 @@
         }
 
         this.move(valueX, valueY);
-
-        var bb = this.get('boundingBox');
-        bb.on('focus', function(e) {
-            bb.one('.close-button').focus();
-        });
-        bb.one('.close-button').focus();
+        this.get('boundingBox').one('.close-button').focus();
     },
 
     /**

=== modified file 'lib/lp/app/javascript/overlay/assets/skins/sam/pretty-overlay-skin.css'
--- lib/lp/app/javascript/overlay/assets/skins/sam/pretty-overlay-skin.css	2012-05-16 02:16:25 +0000
+++ lib/lp/app/javascript/overlay/assets/skins/sam/pretty-overlay-skin.css	2012-06-30 14:09:23 +0000
@@ -31,9 +31,11 @@
     float: right;
     width: 15px;
     height: 15px;
+    overflow: hidden;
     background: url('images/close.gif');
     display: block;
     margin-top: 4px;
+    text-indent: 3em;
 }
 
 .yui3-pretty-overlay .close .clear {

=== modified file 'lib/lp/app/javascript/overlay/overlay.js'
--- lib/lp/app/javascript/overlay/overlay.js	2012-01-06 12:34:37 +0000
+++ lib/lp/app/javascript/overlay/overlay.js	2012-06-30 14:09:23 +0000
@@ -9,10 +9,12 @@
  */
 
 var ESCAPE = 27,
+    TAB = 9,
     CANCEL = 'cancel',
     BOUNDING_BOX = 'boundingBox',
     CONTENT_BOX = 'contentBox',
     BINDUI = "bindUI";
+    TABBABLE_SELECTOR = 'a, button, input, select, textarea';
 
 
    /**
@@ -31,7 +33,8 @@
     * @class PrettyOverlay
     * @namespace lazr
     */
-    var PrettyOverlay = function(cfg) {
+    var PrettyOverlay;
+    PrettyOverlay = function(cfg) {
         // Check whether the callsite has set a zIndex... if not, set it
         // to 1000, as the YUI.overlay default is zero.
         if (cfg && arguments[0].zIndex === undefined){
@@ -283,6 +286,8 @@
             bounding_box.on('click', function(e) {
                 bounding_box.blur();
             });
+            bounding_box.delegate(
+                "keydown", this._handleTab, TABBABLE_SELECTOR, this);
             this.after('steptitleChange', this._afterSteptitleChange);
             this.after('progressChange', this._afterProgressChange);
         },
@@ -299,6 +304,53 @@
         },
 
         /**
+         * Return a NodeList of elements that the user can navigate
+         * to using the Tab key. Browser allow users to tab to:
+         * a, button, input, select, and textarea. The NodeList only
+         * contains the elements the user can see and interact with.
+         *
+         * @private
+         * @method _getTabNodes
+         */
+         _getTabNodes: function() {
+            var bounding_box = this.get(BOUNDING_BOX);
+            var all_tab_nodes = bounding_box.all(TABBABLE_SELECTOR);
+            var tab_nodes = new Y.NodeList([]);
+            all_tab_nodes.each(function(item, index, node_list) {
+                if (item.get('region').height > 0) {
+                    // The node takes up space on the page.
+                    // This rule will skip empty links which are invalid.
+                    tab_nodes.push(item);
+                }
+            });
+            return tab_nodes;
+        },
+
+        /**
+         * An event handler to ensure the Tab key cycles through
+         * The visible elements that the user can interact with.
+         * The user cannot tab out of the modal overlay.
+         *
+         * @private
+         * @method _handleTab
+         */
+        _handleTab: function(e) {
+            if (e.keyCode === TAB) {
+                var tab_nodes = this._getTabNodes();
+                var max_index = tab_nodes.size() - 1;
+                var next = (e.shiftKey) ? -1 : 1;
+                var next_index = tab_nodes.indexOf(e.currentTarget) + next;
+                if (next_index > max_index) {
+                    tab_nodes.item(0).focus();
+                    e.halt();
+                } else if (next_index < 0) {
+                    tab_nodes.item(max_index).focus();
+                    e.halt();
+                }
+            }
+        },
+
+        /**
          * Overrides the method from WidgetStdMod which creates the separate
          * sections in the contentBox to also add the progressbar widget
          * after headerContent.
@@ -339,7 +391,7 @@
         '<div class="pretty-overlay-window">',
         '<div class="content_box_container" id="yui3-pretty-overlay-modal">',
         '<div class="close">',
-        '<a href="#" title="Close" class="close-button"></a>',
+        '<a href="#" title="Close" class="close-button">(x)</a>',
         '</div>',
         '</div>',
         '</div>'].join('');
@@ -348,4 +400,5 @@
 
     Y.lazr.PrettyOverlay = PrettyOverlay;
 
-}, "0.1", {"skinnable": true, "requires": ["oop", "overlay", "event", "widget", "widget-stack", "widget-position"]});
+}, "0.1", {"skinnable": true, "requires": [
+    "oop", "overlay", "event", "widget", "widget-stack", "widget-position"]});

=== modified file 'lib/lp/app/javascript/overlay/tests/test_overlay.js'
--- lib/lp/app/javascript/overlay/tests/test_overlay.js	2012-04-06 17:28:25 +0000
+++ lib/lp/app/javascript/overlay/tests/test_overlay.js	2012-06-30 14:09:23 +0000
@@ -228,8 +228,112 @@
                     one('.steps .step-on').
                     getStyle('width')
             );
+        },
+
+        test_getTabNodes_types: function() {
+            // Tabbable nodes include <a>, <button>, <input>, <select>.
+            // Remember that the 0 button is the close button in the header.
+            this.overlay = new Y.lazr.PrettyOverlay({
+                headerContent: 'Fnord',
+                bodyContent: [
+                    '<div>',
+                    '<b>ignored</b>',
+                    '<input type="text" value="1"/>',
+                    '<select><option>2</opton></select>',
+                    '<a href="#">3</a>',
+                    '<button>4</button>',
+                    '<textarea rows="2" cols="5">5</textarea>',
+                    '</div>'].join(' ')
+            });
+            this.overlay.render();
+            var tab_nodes = this.overlay._getTabNodes();
+            Y.Assert.areEqual(6, tab_nodes.size());
+        },
+
+        test_getTabNodes_visibility: function() {
+            // Hidden nodes are exluded because tab ignores them.
+            // Remember that the 0 button is the close button in the header.
+            this.overlay = new Y.lazr.PrettyOverlay({
+                headerContent: 'Fnord',
+                bodyContent: [
+                    '<div>',
+                    '<a href="#">1</a>',
+                    '<input type="text" value="2"/>',
+                    '<a style="display:none;">ignored</a>',
+                    '<span style="display:none;"><button>nil</button></span>',
+                    '',
+                    '</div>'].join(' ')
+            });
+            this.overlay.render();
+            var tab_nodes = this.overlay._getTabNodes();
+            Y.Assert.areEqual(3, tab_nodes.size());
+        },
+
+        test_handleTab_last_to_first: function() {
+            // Tabbing from the last navigatable node moves focus to the first.
+            // Remember that the 0 button is the close button in the header.
+            this.overlay = new Y.lazr.PrettyOverlay({
+                headerContent: 'Fnord',
+                bodyContent: '<div><a href="#">1</a> <a href="#">2</a></div>'
+            });
+            this.overlay.render();
+            var nodes = this.overlay.get('boundingBox').all('a');
+            var focused = false;
+            nodes.item(0).after('focus', function(e) {
+                focused = true;});
+            var halted = false;
+            var fake_e = {
+                keyCode: 9, shiftKey: false, currentTarget: nodes.item(2),
+                halt: function() {halted = true;}};
+            this.overlay._handleTab(fake_e);
+            Y.Assert.isTrue(halted);
+            Y.Assert.isTrue(focused);
+        },
+
+        test_handleTab_first_to_last: function() {
+            // Shift+Tab from the first navigatable node moves focus
+            // to the last.
+            // Remember that the 0 button is the close button in the header.
+            this.overlay = new Y.lazr.PrettyOverlay({
+                headerContent: 'Fnord',
+                bodyContent: '<div><a href="#">1</a> <a href="#">2</a></div>'
+            });
+            this.overlay.render();
+            var nodes = this.overlay.get('boundingBox').all('a');
+            var focused = false;
+            nodes.item(2).after('focus', function(e) {
+                focused = true;});
+            var halted = false;
+            var fake_e = {
+                keyCode: 9, shiftKey: true, currentTarget: nodes.item(0),
+                halt: function() {halted = true;}
+            };
+            this.overlay._handleTab(fake_e);
+            Y.Assert.isTrue(halted);
+            Y.Assert.isTrue(focused);
+        },
+
+        test_handleTab_within_tab_range: function() {
+            // Tab from one element to another that is not beyond the
+            // first or last element does nothing special.
+            // Remember that the 0 button is the close button in the header.
+            this.overlay = new Y.lazr.PrettyOverlay({
+                headerContent: 'Fnord',
+                bodyContent: '<div><a href="#">1</a> <a href="#">2</a></div>'
+            });
+            this.overlay.render();
+            var nodes = this.overlay.get('boundingBox').all('a');
+            var focused = false;
+            nodes.after('focus', function(e) {focused = true;});
+            var halted = false;
+            var fake_e = {
+                keyCode: 9, shiftKey: false, currentTarget: nodes.item(0),
+                halt: function() {halted = true;}
+            };
+            this.overlay._handleTab(fake_e);
+            Y.Assert.isFalse(halted);
+            Y.Assert.isFalse(halted);
         }
-
     }));
 
 }, '0.1', {


Follow ups