メインコンテンツに移動

スタイル付きコンポーネントとは?リアクトでCSSをコンポーネント化する手法を解説

リアクト開発では、画面を小さなコンポーネントに分割し、それぞれを組み合わせながらUIを構築します。そのため、CSSの管理方法も従来のようにページ全体へまとめて書く方法だけではなく、コンポーネント単位で整理する考え方が重要になります。ボタン、カード、入力フォーム、モーダル、ヘッダーなどのUI部品が増えていくと、どのCSSがどの部品に影響しているのか分かりにくくなり、クラス名の衝突や不要なスタイルの残存が発生しやすくなります。

スタイル付きコンポーネントは、リアクトコンポーネントとスタイルを近い場所で管理できるライブラリです。CSSをJavaScriptの中に記述し、スタイルを持ったコンポーネントとして定義できるため、UI部品ごとに見た目と構造をまとめやすくなります。たとえば、ボタン用のスタイルをボタンコンポーネントとして定義すれば、そのコンポーネントを使うだけで同じ見た目のボタンを再利用できます。これにより、リアクトのコンポーネント指向とCSS設計を自然に結び付けることができます。

一方で、スタイル付きコンポーネントは便利な反面、すべてのプロジェクトに必ず適しているわけではありません。JavaScript内にCSSを書くため、従来のCSSファイルとは管理感覚が異なり、チームによっては学習コストを感じる場合もあります。また、使い方を誤るとスタイル定義がコンポーネント内に増えすぎ、かえって可読性が下がることもあります。そのため、スタイル付きコンポーネントを導入する際は、基本構文だけでなく、実務での設計方針やメリット・デメリット、テーマ管理の考え方まで理解しておくことが重要です。

本記事では、スタイル付きコンポーネントの基本から、インストール方法、基本構文、プロパティを使った動的スタイル、スタイルの再利用、グローバルスタイル、テーマ管理、実務での使い方、注意点までを体系的に解説します。リアクトでCSSをどのように管理すべきか悩んでいる方や、コンポーネント単位でUIを整理したい方に向けて、実務で使いやすい考え方をまとめます。

1. スタイル付きコンポーネントとは?

スタイル付きコンポーネントとは、リアクトコンポーネントに直接スタイルを付与できるライブラリです。CSSをJavaScript内に記述し、スタイル付きのリアクトコンポーネントとして扱える点が特徴です。従来のCSSでは、HTML要素にクラス名を付け、そのクラスに対して外部CSSでスタイルを指定する方法が一般的でした。しかしスタイル付きコンポーネントでは、buttondiv などのHTML要素をベースに、スタイルを持ったコンポーネントを作成できます。

この仕組みにより、スタイルの適用範囲をコンポーネント単位で閉じ込めやすくなります。通常のCSSでは、クラス名が同じだったり、セレクタの指定が広すぎたりすると、意図しない要素にスタイルが影響することがあります。スタイル付きコンポーネントでは、生成されるクラス名が自動的に管理されるため、スタイルの衝突を防ぎやすくなります。リアクトのようにUIを部品化して作る開発では、このスコープ管理のしやすさが大きなメリットになります。

主な目的

項目内容
コンポーネント化UIとCSSを同じ単位で管理する
スコープ管理CSSの衝突や意図しない影響を防ぐ
再利用性共通スタイルを持つUI部品を作りやすくする
動的スタイルプロパティに応じて見た目を変更する
テーマ管理色や余白などを全体で統一しやすくする

1.1 コンポーネント単位でCSSを管理できる

スタイル付きコンポーネントの大きな特徴は、スタイルをコンポーネント単位で管理できることです。たとえば、ボタンの見た目を定義したい場合、通常のCSSでは .button のようなクラスを作成し、リアクト側で className を指定する必要があります。しかしスタイル付きコンポーネントでは、スタイルを持ったボタンコンポーネントを直接作成できるため、見た目と利用箇所の対応関係が分かりやすくなります。

コンポーネント単位でスタイルを閉じ込めると、UI部品の再利用もしやすくなります。ボタン、カード、ラベル、入力欄などをそれぞれスタイル付きコンポーネントとして定義しておけば、画面ごとに同じCSSを書き直す必要がありません。さらに、コンポーネント名そのものがUIの意味を表すため、コードを読んだときに「これはボタンである」「これはカードの見出しである」と理解しやすくなります。

1.2 CSSの衝突を防ぎやすい

通常のCSSでは、クラス名やセレクタの管理が大きな課題になります。たとえば、別々の画面で同じ .title というクラス名を使っていると、一方のスタイル変更がもう一方に影響してしまう可能性があります。プロジェクトが大きくなるほど、クラス名の命名規則やスタイルの影響範囲を慎重に管理する必要があります。

