土曜日, 8月 2, 2025
土曜日, 8月 2, 2025
- Advertisment -
ホームニューステックニュースReact.js UI/Container分離 × Hooks × Composition で責務を整理してみた話

React.js UI/Container分離 × Hooks × Composition で責務を整理してみた話



1 年以上前から、チームメンバーの提案で Container/Presentational パターンを現代的にアレンジした設計を導入し、徐々に改良を重ねてきました。

Hooks が普及して、関数コンポーネント内で状態管理や API 呼び出しができるようになって便利になった一方で、いくつかの困りごとがありました。

「見た目だけテストしたいのに、なんで API モックの準備が必要なんだろう…?」
「このコンポーネント、何の責務を持ってるんだっけ…?」
「新しいメンバーが入ったとき、どこを変更すればいいか分からない…」

確かにファイル数は Hooks だけの時より増えてしまいますが、責務分離により各ファイルが単純になり、特にチーム開発では「単純明快なルール」の方がうまく機能することが分かってきました。

そこで、一時期「もう古い」と言われがちだった Container/Presentational パターンを、Hooks と組み合わせて現代的に活用する設計を実践してきました。

その実践記録を共有してみます。

Container/Presentational パターンについて振り返ってみる

React を書いている方なら、Presentational/Container パターンについて聞いたことがあるかもしれません。

2015 年頃から広まったこのパターンですが、2019 年に React Hooks のコアチームメンバーである Dan Abramov が自身のブログ記事で「現在は推奨していない」と更新したことで、使われる機会が減ってきました。

彼が指摘したのは、Hooks の登場により「関数コンポーネントでも状態管理やライフサイクル処理ができるようになったため、パフォーマンス最適化以外の理由で無理に分ける必要がなくなった」ということでした。

確かにその通りで、Hooks があれば一つのコンポーネント内で多くのことができるようになりました。

よくありそうな困りごとの例

たとえば、ユーザー編集モーダルの VRT(Visual Regression Test)を追加したい場面を考えてみましょう。


UserEditModal isOpen={true} userId="123" />

このコンポーネントは一見シンプルに見えますが、内部では:

  • ユーザー情報の API 取得
  • フォームバリデーション
  • 更新 API 呼び出し
  • 楽観的更新

など、複数の処理が動いているとします。

そうすると、純粋に「見た目が崩れていないか」を確認したいだけなのに:

  • API サーバーのモック設定
  • 認証状態のセットアップ
  • 各種ビジネスロジックのテストデータ準備

これらの準備作業が必要になってしまいます。

Storybook を作成する場合も同様で、「ちょっとレイアウト確認したいだけなのに、なんでこんなに準備が必要なんだろう…」といった声をよく聞きます。

Hooks を使ったよくある実装例を見てみる

Hooks が便利なので、機能を一つのコンポーネントにまとめることがよくあります。例えばこんな感じです:


const UserManagementPage = () => {
  
  const [users, setUsers] = useState([]);
  const [selectedUser, setSelectedUser] = useState(null);
  const [isEditModalOpen, setIsEditModalOpen] = useState(false);
  

  
  

  return (
    div>
      SearchBox value={searchQuery} onChange={setSearchQuery} />
      UserList users={filteredUsers} onEditClick={handleEditClick} />
      {}
      UserEditModal
        isOpen={isEditModalOpen}
        user={selectedUser}
        onClose={() => setIsEditModalOpen(false)}
      />
    div>
  );
};

こういう実装はよく見かけますし、動作もします。

子コンポーネントでも同様の構造が続く


const UserEditModal = ({ isOpen, user, onClose }) => {
  
  const [formData, setFormData] = useState({});
  const [validationErrors, setValidationErrors] = useState({});
  const [isSubmitting, setIsSubmitting] = useState(false);

  
  

  return (
    Modal isOpen={isOpen}>
      UserForm
        initialData={user}
        onSubmit={handleSubmit}
        errors={validationErrors}
        isSubmitting={isSubmitting}
      />
    Modal>
  );
};

UserForm 内でも同様に


const UserForm = ({ initialData, onSubmit, errors, isSubmitting }) => {
  
  const [name, setName] = useState(initialData.name);
  const [email, setEmail] = useState(initialData.email);
  

  
  

  return form onSubmit={handleSubmit}>{}form>;
};

この構造で感じたちょっとした不便さ

このような実装でも十分動作しますが、開発していていくつか気になることがありました:

テストを書くとき

  • UI の見た目だけテストしたいのに、API モックの設定が必要
  • 子コンポーネントの動作を制御するための準備が多い

Storybook を作成するとき

  • レイアウトだけ確認したいのに、ビジネスロジックの依存関係を準備する必要

機能を再利用したいとき

  • 同じ UI で異なるビジネスロジックを使いたい場合に対応しにくい

