isirmtブログ
タグシリーズ投稿
お気に入りのタグ

ブログを書いている人

isirmt

ホームページ

GitHub
github_user_icon
isirmt
Software & Web
@isirmt

(管理者用)ダッシュボード

© isirmtBuild with NextjsBlogWithGitPAT  (bbf84c2)
  1. 【Next.js】両サイドにメニューがある場合のCSS設計
  2. 目標
  3. どうHTML要素を配置するか
  4. メニュー画面エリア内の設計
  5. position: sticky を利用する
  6. 画面全体の要素配置
  7. コンポーネントとして扱う
共有
2024
8
10

【Next.js】両サイドにメニューがある場合のCSS設計

コメント: 0件
Tailwind CSS
CSS
Next.js
当ブログのCSS設計 (2件)

前の投稿

最初の投稿です

次の投稿

2024
9
13

シリーズ【Next.js】日付カードコンポーネントとHydrationWarning

完成物 完成物 上のようなものを作ります。当ブログにもある年・月・日を表示します。 作成 イメージ イメージは「日めく...

読み込み中
シリーズを表示
Next.js
Tailwind CSS
HydrationWarning

目標

  • 左 :共通メニュー
  • 中央:メインコンテンツ
  • 右 :関連メニュー

左・右のどちらかは、お気に入りの表示や履歴といった常に表示することが目的となるようなコンテンツを置きたいと考えていました。

今回の設計では左にそれを配置し、右に目次等を置くことにしました。

どうHTML要素を配置するか

メニュー画面エリア内の設計

最初にメニュー画面(右・左)のエリア内を考えます。

position: fixedはスクロール位置に関係なく配置することに適しています。ヘッダーの作成で利用される方も多いのではないでしょうか。

しかし、レスポンシブデザイン設計の際に位置をrem単位やpx単位等の微調整する作業が求められることと思います。

position: sticky を利用する

position: stickyもスクロール中にtopを指定すると固定が可能です。<div>や<section>の中から出る際にコンテンツと共にスクロールされ、HTML要素の親子関係が維持された設計が可能になるため今回採用しました。

画面全体の要素配置

次に、画面全体の配置を考えます。

display: flexを複数回使うことで全体のレイアウトを構築しました。

まずは、下の画像をご覧ください。

  1. 外枠はflexプロパティとし、子要素に2個divを配置します。
  2. 1個目を共通メニューとし、2個目をflex flex-row-reverseプロパティにし、子要素に2個divを配置します。
  3. 1個目を関連メニューとし、2個目をメインコンテンツにして完成です。

flex-row-reverseを使いました。通常のCSS表記は下の通りです。

flex-row-reverse
.class {
  flex-direction: row-reverse;
}

これは、子要素を右から横に並べて配置する宣言です。flex-rowを使わないことで、レスポンシブデザイン適応時にdisplay: blockと宣言するだけで画像のオレンジ色の要素が「関連メニュー」→「メインコンテンツ」の順で縦に並びます。

コンポーネントとして扱う

Next.js環境下なのでこれらをコンポーネント化します。

layout.tsx(簡略)
export default function RootLayout({ children }: Readonly<{ children: React.ReactNode }>) {
  return (
    <html lang="ja">
      {/* 略 */}
      <body className="min-h-vh">
        <Header />

        {/* (画像の黒色の要素) */}
        <div className="md:flex justify-center">
          <Menu />
          {children}
        </div>

        <Footer />
      </body>
    </html>
  );
}
Menu.tsx
// (画像の青色1個目の要素)
export default function Menu() {
  return <div className="relative flex-grow-0 min-w-72 hidden xl:block">
    <div className="w-full sticky top-14">
      {/* 要素を記述 */}
    </div>
  </div>
}
PageLayout.tsx(簡易)
// (画像の青色2個目の要素)
export function Main({ children }: { children?: React.ReactNode }) {
  return <main className="flex-row-reverse flex-grow md:flex md:flex-grow-0">
    {children}
  </main>
}

// (画像のオレンジ色1個目の要素)
// md以上で要素を表示
export function Side({ children }: { children?: React.ReactNode }) {
  return <div className="hidden w-44 md:block lg:min-w-64">
    <div className="w-full sticky top-14">
      {children}
    </div>
  </div>
}

// (画像のオレンジ色1個目の要素)
// md未満ではstickyとなり画面トップにくっつく
export function SideMDShown({ children }: { children?: React.ReactNode }) {
  return <div className="sticky md:relative md:w-44 lg:min-w-64 top-0 z-20 md:z-auto">
    <div className="w-full sticky top-14 flex flex-row-reverse md:block">
      {children}
    </div>
  </div>
}

// (画像のオレンジ色2個目の要素)
export function Section({ children }: { children?: React.ReactNode }) {
  return <section className="p-8 rounded-3xl w-full md:w-[34rem] lg:w-[44rem] mx-auto xl:m-0">
    {children}
  </section>
}

大きさ指定は残した状態で記述してみました。右メニューは、必須でない場合にレスポンシブで非表示にするパターンも作りました。

これを使ったpage.tsxの記述方法も記載します。

page.tsx(例)
// import は省略
export default function TagList() {
  return <Main>
    <Side>
      右に書く内容
    </Side>
    <Section>
      メインコンテンツはここに
    </Section>
  </Main>
}

スッキリと書けました。編集する際もページ毎のレイアウトのズレも心配することなく、記述できるのではないでしょうか。

コメント

自動更新
コメントはまだありません
This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.
layout