sidenavコンポーネントの構築

応答性の高いスライドアウトsidenavを作成する方法の基本的な概要

この投稿では、応答性が高く、ステートフルで、キーボードナビゲーションをサポートし、JavaScriptの有無にかかわらず動作し、ブラウザー間で動作するWeb用のSidenavコンポーネントのプロトタイプを作成した方法について紹介します。デモを試しください。

ビデオがお好みの場合は、この投稿のYouTubeバージョンをご覧ください。

概要

応答性の高いナビゲーションシステムを構築するのは難しいです。多少のユーザーはキーボードを使用し、多少のユーザーは強力なデスクトップを使用し、一部のユーザーは小さなモバイルデバイスからアクセスします。訪問する全員がメニューを開いたり閉じたりできる必要があります。

デスクトップからモバイルへのレスポンシブ対応のレイアウトのデモ
iOSとAndroidでの明るいテーマと暗いテーマ

Web戦術

このコンポーネントの調査では、いくつかの重要なWebプラットフォーム機能を組み合わせることができました。

  1. CSS :target
  2. CSSグリッド
  3. CSS変換
  4. ビューポートとユーザー設定に関するCSSメディアクエリ
  5. focus UX拡張のためのJS

私のソリューションとしてはサイドバーを1つ準備し、 540px以下の「モバイル」ビューポートになる時だけで切り替わります。540pxは、モバイル向けのレイアウトと静的デスクトップ向けのレイアウトを切り替えるためのブレークポイントになります。

CSS :target pseudo-class

一つの<a>リンクはURLハッシュを#sidenav-openに設定し、別の1つは空( '' )に設定します。最後に、要素の一個にはハッシュと一致するidを持つようになります。

<a href="#sidenav-open" id="sidenav-button" title="Open Menu" aria-label="Open Menu">

<a href="#" id="sidenav-close" title="Close Menu" aria-label="Close Menu"></a>

<aside id="sidenav-open">
  …
</aside>

これらの各リンクをクリックすると、ページURLのハッシュ状態が変更され、疑似クラスを使用してサイドナビーゲションを表示したり、非表示にしたりします。

@media (max-width: 540px) {
  #sidenav-open {
    visibility: hidden;
  }

  #sidenav-open:target {
    visibility: visible;
  }
}

CSSグリッド

以前、絶対に位置又は固定位置のサイドナビゲーションのレイアウトとコンポーネントのみを使用しました。ただし、グリッドでは、 grid-area構文を使用して、同行または同列に複数の要素を割り当てることができます。

スタック

主要なレイアウト要素#sidenav-containerは、1行と2列を作成するグリッドであり、それぞれ1つがstackと呼ばれます。スペースが制限されている場合、CSSはすべての<main>要素の子を同じグリッド名に割り当て、すべての要素を同じスペースに配置してスタックを作成します。

#sidenav-container {
  display: grid;
  grid: [stack] 1fr / min-content [stack] 1fr;
  min-height: 100vh;
}

@media (max-width: 540px) {
  #sidenav-container > * {
    grid-area: stack;
  }
}

メニューの背景

<aside>は、サイドナビゲーションを含むアニメーション要素です。 2つの子が[nav]という名前を持つ<nav>と、メニューを閉じるために使用される[escape]という名前を持つ背景<a>となります。

#sidenav-open {
  display: grid;
  grid-template-columns: [nav] 2fr [escape] 1fr;
}

メニューオーバーレイとボタンに近くのネガティブスペースに適した比率を見つけるために、2fr1frを調整してください。

比率を変更する結果のデモ。

CSS 3D変換と遷移

これで、レイアウトはモバイルビューポートのサイズに応じてスタックされます。新規のスタイルを追加するまで、デフォルトで記事をオーバーレイしています。この次のセクションで狙っているUXは次のとおりです。

  • 開いたり閉じたりするアニメーションを作ります
  • ユーザーの同意を得るに限り、モーションでアニメートします
  • キーボードのフォーカスが画面外の要素に入らないようにvisibilityアニメートします

モーションアニメーションの実装を開始する頃には、アクセシビリティを念頭に置いて始めたいと思います。

アクセス可能なモーション

誰もスライドアウトモーションを体験したいわけではありません。私たちのソリューションでは、この設定がメディアクエリ内の--durationCSS変数を調整することにより適用されています。このメディアクエリ値は、(使用可能な場合)モーションに対するユーザーのオペレーティングシステムの設定を表します。

#sidenav-open {
  --duration: .6s;
}

@media (prefers-reduced-motion: reduce) {
  #sidenav-open {
    --duration: 1ms;
  }
}
durationが適用されている場合と適用されていない場合のインタラクションのデモ。