スタイル付きコンポーネントでは、定義したスタイルに対して自動的に一意のクラス名が生成されるため、意図しないCSSの衝突を防ぎやすくなります。これにより、特定のコンポーネントのスタイル変更が他のコンポーネントへ影響するリスクを減らせます。特に複数人で開発するリアクトプロジェクトでは、CSSの影響範囲を限定できることが保守性の向上につながります。

1.3 動的スタイルを扱いやすい

スタイル付きコンポーネントでは、リアクトのプロパティを使ってスタイルを動的に変更できます。たとえば、主要ボタンである場合だけ背景色を変える、無効状態の場合だけ透明度を下げる、サイズ指定に応じて余白や文字サイズを変える、といった実装が可能です。JavaScriptの値を使ってCSSを制御できるため、状態に応じたUI表現を作りやすくなります。

この動的スタイルは、リアクトの状態管理と相性が良いです。ボタンの種類、入力エラーの有無、選択状態、テーマの切り替えなど、UIは状態によって見た目が変わることが多くあります。スタイル付きコンポーネントを使うと、状態とスタイルの関係をコンポーネント内で表現できるため、条件分岐のあるスタイル管理が分かりやすくなります。

2. 基本構文

スタイル付きコンポーネントを使うには、まずライブラリをインストールし、リアクトファイル内で styled を読み込みます。その後、styled.buttonstyled.div のように、HTML要素をベースにしたスタイル付きコンポーネントを作成します。CSSの記述にはテンプレートリテラルを使うため、通常のCSSに近い感覚でスタイルを書くことができます。

基本構文は比較的シンプルですが、通常のCSSとは異なる点もあります。たとえば、スタイル定義がJavaScriptファイル内に入るため、コンポーネント構造とスタイルの関係が近くなります。また、スタイル付きコンポーネントはリアクトコンポーネントとして扱えるため、通常のJSX内で <Button> のように利用できます。これにより、スタイルを持つUI部品を自然に再利用できます。

2.1 インストール

スタイル付きコンポーネントを使うには、まずプロジェクトにライブラリを追加します。npmを使う場合は、以下のコマンドでインストールできます。リアクトプロジェクトであれば、通常は既存のプロジェクトディレクトリ内で実行します。

 

npm install styled-components

 

Yarnを使っている場合は、以下のようにインストールできます。チーム開発では、プロジェクトで利用しているパッケージ管理ツールに合わせてコマンドを選びます。

 

yarn add styled-components

 

インストール後は、リアクトコンポーネント内で styled を読み込んで使用します。スタイル付きコンポーネントはCSSをJavaScript内で扱うため、通常のCSSファイルを別途作成しなくても、コンポーネントファイル内でスタイル定義を完結できます。ただし、すべてのスタイルを同じファイルに詰め込むと読みにくくなるため、コンポーネントの規模に応じてファイル分割も検討する必要があります。

2.2 スタイル付きコンポーネントの作成

以下は、スタイル付きコンポーネントでボタンを作成する基本例です。styled.button を使うことで、HTMLの button 要素をベースにしたスタイル付きコンポーネントを定義しています。定義した Button は、通常のリアクトコンポーネントと同じようにJSX内で使用できます。

 

import styled from "styled-components";

const Button = styled.button`
  padding: 12px 20px;
  border: none;
  border-radius: 8px;
  background: #2563eb;
  color: #ffffff;
  font-size: 16px;
  cursor: pointer;
`;

export default function App() {
  return <Button>送信する</Button>;
}

 

この例では、ボタンの余白、角丸、背景色、文字色、文字サイズ、カーソルなどをスタイル付きコンポーネント内に直接記述しています。通常のCSSと似た書き方でありながら、Button というコンポーネントとして再利用できる点が特徴です。画面内で同じボタンを複数使いたい場合も、<Button> を呼び出すだけで同じスタイルを適用できます。

2.3 HTML要素をベースにできる

スタイル付きコンポーネントでは、button だけでなく、divsectionheaderinputa など、さまざまなHTML要素をベースにできます。たとえば、カードUIを作りたい場合は styled.div を使い、見出しを作りたい場合は styled.h2 を使うことができます。要素の意味を保ちながらスタイルを付けられるため、構造と見た目を整理しやすくなります。

 

const Card = styled.div`
  padding: 24px;
  border-radius: 12px;
  background: #ffffff;
  box-shadow: 0 8px 24px rgba(0, 0, 0, 0.08);
`;

const Title = styled.h2`
  margin: 0 0 12px;
  font-size: 24px;
  color: #111827;
`;

 

