スケルトンスクリーン

読み込み中はプレースホルダを表示し、データ取得後に実コンテンツへフェードで差し替えるカードUI。商品リストやプロフィールの体感速度向上に使えます。

#css#javascript#skeleton#ux

ライブデモ

使用例(お題: カフェ MOON BREW)

この技法を「カフェ MOON BREW」というテーマのダミーサイトで実際に使った例です。

HTML
<!-- MOON BREW:本日のおすすめメニュー。読み込み中はスケルトン→完了で実カードへ -->
<div class="mb-menu">
  <header class="mb-menu__bar">
    <span class="mb-menu__logo"><span class="mb-menu__moon">☾</span> MOON BREW</span>
    <span class="mb-menu__sub">本日のおすすめ</span>
  </header>

  <div class="mb-grid" id="mbGrid" data-state="loading">
    <!-- カードは状態でスケルトン/実体を切り替え -->
    <article class="mb-card">
      <div class="mb-skel" aria-hidden="true">
        <div class="mb-skel__thumb shimmer"></div>
        <div class="mb-skel__line shimmer" style="--w:80%"></div>
        <div class="mb-skel__line shimmer" style="--w:55%"></div>
        <div class="mb-skel__price shimmer"></div>
      </div>
      <div class="mb-real">
        <div class="mb-real__thumb" style="--img:url('https://picsum.photos/240/180?random=51')"></div>
        <h3 class="mb-real__name">ハニーカフェラテ</h3>
        <p class="mb-real__desc">深煎りエスプレッソに国産はちみつのコク。</p>
        <p class="mb-real__price">¥620</p>
      </div>
    </article>

    <article class="mb-card">
      <div class="mb-skel" aria-hidden="true">
        <div class="mb-skel__thumb shimmer"></div>
        <div class="mb-skel__line shimmer" style="--w:75%"></div>
        <div class="mb-skel__line shimmer" style="--w:60%"></div>
        <div class="mb-skel__price shimmer"></div>
      </div>
      <div class="mb-real">
        <div class="mb-real__thumb" style="--img:url('https://picsum.photos/240/180?random=52')"></div>
        <h3 class="mb-real__name">琥珀カプチーノ</h3>
        <p class="mb-real__desc">きめ細かなフォームと焦がしカラメルの香り。</p>
        <p class="mb-real__price">¥580</p>
      </div>
    </article>
  </div>

  <button class="mb-reload" id="mbReload" type="button">メニューを更新</button>
</div>
CSS
/* MOON BREW:メニュー読み込み(スケルトン→実カード) */
:root {
  --cream: #f5ede1;
  --brown: #2b1d12;
  --amber: #c98a3b;
  --base: #e7dccb;
  --hi: #f7f1e6;
}

* { box-sizing: border-box; }

body {
  margin: 0;
  height: 400px;
  display: grid;
  place-items: center;
  font-family: "Hiragino Kaku Gothic ProN", system-ui, sans-serif;
  background: var(--cream);
  color: var(--brown);
  overflow: hidden;
}

.mb-menu {
  width: 340px;
  display: flex;
  flex-direction: column;
  gap: 14px;
  padding: 18px;
}

/* ヘッダー */
.mb-menu__bar {
  display: flex;
  align-items: baseline;
  justify-content: space-between;
}
.mb-menu__logo {
  font-family: "Hiragino Mincho ProN", serif;
  font-size: 15px;
  font-weight: 700;
  letter-spacing: 0.08em;
}
.mb-menu__moon { color: var(--amber); }
.mb-menu__sub { font-size: 11px; color: var(--amber); letter-spacing: 0.12em; }

/* カードグリッド */
.mb-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 14px; }
.mb-card {
  background: #fff;
  border-radius: 16px;
  padding: 10px;
  box-shadow: 0 8px 20px rgba(43, 29, 18, 0.08);
}