これで、サイドナビゲーションがスライドして開閉している間に、ユーザーが動きを抑えたい場合は、要素を即座にビューに移動して、動きのない状態を維持します。

移行、変換、翻訳

Sidenav out(デフォルト)

モバイルでのサイドナビゲーションのデフォルト状態をオフスクリーン状態に設定するには、変換を使用して要素transform: translateX(-110vw)を配置します。

box-shadowは非表示するときにメインビューポートを覗き込まないように確保するために、通常のオフスクリーンコードの-100vw10vwを追加したことにご注意ください。

@media (max-width: 540px) {
  #sidenav-open {
    visibility: hidden;
    transform: translateX(-110vw);
    will-change: transform;
    transition:
      transform var(--duration) var(--easeOutExpo),
      visibility 0s linear var(--duration);
  }
}
Sidenav in

#sidenav要素が:targetと一致したら、 translateX()の位置をhomebase 0に設定し、URLハッシュが変更されたときにCSSが要素を-110vw の「out」位置からvar(--duration)0 の「in」位置にスライドするのを確認します。

@media (max-width: 540px) {
  #sidenav-open:target {
    visibility: visible;
    transform: translateX(0);
    transition:
      transform var(--duration) var(--easeOutExpo);
  }
}

可視性の移行

現在の目標は、メニューが表示されていない時にスクリーンリーダーからメニューを非表示にすることため、システムはオフスクリーンのメニューに注目していません。 :targetが変わる時に可視性の移行を設定することで実現します。

  • 入ったら、可視性を移行しないでください。すぐに表示するので、要素がスライドインしてフォーカスを受け入れるのを確認できます。
  • 出る際に、可視性を移行するが、遅延させるため、移行の終了時hiddenに反転します。

アクセシビリティUXの機能強化

このソリューションは、状態を管理することを目的とするURLの変更に依存しています。もちろん、<a>要素がここで利用される必要があり、いくつかの優れたアクセシビリティ機能を無料で取得します。インタラクティブな要素を、意図を明確に表現するラベルで飾りましょう。

<a href="#" id="sidenav-close" title="Close Menu" aria-label="Close Menu"></a>

<a href="#sidenav-open" id="sidenav-button" class="hamburger" title="Open Menu" aria-label="Open Menu">
  <svg>...</svg>
</a>
ナレーションとキーボードインタラクションUXのデモ。

これで、主要なインタラクションボタンは、マウスとキーボードの両方に対する意図を明確に示しています。

:is(:hover, :focus)

この便利なCSS機能疑似セレクターを使用すると、ホバースタイルをフォーカスと共に共有することで、ホバースタイルをすばやくこなせます。

.hamburger:is(:hover, :focus) svg > line {
  stroke: hsl(var(--brandHSL));
}

JavaScriptを振りかける

閉じるにはescapeを押してください

キーボードのEscapeキーでメニューを閉じる必要がありますか?それを配線しましょう。

const sidenav = document.querySelector('#sidenav-open');

sidenav.addEventListener('keyup', event => {
  if (event.code === 'Escape') document.location.hash = '';
});
ブラウザの履歴

開閉する操作によって複数のエントリがブラウザの履歴にスタックされないようにするには、閉じるボタンに次のインラインにあるJavaScriptを追加します。

<a href="#" id="sidenav-close" title="Close Menu" aria-label="Close Menu" onchange="history.go(-1)"></a>

これにより、閉じるときにURL履歴エントリが削除され、メニューが開かれなかったかのようになります。

フォーカスUX

次のスニペットは、開いた後または閉じた後の開くボタンと閉じるボタンにfocusするのに役立ちます。切り替えを簡単にしたい。

sidenav.addEventListener('transitionend', e => {
  const isOpen = document.location.hash === '#sidenav-open';

  isOpen
      ? document.querySelector('#sidenav-close').focus()
      : document.querySelector('#sidenav-button').focus();
})

サイドナビが開いたら、閉じるボタンにfocusします。サイドナビが閉じたら、開くボタンにfocusします。JavaScriptの要素でfocus()を呼び出すことにより、それをできます。

結論

以上で私のやり方を理解したと思うが、あなたはどうしますか?これにより、面白いコンポーネントアーキテクチャが実現します。最初のバージョンを作る方は何方ですか? 🙂

私たちのやり方を多様化し、ウェブ上でビルドするためのすべての方法を調べましょう。Glitchを作成し、バージョンを私へツイートしてください。下のコミュニティリミックス セクションに追加します。

コミュニティリミックス