このように、UIの部品ごとにスタイル付きコンポーネントを作ることで、リアクトコンポーネントの中身が読みやすくなります。div に大量のクラス名を付けるよりも、CardTitle のように意味のある名前を付けた方が、UI構造を理解しやすくなります。実務では、この命名の分かりやすさが保守性に大きく影響します。

3. プロパティを使った動的スタイル

スタイル付きコンポーネントでは、リアクトコンポーネントに渡されたプロパティをもとにスタイルを変更できます。これにより、同じコンポーネントでも状態や用途に応じて見た目を変えることができます。たとえば、主要ボタンと通常ボタンを同じボタンコンポーネントで表現したり、エラー状態の入力欄だけ枠線を赤くしたりできます。

動的スタイルを使うと、スタイルのバリエーションをコンポーネント内で管理しやすくなります。従来のCSSでは、.button-primary.button-secondary のように複数のクラスを用意することが多くありますが、スタイル付きコンポーネントではプロパティによって条件分岐できます。これにより、UIの状態とスタイルの関係をより直接的に表現できます。

3.1 ボタンの種類を切り替える

以下の例では、primary というプロパティが渡された場合に背景色を変更しています。主要なアクションには青色のボタンを使い、それ以外の場合はグレーのボタンにするような設計です。

 

const Button = styled.button`
  padding: 12px 20px;
  border: none;
  border-radius: 8px;
  color: #ffffff;
  background: ${(props) => (props.primary ? "#2563eb" : "#6b7280")};
  cursor: pointer;
`;

export default function App() {
  return (
    <>
      <Button primary>保存する</Button>
      <Button>キャンセル</Button>
    </>
  );
}

 

このようにプロパティを使うと、同じコンポーネントを使いながら見た目だけを切り替えられます。実務では、ボタンの種類、サイズ、状態、重要度などをプロパティで管理することが多くあります。ただし、プロパティによる分岐が増えすぎるとコンポーネントが複雑になるため、バリエーションが多い場合は設計を見直すことも必要です。

3.2 入力エラー状態を表現する

フォームでは、入力エラーの有無によって枠線や背景色を変えることがあります。スタイル付きコンポーネントを使うと、error というプロパティに応じてスタイルを変更できます。これにより、UIの状態と見た目の関係を分かりやすく表現できます。

 

const Input = styled.input`
  width: 100%;
  padding: 10px 12px;
  border: 1px solid ${(props) => (props.error ? "#dc2626" : "#d1d5db")};
  border-radius: 6px;
  outline: none;
`;

export default function App() {
  return <Input error placeholder="メールアドレスを入力してください" />;
}

 

このような動的スタイルは、フォーム、タブ、メニュー、アコーディオン、カード、通知など、状態によって見た目が変わるUIで役立ちます。ただし、状態の数が増えると条件式が読みにくくなるため、必要に応じてヘルパー関数を分けたり、バリエーションごとに別コンポーネントを作ったりすると保守しやすくなります。

3.3 サイズをプロパティで切り替える

UIコンポーネントでは、同じボタンでも小さいサイズ、標準サイズ、大きいサイズを使い分けることがあります。スタイル付きコンポーネントでは、size のようなプロパティを使って余白や文字サイズを変更できます。

 

const Button = styled.button`
  padding: ${(props) => {
    if (props.size === "large") return "14px 24px";
    if (props.size === "small") return "8px 14px";
    return "10px 18px";
  }};
  font-size: ${(props) => {
    if (props.size === "large") return "18px";
    if (props.size === "small") return "14px";
    return "16px";
  }};
  border: none;
  border-radius: 8px;
  background: #111827;
  color: #ffffff;
`;

 

この方法を使うと、同じコンポーネントを複数の画面で柔軟に使い回せます。ただし、サイズや色のバリエーションをその場で自由に増やしすぎると、デザインの一貫性が失われる可能性があります。実務では、あらかじめ許可するサイズや色を決めておき、デザインシステムのルールに沿ってプロパティを設計することが重要です。

4. スタイル付きコンポーネントのメリット

スタイル付きコンポーネントのメリットは、リアクトのコンポーネント設計とCSS管理を自然に結び付けられる点にあります。スタイルをコンポーネント単位で定義できるため、UI部品ごとの責務が明確になり、スタイルの影響範囲も分かりやすくなります。特に大規模なリアクトアプリケーションでは、CSSの衝突や不要なスタイルの蓄積を防ぐことが保守性に大きく関わります。

また、JavaScriptの値を使ってスタイルを制御できる点も大きな特徴です。リアクトでは状態やプロパティによってUIが変化するため、その変化に合わせてスタイルも動的に変えられると実装が自然になります。ボタンの状態、フォームのエラー、テーマの切り替え、画面幅に応じた調整など、状態に応じたUI表現を実装しやすくなります。