/* 状態で表示切り替え */
.mb-skel, .mb-real { display: contents; }
.mb-grid[data-state="loading"] .mb-real { display: none; }
.mb-grid[data-state="ready"] .mb-skel { display: none; }

/* スケルトン */
.mb-skel__thumb {
  height: 84px;
  border-radius: 12px;
  background: var(--base);
}
.mb-skel__line {
  height: 11px;
  width: var(--w, 100%);
  margin-top: 9px;
  border-radius: 6px;
  background: var(--base);
}
.mb-skel__price {
  height: 14px;
  width: 40%;
  margin-top: 10px;
  border-radius: 6px;
  background: var(--base);
}

/* シマー(光沢が流れる) */
.shimmer { position: relative; overflow: hidden; }
.shimmer::after {
  content: "";
  position: absolute;
  inset: 0;
  transform: translateX(-100%);
  background: linear-gradient(90deg, transparent, var(--hi), transparent);
  animation: mb-shimmer 1.4s infinite;
}
@keyframes mb-shimmer { 100% { transform: translateX(100%); } }

/* 実コンテンツ */
.mb-real__thumb {
  height: 84px;
  border-radius: 12px;
  background: var(--img) center/cover no-repeat, var(--base);
}
.mb-real__name { margin: 9px 0 4px; font-size: 14px; }
.mb-real__desc { margin: 0; font-size: 11px; line-height: 1.5; color: #6d5b49; }
.mb-real__price { margin: 8px 0 2px; font-size: 15px; font-weight: 700; color: var(--amber); }

.mb-grid[data-state="ready"] .mb-real { animation: mb-fade 0.5s ease both; }
@keyframes mb-fade { from { opacity: 0; transform: translateY(5px); } }

/* 更新ボタン */
.mb-reload {
  align-self: center;
  border: 1px solid rgba(43, 29, 18, 0.18);
  background: #fff;
  color: var(--brown);
  padding: 9px 20px;
  border-radius: 999px;
  font-size: 12px;
  cursor: pointer;
  transition: background 0.2s, transform 0.1s;
}
.mb-reload:hover { background: var(--amber); color: #fff; border-color: var(--amber); }
.mb-reload:active { transform: scale(0.96); }

@media (prefers-reduced-motion: reduce) {
  .shimmer::after { animation: none; opacity: 0.5; }
  .mb-grid[data-state="ready"] .mb-real { animation: none; }
}
JavaScript
// MOON BREW メニュー:スケルトン → 実カードへ(疑似フェッチ・ループ可)
const grid = document.getElementById('mbGrid');
const reload = document.getElementById('mbReload');

// 一定時間後にデータ取得完了とみなす
function loadMenu() {
  if (!grid) return;
  grid.dataset.state = 'loading';
  setTimeout(() => { grid.dataset.state = 'ready'; }, 1700);
}

// 更新ボタンで再読み込み
reload?.addEventListener('click', loadMenu);

// 初回起動
loadMenu();

コード

HTML
<!-- スケルトンスクリーン: 読み込み中はプレースホルダ、完了で実コンテンツへ差し替え -->
<div class="sk-stage">
  <div class="sk-card" id="skCard" data-state="loading">
    <!-- ローディング表示(スケルトン) -->
    <div class="sk-skeleton" aria-hidden="true">
      <div class="sk-thumb shimmer"></div>
      <div class="sk-lines">
        <div class="sk-line shimmer" style="--w:90%"></div>
        <div class="sk-line shimmer" style="--w:70%"></div>
        <div class="sk-line shimmer" style="--w:55%"></div>
        <div class="sk-chips">
          <span class="sk-chip shimmer"></span>
          <span class="sk-chip shimmer"></span>
        </div>
      </div>
    </div>
    <!-- 実コンテンツ(読み込み完了後) -->
    <div class="sk-real">
      <div class="sk-thumb sk-thumb--real"></div>
      <div class="sk-lines">
        <h3 class="sk-title">北欧デザインの椅子</h3>
        <p class="sk-text">職人が手仕上げしたオーク材のチェア。やわらかな曲線が長時間の座り心地を支えます。</p>
        <div class="sk-chips">
          <span class="sk-chip sk-chip--real">送料無料</span>
          <span class="sk-chip sk-chip--real">在庫あり</span>
        </div>
      </div>
    </div>
  </div>
  <button class="sk-reload" id="skReload" type="button">再読み込み</button>
</div>
CSS
:root {
  --bg: #0f1226;
  --card: #1a1f3c;
  --line: #2a3160;
  --base: #232a55;
  --hi: #313a72;
  --accent: #7c8cff;
  --txt: #e8ebff;
  --muted: #9aa3d8;
}
* { box-sizing: border-box; }
body {
  margin: 0;
  min-height: 100vh;
  display: grid;
  place-items: center;
  font-family: "Segoe UI", system-ui, sans-serif;
  background:
    radial-gradient(1200px 400px at 50% -10%, #2a2f63 0%, transparent 60%),
    var(--bg);
  color: var(--txt);
}
.sk-stage {
  display: flex;
  flex-direction: column;
  align-items: center;
  gap: 18px;
  padding: 24px;
}
.sk-card {
  width: 340px;
  background: var(--card);
  border: 1px solid var(--line);
  border-radius: 18px;
  padding: 18px;
  box-shadow: 0 18px 40px rgba(0, 0, 0, .45);
  display: grid;
  grid-template-columns: 96px 1fr;
  gap: 16px;
}
/* スケルトンと実体を状態で切り替え */
.sk-skeleton, .sk-real { display: contents; }
.sk-card[data-state="loading"] .sk-real { display: none; }
.sk-card[data-state="ready"]   .sk-skeleton { display: none; }

.sk-thumb {
  width: 96px;
  height: 96px;
  border-radius: 14px;
  background: var(--base);
}
.sk-thumb--real {
  background:
    conic-gradient(from 200deg, #ffb86b, #ff7eb6, #7c8cff, #ffb86b);
}
.sk-lines { display: flex; flex-direction: column; gap: 10px; min-width: 0; }
.sk-line {
  height: 12px;
  width: var(--w, 100%);
  border-radius: 6px;
  background: var(--base);
}
.sk-chips { display: flex; gap: 8px; margin-top: 6px; }
.sk-chip {
  width: 64px;
  height: 22px;
  border-radius: 999px;
  background: var(--base);
}
/* シマー(光沢が流れるアニメーション) */
.shimmer {
  position: relative;
  overflow: hidden;
}
.shimmer::after {
  content: "";
  position: absolute;
  inset: 0;
  transform: translateX(-100%);
  background: linear-gradient(90deg, transparent, var(--hi), transparent);
  animation: sk-shimmer 1.4s infinite;
}
@keyframes sk-shimmer { 100% { transform: translateX(100%); } }

.sk-title { margin: 0; font-size: 16px; letter-spacing: .02em; }
.sk-text { margin: 0; font-size: 13px; line-height: 1.6; color: var(--muted); }
.sk-chip--real {
  width: auto;
  padding: 0 12px;
  display: inline-grid;
  place-items: center;
  font-size: 11px;
  color: var(--accent);
  background: rgba(124, 140, 255, .14);
  border: 1px solid rgba(124, 140, 255, .35);
}
.sk-card[data-state="ready"] .sk-real { animation: sk-fade .5s ease both; }
@keyframes sk-fade { from { opacity: 0; transform: translateY(4px); } }

.sk-reload {
  border: 1px solid var(--line);
  background: var(--card);
  color: var(--txt);
  padding: 9px 18px;
  border-radius: 999px;
  font-size: 13px;
  cursor: pointer;
  transition: background .2s, transform .1s;
}
.sk-reload:hover { background: var(--hi); }
.sk-reload:active { transform: scale(.96); }

@media (prefers-reduced-motion: reduce) {
  .shimmer::after { animation: none; opacity: .5; }
  .sk-card[data-state="ready"] .sk-real { animation: none; }
}
JavaScript
// スケルトン → 実コンテンツへ自動で切り替えるデモ
const card = document.getElementById('skCard');
const reload = document.getElementById('skReload');

// 一定時間後にデータ取得完了とみなす(疑似フェッチ)
function loadContent() {
  if (!card) return;
  card.dataset.state = 'loading';
  // 1.8秒後に ready へ
  setTimeout(() => { card.dataset.state = 'ready'; }, 1800);
}

// 再読み込みボタン
reload?.addEventListener('click', loadContent);

// 初回起動
loadContent();

🤖 AIエージェント用プロンプト

このままコピーしてAIに貼り付け「追加する場所」だけ書き換えればOK
あなたは熟練のフロントエンドエンジニアです。私のWebサイトに「スケルトンスクリーン」の効果を追加してください。

# 追加してほしい効果
スケルトンスクリーン(ローダー & スケルトン)
読み込み中はプレースホルダを表示し、データ取得後に実コンテンツへフェードで差し替えるカードUI。商品リストやプロフィールの体感速度向上に使えます。

# 追加する場所
👉【ここに対象箇所を記入:例「トップのヒーローセクション」「お問い合わせボタン」「記事カードの一覧」など】

# 参考実装(この見た目・挙動を再現してください)
【HTML】
<!-- スケルトンスクリーン: 読み込み中はプレースホルダ、完了で実コンテンツへ差し替え -->
<div class="sk-stage">
  <div class="sk-card" id="skCard" data-state="loading">
    <!-- ローディング表示(スケルトン) -->
    <div class="sk-skeleton" aria-hidden="true">
      <div class="sk-thumb shimmer"></div>
      <div class="sk-lines">
        <div class="sk-line shimmer" style="--w:90%"></div>
        <div class="sk-line shimmer" style="--w:70%"></div>
        <div class="sk-line shimmer" style="--w:55%"></div>
        <div class="sk-chips">
          <span class="sk-chip shimmer"></span>
          <span class="sk-chip shimmer"></span>
        </div>
      </div>
    </div>
    <!-- 実コンテンツ(読み込み完了後) -->
    <div class="sk-real">
      <div class="sk-thumb sk-thumb--real"></div>
      <div class="sk-lines">
        <h3 class="sk-title">北欧デザインの椅子</h3>
        <p class="sk-text">職人が手仕上げしたオーク材のチェア。やわらかな曲線が長時間の座り心地を支えます。</p>
        <div class="sk-chips">
          <span class="sk-chip sk-chip--real">送料無料</span>
          <span class="sk-chip sk-chip--real">在庫あり</span>
        </div>
      </div>
    </div>
  </div>
  <button class="sk-reload" id="skReload" type="button">再読み込み</button>
</div>

【CSS】
:root {
  --bg: #0f1226;
  --card: #1a1f3c;
  --line: #2a3160;
  --base: #232a55;
  --hi: #313a72;
  --accent: #7c8cff;
  --txt: #e8ebff;
  --muted: #9aa3d8;
}
* { box-sizing: border-box; }
body {
  margin: 0;
  min-height: 100vh;
  display: grid;
  place-items: center;
  font-family: "Segoe UI", system-ui, sans-serif;
  background:
    radial-gradient(1200px 400px at 50% -10%, #2a2f63 0%, transparent 60%),
    var(--bg);
  color: var(--txt);
}
.sk-stage {
  display: flex;
  flex-direction: column;
  align-items: center;
  gap: 18px;
  padding: 24px;
}
.sk-card {
  width: 340px;
  background: var(--card);
  border: 1px solid var(--line);
  border-radius: 18px;
  padding: 18px;
  box-shadow: 0 18px 40px rgba(0, 0, 0, .45);
  display: grid;
  grid-template-columns: 96px 1fr;
  gap: 16px;
}
/* スケルトンと実体を状態で切り替え */
.sk-skeleton, .sk-real { display: contents; }
.sk-card[data-state="loading"] .sk-real { display: none; }
.sk-card[data-state="ready"]   .sk-skeleton { display: none; }

.sk-thumb {
  width: 96px;
  height: 96px;
  border-radius: 14px;
  background: var(--base);
}
.sk-thumb--real {
  background:
    conic-gradient(from 200deg, #ffb86b, #ff7eb6, #7c8cff, #ffb86b);
}
.sk-lines { display: flex; flex-direction: column; gap: 10px; min-width: 0; }
.sk-line {
  height: 12px;
  width: var(--w, 100%);
  border-radius: 6px;
  background: var(--base);
}
.sk-chips { display: flex; gap: 8px; margin-top: 6px; }
.sk-chip {
  width: 64px;
  height: 22px;
  border-radius: 999px;
  background: var(--base);
}
/* シマー(光沢が流れるアニメーション) */
.shimmer {
  position: relative;
  overflow: hidden;
}
.shimmer::after {
  content: "";
  position: absolute;
  inset: 0;
  transform: translateX(-100%);
  background: linear-gradient(90deg, transparent, var(--hi), transparent);
  animation: sk-shimmer 1.4s infinite;
}
@keyframes sk-shimmer { 100% { transform: translateX(100%); } }

.sk-title { margin: 0; font-size: 16px; letter-spacing: .02em; }
.sk-text { margin: 0; font-size: 13px; line-height: 1.6; color: var(--muted); }
.sk-chip--real {
  width: auto;
  padding: 0 12px;
  display: inline-grid;
  place-items: center;
  font-size: 11px;
  color: var(--accent);
  background: rgba(124, 140, 255, .14);
  border: 1px solid rgba(124, 140, 255, .35);
}
.sk-card[data-state="ready"] .sk-real { animation: sk-fade .5s ease both; }
@keyframes sk-fade { from { opacity: 0; transform: translateY(4px); } }

.sk-reload {
  border: 1px solid var(--line);
  background: var(--card);
  color: var(--txt);
  padding: 9px 18px;
  border-radius: 999px;
  font-size: 13px;
  cursor: pointer;
  transition: background .2s, transform .1s;
}
.sk-reload:hover { background: var(--hi); }
.sk-reload:active { transform: scale(.96); }

@media (prefers-reduced-motion: reduce) {
  .shimmer::after { animation: none; opacity: .5; }
  .sk-card[data-state="ready"] .sk-real { animation: none; }
}

【JavaScript】
// スケルトン → 実コンテンツへ自動で切り替えるデモ
const card = document.getElementById('skCard');
const reload = document.getElementById('skReload');

// 一定時間後にデータ取得完了とみなす(疑似フェッチ)
function loadContent() {
  if (!card) return;
  card.dataset.state = 'loading';
  // 1.8秒後に ready へ
  setTimeout(() => { card.dataset.state = 'ready'; }, 1800);
}

// 再読み込みボタン
reload?.addEventListener('click', loadContent);

// 初回起動
loadContent();

# 外部ライブラリ
なし(追加ライブラリ不要)

# 守ってほしいこと
- 既存のHTML構造・レイアウト・デザインを壊さないこと。必要に応じてクラス名・色・サイズを私のサイトに合わせて調整してよい。
- クラス名やidが既存と衝突しないよう、必要なら接頭辞で名前空間を分けること。
- レスポンシブ対応と prefers-reduced-motion への配慮を入れること。
- 私のサイトのフレームワーク(React / Vue / 素のHTML など)に合わせて実装すること。不明な場合は素のHTML/CSS/JSで提示し、組み込み手順も説明すること。
- 変更後の確認手順も簡潔に教えてください。