Una descripción general fundamental de cómo compilar un componente de configuración de controles deslizantes y casillas de verificación.
En esta publicación, quiero compartir ideas sobre cómo compilar un componente de configuración para la Web que sea responsivo, admita varias entradas de dispositivos y funcione en distintos navegadores. Prueba la demostración.
Si prefieres ver un video o quieres obtener una vista previa de la IU/UX de lo que estamos creando, aquí tienes una explicación más breve en YouTube:
Descripción general
Dividí los aspectos de este componente en las siguientes secciones:
- Diseños
- Color
- Entrada de rango personalizado
- Entrada de casilla de verificación personalizada
- Consideraciones de accesibilidad
- JavaScript
Diseños
Esta es la primera demostración del Desafío de GUI que es completamente de cuadrícula de CSS. A continuación, se muestra cada cuadrícula destacada con las Herramientas para desarrolladores de Chrome para cuadrículas:
Solo para el espacio
El diseño más común:
foo {
display: grid;
gap: var(--something);
}
A este diseño lo llamo "solo para espacios" porque solo usa una cuadrícula para agregar espacios entre los bloques.
Cinco diseños usan esta estrategia, aquí se muestran todos:
El elemento fieldset
, que contiene cada grupo de entrada (.fieldset-item
), usa gap: 1px
para crear los bordes de línea fina entre los elementos. No hay una solución de borde complicada.
.grid { display: grid; gap: 1px; background: var(--bg-surface-1); & > .fieldset-item { background: var(--bg-surface-2); } }
.grid { display: grid; & > .fieldset-item { background: var(--bg-surface-2); &:not(:last-child) { border-bottom: 1px solid var(--bg-surface-1); } } }
Ajuste de cuadrícula natural
El diseño más complejo terminó siendo el diseño de macro, el sistema de diseño lógico entre <main>
y <form>
.
Cómo centrar el contenido de unión
Flexbox y la cuadrícula proporcionan capacidades a align-items
o align-content
y, cuando se trata de elementos de unión, las alineaciones de diseño content
distribuirán el espacio entre los elementos secundarios como un grupo.
main {
display: grid;
gap: var(--space-xl);
place-content: center;
}
El elemento principal usa la sintaxis de alineación place-content: center
para que los elementos secundarios se centren vertical y horizontalmente en diseños de una y dos columnas.
En el video anterior, observa cómo el "contenido" se mantiene centrado, a pesar de que se produjo el encadenamiento.
Repite el ajuste automático de minmax
El <form>
usa un diseño de cuadrícula adaptable para cada sección.
Este diseño cambia de una a dos columnas según el espacio disponible.
form {
display: grid;
gap: var(--space-xl) var(--space-xxl);
grid-template-columns: repeat(auto-fit, minmax(min(10ch, 100%), 35ch));
align-items: flex-start;
max-width: 89vw;
}
Esta cuadrícula tiene un valor diferente para row-gap
(--space-xl) que para column-gap
(--space-xxl) para darle ese toque personalizado al diseño responsivo. Cuando las columnas se apilan, queremos un espacio grande, pero no tan grande como si estuviéramos en una pantalla ancha.
La propiedad grid-template-columns
usa 3 funciones de CSS: repeat()
, minmax()
y min()
. Una Kravets tiene una excelente entrada de blog sobre el diseño sobre este tema, y la llama RAM.
Si lo comparas con el de Una, verás que nuestro diseño tiene 3 incorporaciones especiales:
- Pasamos una función
min()
adicional. - Especificamos
align-items: flex-start
. - Hay un estilo
max-width: 89vw
.
Evan Minto describe bien la función adicional min()
en su blog en la entrada Cuadrícula de CSS intrínsecamente responsiva con minmax() y min(). Te recomiendo que leas. La corrección de alineación de flex-start
quita el efecto de estiramiento predeterminado, de modo que los elementos secundarios de este diseño no tengan que tener la misma altura, sino que pueden tener alturas intrínsecas y naturales. El video de YouTube incluye un desglose rápido de esta adición de alineación.
Vale la pena hacer un pequeño desglose de max-width: 89vw
en esta publicación.
Te mostraré el diseño con y sin el estilo aplicado:
¿Qué sucede? Cuando se especifica max-width
, se proporciona contexto, tamaño explícito o tamaño definido para que el algoritmo de diseño auto-fit
sepa cuántas repeticiones puede ajustar en el espacio. Si bien parece obvio que el espacio es de "ancho completo", según las especificaciones de la cuadrícula de CSS, se debe proporcionar un tamaño definido o un tamaño máximo. Proporcionamos un tamaño máximo.
Entonces, ¿por qué elegir 89vw
? Porque "funcionó" para mi diseño.
Yo y otras personas del equipo de Chrome estamos investigando por qué un valor más razonable, como 100vw
, no es suficiente y si, de hecho, se trata de un error.
Espaciado
La mayor parte de la armonía de este diseño proviene de una paleta limitada de espaciado, 7 para ser exactos.
:root {
--space-xxs: .25rem;
--space-xs: .5rem;
--space-sm: 1rem;
--space-md: 1.5rem;
--space-lg: 2rem;
--space-xl: 3rem;
--space-xxl: 6rem;
}
El uso de estos flujos se ejecuta muy bien con cuadrícula, CSS @nest y sintaxis de nivel 5 de @media. Este es un ejemplo, el conjunto de estilos de diseño <main>
completo.
main {
display: grid;
gap: var(--space-xl);
place-content: center;
padding: var(--space-sm);
@media (width >= 540px) {
& {
padding: var(--space-lg);
}
}
@media (width >= 800px) {
& {
padding: var(--space-xl);
}
}
}
Una cuadrícula con contenido centrado y relleno moderado de forma predeterminada (como en dispositivos móviles). Sin embargo, a medida que hay más espacio de viewport disponible, se distribuye aumentando el padding. El CSS de 2021 se ve muy bien.
¿Recuerdas el diseño anterior, "solo para la brecha"? Aquí hay una versión más completa de cómo se ven en este componente:
header {
display: grid;
gap: var(--space-xxs);
}
section {
display: grid;
gap: var(--space-md);
}
Color
Un uso controlado del color ayudó a que este diseño se destacara como expresivo y, al mismo tiempo, mínimo. Lo hago de la siguiente manera:
:root {
--surface1: lch(10 0 0);
--surface2: lch(15 0 0);
--surface3: lch(20 0 0);
--surface4: lch(25 0 0);
--text1: lch(95 0 0);
--text2: lch(75 0 0);
}
Asigno nombres a mis colores de superficie y texto con números en lugar de nombres como surface-dark
y surface-darker
porque, en una consulta de contenido multimedia, los cambiaré, y los colores claro y oscuro no serán significativos.
Los cambio en una consulta de contenido multimedia de preferencia de la siguiente manera:
:root {
...
@media (prefers-color-scheme: light) {
& {
--surface1: lch(90 0 0);
--surface2: lch(100 0 0);
--surface3: lch(98 0 0);
--surface4: lch(85 0 0);
--text1: lch(20 0 0);
--text2: lch(40 0 0);
}
}
}
Es importante obtener una vista rápida del panorama general y la estrategia antes de adentrarnos en los detalles de la sintaxis de colores. Pero, como me adelantaba un poco, permítanme regresar un poco.
¿LCH?
Sin entrar en detalles sobre la teoría del color, LCH es una sintaxis orientada a las personas, que se adapta a la forma en que percibimos el color, no a la forma en que lo medimos con matemáticas (como 255). Esto le brinda una ventaja distintiva, ya que las personas pueden escribirlo con mayor facilidad y otras personas estarán en sintonía con estos ajustes.
Hoy, en esta demostración, enfoquémonos en la sintaxis y los valores que estoy cambiando para crear los colores claro y oscuro. Veamos 1 superficie y 1 color de texto:
:root {
--surface1: lch(10 0 0);
--text1: lch(95 0 0);
@media (prefers-color-scheme: light) {
& {
--surface1: lch(90 0 0);
--text1: lch(40 0 0);
}
}
}
--surface1: lch(10 0 0)
se traduce en una luminosidad de 10%
, 0 croma y 0 tono: un gris sin color muy oscuro. Luego, en la consulta de medios para el modo claro, la luminosidad se cambia a 90%
con --surface1: lch(90 0 0);
. Y esa es la esencia de la estrategia. Comienza por cambiar solo la luminosidad entre los 2 temas y mantén las relaciones de contraste que requiere el diseño o lo que puede mantener la accesibilidad.
La ventaja de lch()
aquí es que la ligereza está orientada a las personas, y podemos sentirnos bien con un cambio de %
, que será perceptual y coherentemente esa %
diferente. Por ejemplo, hsl()
no es tan confiable.
Puedes obtener más información sobre los espacios de color y lch()
si te interesa. ¡Ya está disponible!
Actualmente, el CSS no puede acceder a estos colores en absoluto. Repito: No tenemos acceso a un tercio de los colores de la mayoría de los monitores modernos. Estos no son todos colores, sino los colores más brillantes que puede mostrar la pantalla. Nuestros sitios web se desvanecen porque el hardware de los monitores evolucionó más rápido que las especificaciones de CSS y las implementaciones de navegadores.
Lea Verou
Controles de formulario adaptables con esquema de colores
Muchos navegadores envían controles de tema oscuro, actualmente Safari y Chromium, pero debes especificar en CSS o HTML que tu diseño los usa.
En lo anterior, se muestra el efecto de la propiedad desde el panel de estilos de DevTools. La demostración usa la etiqueta HTML, que, en mi opinión, es una mejor ubicación en general:
<meta name="color-scheme" content="dark light">
Obtén toda la información al respecto en este artículo color-scheme
de Thomas Steiner. Hay mucho más que ganar que las entradas de casilla de verificación oscuras.
CSS accent-color
Hubo actividad reciente en torno a accent-color
en los elementos del formulario, ya que hay un único estilo de CSS que puede cambiar el color del tono que se usa en el elemento de entrada del navegador. Obtén más información aquí en GitHub. Lo incluí en mis estilos para este componente. Como los navegadores lo admiten, mis casillas de verificación
serán más específicas con los colores rosa y púrpura.
input[type="checkbox"] {
accent-color: var(--brand);
}
Resalta los colores con degradados fijos y enfoque dentro
El color se destaca más cuando se usa con moderación, y una de las formas en que me gusta lograrlo es a través de interacciones coloridas de la IU.
En el video anterior, hay muchas capas de interacción y comentarios de la IU que ayudan a darle personalidad a la interacción de las siguientes maneras:
- Destaca el contexto.
- Proporcionar comentarios en la IU sobre qué tan lleno está el valor en el rango
- Proporcionar comentarios a la IU que indiquen que un campo acepta entradas
Para proporcionar comentarios cuando se interactúa con un elemento, CSS usa la seudoclase :focus-within
para cambiar la apariencia de varios elementos. Analicemos la .fieldset-item
, es muy interesante:
.fieldset-item {
...
&:focus-within {
background: var(--surface2);
& svg {
fill: white;
}
& picture {
clip-path: circle(50%);
background: var(--brand-bg-gradient) fixed;
}
}
}
Cuando uno de los elementos secundarios de este elemento tiene el foco dentro:
- Al fondo
.fieldset-item
se le asigna un color de superficie de mayor contraste. - El
svg
anidado se completa en blanco para obtener un mayor contraste. - El
<picture>
clip-path
anidado se expande a un círculo completo y el fondo se rellena con el gradiente fijo y brillante.
Período personalizado
Dado el siguiente elemento de entrada HTML, te mostraré cómo personalicé su apariencia:
<input type="range">
Este elemento tiene 3 partes que debemos personalizar:
Estilos de elementos de rango
input[type="range"] {
/* style setting variables */
--track-height: .5ex;
--track-fill: 0%;
--thumb-size: 3ex;
--thumb-offset: -1.25ex;
--thumb-highlight-size: 0px;
appearance: none; /* clear styles, make way for mine */
display: block;
inline-size: 100%; /* fill container */
margin: 1ex 0; /* ensure thumb isn't colliding with sibling content */
background: transparent; /* bg is in the track */
outline-offset: 5px; /* focus styles have space */
}
Las primeras líneas de CSS son las partes personalizadas de los estilos, y espero que etiquetarlas de forma clara te ayude. El resto de los estilos son en su mayoría estilos de restablecimiento para proporcionar una base coherente para compilar las partes difíciles del componente.
Estilos de pista
input[type="range"]::-webkit-slider-runnable-track {
appearance: none; /* clear styles, make way for mine */
block-size: var(--track-height);
border-radius: 5ex;
background:
/* hard stop gradient:
- half transparent (where colorful fill we be)
- half dark track fill
- 1st background image is on top
*/
linear-gradient(
to right,
transparent var(--track-fill),
var(--surface1) 0%
),
/* colorful fill effect, behind track surface fill */
var(--brand-bg-gradient) fixed;
}
El truco para esto es “revelar” el color de relleno vibrante. Esto se hace con el gradiente de parada dura en la parte superior. El gradiente es transparente hasta el porcentaje de relleno y, luego, usa el color de la superficie de la pista sin relleno. Detrás de esa superficie sin relleno, hay un color de ancho completo que espera a que la transparencia lo revele.
Estilo de relleno de la pista
Mi diseño requiere JavaScript para mantener el estilo de relleno. Existen estrategias solo de CSS, pero requieren que el elemento de miniatura tenga la misma altura que la pista, y no pude encontrar una armonía dentro de esos límites.
/* grab sliders on page */
const sliders = document.querySelectorAll('input[type="range"]')
/* take a slider element, return a percentage string for use in CSS */
const rangeToPercent = slider => {
const max = slider.getAttribute('max') || 10;
const percent = slider.value / max * 100;
return `${parseInt(percent)}%`;
};
/* on page load, set the fill amount */
sliders.forEach(slider => {
slider.style.setProperty('--track-fill', rangeToPercent(slider));
/* when a slider changes, update the fill prop */
slider.addEventListener('input', e => {
e.target.style.setProperty('--track-fill', rangeToPercent(e.target));
})
})
Creo que esto es una buena actualización visual. El control deslizante funciona muy bien sin JavaScript. No se requiere el atributo --track-fill
, simplemente no tendrá un estilo de relleno si no está presente. Si JavaScript está disponible, propaga la propiedad personalizada y, al mismo tiempo, observa los cambios del usuario y sincroniza la propiedad personalizada con el valor.
Esta es una excelente publicación sobre CSS-Tricks de Ana Tudor, en la que se demuestra una solución exclusiva de CSS para el relleno de segmentos. También me resultó muy inspirador este elemento range
.
Estilos de miniaturas
input[type="range"]::-webkit-slider-thumb {
appearance: none; /* clear styles, make way for mine */
cursor: ew-resize; /* cursor style to support drag direction */
border: 3px solid var(--surface3);
block-size: var(--thumb-size);
inline-size: var(--thumb-size);
margin-top: var(--thumb-offset);
border-radius: 50%;
background: var(--brand-bg-gradient) fixed;
}
La mayoría de estos estilos son para hacer un círculo agradable.
Una vez más, verás el gradiente de fondo fijo que unifica los colores dinámicos de las miniaturas, las pistas y los elementos SVG asociados.
Separamos los estilos de la interacción para ayudar a aislar la técnica box-shadow
que se usa para el elemento destacado de desplazamiento del mouse:
@custom-media --motionOK (prefers-reduced-motion: no-preference);
::-webkit-slider-thumb {
…
/* shadow spread is initally 0 */
box-shadow: 0 0 0 var(--thumb-highlight-size) var(--thumb-highlight-color);
/* if motion is OK, transition the box-shadow change */
@media (--motionOK) {
& {
transition: box-shadow .1s ease;
}
}
/* on hover/active state of parent, increase size prop */
@nest input[type="range"]:is(:hover,:active) & {
--thumb-highlight-size: 10px;
}
}
El objetivo era crear un elemento destacado visual animado y fácil de administrar para los comentarios de los usuarios. Si uso una sombra de cuadro, puedo evitar activar el diseño con el efecto. Para ello, creo una sombra que no está desenfocada y que coincide con la forma circular del elemento de miniatura. Luego, cambio y transfiero su tamaño de propagación cuando se coloca el cursor sobre él.
Ojalá el efecto de resaltado fuera tan fácil en las casillas de verificación…
Selectores multinavegador
Descubrí que necesitaba estos selectores -webkit-
y -moz-
para lograr la coherencia entre navegadores:
input[type="range"] {
&::-webkit-slider-runnable-track {}
&::-moz-range-track {}
&::-webkit-slider-thumb {}
&::-moz-range-thumb {}
}
Casilla de verificación personalizada
Dado el siguiente elemento de entrada HTML, te mostraré cómo personalicé su apariencia:
<input type="checkbox">
Este elemento tiene 3 partes que debemos personalizar:
Elemento de casilla de verificación
input[type="checkbox"] {
inline-size: var(--space-sm); /* increase width */
block-size: var(--space-sm); /* increase height */
outline-offset: 5px; /* focus style enhancement */
accent-color: var(--brand); /* tint the input */
position: relative; /* prepare for an absolute pseudo element */
transform-style: preserve-3d; /* create a 3d z-space stacking context */
margin: 0;
cursor: pointer;
}
Los estilos transform-style
y position
se preparan para el pseudoelemento que presentaremos más adelante
para aplicar diseño al elemento destacado. De lo contrario, se trata principalmente de aspectos de estilo menores. Me gusta que el cursor sea un puntero, me gustan los desfases de contorno, las casillas de verificación predeterminadas son demasiado pequeñas y, si accent-color
es compatible, lleva estas casillas de verificación al esquema de colores de la marca.
Etiquetas de casillas de verificación
Es importante proporcionar etiquetas para las casillas de verificación por 2 motivos. El primero es representar para qué se usa el valor de la casilla de verificación, para responder "¿para qué se activa o desactiva?". En segundo lugar, en el caso de la UX, los usuarios web se acostumbraron a interactuar con las casillas de verificación a través de sus etiquetas asociadas.
<input type="checkbox" id="text-notifications" name="text-notifications" >
<label for="text-notifications"> <h3>Text Messages</h3> <small>Get notified about all text messages sent to your device</small> </label>
En la etiqueta, coloca un atributo for
que haga referencia a una casilla de verificación por ID: <label for="text-notifications">
. En la casilla de verificación, duplica el nombre y el ID para asegurarte de que se encuentre con diferentes herramientas y tecnologías, como un mouse o un lector de pantalla: <input type="checkbox" id="text-notifications" name="text-notifications">
.
:hover
, :active
y muchos más son gratuitos con la conexión, lo que aumenta las formas en que se puede interactuar con tu formulario.
Destacado de casilla de verificación
Quiero mantener la coherencia de mis interfaces, y el elemento del control deslizante tiene un buen elemento destacado de miniatura que me gustaría usar con la casilla de verificación. La miniatura pudo usar box-shadow
y su propiedad spread
para escalar una sombra hacia arriba y hacia abajo. Sin embargo, ese efecto no funciona aquí porque nuestras casillas de verificación son, y deben ser, cuadradas.
Pude lograr el mismo efecto visual con un pseudoelemento y una cantidad desdichada de CSS complicado:
@custom-media --motionOK (prefers-reduced-motion: no-preference);
input[type="checkbox"]::before {
--thumb-scale: .01; /* initial scale of highlight */
--thumb-highlight-size: var(--space-xl);
content: "";
inline-size: var(--thumb-highlight-size);
block-size: var(--thumb-highlight-size);
clip-path: circle(50%); /* circle shape */
position: absolute; /* this is why position relative on parent */
top: 50%; /* pop and plop technique (https://web.dev/centering-in-css#5-pop-and-plop) */
left: 50%;
background: var(--thumb-highlight-color);
transform-origin: center center; /* goal is a centered scaling circle */
transform: /* order here matters!! */
translateX(-50%) /* counter balances left: 50% */
translateY(-50%) /* counter balances top: 50% */
translateZ(-1px) /* PUTS IT BEHIND THE CHECKBOX */
scale(var(--thumb-scale)) /* value we toggle for animation */
;
will-change: transform;
@media (--motionOK) { /* transition only if motion is OK */
& {
transition: transform .2s ease;
}
}
}
/* on hover, set scale custom property to "in" state */
input[type="checkbox"]:hover::before {
--thumb-scale: 1;
}
Crear un pseudoelemento de círculo es un trabajo sencillo, pero colocarlo detrás del elemento al que está unido fue más difícil. A continuación, se muestran las imágenes del problema antes y después de solucionarlo:
Sin duda, es una microinteracción, pero es importante para mí mantener la coherencia visual. La técnica de escalamiento de animación es la misma que hemos usado en
otros lugares. Establecemos una propiedad personalizada en un valor nuevo y dejamos que CSS la transfiera según las preferencias de movimiento. La función clave aquí es translateZ(-1px)
. El elemento superior creó un espacio 3D y este pseudoelemento secundario se conectó a él colocándose un poco hacia atrás en el espacio z.
Accesibilidad
En el video de YouTube, se muestra una excelente demostración de las interacciones del mouse, el teclado y el lector de pantalla para este componente de configuración. Mencionaré algunos de los detalles aquí.
Opciones de elementos HTML
<form>
<header>
<fieldset>
<picture>
<label>
<input>
Cada uno de ellos contiene sugerencias para la herramienta de navegación del usuario. Algunos elementos proporcionan sugerencias de interacción, algunos conectan la interactividad y otros ayudan a dar forma al árbol de accesibilidad por el que navega un lector de pantalla.
Atributos HTML
Podemos ocultar los elementos que los lectores de pantalla no necesitan; en este caso, el ícono junto al control deslizante:
<picture aria-hidden="true">
En el video anterior, se muestra el flujo del lector de pantalla en macOS. Observa cómo el enfoque de entrada se mueve directamente de un control deslizante al siguiente. Esto se debe a que ocultamos el ícono que podría haber sido una parada en el camino al siguiente control deslizante. Sin este atributo, el usuario tendría que detenerse, escuchar y pasar por alto la imagen que podría no ver.
El SVG es un montón de matemáticas. Agreguemos un elemento <title>
para un título de desplazamiento del mouse libre y un comentario legible sobre lo que crean los cálculos:
<svg viewBox="0 0 24 24">
<title>A note icon</title>
<path d="M12 3v10.55c-.59-.34-1.27-.55-2-.55-2.21 0-4 1.79-4 4s1.79 4 4 4 4-1.79 4-4V7h4V3h-6z"/>
</svg>
Aparte de eso, usamos suficiente HTML marcado de forma clara, por lo que el formulario se prueba muy bien en mouse, teclado, controles de videojuegos y lectores de pantalla.
JavaScript
Ya analizamos cómo se administraba el color de relleno de la pista desde JavaScript. Ahora, veamos el JavaScript relacionado con <form>
:
const form = document.querySelector('form');
form.addEventListener('input', event => {
const formData = Object.fromEntries(new FormData(form));
console.table(formData);
})
Cada vez que se interactúa con el formulario y se lo cambia, la consola lo registra como un objeto en una tabla para facilitar su revisión antes de enviarlo a un servidor.
Conclusión
Ahora que sabes cómo lo hice, ¿cómo lo harías? Esto genera una arquitectura de componentes divertida. ¿Quién va a hacer la 1a versión con ranuras en su framework favorito? 🙂
Diversifiquemos nuestros enfoques y aprendamos todas las formas de desarrollar en la Web. Crea una demostración, twittea los vínculos y los agregaré a la sección Remixes de la comunidad a continuación.