4.1 スタイルの影響範囲を限定できる

スタイル付きコンポーネントでは、定義したスタイルが基本的にそのコンポーネントに紐づくため、他のコンポーネントへ意図せず影響するリスクを減らせます。通常のCSSでは、同じクラス名や広すぎるセレクタによってスタイルが衝突する可能性がありますが、スタイル付きコンポーネントでは自動生成されるクラス名によってスコープが管理されます。

このスコープ管理は、プロジェクトが大きくなるほど効果を発揮します。複数の開発者が同時にUIを実装する場合でも、コンポーネントごとにスタイルを閉じ込められるため、他人のCSSを壊すリスクが下がります。特に、共通UI部品を多く扱う管理画面や業務系アプリでは、スタイルの衝突防止が大きなメリットになります。

4.2 コンポーネントの再利用性が高まる

スタイル付きコンポーネントを使うと、見た目を含めたUI部品を再利用しやすくなります。たとえば、ボタン、カード、バッジ、モーダルのようなスタイル付きコンポーネントを作っておけば、複数の画面で同じUIを簡単に使い回せます。スタイルと構造がまとまっているため、再利用時に必要なクラス名を覚える負担も減ります。

再利用性が高まると、デザインの一貫性も保ちやすくなります。画面ごとに別々のCSSを書くと、少しずつ余白や色、角丸、文字サイズがずれていくことがあります。スタイル付きコンポーネントで共通コンポーネントを作っておけば、UIルールを一箇所で管理しやすくなり、アプリ全体の見た目を統一しやすくなります。

4.3 JavaScriptと連携しやすい

スタイル付きコンポーネントはJavaScript内でスタイルを記述するため、変数、関数、条件分岐などを活用できます。これにより、テーマカラーを参照したり、プロパティに応じてスタイルを切り替えたり、状態に応じてアニメーションを変更したりできます。リアクトの状態管理とCSSを近い場所で扱えるため、動的なUIを実装しやすくなります。

ただし、JavaScriptと連携しやすいからといって、すべてのスタイルを複雑な条件式で書くべきではありません。条件分岐が多すぎるスタイルは読みづらくなり、変更時の影響も追いにくくなります。実務では、単純なバリエーションはプロパティで管理し、複雑なスタイル変更はコンポーネント分割や設計見直しで対応することが重要です。

5. スタイル付きコンポーネントのデメリット

スタイル付きコンポーネントには多くのメリットがありますが、デメリットもあります。まず、CSSをJavaScript内に書くため、従来のCSSファイルに慣れている開発者にとっては書き方や管理方法が変わります。また、コンポーネント内にスタイル定義が増えると、ファイルが長くなり、ロジックとスタイルの境界が分かりにくくなることもあります。

さらに、チーム全体で設計ルールを決めずに使うと、似たようなスタイル付きコンポーネントが増えたり、命名がばらばらになったりする可能性があります。スタイル付きコンポーネントは便利ですが、自由度が高い分、プロジェクト内でのルール作りが重要です。導入する際は、メリットだけでなく、運用面の課題も理解しておく必要があります。

5.1 学習コストがある

スタイル付きコンポーネントは通常のCSSに近い書き方ができますが、JavaScript内でCSSを書くという点では独自の考え方があります。テンプレートリテラル、プロパティによる条件分岐、テーマ参照、グローバルスタイルなど、通常のCSSだけでは使わない概念も登場します。そのため、初めて使う場合は一定の学習コストがあります。

特に、CSSファイルを分けて管理する方法に慣れているチームでは、最初は違和感を持つ場合があります。どこにスタイルを書くのか、どの単位でコンポーネント化するのか、共通スタイルをどう管理するのかを理解する必要があります。学習コストを下げるには、最初にプロジェクト内の書き方ルールやコンポーネント設計方針を決めておくことが有効です。

5.2 ファイルが長くなりやすい

コンポーネント内にスタイル定義を直接書けることは便利ですが、コンポーネントが大きくなるとファイルが長くなりやすいです。ロジック、JSX、スタイルが同じファイルに大量に存在すると、どこを修正すればよいか分かりにくくなることがあります。特に画面単位の大きなコンポーネントでは、スタイル付きコンポーネントの定義が増えすぎると可読性が下がります。

この問題を避けるためには、スタイル付きコンポーネントを適切に分割することが重要です。たとえば、共通UI部品は別ファイルに切り出し、画面固有のスタイルだけを画面コンポーネント内に残す方法があります。また、スタイル定義をファイル下部にまとめる、または専用のスタイルファイルに分けるなど、チームで管理しやすいルールを決めるとよいでしょう。

