Présentation de base sur la création d'une barre de chargement accessible et adaptative aux couleurs à l'aide de l'élément <progress>
.
Dans cet article, je souhaite partager mes réflexions sur la création d'une barre de chargement adaptative et accessible aux couleurs avec l'élément <progress>
. Essayez la démonstration et consultez le code source.
Si vous préférez regarder une vidéo, voici une version YouTube de cet article:
Présentation
L'élément <progress>
fournit aux utilisateurs des commentaires visuels et audibles sur l'avancement. Ce retour visuel est utile dans des scénarios tels que la progression dans un formulaire, l'affichage d'informations de téléchargement ou d'importation, ou même l'indication que le niveau de progression est inconnu, mais que le travail est toujours actif.
Ce défi d'IUG a utilisé l'élément HTML <progress>
existant pour réduire les efforts d'accessibilité. Les couleurs et les mises en page repoussent les limites de la personnalisation de l'élément intégré pour moderniser le composant et l'adapter mieux aux systèmes de conception.
Annoter
J'ai choisi d'encapsuler l'élément <progress>
dans un <label>
afin de pouvoir ignorer les attributs de relation explicites au profit d'une relation implicite.
J'ai également ajouté un libellé à un élément parent affecté par l'état de chargement, afin que les technologies de lecteur d'écran puissent transmettre ces informations à un utilisateur.
<progress></progress>
Si aucun value
n'est défini, la progression de l'élément est indéterminée.
La valeur par défaut de l'attribut max
est 1. La progression est donc comprise entre 0 et 1. Si vous définissez max
sur 100, par exemple, la plage sera comprise entre 0 et 100. J'ai choisi de rester dans les limites de 0 et 1, en traduisant les valeurs de progression en 0,5 ou 50%.
Progression encapsulée par un libellé
Dans une relation implicite, un élément de progression est encapsulé par un libellé comme suit:
<label>Loading progress<progress></progress></label>
Dans ma démonstration, j'ai choisi d'inclure le libellé pour les lecteurs d'écran uniquement.
Pour ce faire, encapsulez le texte du libellé dans un <span>
et appliquez-lui des styles afin qu'il soit effectivement hors écran:
<label>
<span class="sr-only">Loading progress</span>
<progress></progress>
</label>
Avec le code CSS suivant de WebAIM:
.sr-only {
clip: rect(1px, 1px, 1px, 1px);
clip-path: inset(50%);
height: 1px;
width: 1px;
margin: -1px;
overflow: hidden;
padding: 0;
position: absolute;
}
Zone affectée par la progression du chargement
Si vous avez une bonne vision, il peut être facile d'associer un indicateur de progression aux éléments et aux zones de page associés, mais ce n'est pas aussi clair pour les utilisateurs ayant une déficience visuelle. Pour améliorer ce point, attribuez l'attribut aria-busy
à l'élément le plus haut qui changera une fois le chargement terminé.
De plus, indiquez une relation entre la progression et la zone de chargement avec aria-describedby
.
<main id="loading-zone" aria-busy="true">
…
<progress aria-describedby="loading-zone"></progress>
</main>
À partir de JavaScript, basculez aria-busy
sur true
au début de la tâche et sur false
une fois terminé.
Ajout d'attributs ARIA
Bien que le rôle implicite d'un élément <progress>
soit progressbar
, je l'ai rendu explicite pour les navigateurs qui ne disposent pas de ce rôle implicite. J'ai également ajouté l'attribut indeterminate
pour définir explicitement l'état "inconnu" de l'élément, ce qui est plus clair que de constater que l'élément n'a pas de value
défini.
<label>
Loading
<progress
indeterminate
role="progressbar"
aria-describedby="loading-zone"
tabindex="-1"
>unknown</progress>
</label>
Utilisez tabindex="-1"
pour rendre l'élément de progression sélectionnable à partir de JavaScript. Cela est important pour la technologie de lecteur d'écran, car en donnant la priorité à la progression à mesure qu'elle évolue, l'utilisateur sera informé du niveau de progression actualisé.
Styles
L'élément de progression est un peu délicat à styliser. Les éléments HTML intégrés comportent des parties masquées spéciales qui peuvent être difficiles à sélectionner et n'offrent souvent qu'un ensemble limité de propriétés à définir.
Mise en page
Les styles de mise en page sont destinés à offrir une certaine flexibilité concernant la taille et la position de l'élément de progression. Un état de finalisation spécial est ajouté, qui peut être un indice visuel supplémentaire utile, mais non obligatoire.
Mise en page <progress>
La largeur de l'élément de progression n'est pas modifiée afin qu'il puisse se réduire et se développer en fonction de l'espace nécessaire dans la conception. Les styles intégrés sont supprimés en définissant appearance
et border
sur none
. Cela permet de normaliser l'élément dans les différents navigateurs, car chacun d'eux a ses propres styles pour ses éléments.
progress {
--_track-size: min(10px, 1ex);
--_radius: 1e3px;
/* reset */
appearance: none;
border: none;
position: relative;
height: var(--_track-size);
border-radius: var(--_radius);
overflow: hidden;
}
La valeur 1e3px
pour _radius
utilise la notation scientifique pour exprimer un grand nombre. border-radius
est donc toujours arrondi. Cela équivaut à 1000px
. J'aime utiliser cette méthode, car mon objectif est d'utiliser une valeur suffisamment élevée pour pouvoir la définir et l'oublier (et c'est plus court à écrire que 1000px
). Il est également facile de l'augmenter encore si nécessaire: il suffit de remplacer le chiffre 3 par un chiffre 4, puis 1e4px
est équivalent à 10000px
.
overflow: hidden
est utilisé et a toujours été un style controversé. Cela a simplifié certaines choses, par exemple le fait de ne pas avoir à transmettre des valeurs border-radius
à la piste et aux éléments de remplissage de la piste. En revanche, cela signifiait qu'aucun enfant de la progression ne pouvait se trouver en dehors de l'élément. Une autre itération de cet élément de progression personnalisé peut être effectuée sans overflow: hidden
, ce qui peut ouvrir des possibilités d'animations ou de meilleurs états de progression.
Processus terminé
Les sélecteurs CSS effectuent la tâche difficile en comparant la valeur maximale à la valeur. Si elles correspondent, la progression est terminée. Une fois l'opération terminée, un pseudo-élément est généré et ajouté à la fin de l'élément de progression, ce qui fournit un bon indice visuel supplémentaire pour la finalisation.
progress:not([max])[value="1"]::before,
progress[max="100"][value="100"]::before {
content: "✓";
position: absolute;
inset-block: 0;
inset-inline: auto 0;
display: flex;
align-items: center;
padding-inline-end: max(calc(var(--_track-size) / 4), 3px);
color: white;
font-size: calc(var(--_track-size) / 1.25);
}
Couleur
Le navigateur apporte ses propres couleurs pour l'élément de progression et s'adapte au mode clair et sombre avec une seule propriété CSS. Vous pouvez l'utiliser avec certains sélecteurs spécifiques au navigateur.
Styles de navigateur clair et sombre
Pour activer un élément <progress>
adaptatif sombre et clair sur votre site, il vous suffit d'utiliser color-scheme
.
progress {
color-scheme: light dark;
}
Couleur de remplissage de la progression d'une seule propriété
Pour teinter un élément <progress>
, utilisez accent-color
.
progress {
accent-color: rebeccapurple;
}
Notez que la couleur d'arrière-plan du canal passe du clair au sombre en fonction de l'accent-color
. Le navigateur assure un contraste approprié: c'est plutôt pratique.
Couleurs claires et sombres entièrement personnalisées
Définissez deux propriétés personnalisées sur l'élément <progress>
, l'une pour la couleur du canal et l'autre pour la couleur de la progression du canal. Dans la requête multimédia prefers-color-scheme
, fournissez de nouvelles valeurs de couleur pour le titre et la progression du titre.
progress {
--_track: hsl(228 100% 90%);
--_progress: hsl(228 100% 50%);
}
@media (prefers-color-scheme: dark) {
progress {
--_track: hsl(228 20% 30%);
--_progress: hsl(228 100% 75%);
}
}
Styles de mise au point
Auparavant, nous avons attribué à l'élément un indice de tabulation négatif afin qu'il puisse être mis au premier plan par programmation. Utilisez :focus-visible
pour personnaliser la sélection et activer le style d'anneau de sélection plus intelligent. Dans ce cas, un clic de souris et la sélection ne déclenchent pas l'anneau de sélection, mais les clics au clavier le font. La vidéo YouTube explique cela plus en détail et vaut la peine d'être regardée.
progress:focus-visible {
outline-color: var(--_progress);
outline-offset: 5px;
}
Styles personnalisés dans les navigateurs
Personnalisez les styles en sélectionnant les parties d'un élément <progress>
que chaque navigateur expose. L'utilisation de l'élément de progression ne nécessite qu'une seule balise, mais elle est composée de quelques éléments enfants exposés via des pseudo-sélecteurs CSS. Les outils de développement Chrome vous montreront ces éléments si vous activez le paramètre:
- Effectuez un clic droit sur votre page, puis sélectionnez Inspecter l'élément pour ouvrir les outils pour les développeurs.
- Cliquez sur l'icône en forme de roue dentée des paramètres en haut à droite de la fenêtre DevTools.
- Sous l'en-tête Éléments, recherchez et cochez la case Afficher le DOM de l'ombre de l'agent utilisateur.
Styles Safari et Chromium
Les navigateurs basés sur WebKit, tels que Safari et Chromium, exposent ::-webkit-progress-bar
et ::-webkit-progress-value
, qui permettent d'utiliser un sous-ensemble de CSS. Pour l'instant, définissez background-color
à l'aide des propriétés personnalisées créées précédemment, qui s'adaptent aux modes clair et sombre.
/* Safari/Chromium */
progress[value]::-webkit-progress-bar {
background-color: var(--_track);
}
progress[value]::-webkit-progress-value {
background-color: var(--_progress);
}
Styles Firefox
Firefox n'expose que le pseudo-sélecteur ::-moz-progress-bar
sur l'élément <progress>
. Cela signifie également que nous ne pouvons pas teinter le canal directement.
/* Firefox */
progress[value]::-moz-progress-bar {
background-color: var(--_progress);
}
Notez que Firefox utilise la couleur de la piste définie à partir de accent-color
, tandis qu'iOS Safari utilise une piste bleu clair. Il en va de même en mode sombre: Firefox affiche une piste sombre, mais pas la couleur personnalisée que nous avons définie. Cela fonctionne dans les navigateurs basés sur Webkit.
Animation
Lorsque vous travaillez avec des pseudo-sélecteurs intégrés au navigateur, vous utilisez souvent un ensemble limité de propriétés CSS autorisées.
Animer le remplissage du canal
L'ajout d'une transition à l'inline-size
de l'élément de progression fonctionne pour Chromium, mais pas pour Safari. Firefox n'utilise pas non plus de propriété de transition sur son ::-moz-progress-bar
.
/* Chromium Only 😢 */
progress[value]::-webkit-progress-value {
background-color: var(--_progress);
transition: inline-size .25s ease-out;
}
Animer l'état :indeterminate
Je vais être un peu plus créatif pour pouvoir fournir une animation. Un pseudo-élément pour Chromium est créé et un dégradé est appliqué, qui est animé en avant et en arrière pour les trois navigateurs.
Propriétés personnalisées
Les propriétés personnalisées sont utiles pour de nombreuses choses, mais l'une de mes préférées consiste simplement à donner un nom à une valeur CSS qui semble magique. Voici un linear-gradient
assez complexe, mais avec un nom agréable. Son objectif et ses cas d'utilisation sont clairement compris.
progress {
--_indeterminate-track: linear-gradient(to right,
var(--_track) 45%,
var(--_progress) 0%,
var(--_progress) 55%,
var(--_track) 0%
);
--_indeterminate-track-size: 225% 100%;
--_indeterminate-track-animation: progress-loading 2s infinite ease;
}
Les propriétés personnalisées permettent également de garder le code DRY, car nous ne pouvons pas regrouper ces sélecteurs spécifiques au navigateur.
Les images clés
L'objectif est d'obtenir une animation infinie qui va et vient. Les images clés de début et de fin seront définies en CSS. Une seule image clé est nécessaire, l'image clé du milieu à 50%
, pour créer une animation qui revient à son point de départ, encore et encore.
@keyframes progress-loading {
50% {
background-position: left;
}
}
Cibler chaque navigateur
Tous les navigateurs ne permettent pas de créer de pseudo-éléments sur l'élément <progress>
lui-même ni d'animer la barre de progression. Plus de navigateurs sont compatibles avec l'animation de la piste qu'avec un pseudo-élément. Je passe donc des pseudo-éléments de base aux barres d'animation.
Pseudo-élément Chromium
Chromium autorise le pseudo-élément ::after
utilisé avec une position pour recouvrir l'élément. Les propriétés personnalisées indéterminées sont utilisées, et l'animation de va-et-vient fonctionne très bien.
progress:indeterminate::after {
content: "";
inset: 0;
position: absolute;
background: var(--_indeterminate-track);
background-size: var(--_indeterminate-track-size);
background-position: right;
animation: var(--_indeterminate-track-animation);
}
Barre de progression Safari
Pour Safari, les propriétés personnalisées et une animation sont appliquées à la barre de progression du pseudo-élément:
progress:indeterminate::-webkit-progress-bar {
background: var(--_indeterminate-track);
background-size: var(--_indeterminate-track-size);
background-position: right;
animation: var(--_indeterminate-track-animation);
}
Barre de progression de Firefox
Pour Firefox, les propriétés personnalisées et une animation sont également appliquées à la barre de progression du pseudo-élément:
progress:indeterminate::-moz-progress-bar {
background: var(--_indeterminate-track);
background-size: var(--_indeterminate-track-size);
background-position: right;
animation: var(--_indeterminate-track-animation);
}
JavaScript
JavaScript joue un rôle important avec l'élément <progress>
. Il contrôle la valeur envoyée à l'élément et s'assure qu'il y a suffisamment d'informations dans le document pour les lecteurs d'écran.
const state = {
val: null
}
La démonstration propose des boutons pour contrôler la progression. Ils mettent à jour state.val
, puis appellent une fonction pour mettre à jour le DOM.
document.querySelector('#complete').addEventListener('click', e => {
state.val = 1
setProgress()
})
setProgress()
C'est dans cette fonction que l'orchestration de l'UI/UX se produit. Commencez par créer une fonction setProgress()
. Aucun paramètre n'est nécessaire, car il a accès à l'objet state
, à l'élément de progression et à la zone <main>
.
const setProgress = () => {
}
Définir l'état de chargement dans la zone <main>
Selon que la progression est terminée ou non, l'élément <main>
associé doit mettre à jour l'attribut aria-busy
:
const setProgress = () => {
zone.setAttribute('aria-busy', state.val < 1)
}
Supprimer les attributs si la quantité de chargement est inconnue
Si la valeur est inconnue ou non définie, null
dans cet usage, supprimez les attributs value
et aria-valuenow
. L'état <progress>
devient alors indéterminé.
const setProgress = () => {
zone.setAttribute('aria-busy', state.val < 1)
if (state.val === null) {
progress.removeAttribute('aria-valuenow')
progress.removeAttribute('value')
progress.focus()
return
}
}
Résoudre les problèmes de calcul décimal JavaScript
Étant donné que j'ai choisi de conserver la valeur maximale par défaut de 1 pour la progression, les fonctions d'incrémentation et de décroissance de la démonstration utilisent les mathématiques décimales. JavaScript et d'autres langages ne sont pas toujours très bons pour cela.
Voici une fonction roundDecimals()
qui supprime l'excédent du résultat mathématique:
const roundDecimals = (val, places) =>
+(Math.round(val + "e+" + places) + "e-" + places)
Arrondissez la valeur pour qu'elle puisse être présentée et lisible:
const setProgress = () => {
zone.setAttribute('aria-busy', state.val < 1)
if (state.val === null) {
progress.removeAttribute('aria-valuenow')
progress.removeAttribute('value')
progress.focus()
return
}
const val = roundDecimals(state.val, 2)
const valPercent = val * 100 + "%"
}
Définir la valeur pour les lecteurs d'écran et l'état du navigateur
La valeur est utilisée à trois endroits dans le DOM:
- Attribut
value
de l'élément<progress>
. - Attribut
aria-valuenow
. - Contenu du texte interne
<progress>
.
const setProgress = () => {
zone.setAttribute('aria-busy', state.val < 1)
if (state.val === null) {
progress.removeAttribute('aria-valuenow')
progress.removeAttribute('value')
progress.focus()
return
}
const val = roundDecimals(state.val, 2)
const valPercent = val * 100 + "%"
progress.value = val
progress.setAttribute('aria-valuenow', valPercent)
progress.innerText = valPercent
}
Mettre en avant la progression
Une fois les valeurs mises à jour, les utilisateurs voyants verront le changement de progression, mais les utilisateurs de lecteurs d'écran ne recevront pas encore l'annonce du changement. Mettez en surbrillance l'élément <progress>
, et le navigateur annoncera la mise à jour.
const setProgress = () => {
zone.setAttribute('aria-busy', state.val < 1)
if (state.val === null) {
progress.removeAttribute('aria-valuenow')
progress.removeAttribute('value')
progress.focus()
return
}
const val = roundDecimals(state.val, 2)
const valPercent = val * 100 + "%"
progress.value = val
progress.setAttribute('aria-valuenow', valPercent)
progress.innerText = valPercent
progress.focus()
}
Conclusion
Maintenant que vous savez comment j'ai fait, comment procéderiez-vous ? 🙂
J'aimerais apporter quelques modifications si j'ai la chance de retenter ma chance. Je pense qu'il est possible de nettoyer le composant actuel et d'essayer d'en créer un sans les limites de style de la pseudo-classe de l'élément <progress>
. Cela vaut la peine d'explorer !
Diversifions nos approches et découvrons toutes les façons de créer sur le Web.
Créez une démo, tweetez-moi des liens et je les ajouterai à la section "Remix de la communauté" ci-dessous.