liquiddesign

Wzorzec 18

Scroll jako oś czasu

Pasek postępu czytania i elementy wjeżdżające przy scrollu wymagały dotąd nasłuchu zdarzeń, throttlingu i pracy w każdej klatce. Scroll-driven animations oddają oś czasu przeglądarce: scroll() mierzy postęp kontenera, view() pozycję elementu w oknie, a animacja jedzie na kompozytorze.

Proces studia, przewijany własnym tempem

Przewiń ramkę poniżej. Pasek postępu i wjazd kart czytają pozycję scrolla wprost z CSS: zero nasłuchu zdarzeń, zero throttlingu, animacja jedzie na kompozytorze.

  1. Krok 01

    Brief i inwentarz treści

    Zanim powstanie pierwszy piksel, spisujemy prawdziwe treści: najdłuższy tytuł, najkrótszy opis i wszystkie stany pośrednie.

  2. Krok 02

    Szkielet informacji

    Struktura widoku powstaje z samych kontenerów. Każdy z nich zna swoje minimalne wymiary, zanim dostanie treść.

  3. Krok 03

    Skala typograficzna

    Nazwane kroki clamp() zamiast rozmiarów per widok. Tekst płynie z kontenerem od zegarka po billboard.

  4. Krok 04

    Stany widoku

    Ładowanie, treść, pustka i błąd projektowane razem, w jednym kontenerze o wspólnej geometrii.

  5. Krok 05

    Prototyp w kodzie

    Zamiast klikalnej makiety: prawdziwy komponent w przeglądarce, z prawdziwym zawijaniem tekstu i prawdziwym fokusem.

  6. Krok 06

    Review na checkliście

    Design i kod przechodzą przez tę samą listę pytań o stabilność, dostępność i płynność.

  7. Krok 07

    Pomiar po wdrożeniu

    CLS i INP z prawdziwych sesji wracają do zespołu. Regres stabilności traktujemy jak błąd krytyczny.

Ta przeglądarka nie wspiera jeszcze scroll-driven animations, więc widzisz stan końcowy: karty stoją widoczne, a pasek jest pełny. Dokładnie tak wygląda tu poprawna degradacja.

Reguły

  • Postęp przewijania czyta animation-timeline: scroll(), nie zdarzenie scroll z throttlingiem.
  • Wjazd elementu wiąże view() z animation-range, więc dzieje się dokładnie tam, gdzie element wchodzi w kadr.
  • Animowane są transform i opacity, więc oś czasu ze scrolla nie dotyka layoutu.
  • Stan końcowy animacji jest stanem spoczynkowym: bez wsparcia i przy reduced motion treść po prostu stoi widoczna.

Wzorzec w kodzie

Postęp czytania bez nasłuchu zdarzenia scroll
.pasek-postepu {
  transform-origin: left center;
  animation: postep auto linear both;
  animation-timeline: scroll(nearest block);
}
@keyframes postep {
  from { scale: 0 1; }
  to { scale: 1 1; }
}
view() wie, kiedy element wchodzi w kadr
.karta {
  animation: wjazd auto linear both;
  animation-timeline: view(block);
  /* animacja trwa, póki karta wchodzi w kadr */
  animation-range: entry 5% entry 85%;
}
@keyframes wjazd {
  from { opacity: 0.15; translate: 0 1rem; }
  to { opacity: 1; translate: 0 0; }
}
@media (prefers-reduced-motion: reduce) {
  .karta { animation: none; }
}