5.3 設計ルールがないと乱れやすい

スタイル付きコンポーネントは自由度が高いため、ルールを決めずに使うとスタイル定義がばらばらになりやすいです。ある画面では主要ボタン、別の画面ではメインボタン、さらに別の画面では送信ボタンのように似た役割のコンポーネントが増えると、どれを使うべきか分かりにくくなります。結果として、再利用性が下がり、保守が難しくなります。

実務では、共通コンポーネントの命名、テーマの使い方、色や余白の定義、ファイル分割、プロパティの命名などをあらかじめ決めておくことが重要です。スタイル付きコンポーネントは設計がうまくいけば非常に便利ですが、ルールがない状態で使うと、プロジェクト全体のスタイル管理が複雑になる可能性があります。

6. スタイルの再利用

スタイル付きコンポーネントでは、共通スタイルを再利用する方法がいくつかあります。最も基本的なのは、スタイル付きコンポーネントをそのまま複数の場所で使う方法です。また、既存のスタイル付きコンポーネントを拡張して、新しいコンポーネントを作ることもできます。これにより、基本デザインを保ちながら、一部だけ変更したバリエーションを作れます。

再利用を意識すると、UI全体の一貫性を保ちやすくなります。たとえば、共通のボタンスタイルを作り、主要ボタン、危険操作ボタン、アウトラインボタンなどを派生として管理すれば、見た目の統一感を維持しながら用途に応じた表現ができます。実務では、同じようなスタイルを何度も書かないことが保守性の向上につながります。

6.1 既存コンポーネントを拡張する

スタイル付きコンポーネントでは、既存のスタイル付きコンポーネントをベースにして、新しいスタイルを追加できます。以下の例では、基本のボタンを作成し、それを拡張して危険操作用ボタンを作成しています。

 

const Button = styled.button`
  padding: 12px 20px;
  border: none;
  border-radius: 8px;
  background: #2563eb;
  color: #ffffff;
  cursor: pointer;
`;

const DangerButton = styled(Button)`
  background: #dc2626;
`;

 

この方法を使うと、共通部分を保ちながら一部だけ変更できます。危険操作用ボタンは基本ボタンの余白、角丸、文字色などを引き継ぎつつ、背景色だけを赤に変更しています。実務では、基本コンポーネントを作ってから用途別に拡張することで、重複を減らしながら一貫したUIを作れます。

6.2 共通スタイルを関数として切り出す

複数のコンポーネントで同じCSSを使いたい場合は、共通スタイルを切り出すこともできます。スタイル付きコンポーネントでは、css ヘルパーを使ってスタイルのまとまりを定義できます。これにより、同じスタイルを複数のコンポーネントに安全に再利用できます。

 

import styled, { css } from "styled-components";

const roundedBox = css`
  border-radius: 12px;
  box-shadow: 0 8px 24px rgba(0, 0, 0, 0.08);
`;

const Card = styled.div`
  ${roundedBox}
  padding: 24px;
  background: #ffffff;
`;

const Panel = styled.section`
  ${roundedBox}
  padding: 32px;
  background: #f9fafb;
`;

 

このように共通スタイルを切り出すと、同じデザインルールを複数のUI部品で使いやすくなります。ただし、共通化しすぎると逆に変更の影響範囲が分かりにくくなる場合があります。共通化するべきスタイルと、各コンポーネント固有に持たせるべきスタイルを分けて考えることが重要です。

6.3 共通UIコンポーネントとして管理する

実務では、ボタン、入力欄、カード、見出し、ラベル、バッジなどを共通UIコンポーネントとして管理することが多くあります。スタイル付きコンポーネントを使うと、これらのUI部品に必要なスタイルをまとめて定義できるため、プロジェクト全体で同じ見た目を再利用しやすくなります。

共通UIコンポーネントを管理するときは、単に見た目を共通化するだけでなく、使い方も分かりやすくする必要があります。どのプロパティでサイズを変えるのか、どのプロパティで色を変えるのか、どの状態をサポートするのかを整理しておくと、他の開発者も使いやすくなります。スタイル付きコンポーネントは、デザインシステムの実装にも活用しやすいライブラリです。

7. テーマ管理

スタイル付きコンポーネントでは、テーマ管理機能を使うことで、色、余白、フォントサイズ、角丸、影などのデザイン値を一元管理できます。たとえば、ブランドカラーや背景色をテーマとして定義し、各コンポーネントから参照することで、アプリ全体のデザインを統一しやすくなります。ダークモードや複数ブランド対応にも活用できます。

