Ringkasan dasar tentang cara membuat status pemuatan yang adaptif dan mudah diakses dengan elemen <progress>
.
Dalam postingan ini, saya ingin membagikan pemikiran tentang cara membuat status pemuatan yang adaptif dan
dapat diakses dengan elemen <progress>
. Coba demo dan lihat sumbernya.
Jika Anda lebih suka video, berikut versi YouTube dari postingan ini:
Ringkasan
Elemen
<progress>
memberikan masukan visual dan audio kepada pengguna tentang penyelesaian. Masukan
visual ini sangat berharga untuk skenario seperti: progres melalui formulir,
menampilkan informasi download atau upload, atau bahkan menunjukkan bahwa
jumlah progres tidak diketahui, tetapi pekerjaan masih aktif.
Tantangan GUI ini berfungsi dengan
elemen <progress>
HTML yang ada untuk menghemat beberapa upaya dalam aksesibilitas. Warna
dan tata letak mendorong batas penyesuaian untuk elemen bawaan, untuk
memodernisasi komponen dan membuatnya lebih cocok dalam sistem desain.
Markup
Saya memilih untuk menggabungkan elemen <progress>
dalam
<label>
sehingga
saya dapat melewati atribut hubungan eksplisit dan memilih hubungan
implisit.
Saya juga telah memberi label pada elemen induk yang terpengaruh oleh status pemuatan, sehingga teknologi pembaca
layar dapat menyampaikan kembali informasi tersebut kepada pengguna.
<progress></progress>
Jika tidak ada value
, progres elemen akan
tidak ditentukan.
Atribut max
ditetapkan secara default ke 1, sehingga progresnya antara 0 dan 1. Misalnya, menetapkan max
ke 100 akan menetapkan rentang ke 0-100. Saya memilih untuk tetap berada dalam batas 0
dan 1, yang menerjemahkan nilai progres menjadi 0,5 atau 50%.
Progres yang digabungkan dengan label
Dalam hubungan implisit, elemen progres digabungkan dengan label seperti ini:
<label>Loading progress<progress></progress></label>
Dalam demo, saya memilih untuk menyertakan label untuk pembaca layar saja.
Hal ini dilakukan dengan menggabungkan teks label dalam <span>
dan menerapkan beberapa gaya
kepadanya sehingga secara efektif tidak ditampilkan di layar:
<label>
<span class="sr-only">Loading progress</span>
<progress></progress>
</label>
Dengan CSS yang menyertainya berikut dari 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;
}
Area yang terpengaruh oleh progres pemuatan
Jika Anda memiliki penglihatan yang sehat, mudah untuk mengaitkan indikator progres
dengan elemen dan area halaman terkait, tetapi bagi pengguna penyandang gangguan penglihatan, hal ini tidak
begitu jelas. Tingkatkan hal ini dengan menetapkan atribut
aria-busy
ke elemen paling atas yang akan berubah saat pemuatan selesai.
Selain itu, tunjukkan hubungan antara progres dan zona pemuatan
dengan
aria-describedby
.
<main id="loading-zone" aria-busy="true">
…
<progress aria-describedby="loading-zone"></progress>
</main>
Dari JavaScript, alihkan aria-busy
ke true
di awal tugas, dan ke
false
setelah selesai.
Penambahan atribut Aria
Meskipun peran implisit elemen <progress>
adalah
progressbar
, saya telah membuatnya eksplisit
untuk browser yang tidak memiliki peran implisit tersebut. Saya juga telah menambahkan atribut
indeterminate
untuk secara eksplisit menempatkan elemen ke dalam status tidak diketahui, yang
lebih jelas daripada mengamati elemen yang tidak memiliki value
yang ditetapkan.
<label>
Loading
<progress
indeterminate
role="progressbar"
aria-describedby="loading-zone"
tabindex="-1"
>unknown</progress>
</label>
Gunakan
tabindex="-1"
agar elemen progres dapat difokuskan dari JavaScript. Hal ini penting bagi
teknologi pembaca layar, karena memberikan fokus progres saat progres berubah,
akan mengumumkan kepada pengguna sejauh mana progres yang diperbarui telah dicapai.
Gaya
Elemen progres sedikit rumit dalam hal gaya visual. Elemen HTML bawaan memiliki bagian khusus tersembunyi yang mungkin sulit dipilih dan sering kali hanya menawarkan serangkaian properti terbatas.
Tata Letak
Gaya tata letak dimaksudkan untuk memungkinkan beberapa fleksibilitas dalam ukuran dan posisi label elemen progres. Status penyelesaian khusus ditambahkan yang dapat menjadi isyarat visual tambahan yang berguna, tetapi tidak diperlukan.
Tata Letak <progress>
Lebar elemen progres tidak disentuh sehingga dapat menyusut dan membesar
sesuai ruang yang dibutuhkan dalam desain. Gaya bawaan dihilangkan dengan menetapkan appearance
dan border
ke none
. Hal ini dilakukan agar elemen dapat
dinormalisasi di seluruh browser, karena setiap browser memiliki gayanya sendiri untuk elemennya.
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;
}
Nilai 1e3px
untuk _radius
menggunakan notasi angka ilmiah untuk menyatakan
angka besar sehingga border-radius
selalu dibulatkan. Setara dengan
1000px
. Saya suka menggunakan ini karena tujuan saya adalah menggunakan nilai yang cukup besar sehingga
saya dapat menyetelnya dan melupakannya (dan lebih pendek untuk menulisnya daripada 1000px
). Selain itu,
mudah untuk membuatnya lebih besar lagi jika perlu: cukup ubah 3 menjadi 4, lalu 1e4px
setara dengan 10000px
.
overflow: hidden
digunakan dan telah menjadi gaya yang diperdebatkan. Hal ini memudahkan beberapa
hal, seperti tidak perlu meneruskan nilai border-radius
ke
jalur, dan melacak elemen isi; tetapi hal ini juga berarti tidak ada turunan progres
yang dapat berada di luar elemen. Iterasi lain pada elemen progres
kustom ini dapat dilakukan tanpa overflow: hidden
dan dapat membuka beberapa
peluang untuk animasi atau status penyelesaian yang lebih baik.
Proses selesai
Pemilih CSS melakukan pekerjaan berat di sini dengan membandingkan maksimum dengan nilai, dan jika cocok, progres akan selesai. Setelah selesai, elemen pseudo akan dibuat dan ditambahkan ke akhir elemen progres, sehingga memberikan isyarat visual tambahan yang bagus untuk penyelesaian.
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);
}
Warna
Browser menghadirkan warnanya sendiri untuk elemen progres, dan adaptif terhadap terang dan gelap hanya dengan satu properti CSS. Hal ini dapat dibuat dengan beberapa pemilih khusus browser khusus.
Gaya browser terang dan gelap
Untuk mengikutsertakan situs Anda ke elemen <progress>
adaptif gelap dan terang,
color-scheme
adalah satu-satunya yang diperlukan.
progress {
color-scheme: light dark;
}
Warna terisi progres properti tunggal
Untuk memberi tint pada elemen <progress>
, gunakan accent-color
.
progress {
accent-color: rebeccapurple;
}
Perhatikan warna latar belakang jalur yang berubah dari terang menjadi gelap, bergantung pada
accent-color
. Browser memastikan kontras yang tepat: cukup rapi.
Warna terang dan gelap yang sepenuhnya kustom
Tetapkan dua properti kustom pada elemen <progress>
, satu untuk warna jalur
dan satu lagi untuk warna progres jalur. Di dalam
kueri media
prefers-color-scheme
, berikan nilai warna baru untuk progres lagu dan trek.
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%);
}
}
Gaya fokus
Sebelumnya, kita memberi elemen indeks tab negatif sehingga dapat difokuskan secara
terprogram. Gunakan
:focus-visible
untuk
menyesuaikan fokus agar dapat memilih gaya cincin fokus yang lebih cerdas. Dengan ini, klik mouse
dan fokus tidak akan menampilkan lingkaran fokus, tetapi klik keyboard akan melakukannya. Video YouTube ini membahasnya secara lebih mendalam dan
layak untuk ditinjau.
progress:focus-visible {
outline-color: var(--_progress);
outline-offset: 5px;
}
Gaya kustom di seluruh browser
Sesuaikan gaya dengan memilih bagian elemen <progress>
yang ditampilkan setiap browser. Penggunaan elemen progres terdiri dari satu tag, tetapi terdiri dari
beberapa elemen turunan yang diekspos melalui pemilih pseudo CSS. Chrome DevTools
akan menampilkan elemen ini kepada Anda jika Anda mengaktifkan setelan:
- Klik kanan pada halaman, lalu pilih Periksa Elemen untuk membuka DevTools.
- Klik roda gigi Settings di sudut kanan atas jendela DevTools.
- Di bagian judul Elemen, temukan dan aktifkan kotak centang Tampilkan DOM bayangan agen pengguna.
Gaya Safari dan Chromium
Browser berbasis WebKit seperti Safari dan Chromium mengekspos
::-webkit-progress-bar
dan ::-webkit-progress-value
, yang memungkinkan sebagian
CSS digunakan. Untuk saat ini, tetapkan background-color
menggunakan properti kustom
yang dibuat sebelumnya, yang beradaptasi dengan terang dan gelap.
/* Safari/Chromium */
progress[value]::-webkit-progress-bar {
background-color: var(--_track);
}
progress[value]::-webkit-progress-value {
background-color: var(--_progress);
}
Gaya Firefox
Firefox hanya mengekspos pemilih pseudo ::-moz-progress-bar
pada elemen <progress>
. Hal ini juga berarti kita tidak dapat langsung mewarnai jalur.
/* Firefox */
progress[value]::-moz-progress-bar {
background-color: var(--_progress);
}
Perhatikan bahwa Firefox memiliki warna jalur yang ditetapkan dari accent-color
, sedangkan iOS Safari
memiliki jalur biru muda. Hal ini sama dengan mode gelap: Firefox memiliki jalur gelap, tetapi
bukan warna kustom yang telah kita tetapkan, dan berfungsi di browser berbasis Webkit.
Animasi
Saat menggunakan pemilih pseudo bawaan browser, sering kali dengan kumpulan properti CSS yang diizinkan terbatas.
Menganimasikan trek yang terisi
Menambahkan transisi ke
inline-size
elemen progres berfungsi untuk Chromium, tetapi tidak untuk Safari. Firefox juga
tidak menggunakan properti transisi pada ::-moz-progress-bar
-nya.
/* Chromium Only 😢 */
progress[value]::-webkit-progress-value {
background-color: var(--_progress);
transition: inline-size .25s ease-out;
}
Menganimasikan status :indeterminate
Di sini saya menjadi sedikit lebih kreatif sehingga dapat memberikan animasi. Elemen pseudo untuk Chromium dibuat dan gradien diterapkan yang dianimasikan bolak-balik untuk ketiga browser.
Properti kustom
Properti kustom sangat bagus untuk banyak hal, tetapi salah satu favorit saya adalah
memberi nama ke nilai CSS yang terlihat ajaib. Berikut adalah linear-gradient
yang cukup kompleks, tetapi dengan nama yang bagus. Tujuan dan kasus penggunaannya dapat dipahami dengan jelas.
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;
}
Properti kustom juga akan membantu kode tetap DRY karena sekali lagi, kita tidak dapat menggabungkan pemilih khusus browser ini.
Keyframe
Tujuannya adalah animasi tanpa batas yang bolak-balik. Keyframe awal dan akhir
akan ditetapkan di CSS. Hanya satu keyframe yang diperlukan, yaitu keyframe tengah
di 50%
, untuk membuat animasi yang kembali ke tempat asalnya, berulang kali.
@keyframes progress-loading {
50% {
background-position: left;
}
}
Menargetkan setiap browser
Tidak semua browser mengizinkan pembuatan elemen pseudo pada elemen <progress>
itu sendiri atau memungkinkan animasi status progres. Lebih banyak browser yang mendukung
animasi jalur dibandingkan elemen pseudo, jadi saya mengupgrade dari elemen semu sebagai
dasar dan menjadi batang animasi.
Pseudo-elemen Chromium
Chromium mengizinkan elemen pseudo: ::after
yang digunakan dengan posisi untuk menutupi elemen. Properti kustom yang tidak ditentukan digunakan, dan animasi bolak-balik berfungsi dengan sangat baik.
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);
}
Status progres Safari
Untuk Safari, properti kustom dan animasi diterapkan ke status progres elemen pseudo:
progress:indeterminate::-webkit-progress-bar {
background: var(--_indeterminate-track);
background-size: var(--_indeterminate-track-size);
background-position: right;
animation: var(--_indeterminate-track-animation);
}
Status progres Firefox
Untuk Firefox, properti khusus dan animasi juga diterapkan ke status progres elemen pseudo:
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 memainkan peran penting dengan elemen <progress>
. Fungsi ini mengontrol
nilai yang dikirim ke elemen dan memastikan informasi yang memadai ada dalam
dokumen untuk pembaca layar.
const state = {
val: null
}
Demo ini menawarkan tombol untuk mengontrol progres; tombol tersebut memperbarui state.val
,
lalu memanggil fungsi untuk memperbarui
DOM.
document.querySelector('#complete').addEventListener('click', e => {
state.val = 1
setProgress()
})
setProgress()
Fungsi ini adalah tempat orkestrasi UI/UX terjadi. Mulai dengan membuat fungsi setProgress()
. Tidak ada parameter yang diperlukan karena memiliki akses ke
objek state
, elemen progres, dan zona <main>
.
const setProgress = () => {
}
Menetapkan status pemuatan di zona <main>
Bergantung pada apakah progres sudah selesai atau belum, elemen <main>
terkait memerlukan pembaruan pada
atribut
aria-busy
:
const setProgress = () => {
zone.setAttribute('aria-busy', state.val < 1)
}
Menghapus atribut jika jumlah pemuatan tidak diketahui
Jika nilai tidak diketahui atau tidak disetel, null
dalam penggunaan ini, hapus atribut value
dan
aria-valuenow
. Tindakan ini akan mengubah <progress>
menjadi tidak ditentukan.
const setProgress = () => {
zone.setAttribute('aria-busy', state.val < 1)
if (state.val === null) {
progress.removeAttribute('aria-valuenow')
progress.removeAttribute('value')
progress.focus()
return
}
}
Memperbaiki masalah matematika desimal JavaScript
Karena saya memilih untuk tetap menggunakan nilai default progres maksimum 1, fungsi
penambahan dan pengurangan demo menggunakan matematika desimal. JavaScript, dan bahasa
lainnya, tidak selalu baik dalam hal
ini.
Berikut adalah fungsi roundDecimals()
yang akan memangkas kelebihan dari hasil
matematika:
const roundDecimals = (val, places) =>
+(Math.round(val + "e+" + places) + "e-" + places)
Membulatkan nilai agar dapat ditampilkan dan dapat dibaca:
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 + "%"
}
Menetapkan nilai untuk pembaca layar dan status browser
Nilai ini digunakan di tiga lokasi di DOM:
- Atribut
value
elemen<progress>
. - Atribut
aria-valuenow
. - Konten teks dalam
<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
}
Memberikan fokus progres
Dengan nilai yang diperbarui, pengguna yang normal akan melihat perubahan progres, tetapi pengguna
pembaca layar belum diberi pengumuman perubahan. Fokuskan
elemen <progress>
dan browser akan mengumumkan update.
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()
}
Kesimpulan
Setelah Anda tahu cara saya melakukannya, bagaimana Anda melakukannya‽ 🙂
Tentu saja ada beberapa perubahan yang ingin saya lakukan jika diberi kesempatan lagi. Menurut saya, ada ruang untuk membersihkan komponen saat ini, dan ada ruang untuk mencoba membuat komponen tanpa batasan gaya class semu elemen <progress>
. Sebaiknya Anda mencobanya.
Mari kita diversifikasi pendekatan dan pelajari semua cara untuk mem-build di web.
Buat demo, link tweet saya, dan saya akan menambahkannya ke bagian remix komunitas di bawah.