Cómo crear la navegación principal para un sitio web

En este tutorial se describe cómo crear una navegación principal accesible de un sitio web. Aprenderás sobre HTML semántico, accesibilidad y cómo el uso de atributos de ARIA a veces puede hacer más daño que bien.

Manuel Matuzović
Manuel Matuzović

Existen muchas formas diferentes de crear la navegación principal de un sitio web en términos de estilo, funcionalidad y el lenguaje de marcado subyacente y la información semántica subyacente. Si la implementación es demasiado minimalista, funcionará para la mayoría de las personas, pero es posible que la experiencia del usuario (UX) no sea buena. Si tiene demasiado diseño, podría confundir a los usuarios o incluso impedirles poder acceder.

Para la mayoría de los sitios web, lo mejor es crear algo que no sea ni demasiado simple ni demasiado complicado.

Compilación capa por capa

En este instructivo, comenzarás con una configuración básica y agregarás funciones capa por capa hasta el punto en que proporciones la información, el estilo y la funcionalidad suficientes para complacer a la mayoría de los usuarios. Para lograrlo, usa el principio de mejora progresiva, que establece que debes comenzar con la solución más fundamental y sólida, y agregar capas de funcionalidad de manera progresiva. Si una capa no funciona por algún motivo, la navegación seguirá funcionando porque recurre a la capa subyacente con facilidad.

Estructura básica

Para una navegación básica, necesitas dos elementos: elementos <a> y algunas líneas de CSS para mejorar el estilo y el diseño predeterminados de tus vínculos.

<a href="/home">Home</a>
<a href="/about-us">About us</a>
<a href="/pricing">Pricing</a>
<a href="/contact">Contact</a>
/* Define variables for your colors */
:root {
  --color-shades-dark: rgb(25, 25, 25);
}

/* Use the alternative box model
Details: <https://web.dev/learn/css/box-model/> */
*{
  box-sizing: border-box;
}

/* Basic font styling */
body {
  font-family: Segoe UI, system-ui, -apple-system, sans-serif;
  font-size: 1.6rem;
}

/* Link styling */
a {
  --text-color: var(--color-shades-dark);
  border-block-end: 3px solid var(--border-color, transparent);
  color: var(--text-color);
  display: inline-block;
  margin-block-end: 0.5rem; /* See note at the bottom of this chapter */
  margin-inline-end: 0.5rem;
  padding: 0.1rem;
  text-decoration: none;
}

/* Change the border-color on :hover and :focus */
a:where(:hover, :focus) {
  --border-color: var(--text-color);
}
Ver Paso 1: HTML y CSS básicos" en CodePen.

Esto funciona bien para la mayoría de los usuarios, independientemente de cómo accedan al sitio. Se puede acceder a la navegación con un mouse, un teclado, un dispositivo táctil o un lector de pantalla, pero hay aspectos para mejorar. Puedes mejorar la experiencia extendiendo este patrón básico con información y funcionalidad adicionales.

Puedes hacer lo siguiente:

  • Destaca la página activa.
  • Anunciar la cantidad de elementos a los usuarios de lectores de pantalla
  • Agrega un punto de referencia y permite que los usuarios de lectores de pantalla accedan a la navegación directamente con un acceso directo.
  • Oculta la navegación en viewports estrechos.
  • Mejora el estilo de enfoque.

Destacar la página activa

Para destacar la página activa, puedes agregar una clase al vínculo correspondiente.

<a href="/about-us" class="active-page">About us</a>

El problema con este enfoque es que transmite la información qué vínculo está activo meramente visualmente. Un usuario ciego de lector de pantalla no podía distinguir entre la página activa y otras. Por suerte, el estándar de Aplicaciones de Internet enriquecidas accesibles (ARIA) ofrece una manera de comunicar esta información también de manera semántica. Utiliza el atributo y el valor aria-current=&quot;page&quot; en lugar de una clase.