テーマ管理を導入すると、デザイン変更にも対応しやすくなります。もし主要カラーを変更する必要がある場合、各コンポーネントのCSSを個別に修正するのではなく、テーマ定義を変更するだけで多くのUIに反映できます。大規模なリアクトアプリでは、この一元管理が保守性に大きく影響します。

7.1 テーマ提供機能の基本

スタイル付きコンポーネントでは、テーマ提供機能を使ってテーマをアプリ全体に渡すことができます。以下の例では、テーマオブジェクトに色を定義し、コンポーネント内で props.theme から参照しています。

 

import styled, { ThemeProvider } from "styled-components";

const theme = {
  colors: {
    primary: "#2563eb",
    text: "#111827",
    background: "#ffffff",
  },
};

const Button = styled.button`
  background: ${(props) => props.theme.colors.primary};
  color: #ffffff;
  padding: 12px 20px;
  border: none;
  border-radius: 8px;
`;

export default function App() {
  return (
    <ThemeProvider theme={theme}>
      <Button>保存する</Button>
    </ThemeProvider>
  );
}

 

このようにテーマを使うと、色や余白などのデザイン値を直接コンポーネント内に書き散らさずに済みます。特に複数のUI部品で同じ色を使う場合、テーマから参照することで一貫性を保ちやすくなります。デザイン変更が発生した場合も、テーマを更新するだけで反映しやすくなります。

7.2 色や余白を一元管理する

テーマには、色だけでなく、余白、フォントサイズ、角丸、影、ブレークポイントなども定義できます。これにより、プロジェクト全体で同じデザインルールを使いやすくなります。たとえば、余白を小、中、大のように定義しておけば、画面ごとにばらばらな余白が生まれにくくなります。

 

const theme = {
  colors: {
    primary: "#2563eb",
    danger: "#dc2626",
    text: "#111827",
    muted: "#6b7280",
  },
  space: {
    sm: "8px",
    md: "16px",
    lg: "24px",
  },
  radius: {
    md: "8px",
    lg: "12px",
  },
};

 

このようなテーマ設計は、デザインシステムの土台になります。デザイナーとエンジニアが同じ色や余白のルールを共有できれば、実装時のズレを減らせます。スタイル付きコンポーネントのテーマ管理は、単なる便利機能ではなく、UI品質を安定させるための重要な仕組みです。

7.3 ダークモードへの活用

テーマ管理は、ダークモードの実装にも向いています。ライトテーマとダークテーマをそれぞれ定義し、状態に応じてテーマ提供機能に渡すテーマを切り替えれば、コンポーネント側の記述を大きく変えずに見た目を変更できます。

 

const lightTheme = {
  colors: {
    background: "#ffffff",
    text: "#111827",
  },
};

const darkTheme = {
  colors: {
    background: "#111827",
    text: "#ffffff",
  },
};

const Page = styled.main`
  min-height: 100vh;
  background: ${(props) => props.theme.colors.background};
  color: ${(props) => props.theme.colors.text};
`;

 

ダークモードを実装する場合、背景色と文字色だけでなく、境界線、影、ボタン、入力欄、カード、リンク色などもテーマとして設計する必要があります。部分的に色を直接書いてしまうと、テーマ切り替え時に不自然な表示になることがあります。最初からテーマを前提に設計しておくと、後から表示モードを増やすときにも対応しやすくなります。

8. グローバルスタイル

スタイル付きコンポーネントはコンポーネント単位のスタイル管理に向いていますが、すべてのスタイルをコンポーネントごとに書くわけではありません。ページ全体に適用するリセットCSS、基本フォント、背景色、body の余白、リンクの基本スタイルなどは、グローバルスタイルとして定義することがあります。

スタイル付きコンポーネントでは、createGlobalStyle を使ってグローバルスタイルを作成できます。これにより、通常のCSSファイルを使わずに、アプリ全体の基本スタイルをJavaScript内で管理できます。ただし、グローバルスタイルはアプリ全体に影響するため、範囲を広げすぎないことが重要です。

8.1 グローバルスタイル作成機能の基本

以下は、createGlobalStyle を使ってアプリ全体の基本スタイルを定義する例です。body の余白をなくし、フォントを指定し、背景色と文字色をテーマから参照しています。

 

import { createGlobalStyle } from "styled-components";

const GlobalStyle = createGlobalStyle`
  body {
    margin: 0;
    font-family: system-ui, sans-serif;
    background: ${(props) => props.theme.colors.background};
    color: ${(props) => props.theme.colors.text};
  }

  * {
    box-sizing: border-box;
  }
`;

 

