liquiddesign

Zasada 02

Skeleton wierny treści

Skeleton loader ma sens tylko wtedy, gdy jego geometria jest identyczna z treścią, którą zapowiada. Najprostsza droga do wierności to jeden komponent, który renderuje ten sam layout w dwóch stanach, zamiast osobnego pliku ze zgadywanymi prostokątami.

List artykułów: szkielet i treść to ten sam layout

Przeładuj i patrz na krawędzie kart. Żadna nie drgnie, bo placeholdery dziedziczą typografię treści.

Maja Lis · 2 lip 2026

Aspect-ratio zamiast modlitwy o wymiary

Każdy element ładowany asynchronicznie może mieć zadeklarowany kształt, zanim przyjdą dane. Pokazuję, jak robić to systemowo, a nie przypadkiem od obrazka do obrazka.

Igor Wrona · 28 cze 2026

Jeden komponent, dwa stany

Osobny plik ze skeletonem rozjeżdża się z treścią po pierwszym refaktorze. Wspólny layout renderowany w dwóch stanach nie ma jak się rozjechać.

Nina Cios · 21 cze 2026

Line-clamp po obu stronach lustra

Skoro excerpt ma maksymalnie dwie linie, skeleton też ma dokładnie dwie. Typografia dyktuje wysokość placeholdera, nigdy odwrotnie.

Reguły

  • Skeleton i treść to jeden komponent i jeden layout, różni się tylko wypełnienie.
  • Placeholdery dziedziczą typografię, więc wysokość linii szkieletu równa się wysokości linii tekstu.
  • Linia szkieletu zajmuje dokładnie 1lh, a widoczny pasek jest niższy i wyśrodkowany, więc między liniami zostaje światło jak w prawdziwym tekście.
  • Liczba linii szkieletu odpowiada liczbie linii docelowej treści (line-clamp po obu stronach).
  • Podmiana szkieletu na treść nie zmienia ani jednego wymiaru kontenera.

Wzorzec w kodzie

Jeden komponent, w którym treść i szkielet dzielą layout
function ArticleCard({ article }: { article?: Article }) {
  const loading = !article;
  return (
    <article className="flex gap-4 p-5">
      <div
        className={
          loading
            ? "skeleton size-12 rounded-full"
            : "size-12 rounded-full bg-cover"
        }
      />
      <div className="min-w-0 flex-1">
        <h3 className="truncate text-lg font-semibold">
          {loading ? <SkeletonLine className="w-4/5" /> : article.title}
        </h3>
        <p className="mt-1 line-clamp-2 text-sm">
          {loading ? (
            <>
              <SkeletonLine className="w-full" />
              <SkeletonLine className="w-2/3" />
            </>
          ) : (
            article.excerpt
          )}
        </p>
      </div>
    </article>
  );
}
Linia szkieletu zajmuje dokładnie 1lh, a pasek jest niższy i wyśrodkowany
function SkeletonLine({ className = "" }: { className?: string }) {
  return (
    <span aria-hidden className="flex h-[1lh] items-center">
      <span className={`skeleton h-[0.75em] ${className}`} />
    </span>
  );
}