こういったことから、「もしかして Container/Presentational パターンって、今でも使い道あるんじゃない?」と思うようになりました。

試してみた提案:UI/Container + Composition + Hooks の組み合わせ

こんな困りごとを解決するために、Container/Presentational パターンを Hooks や Composition と組み合わせることを試してみました。

4 つの要素で責務を整理するアイデア

先ほどの実装例を、4 つの要素で整理してみることを考えました:

1. UI 層(.ui.tsx):純粋な見た目とインタラクション
2. Container 層(.container.tsx):データ取得とロジック統合
3. Composition:外部からのコンポーネント合成・注入
4. Hooks(useXxx.ts):再利用可能なビジネスロジック

実際に書き換えてみた例

1. Hooks でロジックを切り出し


export const useUserManagement = () => {
  const [users, setUsers] = useState([]);
  const [searchQuery, setSearchQuery] = useState("");

  
  useEffect(() => {
    fetchUsers().then(setUsers);
  }, []);

  
  const filteredUsers = useMemo(() => {
    return users.filter(
      (user) =>
        user.name.includes(searchQuery) || user.email.includes(searchQuery)
    );
  }, [users, searchQuery]);

  return {
    users: filteredUsers,
    searchQuery,
    setSearchQuery,
  };
};


export const useUserEdit = () => {
  const [selectedUser, setSelectedUser] = useState(null);
  const [isModalOpen, setIsModalOpen] = useState(false);

  const openEditModal = (user) => {
    setSelectedUser(user);
    setIsModalOpen(true);
  };

  const closeEditModal = () => {
    setSelectedUser(null);
    setIsModalOpen(false);
  };

  return {
    selectedUser,
    isModalOpen,
    openEditModal,
    closeEditModal,
  };
};

2. UI 層は純粋な見た目のみ


export const UserManagementPageUI = ({
  users,
  searchQuery,
  onSearchChange,
  onEditClick,
  editModalElement, 
}) => {
  return (
    div>
      SearchBox value={searchQuery} onChange={onSearchChange} />
      UserList users={users} onEditClick={onEditClick} />
      {editModalElement} {}
    div>
  );
};


export const UserEditModalUI = ({
  isOpen,
  onClose,
  userFormElement, 
}) => {
  return (
    Modal isOpen={isOpen} onClose={onClose}>
      {userFormElement} {}
    Modal>
  );
};


export const UserFormUI = ({
  name,
  email,
  department,
  onNameChange,
  onEmailChange,
  onDepartmentChange,
  onSubmit,
  errors,
  isSubmitting,
}) => {
  return (
    form onSubmit={onSubmit}>
      input value={name} onChange={onNameChange} error={errors.name} />
      input value={email} onChange={onEmailChange} error={errors.email} />
      {}
      button type="submit" disabled={isSubmitting}>
        更新
      button>
    form>
  );
};

3. Container 層でロジックを統合


export const UserManagementPage = () => {
  const userManagement = useUserManagement();
  const userEdit = useUserEdit();

  return (
    UserManagementPageUI
      users={userManagement.users}
      searchQuery={userManagement.searchQuery}
      onSearchChange={userManagement.setSearchQuery}
      onEditClick={userEdit.openEditModal}
      editModalElement={
        UserEditModalContainer
          isOpen={userEdit.isModalOpen}
          user={userEdit.selectedUser}
          onClose={userEdit.closeEditModal}
        />
      }
    />
  );
};


export const UserEditModalContainer = ({ isOpen, user, onClose }) => {
  const userForm = useUserForm(user);

  return (
    UserEditModalUI
      isOpen={isOpen}
      onClose={onClose}
      userFormElement={
        UserFormUI
          {...userForm.formData}
          {...userForm.handlers}
          errors={userForm.errors}
          isSubmitting={userForm.isSubmitting}
        />
      }
    />
  );
};

こんな変化がありました

以前の実装

  • UI とビジネスロジックが同じコンポーネント内に混在
  • コンポーネントが深くネストされる構造
  • テストで子コンポーネントの動作を制御する必要

書き換え後

  • UI 層は純粋に props を受け取って表示するだけ
  • Composition で外部からコンポーネントを注入する構造
  • Hooks で切り出したロジックを他のコンポーネントでも使える
  • UI のテストとロジックのテストを分離できる

実践するために考えたルール

この設計を試すために、チームで考えたルールを紹介します:

ファイル種別 役割 命名例 やっていいこと やってはダメなこと
xxx.ui.tsx 見た目専用 UserForm.ui.tsx ・props の表示
・UI インタラクション
・レイアウト
・useState
・useEffect
・API 呼び出し
xxx.container.tsx ロジック統合 UserForm.container.tsx ・Hooks の呼び出し
・UI への props 渡し
・Composition
・直接的な DOM 操作
・スタイリング
useXxx.ts ビジネスロジック useUserForm.ts ・状態管理
・API 呼び出し
・計算処理
・JSX の return
・UI 固有の処理