このようにグローバルスタイルを使うと、アプリ全体の基本ルールを一箇所で管理できます。特に box-sizingbody の余白、基本フォントなどは、多くのコンポーネントに共通するため、グローバルに設定する方が自然です。一方で、ボタンやカードなどの具体的なUIスタイルまでグローバルに書きすぎると、スタイル付きコンポーネントのスコープ管理のメリットが薄れるため注意が必要です。

8.2 リセットCSSとの関係

Webブラウザには、要素ごとに標準スタイルが用意されています。たとえば、body には余白があり、見出しには大きな文字サイズや上下余白があり、ボタンにも標準の装飾があります。リセットCSSや標準化CSSは、これらの差を整え、ブラウザ間の表示差を減らすために使われます。

スタイル付きコンポーネントを使う場合でも、基本的なリセットや標準化は必要です。グローバルスタイル作成機能を使えば、プロジェクトに合わせたリセットCSSを定義できます。ただし、すべての要素の見た目を強くリセットしすぎると、アクセシビリティや標準の操作感に影響することがあります。必要な範囲だけを整え、コンポーネントごとのスタイルと役割を分けることが大切です。

8.3 グローバルに書きすぎない

グローバルスタイルは便利ですが、書きすぎると通常のCSSと同じように影響範囲が広くなります。たとえば、すべての button に強いスタイルを指定すると、個別のボタンコンポーネントのスタイルと競合する可能性があります。スタイル付きコンポーネントを使うメリットは、スタイルをコンポーネント単位で管理できることなので、グローバルスタイルは最小限に抑える方が安全です。

実務では、グローバルスタイルにはリセット、基本フォント、背景、共通のボックスサイズ設定などを置き、具体的なUIの見た目はコンポーネント側で管理するのが分かりやすいです。アプリ全体に影響するルールと、個別コンポーネントのルールを明確に分けることで、保守性を高められます。

9. 実務での使い方

実務でスタイル付きコンポーネントを使う場合、単にスタイルをJavaScript内に書くだけでは不十分です。どの単位でコンポーネントを作るのか、どのスタイルを共通化するのか、テーマをどのように管理するのか、画面固有のスタイルと共通UIをどう分けるのかを考える必要があります。設計方針がないまま使うと、似たようなコンポーネントが増え、保守しにくくなる可能性があります。

スタイル付きコンポーネントは、コンポーネント指向のUI設計と相性が良いです。たとえば、デザインシステムに基づいてボタンやカードを共通化し、画面ごとのレイアウトは別のスタイル付きコンポーネントで管理する方法があります。共通部品と画面固有部品を分けることで、再利用性と柔軟性のバランスを取りやすくなります。

9.1 共通コンポーネントを先に設計する

スタイル付きコンポーネントを導入する場合、まず共通で使うUI部品を整理すると効果的です。ボタン、入力欄、ラベル、カード、見出し、モーダル、アラートなど、複数画面で使われる部品を共通コンポーネントとして作成します。これにより、画面ごとに同じようなスタイルを繰り返し書く必要がなくなります。

共通コンポーネントを作るときは、どの程度まで柔軟にするかが重要です。すべてのバリエーションを一つのコンポーネントに詰め込むと、プロパティが増えすぎて使いにくくなります。一方で、少し違うだけのコンポーネントを大量に作ると、再利用性が下がります。実務では、基本形を明確にし、必要なバリエーションだけをプロパティで切り替える設計が使いやすいです。

9.2 画面固有のスタイルと分ける

共通UI部品とは別に、画面固有のレイアウトや余白を管理するスタイルも必要です。たとえば、一覧画面のコンテナ、詳細画面のセクション、フォーム画面のレイアウトなどは、共通部品ではなく画面固有のスタイルとして扱う方が自然です。スタイル付きコンポーネントでは、こうした画面固有のラッパーコンポーネントも作成できます。

画面固有のスタイルを共通コンポーネントに混ぜすぎると、共通部品の責務が曖昧になります。たとえば、ボタンコンポーネントに画面ごとの外側余白まで含めると、別の画面で使いにくくなることがあります。共通コンポーネントは部品そのものの見た目に集中し、配置やレイアウトは画面側で管理する方が保守しやすくなります。

9.3 デザインシステムと連携する

スタイル付きコンポーネントは、デザインシステムと組み合わせることでより効果を発揮します。テーマに色、余白、フォント、角丸、影などを定義し、共通コンポーネントがそれを参照するようにすれば、UI全体の一貫性を保ちやすくなります。デザイナーとエンジニアが共通のデザイントークンを使えるようにすることで、実装とデザインのズレも減らせます。

デザインシステムとの連携では、自由に値を書くのではなく、定義済みの値を使うことが大切です。たとえば、余白を毎回 13px17px のように個別指定すると、UI全体の統一感が崩れます。テーマに定義した余白や色の値を使うことで、見た目の一貫性と保守性を両立できます。