aria-current (estado) indica el elemento que representa el elemento actual dentro de un contenedor o conjunto de elementos relacionados. Es un token de página que se usa para indicar un vínculo dentro de un conjunto de vínculos de paginación, en el que el vínculo tiene un estilo visual para representar la página que se muestra actualmente. [Aplicaciones de Internet Enriquecidas y accesibles (WAI-ARIA) 1.1](https://www.w3.org/TR/wai-aria/#aria-current)

Con el atributo adicional, un lector de pantalla ahora anuncia algo como "página actual, vínculo, Acerca de nosotros" en lugar de "enlace, Acerca de nosotros".

<a href="/about-us" aria-current="page" class="active-page">About us</a>

Un efecto secundario conveniente es que puedes usar el atributo para seleccionar el vínculo activo en CSS, lo que hace que la clase active-page quede obsoleta.

<a href="/home">Home</a>
<a href="/about-us" aria-current="page">About us</a>
<a href="/pricing">Pricing</a>
<a href="/contact">Contact</a>
/* Change border-color and color for the active page */
[aria-current="page"] {
  --border-color: var(--color-highlight);
  --text-color: var(--color-highlight);
}
Consulta el Paso 2: Destaca la página activa en CodePen.

Anuncia la cantidad de elementos

Al mirar la navegación, los usuarios videntes se dan cuenta de que solo contiene cuatro enlaces. Un usuario de lector de pantalla ciego no puede obtener esta información tan rápido. Es posible que deban trabajar con toda la lista de vínculos. Esto podría no ser un problema si la lista es corta como en este ejemplo, pero, si contiene 40 vínculos, esta tarea puede resultar engorrosa. Si el usuario de un lector de pantalla sabe de antemano que la navegación contiene muchos vínculos, es posible que decida usar una forma de navegación diferente y más eficiente, como la búsqueda en el sitio.
Una buena manera de comunicar la cantidad de elementos por adelantado es unir cada vínculo en un elemento de lista (<li>), anidado en una lista no ordenada (<ul>).

<ul>
  <li>
     <a href="/home">Home</a>
  </li>
  <li>
    <a href="/about-us" aria-current="page">About us</a>
  </li>
  <li>
    <a href="/pricing">Pricing</a>
  </li>
  <li>
    <a href="/contact">Contact</a>
  </li>
</ul>

Cuando un usuario de lector de pantalla encuentre la lista, el software anunciará algo como "lista, 4 elementos".

Esta es una demostración de la navegación que se usa con el lector de pantalla NVDA en Windows.

Ahora debes adaptar el estilo para que se vea como antes.

/* Remove the default list styling and create a flexible layout for the list */
ul {
  display: flex;
  flex-wrap: wrap;
  gap: 1rem;
  list-style: none;
  margin: 0;
  padding: 0;
}

/* Basic link styling */
a {
  --text-color: var(--color-shades-dark);

  border-block-end: 3px solid var(--border-color, transparent);
  color: var(--text-color);
  padding: 0.1rem;
  text-decoration: none;
}

Usar listas puede tener muchas ventajas para los usuarios de lectores de pantalla:

  • Pueden obtener la cantidad total de elementos antes de interactuar con ellos.
  • Pueden usar combinaciones de teclas para ir de un elemento a otro.
  • Pueden usar combinaciones de teclas para pasar de una lista a otra.
  • El lector de pantalla puede anunciar el índice del elemento actual (por ejemplo, "elemento de la lista, dos de cuatro").

Además, si la página se presenta sin CSS, la lista muestra los vínculos como un grupo coherente de elementos en lugar de solo una pila de vínculos.

Un detalle notable sobre VoiceOver en Safari es que pierdes todas estas ventajas cuando configuras list-style: none. Se diseñó de este modo. El equipo de WebKit decidió quitar la semántica de lista cuando una lista no parece una. Según la complejidad de tu navegación, esto puede o no ser un problema. Por un lado, la navegación todavía se puede utilizar y solo afecta a VoiceOver en Safari. VoiceOver con Chrome o Firefox aún anuncia la cantidad de elementos, así como otros lectores de pantalla, como NVDA. Por otro lado, la información semántica podría ser muy útil en algunas situaciones. Para tomar esa decisión, debes probar la navegación con usuarios reales de lectores de pantalla y obtener sus comentarios. Si decides que necesitas que VoiceOver en Safari se comporte como el resto de los lectores de pantalla, puedes establecer el rol de la lista ARIA de forma explícita en <ul> para solucionar el problema. De esta manera, se revierte el comportamiento al estado anterior a la eliminación del diseño de lista. Visualmente, la lista se ve igual.

<ul role="list">
  <li>
     <a href="/home">Home</a>
  </li>
  ...
</ul>
Consulta el Paso 3: Anuncia la cantidad de elementos en CodePen.

Agregar un punto de referencia

Con poco esfuerzo, realizaste grandes mejoras para los usuarios de lectores de pantalla, pero hay una cosa más que puedes hacer. Semánticamente, la navegación es solo una lista de vínculos, y resulta difícil determinar si esa lista específica es la navegación principal del sitio web. Puedes convertir esta lista común en una lista de navegación si unes <ul> a un elemento <nav>.

Usar el elemento <nav> tiene varias ventajas. En particular, un lector de pantalla anuncia algo como "navegación", cuando el usuario interactúa con él y agrega un punto de referencia a la página. Los puntos de referencia son regiones especiales de la página, como <header>, <footer> o <main>, a las que puede saltar un lector de pantalla. Tener puntos de referencia en una página puede ser útil, ya que permite que los usuarios de lectores de pantalla accedan directamente a regiones importantes de la página sin tener que interactuar con el resto. Por ejemplo, puedes presionar la tecla D en NVDA para saltar de un punto de referencia a otro. En Voice Over, puedes presionar VO + U para usar el rotor y que se muestren todos los puntos de referencia de la página.

Una lista de cuatro puntos de referencia: banner, navegación, principal e información del contenido.
Rotor en VoiceOver que enumera todos los puntos de referencia en una página.

En esta lista, ves 4 puntos de referencia: banner es el elemento <header>, navigation es el <nav>, main elemento <main> e información del contenido es <footer>. Esta lista no debe ser demasiado larga. Solo debes marcar partes críticas de tu IU como puntos de referencia, como la búsqueda en el sitio, una navegación local o una paginación.

Si tienes una navegación para todo el sitio, una navegación local para la página y una paginación en una sola página, también puedes tener 3 elementos <nav>. Eso está bien, pero ahora hay tres puntos de referencia de navegación y, semánticamente, todos se ven iguales. Es difícil diferenciarlos, a menos que conozcas muy bien la estructura de la página.

Imagen que muestra tres puntos de referencia que dicen &quot;navegación&quot;.
El rotor de VoiceOver muestra tres puntos de referencia de navegación sin etiquetas.

Para que se distingan, debes etiquetarlos con aria-labelledby o aria-label.

<nav aria-label="Main">
    <ul>
      <li>
         <a href="/home">Home</a>
      </li>
      ...
  </ul>
</nav>
...
<nav aria-label="Select page">
    <ul>
      <li>
         <a href="/page-1">1</a>
      </li>
      ...
    </ul>
</nav>

Si la etiqueta que elegiste ya existe en alguna parte de la página, puedes usar aria-labelledby y hacer referencia a la etiqueta existente con el atributo id.

<nav aria-labelledby="pagination_heading">
  <h2 id="pagination_heading">Select a page</h2>
  <ul>
    <li>
       <a href="/page-1">1</a>
    </li>
    ...
  </ul>
</nav>

Una etiqueta concisa es suficiente, no te compliques demasiado. Omite expresiones como "navegación" o "menú" porque el lector de pantalla ya proporciona esta información a los usuarios.

Puntos de referencia
VoiceOver en el que se enumeran los puntos de referencia: “banner”, “navegación principal”, “principal”, “navegación de página” y “selección de navegación de página”. e “información sobre el contenido”.
Consulta el Paso 4: Agrega un punto de referencia en CodePen.

Cómo ocultar la navegación en viewports estrechos

Personalmente, no me gusta mucho ocultar la navegación principal en viewports estrechos, pero si la lista de vínculos es demasiado larga, no hay forma de evitarlo. Si ese es el caso, en lugar de la lista, los usuarios verán un botón con la etiqueta "Menú". o un ícono de hamburguesa o una combinación. Cuando haces clic en el botón, se muestra y se oculta la lista. Si conoces conceptos básicos de JavaScript y CSS, es una tarea viable, pero hay varias cosas en términos de UX y accesibilidad que debes ocuparte.

  • Debes ocultar la lista de forma accesible.
  • La navegación debe ser accesible con el teclado.
  • La navegación debe comunicar si es visible o no.

Agrega un botón de hamburguesa

Dado que estás siguiendo el principio de mejora progresiva, debes asegurarte de que la navegación siga funcionando y tenga sentido incluso con JavaScript desactivado.
Lo primero que necesita tu navegación es un botón de hamburguesa. Se crea en HTML en un elemento de plantilla, se clona en JavaScript y se agrega a la navegación.

Una página en la que se muestra un botón de hamburguesa.
Resultado: En lugar de vínculos, la navegación muestra un botón de hamburguesa en ventanas gráficas estrechas.
<nav id="mainnav">
  ...
</nav>

<template id="burger-template">
  <button type="button" aria-expanded="false" aria-label="Menu" aria-controls="mainnav">
    <svg width="24" height="24" aria-hidden="true">
      <path d="M3 18h18v-2H3v2zm0-5h18v-2H3v2zm0-7v2h18V6H3z">
    </svg>
  </button>
</template>
  1. El atributo aria-expanded le indica al software del lector de pantalla si se expande o no el elemento que controla el botón.
  2. aria-label le da al botón un nombre de accesibilidad, una alternativa de texto para el ícono de hamburguesa.
  3. Ocultas el <svg> de la tecnología de accesibilidad con aria-hidden porque ya tiene una etiqueta de texto proporcionada por aria-label.
  4. aria-controls le indica a la tecnología de accesibilidad, que admite el atributo (por ejemplo, JAWS), qué elemento controla el botón.
const nav = document.querySelector('#mainnav')
const list = nav.querySelector('ul');
const burgerClone = document.querySelector('#burger-template').content.cloneNode(true);
const button = burgerClone.querySelector('button');

// Toggle aria-expanded attribute
button.addEventListener('click', e => {
  // aria-expanded="true" signals that the menu is currently open
  const isOpen = button.getAttribute('aria-expanded') === "true"
  button.setAttribute('aria-expanded', !isOpen);
});

// Hide list on keydown Escape
nav.addEventListener('keyup', e => {
  if (e.code === 'Escape') {
    button.setAttribute('aria-expanded', false);
  }
});

// Add the button to the page
nav.insertBefore(burgerClone, list);
  1. Resulta conveniente que los usuarios puedan cerrar la navegación cuando lo deseen, por ejemplo, presionando la tecla Escape.
  2. Es importante usar insertBefore en lugar de appendChild, ya que el botón debería ser el primer elemento de tu navegación. Si el usuario de un teclado o lector de pantalla presiona Tab después de hacer clic en el botón, espera enfocar el primer elemento de la lista. Si el botón aparece después de la lista, ese no sería el caso.

A continuación, restableces el estilo predeterminado del botón y te aseguras de que solo sea visible en viewports estrechos.

@media (min-width: 48em) {
  nav {
    --nav-button-display: none;
  }
}

/* Reset button styling */
button {
  all: unset;
  display: var(--nav-button-display, flex);
}
Consulta el Paso 5: Agrega un botón de hamburguesa en CodePen.

Cómo ocultar la lista

Antes de ocultar la lista, coloca la navegación y la lista, y aplica un estilo, de modo que el diseño esté optimizado para viewports estrechos, pero aun así se vea bien en pantallas más grandes.
Primero, quita el elemento <nav> del flujo natural de la página y colócalo en la esquina superior final del viewport.

@media (min-width: 48em) {
  nav {
    --nav-button-display: none;
    --nav-position: static;
  }
}

nav {
  position: var(--nav-position, fixed);
  inset-block-start: 1rem;
  inset-inline-end: 1rem;
}

A continuación, agrega una nueva propiedad personalizada (—-nav-list-layout) para cambiar el diseño en viewports estrechos. El diseño es columna de forma predeterminada y cambia a fila en pantallas más grandes.

@media (min-width: 48em) {
  nav {
    --nav-button-display: none;
    --nav-position: static;
  }

  ul {
    --nav-list-layout: row;
  }
}

ul {
  display: flex;
  flex-direction: var(--nav-list-layout, column);
  flex-wrap: wrap;
  gap: 1rem;
  list-style: none;
  margin: 0;
  padding: 0;
}

Tu navegación debería verse de la siguiente manera en viewports estrechos.

La página que muestra la lista de navegación y el botón de hamburguesa.
Tanto el botón de hamburguesa como la lista se colocan en la esquina superior del extremo superior del viewport.

Es obvio que la lista necesita CSS. Lo moveremos hasta la esquina superior, haremos que ocupe toda la pantalla verticalmente y aplicaremos una background-color y una box-shadow.

@media (min-width: 48em) {
  nav {
    --nav-button-display: none;
    --nav-position: static;
  }
  
  ul {
    --nav-list-layout: row;
    --nav-list-position: static;
    --nav-list-padding: 0;
    --nav-list-height: auto;
    --nav-list-width: 100%;
    --nav-list-shadow: none;
  }
}

ul {
  background: rgb(255, 255, 255);
  box-shadow: var(--nav-list-shadow, -5px 0 11px 0 rgb(0 0 0 / 0.2));
  display: flex;
  flex-direction: var(--nav-list-layout, column);
  flex-wrap: wrap;
  gap: 1rem;
  height: var(--nav-list-height, 100vh);
  list-style: none;
  margin: 0;
  padding: var(--nav-list-padding, 2rem);
  position: var(--nav-list-position, fixed);
  inset-block-start: 0; /* Logical property. Equivalent to top: 0; */
  inset-inline-end: 0; /* Logical property. Equivalent to right: 0; */
  width: var(--nav-list-width, min(22rem, 100vw));
}

button {
  all: unset;
  display: var(--nav-button-display, flex);
  position: relative;
  z-index: 1;
}

La lista debería verse así en viewports estrechos, más como una barra lateral que una simple lista.

Se abre la lista de navegación.

Por último, esconde la lista, solo muéstrala cuando el usuario haga clic en el botón una vez y ocúltala cuando vuelva a hacer clic. Es importante ocultar solo la lista y no toda la navegación, ya que ocultarla también implicaría ocultar un punto de referencia importante.

Anteriormente, agregaste un evento de clic al botón para activar o desactivar el valor del atributo aria-expanded. Puedes usar esa información como una condición para mostrar y ocultar la lista en CSS.

@media (min-width: 48em) {
  ul {
    --nav-list-visibility: visible;
  }
}

ul {
  visibility: var(--nav-list-visibility, visible);
}

/* Hide the list on narrow viewports, if it comes after an element with
   aria-expanded set to "false". */
[aria-expanded="false"] + ul {
  visibility: var(--nav-list-visibility, hidden);
}

Es importante usar una declaración de propiedad como visibility: hidden o display: none, en lugar de opacity: 0 o translateX(100%), para ocultar la lista. Estas propiedades garantizan que los vínculos no se puedan enfocar cuando la navegación esté oculta. Si usas opacity o translate, se quitará el contenido de manera visual, de manera que los vínculos sean invisibles y se pueda acceder a ellos con el teclado, lo que sería confuso y frustrante. Si usas visibility o display, se ocultarán visualmente y serán inaccesibles, por lo que se ocultarán para todos los usuarios.

Consulta el Paso 6: Oculta la lista.

Cómo animar la lista

Si te preguntas por qué usar visibility: hidden; en lugar de display: none;, es porque puedes animar la visibilidad. Solo tiene dos estados, hidden y visible, pero puedes combinarlo con otra propiedad, como transform o opacity, para crear un efecto de deslizamiento o fundido de entrada. Eso no funcionaría con display: none porque la propiedad display no puede animarse.

El siguiente CSS realiza la transición de opacity para crear un efecto de atenuación de entrada y de salida.

ul {
  transition: opacity 0.6s linear, visibility 0.3s linear;
  visibility: var(--nav-list-visibility, visible);
}

[aria-expanded="false"] + ul {
  opacity: 0;
  visibility: var(--nav-list-visibility, hidden);
}

En cambio, si deseas animar el movimiento, deberías considerar unir la propiedad transition en una consulta de medios prefers-reduced-motion porque las animaciones pueden provocar náuseas, mareos y dolores de cabeza en algunos usuarios.

ul {
  visibility: var(--nav-list-visibility, visible);
}

@media (prefers-reduced-motion: no-preference) {
  ul {
    transition: transform 0.6s cubic-bezier(.68,-0.55,.27,1.55), visibility 0.3s linear;
  }
}

[aria-expanded="false"] + ul {
  transform: var(--nav-list-transform, translateX(100%));
  visibility: var(--nav-list-visibility, hidden);
}

De este modo, se garantiza que solo las personas que no tienen preferencia por el movimiento reducido vean la animación.

Consulta el Paso 7: Anima la lista en CodePen.

Mejora el estilo de enfoque

Los usuarios del teclado dependen de los estilos de enfoque de los elementos para la orientación y la navegación en una página. Los estilos de enfoque predeterminados son mejores que no usar estilos de enfoque (lo cual sucede si estableces outline: none), pero tener estilos de enfoque personalizados más claramente visibles mejora la experiencia del usuario.

Así se verán los estilos de enfoque predeterminados en el vínculo en Chrome 103.

Un contorno azul de 2 píxeles alrededor de un vínculo enfocado en Chrome 103.

Puedes mejorar eso proporcionando tus propios estilos en tus propios colores. Si usas :focus-visible en lugar de :focus, permites que el navegador decida cuándo es apropiado mostrar estilos de enfoque. Los estilos de :focus estarán visibles para todos los usuarios (mouse, teclado y táctil), independientemente de si los necesitan o no. Con :focus-visible, el navegador utiliza una heurística interna para decidir si se mostrarán solo a los usuarios del teclado o a todos.

/* Remove the default :focus outline */
*:focus {
  outline: none;
}

/* Show a custom outline on :focus-visible */
*:focus-visible {
  outline: 2px solid var(--color-shades-dark);
  outline-offset: 4px;
}

Compatibilidad del navegador con :focus-visible

Navegadores compatibles

  • Chrome: 86.
  • Borde: 86.
  • Firefox: 85.
  • Safari: 15.4.

Origen

Contorno oscuro de 2 px claramente visible con espacio interior.

Existen diferentes maneras de destacar los elementos cuando están enfocados. Se recomienda usar la propiedad outline porque no interrumpe el diseño, lo que podría ocurrir con border, y funciona bien con el modo de contraste alto en Windows. Las propiedades que no funcionan bien son background-color o box-shadow, ya que es posible que no se muestren con una configuración de contraste personalizada.

Un sitio con un fondo oscuro destacado en púrpura
Consulta el Paso 8: Mejora los estilos de enfoque en CodePen.

¡Felicitaciones! Creaste una navegación principal mejorada de forma progresiva, enriquecida desde el punto de vista semántico, accesible y optimizada para dispositivos móviles.

Siempre hay algo que se puede mejorar, por ejemplo:

  • Puedes considerar capturar el enfoque de la navegación o hacer que el resto de la página sea inerte en viewports estrechos.
  • Puedes agregar un vínculo para omitir en la parte superior de la página para permitir que los usuarios del teclado omitan la navegación.

Si recuerdas cómo empezó este artículo, con el objetivo de que la solución "no debería ser ni demasiado simple ni demasiado complicada", ahí es donde estamos ahora. Sin embargo, es posible realizar demasiados diseños para una navegación.

Existe una diferencia clara entre las navegaciones y los menús. Las navegaciones son colecciones de vínculos para navegar por documentos relacionados. Los menús son colecciones de acciones para realizar en un documento. A veces, estas tareas se superponen. Es posible que tengas una navegación que también incluya un botón que realice una acción, como abrir una ventana modal, o que tengas un menú en el que una acción navegue a otra página, como una página de ayuda. Cuando ese sea el caso, es importante que no combines los roles de ARIA, sino que identifiques el objetivo principal de tu componente y elijas el lenguaje de marcado y los roles según corresponda.

El elemento <nav> tiene un rol de ARIA implícito de navegación que es suficiente para comunicar que el elemento es una navegación, pero a menudo ves que los sitios también usan menús, barras y elementos de menú. Dado que a veces usamos estos términos de forma indistinta, pensar que combinarlos para mejorar la experiencia de los usuarios de lectores de pantalla puede tener sentido. Antes de aprender por qué no suele ser el caso, veamos la definición oficial de estos roles.

El rol de Navigation

Es una colección de elementos de navegación (por lo general, vínculos) para navegar por el documento o documentos relacionados.

navigation (role) WAI-ARIA 1.1

El rol del menú

Un menú suele ser una lista de acciones o funciones comunes que el usuario puede invocar. El rol del menú es adecuado cuando una lista de elementos del menú se presenta de manera similar a un menú en una aplicación de escritorio.

menú (función) WAI-ARIA 1.1

El rol de la barra de menú

Es una presentación de un menú que suele verse y se presenta horizontalmente. El rol de la barra de menú se usa para crear una barra de menú similar a las que se encuentran en las aplicaciones de escritorio de Windows, Mac y Gnome. Se usa una barra de menú para crear un conjunto coherente de comandos de uso frecuente. Los autores deben asegurarse de que la interacción de la barra de menú sea similar a la interacción típica de la barra de menú en una interfaz gráfica de usuario de escritorio.

barra de menú (función) WAI-ARIA 1.1

El rol Menuitem

Es una opción en un conjunto de opciones que se incluyen en un menú o una barra de menú.

menuitem (role) WAI-ARIA 1.1

La especificación aquí es muy clara. Usa la navegación para navegar por el documento o los documentos relacionados y el menú solo para una lista de acciones o funciones similares a los menús en aplicaciones de escritorio. Si no estás creando el próximo documento de Google, es probable que no necesites ninguno de los roles del menú para la navegación principal.

¿Cuándo es apropiado un menú?

El uso principal de los elementos del menú no es la navegación, sino la realización de acciones. Digamos que tienes una lista o tabla de datos y los usuarios pueden realizar determinadas acciones en cada elemento de la lista. Podrías agregar un botón a cada fila y mostrar las acciones cuando los usuarios hagan clic en el botón.

<ul>
  <li>
    Product 1

    <button aria-expanded="false" aria-controls="options1">Edit</button>

    <div role="menu" id="options1">
      <button role="menuitem">
        Duplicate
      </button>
      <button role="menuitem">
        Delete
      </button>
      <button role="menuitem">
        Disable
      </button>
    </div>
  </li>
  <li>
    Product 2
    ...
  </li>
</ul>

Implicaciones de usar los roles del menú

Es muy importante usar estos roles del menú con prudencia, ya que muchas pueden salir mal.

Los menús esperan una cierta estructura de DOM. menuitem debe ser un elemento secundario directo de menu. El siguiente código podría romper el comportamiento semántico:

 <!-- Wrong, don't do this -->
<ul role="menu">
  <li>
    <a href="#" role="menuitem">Item 1</a>
  </li>
</ul>

Los usuarios expertos esperan que ciertas combinaciones de teclas funcionen con menús y barras de menú. Según la Guía de prácticas de creación de ARIA (APG), esto incluye lo siguiente:

  • Intro y la barra espaciadora para seleccionar elementos del menú.
  • Usa las teclas de flecha en todas las direcciones para navegar entre elementos.
  • Las teclas Inicio y Fin para enfocar el primer o último elemento, respectivamente.
  • a-z para enfocar el siguiente elemento de menú con una etiqueta que comienza con el carácter escrito.
  • Presiona Esc para cerrar el menú.

Si un lector de pantalla detecta un menú, el software puede cambiar automáticamente el modo de navegación, lo que habilita el uso de las combinaciones de teclas mencionadas anteriormente. Es posible que los usuarios sin experiencia del lector de pantalla no puedan usar el menú porque no conocen estas combinaciones de teclas ni cómo usarlas.

Lo mismo sucede con los usuarios de teclado que podrían esperar que puedan usar Mayúsculas y Mayúsculas + Tab.

Hay mucho que considerar a la hora de crear menús y barras de menús, y si es apropiado usarlos en primer lugar. Al crear un sitio web típico, todo lo que necesitas es el elemento de navegación con una lista y vínculos. Esto también incluye aplicaciones de una sola página (SPA) o aplicaciones web. La pila subyacente no importa. A menos que estés compilando algo muy parecido a una aplicación de escritorio, evita los roles del menú.

Recursos adicionales

Hero image de Mick Haupt