大切にしたポイント

1. UI 層は子の振る舞いを知らない


const PageUI = ({ users }) => {
  return (
    div>
      UserList users={users} />
      {}
      {users.some((u) => u.isEditing) && div>編集中ですdiv>}
    div>
  );
};


const PageUI = ({ users, isAnyUserEditing, editingIndicator }) => {
  return (
    div>
      UserList users={users} />
      {editingIndicator} {}
    div>
  );
};

2. Composition で入れ子を避ける


const ParentUI = () => {
  return (
    div>
      Child1>
        Child2>
          Child3 />
        Child2>
      Child1>
    div>
  );
};


const ParentUI = ({ composedChildElement }) => {
  return (
    div>
      {composedChildElement} {}
    div>
  );
};


const ParentContainer = () => {
  const composedChild = (
    Child1Container>
      Child2Container>
        Child3Container />
      Child2Container>
    Child1Container>
  );

  return ParentUI composedChildElement={composedChild} />;
};

3. Container は UI とロジックの橋渡し役


const UserFormContainer = ({ user }) => {
  const form = useUserForm(user);
  const validation = useFormValidation(form.data);

  return (
    UserFormUI
      {...form.data}
      {...form.handlers}
      errors={validation.errors}
      onSubmit={form.submit}
    />
  );
};

この構成で得られる効果

1. テストの分離と効率化

UI 層のテストでは、Testing Library React を使ったイベント駆動のテストが書きやすくなりました。API モックや複雑な状態管理の準備が不要で、純粋にユーザーインタラクション(fireEvent)と表示確認に集中できます。

ロジックのテストは Hooks の単体テストとして分離。renderHookを使って UI の描画確認不要で、ビジネスロジックに集中できます。

統合テストは Container 層で、UI 層とロジック層が正しく連携しているかを確認。

こうすることで、テストの実行速度も向上し、どこに問題があるか特定しやすくなりました。

2. 開発・デザインレビューの効率化

Storybook での開発も楽になりました。UI 層が純粋なので、API 接続やビジネスロジックなしでレイアウト確認が可能。エラー状態、ローディング状態なども簡単に再現できます。

3. 柔軟性と再利用性の向上

同じ UI で異なるビジネスロジックを使いたい場合に対応しやすくなりました。UI を変更せずにロジック部分だけ差し替えられるのは便利です。

4. 複雑なビジネスロジックの柔軟な構築

複数の Hooks の組み合わせにより、フォーム処理、バリデーションなどを組み合わせた複雑な処理も整理して構築できそうです。各 Hooks が独立しているため、組み合わせの柔軟性が高まります。

5. チーム開発での期待効果

責務が分離されることで、並行開発がしやすくなりそうです。UI の担当者は見た目に集中でき、ロジックの担当者はビジネスロジックに集中できるのではないかと思います。

コードレビューでも、UI の変更は見た目のみ、ロジックの変更はビジネスロジックのみに集中してレビューできるため、レビューポイントが絞りやすくなるかもしれません。

新メンバーの学習でも、ファイル命名規則が統一され、責務が明確なので、どこに何があるか分かりやすくなりそうです。

まとめ:個人的には、まだ使い道があると思っています

Container/Presentational パターンについて「もう古い」という話もありますが、個人的には、Hooks や Composition と組み合わせることで、まだまだ現役で使えるパターンだと感じています。

今回試してみた構成:

1. UI 層(.ui.tsx):純粋な見た目とインタラクション
2. Container 層(.container.tsx):データ取得とロジック統合
3. Composition:外部からのコンポーネント合成・注入
4. Hooks(useXxx.ts):再利用可能なビジネスロジック

感じたメリット

実際に試してみて感じたのは:

  • テストが書きやすくなった:UI のテストは見た目だけ、ロジックのテストは Hooks だけに集中できる
  • Storybook の作成が楽になった:ビジネスロジックの準備なしで UI を確認できる

Hooks 時代だからこそ

Hooks が便利だからこそ、ついつい一つのコンポーネントに全部詰め込みがちです。でも「できる」ことと「やるべき」ことは別かもしれません。

もちろん、全てのプロジェクトでこの構成が最適というわけではありません。チームの規模、プロジェクトの複雑さ、開発期間など、様々な要因を考慮して選択するのが良いと思います。

ただ、「テストが書きにくい」「Storybook の準備が大変」といった困りごとがある場合は、一度試してみる価値があるのではないでしょうか。

UI 層が見た目に集中できるだけで、開発体験はかなり変わります。



Source link

Views: 0

RELATED ARTICLES

返事を書く

あなたのコメントを入力してください。
ここにあなたの名前を入力してください

- Advertisment -