10. スタイル付きコンポーネントを使うときの注意点

スタイル付きコンポーネントを使うと、コンポーネント単位でスタイルを整理しやすくなりますが、運用ルールがないとコードが乱れる可能性があります。特に、スタイル付きコンポーネントをどこに定義するか、どのように命名するか、プロパティでどこまで見た目を変えるかは、プロジェクト全体で統一しておく必要があります。自由に書けるからこそ、設計方針が重要になります。

また、スタイル付きコンポーネントはJavaScript実行時にスタイルを生成するため、プロジェクトの構成やレンダリング方式によってはパフォーマンスや表示タイミングに注意が必要になることもあります。通常の中小規模アプリでは大きな問題にならないことも多いですが、大規模アプリやサーバー側レンダリングを扱う場合は、導入方法を慎重に確認する必要があります。

10.1 命名を分かりやすくする

スタイル付きコンポーネントでは、スタイル付きコンポーネントに意味のある名前を付けることが重要です。BoxWrapper のような名前は便利ですが、使いすぎると何を表しているのか分かりにくくなります。たとえば、カードコンテナ、フォームセクション、主要ボタン、エラーメッセージのように、役割が分かる名前にするとコードの可読性が高まります。

命名が分かりやすいと、JSXを見たときに画面構造を理解しやすくなります。逆に、似たような名前のコンポーネントが大量にあると、どれを使うべきか迷いやすくなります。実務では、共通コンポーネントと画面固有コンポーネントの命名ルールを分けておくと、保守しやすくなります。

10.2 プロパティ分岐を増やしすぎない

スタイル付きコンポーネントではプロパティによってスタイルを切り替えられますが、条件分岐を増やしすぎると読みづらくなります。たとえば、一つのボタンコンポーネントに主要、危険、枠線、軽量、大、小、角丸などを大量に追加すると、スタイル定義が複雑になり、どの組み合わせが正しいのか分かりにくくなります。

このような場合は、バリエーション設計を見直すことが重要です。使用頻度の高いものだけをプロパティで管理し、特殊な見た目は別コンポーネントとして分ける方法もあります。スタイル付きコンポーネントでは柔軟な条件分岐が可能ですが、柔軟にしすぎると保守性が下がるため、シンプルな設計を意識することが大切です。

10.3 共通化と個別化のバランスを取る

スタイル付きコンポーネントでは、共通スタイルを切り出したり、既存コンポーネントを拡張したりできます。しかし、すべてを共通化しようとすると、かえって変更しにくくなることがあります。共通化されたスタイルを修正したときに、どの画面へ影響するのか分かりにくくなるためです。

実務では、明らかに共通で使われるUI部品は共通化し、画面固有の表現は個別に管理する方が安全です。共通化は重複を減らすために重要ですが、過剰な共通化は柔軟性を損ないます。スタイル付きコンポーネントを使う際は、再利用性と変更しやすさのバランスを意識する必要があります。

おわりに

スタイル付きコンポーネントは、リアクトでCSSをコンポーネント単位に管理するための便利なライブラリです。CSSをJavaScript内に記述し、スタイル付きコンポーネントとして扱うことで、UI部品とスタイルの関係を分かりやすく整理できます。CSSの衝突を防ぎやすく、プロパティを使った動的スタイルやテーマ管理にも対応しやすいため、リアクトアプリケーションのスタイル管理において有力な選択肢になります。

特に、ボタン、カード、フォーム、モーダルなどの共通UI部品を多く扱うプロジェクトでは、スタイル付きコンポーネントのメリットを感じやすくなります。コンポーネント単位でスタイルを定義できるため、再利用性を高めながら、デザインの一貫性も保ちやすくなります。また、テーマ提供機能を使えば、色や余白、フォントサイズなどを一元管理できるため、ダークモードやデザインシステムとの連携にも対応しやすくなります。

一方で、スタイル付きコンポーネントは自由度が高い分、設計ルールが重要です。スタイル付きコンポーネントの命名、ファイル分割、プロパティによる分岐、共通化の範囲、テーマの使い方などを整理せずに使うと、コードが複雑になりやすくなります。導入する際は、プロジェクトの規模やチームの開発方針に合わせて、使い方のルールを決めることが大切です。

スタイル付きコンポーネントを適切に活用すれば、リアクトのコンポーネント指向とCSS設計を自然に結び付けることができます。見た目と構造を近い場所で管理しながら、再利用性、保守性、一貫性を高められるため、実務のリアクト開発において非常に有用なスタイル管理手法だと言えるでしょう。

LINE Chat