Keyboard navigation: skip sub-items in horizontal menu?

Created on 12 June 2020, almost 5 years ago
Updated 1 March 2024, about 1 year ago

Keyboard navigation through the RM horizontal menu definitely works, but there is one feature missing which has been identified as a problem by our agency's accessibility team. It's not a complete fail but they really want us to have it and all our d8 sites now use Responsive Menu for their horizontal navs.

When you are tabbing through the horizontal menu, if a top-level menu item has sub-items you cannot skip past them to the next top-level link, but have to tab through all of the sub-items to get to the next top-level item. The desired behavior is pretty much as described in this issue in the Superfish module queue 🐛 Accessibility issue with menu and submenu keyboard navigation Active :

  • a tab to move from one top-level menu item to another;
  • an enter to get into submenus;
  • up and down arrows to move within the submenu.

Any chance this could be included any time soon? (Thank you for a fantastic module, by the way!)

Feature request
Status

Closed: works as designed

Version

4.1

Component

Code

Created by

Live updates comments and jobs are added and updated live.
Sign in to follow issues

Comments & Activities

Not all content is available!

It's likely this issue predates Contrib.social: some issue and comment data are missing.

  • 🇨🇦Canada bisonbleu

    Hi @danheisel, would you mind sharing your code ?

  • 🇨🇦Canada bisonbleu

    Adding the following JS to a custom theme should provide basic navigation: Tab to top menu item, Enter to open sub-menu, Tab to navigate in the sub-menu items, Enter to activate link.

    You may have to adjust the selectors depending on your markup. For more info, see Add JavaScript to Your Theme or Module

    (function () {
    
      'use strict';
    
      /**
       * Provides basic keyboard navigation in the Main menu (desktop viewport).
       *
       * @type {Drupal~behavior}
       *
       * @prop {Drupal~behaviorAttach} attach
       *   Attaches the behavior for the keyboard navigation implementation.
       */
      Drupal.behaviors.responsive_menu_horizontal = {
        attach: function (context) {
          let horizontalMenu = document.querySelector('#horizontal-menu');
          if (horizontalMenu) {
            // Process main menu items with children.
            let menuItems = horizontalMenu.querySelectorAll('.menu-item--expanded');
            menuItems.forEach(function(item) {
              item.addEventListener('keydown', function(e) {
                if (e.key === 'Enter') {
                  e.preventDefault();
                  e.stopImmediatePropagation();
    
                  // Toggle visibility of sub-menu.
                  let submenu = item.querySelector('.sub-nav');
                  if (submenu && !submenu.classList.contains('submenu-open')) {
                    submenu.classList.toggle('submenu-open');
                  } else {
                    // If no submenu or submenu is open, extract URL from the link and navigate to it.
                    let link = item.querySelector('a');
                    if (link) {
                      window.location.href = link.getAttribute('href');
                    }
                  }
                }
              });
    
              // Add event listener to sub-menu items.
              let subMenuItems = item.querySelectorAll('.sub-nav .menu-item');
              subMenuItems.forEach(function(subItem, index) {
                subItem.addEventListener('keydown', function(e) {
                  if (e.key === 'Enter') {
                    e.preventDefault();
                    e.stopImmediatePropagation();
                    
                    // Extract URL from the link and navigate to it.
                    let link = subItem.querySelector('a');
                    if (link) {
                      window.location.href = link.getAttribute('href');
                    }
                  } else if (e.key === 'Tab' && index === subMenuItems.length - 1) {
                    // If 'Tab' is pressed while in the last sub-menu item, close the sub-menu.
                    let submenu = item.querySelector('.sub-nav');
                    if (submenu && submenu.classList.contains('submenu-open')) {
                      submenu.classList.toggle('submenu-open');
                    }
                  }
                });
              });
    
            });
          }
    
        }
    
      }
    
    })();
    
Production build 0.71.5 2024