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
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>
);
}function SkeletonLine({ className = "" }: { className?: string }) {
return (
<span aria-hidden className="flex h-[1lh] items-center">
<span className={`skeleton h-[0.75em] ${className}`} />
</span>
);
}