Design System
Guía completa de tokens de diseño, tipografía y componentes del sistema visual "Engineering Notebook". Todos los valores están disponibles como CSS custom properties.
Color Tokens
La paleta del sistema se organiza en cinco grupos funcionales: Base (fondos oscuros),
Jerarquía de texto, Acentos neón (estados UI), Syntax (resaltado de código)
y Bordes. Cada token se expone como CSS custom property en :root.
Base (The Void)
Fondos del sistema. El color primario --bg-void no es negro puro; contiene un tinte
cálido imperceptible que reduce la fatiga visual en sesiones largas de lectura.
Jerarquía de Texto
Cinco niveles de luminosidad para establecer jerarquía sin recurrir a tamaño o peso tipográfico.
Acentos Neón (Estados UI)
Colores de alta saturación reservados para estados interactivos y comunicación semántica. Cada acento tiene un significado funcional asociado.
| Token | Hex | Estado UI | Significado Semántico |
|---|---|---|---|
--accent-yellow | #fde047 | Focus / Active | Atención primaria, subrayador, advertencias |
--accent-pink | #f472b6 | Hover | Elementos interactivos secundarios, UI decorativa |
--accent-cyan | #22d3ee | Link | Enlaces, arquitectura, flujo de datos |
--accent-green | #86efac | Success | Validaciones, terminal success, “copiado” |
--accent-red | #fb7185 | Error | Errores de compilación, alertas críticas |
--accent-orange | #fb923c | Warning | Advertencias no críticas, deprecaciones |
--accent-purple | #a78bfa | Decorative | Categorías, badges, tags de contenido |
Syntax & Code
Tokens para resaltado de sintaxis en bloques de código. El esquema está basado en una variante personalizada de Dracula/Monokai adaptada al tema oscuro.
Bordes
Variables CSS completas
:root {
/* Base (The Void) */
--bg-void: #0f0f11;
--bg-surface: #18181b;
--bg-surface-hover: #27272a;
--bg-overlay: rgba(15, 15, 17, 0.8);
--bg-code: #0a0a0a;
/* Text Hierarchy */
--text-primary: #f4f4f5;
--text-secondary: #a1a1aa;
--text-muted: #71717a;
--text-dim: #52525b;
--text-ghost: #3f3f46;
/* Neon Accents */
--accent-yellow: #fde047;
--accent-pink: #f472b6;
--accent-cyan: #22d3ee;
--accent-green: #86efac;
--accent-red: #fb7185;
--accent-orange: #fb923c;
--accent-purple: #a78bfa;
/* Syntax Highlighting */
--code-keyword: #c678dd;
--code-func: #61afef;
--code-string: #98c379;
--code-comment: #71717a;
/* Borders */
--border-default: #27272a;
--border-subtle: #1f1f23;
--border-strong: #3f3f46;
} Uso de acentos
Los colores de acento neón deben usarse exclusivamente sobre fondos oscuros
(—bg-void o —bg-surface). Nunca combines dos acentos
como fondo + texto entre sí.
Typography
El sistema tipográfico utiliza cuatro familias con roles claramente diferenciados. La combinación de fuentes técnicas (mono) y manuscritas (hand) refuerza la estética de “cuaderno de ingeniero”.
Familias tipográficas
| Fuente | Variable CSS | Uso principal | Ejemplo |
|---|---|---|---|
Inter | --font-sans | Body text, headings de documentación | The quick brown fox |
JetBrains Mono | --font-mono | Código, labels técnicos, rutas, botones | const x = 42; |
Patrick Hand | --font-hand | Notas al margen, tooltips, comentarios de código | Nota al margen! |
Permanent Marker | --font-marker | Títulos de sección principales del hero | HEADER |
Escala de tipos (Type Scale)
Configuración base: 16px con line-height de 1.6 para lectura prolongada.
| Elemento | Fuente | Tamaño | Peso | Tracking | Line Height | Uso |
|---|---|---|---|---|---|---|
| H1 Hero | JetBrains Mono | 48px — 60px | 700 | -0.02em | 1.1 | Título principal del hero. Estilo técnico impactante. |
| H1 Doc | Inter | 36px | 700 | -0.01em | 1.2 | Título de página de documentación. |
| H2 | Inter | 24px | 600 | -0.01em | 1.3 | Secciones principales. Borde inferior opcional. |
| H3 | Inter | 20px | 600 | Normal | 1.4 | Sub-secciones dentro de un H2. |
| Body | Inter | 16px | 300 / 400 | Normal | 1.6 | Texto de lectura. Color —text-primary. |
| Mono Text | JetBrains Mono | 14px | 400 | Normal | 1.5 | Rutas, variables inline, fragmentos de código. |
| Handwritten | Patrick Hand | 18px — 20px | 400 | 0.02em | 1.4 | Notas al margen, tooltips, comentarios visuales. |
| Label / Tag | JetBrains Mono | 12px | 500 | 0.05em | 1.2 | Uppercase. Tags, badges, categorías. |
Estrategia de carga de fuentes
Todas las fuentes se cargan desde Google Fonts con display=swap para evitar
FOIT (Flash of Invisible Text). El orden de carga prioriza Inter y JetBrains Mono
como fuentes críticas.
<link rel="preconnect" href="https://fonts.googleapis.com" />
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@300;400;500;600;700&family=JetBrains+Mono:wght@400;500;700&family=Patrick+Hand&family=Permanent+Marker&display=swap" rel="stylesheet" /> :root {
--font-sans: 'Inter', system-ui, -apple-system, sans-serif;
--font-mono: 'JetBrains Mono', 'Fira Code', monospace;
--font-hand: 'Patrick Hand', cursive;
--font-marker: 'Permanent Marker', cursive;
} Fallbacks del sistema
Cada variable incluye fallbacks del sistema operativo. Si Google Fonts no carga,
la experiencia se degrada elegantemente a system-ui para sans-serif
y monospace para código.
Spacing & Layout
El sistema de espaciado utiliza una escala de 8 niveles basada en rem para
garantizar consistencia y adaptabilidad a diferentes tamaños de pantalla.
Escala de espaciado
| Token | Valor | Píxeles (base 16) | Uso típico |
|---|---|---|---|
--space-xs | 0.25rem | 4px | Separación entre íconos y texto, padding de badges. |
--space-sm | 0.5rem | 8px | Gaps internos en botones, padding de tags. |
--space-md | 1rem | 16px | Padding estándar de componentes, gap de grids. |
--space-lg | 1.5rem | 24px | Separación entre bloques de contenido. |
--space-xl | 2rem | 32px | Margen entre secciones menores. |
--space-2xl | 3rem | 48px | Separación entre secciones de documentación. |
--space-3xl | 4rem | 64px | Padding vertical de secciones principales. |
--space-4xl | 6rem | 96px | Separación entre el hero y el contenido. |
:root {
--space-xs: 0.25rem; /* 4px */
--space-sm: 0.5rem; /* 8px */
--space-md: 1rem; /* 16px */
--space-lg: 1.5rem; /* 24px */
--space-xl: 2rem; /* 32px */
--space-2xl: 3rem; /* 48px */
--space-3xl: 4rem; /* 64px */
--space-4xl: 6rem; /* 96px */
} Contenedor principal
El contenido de documentación se limita a un ancho máximo para mantener una longitud de línea óptima (65-80 caracteres).
.doc-container {
max-width: 72rem; /* 1152px */
margin-inline: auto;
padding-inline: var(--space-md);
} Breakpoints responsivos
El sistema utiliza un enfoque mobile-first con cuatro breakpoints principales.
| Nombre | Min-width | Uso |
|---|---|---|
sm | 640px | Teléfonos en horizontal, ajustes de texto y padding. |
md | 768px | Tablets. Se activa la sidebar de navegación. |
lg | 1024px | Desktop. Layout de documentación completo (sidebar + contenido). |
xl | 1280px | Desktop amplio. Se muestra el Table of Contents a la derecha. |
/* Ejemplo de uso de breakpoints */
.doc-layout {
display: grid;
grid-template-columns: 1fr;
gap: var(--space-lg);
}
@media (min-width: 768px) {
.doc-layout {
grid-template-columns: 260px 1fr;
}
}
@media (min-width: 1280px) {
.doc-layout {
grid-template-columns: 260px 1fr 240px;
}
} Sistema de grid
El grid interno para tarjetas y galerías utiliza auto-fill con tamaño mínimo
configurable para adaptarse automáticamente al espacio disponible.
.doc-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(280px, 1fr));
gap: var(--space-lg);
} Unidades relativas
Siempre usa rem para spacing y em para ajustes
proporcionales al tamaño de texto. Evita px fijos excepto en
bordes y sombras.
Shadows & Depth
En el sistema “Engineering Notebook” no se utilizan sombras difusas (con blur). Todas las sombras son duras — desplazamiento sólido sin difuminado — para reforzar la estética de pegatina, recorte de papel o sticker sobre el cuaderno.
Escala de sombras
| Token | Valor CSS | Uso |
|---|---|---|
--shadow-sm | 2px 2px 0 0 #27272a | Botones pequeños, tags, badges. |
--shadow-md | 4px 4px 0 0 #27272a | Tarjetas, botones primarios, callouts. |
--shadow-lg | 8px 8px 0 0 #000000 | Modales, ventanas flotantes, elementos emergentes. |
--shadow-neon | 0 0 10px 0 rgba(34,211,238,0.3) | Elementos “encendidos” o activos en diagramas. Única excepción con blur. |
--shadow-white | 4px 4px 0 0 #ffffff | Botones sobre fondo oscuro. Alto contraste visual. |
:root {
--shadow-sm: 2px 2px 0px 0px #27272a;
--shadow-md: 4px 4px 0px 0px #27272a;
--shadow-lg: 8px 8px 0px 0px #000000;
--shadow-neon: 0px 0px 10px 0px rgba(34, 211, 238, 0.3);
--shadow-white: 4px 4px 0px 0px #ffffff;
} Radios de borde (Border Radius)
| Token | Valor | Uso |
|---|---|---|
--radius-sm | 2px | Botones técnicos, inputs, elementos rectangulares precisos. |
--radius-md | 4px | Tarjetas estándar, contenedores de código. |
--radius-lg | 8px | Modales, contenedores grandes. |
--radius-sketch | 2px 255px 3px 25px / 255px 5px 225px 5px | Efecto mano alzada. Para elementos con estilo “dibujado”. |
:root {
--radius-sm: 2px;
--radius-md: 4px;
--radius-lg: 8px;
--radius-sketch: 2px 255px 3px 25px / 255px 5px 225px 5px;
}
/* Ejemplo: tarjeta con efecto sketch */
.card-sketch {
border: 2px solid var(--border-default);
border-radius: var(--radius-sketch);
box-shadow: var(--shadow-md);
} Sin blur
La única excepción a la regla “sin blur” es —shadow-neon, que se
reserva exclusivamente para indicadores activos en diagramas interactivos.
Nunca apliques blur a sombras de componentes estándar.
Components
Especificaciones detalladas de cada componente del sistema de diseño. Incluyen estructura, estilos base y estados interactivos.
Botón Primario “Terminal”
Botón principal con estética de terminal. Fondo sólido con borde de alto contraste.
.btn-terminal {
display: inline-flex;
align-items: center;
gap: var(--space-sm);
padding: var(--space-sm) var(--space-lg);
font-family: var(--font-mono);
font-size: 14px;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.05em;
color: #000000;
background: #ffffff;
border: 2px solid #ffffff;
border-radius: var(--radius-sm);
box-shadow: var(--shadow-white);
cursor: pointer;
transition: all 0.2s ease;
}
/* Hover: desplazamiento con sombra */
.btn-terminal:hover {
transform: translate(-2px, -2px);
box-shadow: 6px 6px 0px 0px #ffffff;
}
/* Active: efecto de pulsación mecánica */
.btn-terminal:active {
transform: translate(0, 0);
box-shadow: none;
}
/* Focus visible: outline de accesibilidad */
.btn-terminal:focus-visible {
outline: 2px solid var(--accent-yellow);
outline-offset: 2px;
} Botón Secundario “Sketch”
Botón transparente con borde estilo dibujado a mano. Menor prominencia visual.
.btn-sketch {
display: inline-flex;
align-items: center;
gap: var(--space-sm);
padding: var(--space-sm) var(--space-lg);
font-family: var(--font-mono);
font-size: 14px;
font-weight: 500;
color: var(--text-primary);
background: transparent;
border: 2px solid var(--border-default);
border-radius: var(--radius-sketch);
cursor: pointer;
transition: all 0.2s ease;
}
/* Hover: borde neón y leve rotación */
.btn-sketch:hover {
border-color: var(--accent-yellow);
color: var(--accent-yellow);
transform: rotate(-0.5deg);
}
/* Active */
.btn-sketch:active {
transform: rotate(0deg) scale(0.98);
}
/* Focus visible */
.btn-sketch:focus-visible {
outline: 2px solid var(--accent-yellow);
outline-offset: 2px;
} Input de búsqueda (Ctrl+K)
Campo de búsqueda global con indicador de atajo de teclado.
<div class="search-input">
<svg class="search-input__icon"><!-- lupa --></svg>
<input type="text" placeholder="Buscar documentación..." />
<kbd class="search-input__kbd">Ctrl+K</kbd>
</div> .search-input {
display: flex;
align-items: center;
gap: var(--space-sm);
padding: var(--space-sm) var(--space-md);
background: var(--bg-code);
border: 1px solid var(--border-default);
border-radius: var(--radius-sm);
transition: border-color 0.2s ease;
}
.search-input:focus-within {
border-color: var(--accent-yellow);
outline: none;
}
.search-input input {
flex: 1;
background: transparent;
border: none;
color: var(--text-primary);
font-family: var(--font-sans);
font-size: 14px;
outline: none;
}
.search-input input::placeholder {
font-family: var(--font-hand);
color: var(--text-dim);
}
.search-input__kbd {
padding: 2px 8px;
font-family: var(--font-mono);
font-size: 11px;
color: var(--text-muted);
background: var(--bg-surface);
border: 1px solid var(--border-default);
border-radius: var(--radius-sm);
box-shadow: var(--shadow-sm);
} Callout / Admonición (Efecto cinta adhesiva)
No se usan los típicos cuadros de colores planos. Se aplica el concepto de “cinta adhesiva” (tape) para reforzar la estética de cuaderno.
<div class="callout callout--warning">
<div class="callout__tape"></div>
<div class="callout__content">
<span class="callout__icon"><!-- icono --></span>
<div>
<strong class="callout__title">Advertencia</strong>
<p>Contenido del callout...</p>
</div>
</div>
</div> .callout {
position: relative;
background: #1f1f23;
border-left: 4px solid var(--accent-yellow);
padding: var(--space-md);
margin-top: var(--space-lg);
}
/* Efecto de cinta adhesiva */
.callout__tape {
position: absolute;
top: -12px;
left: 50%;
transform: translateX(-50%) rotate(1deg) skewX(12deg);
width: 96px;
height: 24px;
background: rgba(253, 224, 71, 0.2);
pointer-events: none;
}
.callout__content {
display: flex;
gap: var(--space-sm);
}
/* Variantes por tipo */
.callout--warning { border-left-color: var(--accent-yellow); }
.callout--info { border-left-color: var(--accent-cyan); }
.callout--tip { border-left-color: var(--accent-green); }
.callout--danger { border-left-color: var(--accent-red); } Tablas (Data Grids)
Estilo minimalista sin bordes verticales. Header con fuente monoespaciada.
.doc-table {
width: 100%;
border-collapse: collapse;
}
.doc-table thead {
background: var(--bg-surface);
}
.doc-table th {
padding: var(--space-sm) var(--space-md);
font-family: var(--font-mono);
font-size: 12px;
font-weight: 500;
text-transform: uppercase;
letter-spacing: 0.05em;
color: var(--text-muted);
text-align: left;
border-bottom: 1px solid var(--border-default);
}
.doc-table td {
padding: var(--space-sm) var(--space-md);
font-family: var(--font-sans);
font-size: 14px;
color: var(--text-primary);
border-bottom: 1px dashed var(--border-default);
}
.doc-table tbody tr:hover {
background: var(--bg-surface);
} Bloques de código (con header y botón de copia)
Cada bloque de código incluye un header con el nombre del lenguaje y un botón para copiar el contenido al portapapeles.
<div class="code-block">
<div class="code-block__header">
<span class="code-block__lang">css</span>
<button class="code-block__copy" aria-label="Copiar código">
<!-- icono copiar -->
</button>
</div>
<pre class="code-block__pre"><code>...</code></pre>
</div> .code-block {
background: var(--bg-code);
border: 1px solid var(--border-default);
border-radius: var(--radius-md);
overflow: hidden;
}
.code-block__header {
display: flex;
justify-content: space-between;
align-items: center;
padding: var(--space-xs) var(--space-md);
background: var(--bg-surface);
border-bottom: 1px solid var(--border-default);
}
.code-block__lang {
font-family: var(--font-mono);
font-size: 12px;
color: var(--text-muted);
text-transform: uppercase;
}
.code-block__copy {
padding: var(--space-xs);
color: var(--text-muted);
background: transparent;
border: none;
cursor: pointer;
transition: color 0.2s ease;
}
.code-block__copy:hover {
color: var(--accent-green);
}
.code-block__pre {
padding: var(--space-md);
overflow-x: auto;
font-family: var(--font-mono);
font-size: 14px;
line-height: 1.5;
} Tarjetas de librería (Library Cards)
Tarjetas con rotación sutil que refuerzan la estética de notas pegadas en un cuaderno. Cada tarjeta tiene una rotación pseudo-aleatoria basada en su posición.
.library-card {
position: relative;
background: var(--bg-surface);
border: 2px solid var(--border-default);
border-radius: var(--radius-md);
padding: var(--space-lg);
box-shadow: var(--shadow-md);
transition: all 0.2s ease;
transform: rotate(-0.5deg);
}
.library-card:nth-child(even) {
transform: rotate(0.75deg);
}
.library-card:nth-child(3n) {
transform: rotate(-0.25deg);
}
/* Hover: se eleva y endereza */
.library-card:hover {
transform: rotate(0deg) translate(-2px, -4px);
box-shadow: var(--shadow-lg);
border-color: var(--accent-cyan);
}
/* Label de categoría */
.library-card__category {
display: inline-block;
font-family: var(--font-mono);
font-size: 12px;
font-weight: 500;
text-transform: uppercase;
letter-spacing: 0.05em;
padding: 2px 8px;
border-radius: var(--radius-sm);
} Rotación de tarjetas
La rotación varía entre -0.75deg y +0.75deg. Para un
efecto más orgánico, asigna rotaciones con :nth-child o genera
valores aleatorios con JavaScript al montar el componente.
Animations & Transitions
Las animaciones del sistema son sutiles y funcionales. Nunca decorativas sin propósito.
La transición estándar es 0.2s ease para todos los estados interactivos.
Transición estándar
/* Valor base para todas las interacciones */
.interactive {
transition: all 0.2s ease;
} Hover de tarjetas
Desplazamiento vertical con incremento de sombra. Simula que la tarjeta se “despega” del cuaderno.
.card-hover {
transition: transform 0.2s ease, box-shadow 0.2s ease;
}
.card-hover:hover {
transform: translate(-2px, -4px);
box-shadow: var(--shadow-lg);
} Efecto de pulsación de botón
El botón se “hunde” al hacer clic, eliminando la sombra para simular una pulsación mecánica.
.btn-press {
box-shadow: var(--shadow-md);
transition: all 0.2s ease;
}
.btn-press:hover {
transform: translate(-2px, -2px);
box-shadow: 6px 6px 0px 0px var(--border-default);
}
.btn-press:active {
transform: translate(0, 0);
box-shadow: none;
transition-duration: 0.05s;
} Fade-in de secciones
Animación de entrada para secciones de contenido al hacer scroll. Se activa mediante Intersection Observer.
@keyframes fadeInUp {
from {
opacity: 0;
transform: translateY(16px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
.section-animate {
opacity: 0;
}
.section-animate.is-visible {
animation: fadeInUp 0.4s ease forwards;
} // Intersection Observer para activar animaciones
const observer = new IntersectionObserver(
(entries) => {
entries.forEach((entry) => {
if (entry.isIntersecting) {
entry.target.classList.add('is-visible');
}
});
},
{ threshold: 0.1 }
);
document.querySelectorAll('.section-animate').forEach((el) => {
observer.observe(el);
}); Rotación estilo sticky-note
Rotación sutil para elementos que simulan notas adhesivas o papeles pegados.
El rango va de -0.75deg a +0.75deg.
/* Rotación estática por posición */
.sticky-rotate:nth-child(odd) { transform: rotate(-0.75deg); }
.sticky-rotate:nth-child(even) { transform: rotate(0.75deg); }
.sticky-rotate:nth-child(3n) { transform: rotate(-0.25deg); }
.sticky-rotate:nth-child(4n) { transform: rotate(0.5deg); }
/* Al hacer hover, se endereza */
.sticky-rotate:hover {
transform: rotate(0deg);
transition: transform 0.2s ease;
} Movimiento reducido (Reduced Motion)
Obligatorio respetar la preferencia del usuario para movimiento reducido. Todas las animaciones se desactivan o simplifican.
@media (prefers-reduced-motion: reduce) {
*,
*::before,
*::after {
animation-duration: 0.01ms !important;
animation-iteration-count: 1 !important;
transition-duration: 0.01ms !important;
scroll-behavior: auto !important;
}
.section-animate {
opacity: 1;
}
.sticky-rotate,
.library-card {
transform: none !important;
}
} Rendimiento
Limita las animaciones a propiedades que el navegador puede optimizar con
el compositor GPU: transform y opacity. Evita
animar width, height, margin o
padding.
Grid Background
El fondo de cuadrícula de ingeniería es un elemento esencial de la atmósfera visual del sistema. Simula el papel milimetrado de un cuaderno técnico.
Especificaciones
- Tamaño de celda: 40px x 40px
- Grosor de línea: 1px
- Color de línea:
#27272a(mismo que--border-default) - Máscara: Degradado radial que desvanece la cuadrícula en los bordes
- Opacidad: 0.15 (muy sutil, apenas perceptible)
CSS listo para copiar
.bg-grid-pattern {
position: fixed;
inset: 0;
z-index: 0;
pointer-events: none;
background-size: 40px 40px;
background-image:
linear-gradient(to right, #27272a 1px, transparent 1px),
linear-gradient(to bottom, #27272a 1px, transparent 1px);
mask-image: radial-gradient(circle at center, black 40%, transparent 100%);
-webkit-mask-image: radial-gradient(circle at center, black 40%, transparent 100%);
opacity: 0.15;
} Uso en el HTML
El patrón se coloca como primer hijo del <body> o del contenedor principal,
con position: fixed para que cubra toda la ventana sin interferir con el scroll.
<body>
<div class="bg-grid-pattern" aria-hidden="true"></div>
<div class="app-content">
<!-- Contenido de la aplicación -->
</div>
</body> Variante con CSS custom properties
Para mayor flexibilidad, se pueden parametrizar los valores con variables:
.bg-grid-pattern {
--grid-size: 40px;
--grid-color: var(--border-default);
--grid-opacity: 0.15;
position: fixed;
inset: 0;
z-index: 0;
pointer-events: none;
background-size: var(--grid-size) var(--grid-size);
background-image:
linear-gradient(to right, var(--grid-color) 1px, transparent 1px),
linear-gradient(to bottom, var(--grid-color) 1px, transparent 1px);
mask-image: radial-gradient(circle at center, black 40%, transparent 100%);
-webkit-mask-image: radial-gradient(circle at center, black 40%, transparent 100%);
opacity: var(--grid-opacity);
} aria-hidden
Siempre incluye aria-hidden=“true” en el elemento del grid.
Es puramente decorativo y no debe ser anunciado por lectores de pantalla.
Además, pointer-events: none asegura que no interfiera con clics.
Accessibility
El sistema “Engineering Notebook” cumple con las pautas WCAG 2.1 nivel AA. A continuación se detallan los ratios de contraste, estrategias de foco y consideraciones para tecnologías asistivas.
Ratios de contraste WCAG AA
Todos los colores de acento neón sobre los fondos oscuros del sistema superan el ratio mínimo de 4.5:1 para texto normal y 3:1 para texto grande.
| Color | Hex | Sobre —bg-void (#0f0f11) | Sobre —bg-surface (#18181b) | AA Normal | AA Grande |
|---|---|---|---|---|---|
--accent-yellow | #fde047 | 13.2:1 | 12.1:1 | Pasa | Pasa |
--accent-pink | #f472b6 | 5.9:1 | 5.4:1 | Pasa | Pasa |
--accent-cyan | #22d3ee | 9.6:1 | 8.8:1 | Pasa | Pasa |
--accent-green | #86efac | 11.5:1 | 10.5:1 | Pasa | Pasa |
--accent-red | #fb7185 | 6.1:1 | 5.6:1 | Pasa | Pasa |
--accent-orange | #fb923c | 7.4:1 | 6.8:1 | Pasa | Pasa |
--accent-purple | #a78bfa | 5.5:1 | 5.0:1 | Pasa | Pasa |
--text-primary | #f4f4f5 | 16.1:1 | 14.7:1 | Pasa | Pasa |
--text-secondary | #a1a1aa | 6.8:1 | 6.2:1 | Pasa | Pasa |
--text-muted | #71717a | 4.0:1 | 3.6:1 | Solo grande | Pasa |
--text-muted
El token —text-muted (#71717a) no alcanza el ratio 4.5:1 para texto
de tamaño normal. Úsalo solo para texto grande (18px+), elementos decorativos
o texto que no sea esencial para la comprensión del contenido.
Estados :focus-visible
Todos los elementos interactivos deben mostrar un indicador de foco visible cuando se navega con teclado. El estilo estándar del sistema es:
/* Estilo de foco global */
:focus-visible {
outline: 2px solid var(--accent-yellow);
outline-offset: 2px;
}
/* Eliminar outline nativo solo cuando focus-visible no aplica */
:focus:not(:focus-visible) {
outline: none;
} Navegación por teclado
| Tecla | Acción |
|---|---|
Tab / Shift+Tab | Navegar entre elementos interactivos en orden lógico del DOM. |
Enter / Space | Activar botones, enlaces y controles. |
Escape | Cerrar modales, menús desplegables y overlays. |
Ctrl+K | Abrir el buscador global de documentación. |
Arrow Up/Down | Navegar resultados de búsqueda y menús. |
Consideraciones para lectores de pantalla
- Todos los íconos decorativos llevan
aria-hidden="true". - Los íconos funcionales incluyen
aria-labeldescriptivo. - El grid de fondo siempre lleva
aria-hidden="true". - Los bloques de código incluyen un
aria-labelcon el lenguaje. - Las imágenes de documentación requieren
altdescriptivo. - El botón de copiar código anuncia “Copiado” con
aria-live="polite". - Los callouts usan
role="note"para ser identificados correctamente.
<!-- Icono decorativo -->
<svg aria-hidden="true">...</svg>
<!-- Icono funcional -->
<button aria-label="Copiar código al portapapeles">
<svg aria-hidden="true">...</svg>
</button>
<!-- Anuncio de estado -->
<span class="sr-only" aria-live="polite" id="copy-status"></span> Movimiento reducido
El sistema respeta la preferencia prefers-reduced-motion del usuario.
Cuando está activa, todas las animaciones se reducen a transiciones
instantáneas y las rotaciones decorativas se eliminan.
@media (prefers-reduced-motion: reduce) {
*,
*::before,
*::after {
animation-duration: 0.01ms !important;
animation-iteration-count: 1 !important;
transition-duration: 0.01ms !important;
scroll-behavior: auto !important;
}
} Clase auxiliar screen-reader-only
Clase utilitaria para contenido visible solo por lectores de pantalla.
.sr-only {
position: absolute;
width: 1px;
height: 1px;
padding: 0;
margin: -1px;
overflow: hidden;
clip: rect(0, 0, 0, 0);
white-space: nowrap;
border-width: 0;
} Testing de accesibilidad
Verifica los ratios de contraste con herramientas como WebAIM Contrast Checker o la pestaña de Accesibilidad de Chrome DevTools. Prueba la navegación completa con teclado antes de cada release.