1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

【React】Reactのstateを学びながら星評価を作ろう!!

Posted at

【React】Reactのstateを学びながら星評価を作ろう!!

はじめに

この記事は初学者が学びながらアウトプットとして残している記事になります。
同じところで詰まった方の力に少しでもなれればいいなと思います。
何度か練習でReactで簡単なアプリ作成していたのでサクッとcreate-react-appで作成した上で実行しています。

実行環境

  • M1 Mac
  • node v16.15.0

ディレクトリ

今回はサクッと動作確認できればと思ったので、下記のようにsrcに全てそのまま入れ込んでいます。ここからcomponentディレクトリに入れ使いやすく自身でしていただければと思います。

create-react-app
    ├ node_modules
    ├ public
    └  src
       ├ Star.js
       ├ StarRating.js
       └ index.js

作成準備

まず今回作成したものはReactハンズオンラーニングの6章ステート管理の部分になります。
create-react-appで雛形作成していただき次に今回使用する星のアイコンの準備で下記を実施ください。
様々なアイコンが使えるようになるライブラリになります。

npm i react-icons

その後、ディレクトリ構成のようにStar.jsファイルとStarRating.jsを作成ください。

とりあえず星を表示してみる

まずインストールしたアイコンを使用し星を表示させます。
先ほど作成したStarRatingファイルに下記を記述してください。
import { FaStar } from "react-icons/fa"この部分が先ほどインストールしたライブラリの読み込みになります。

StarRating.js
import React from "react";
import { FaStar } from "react-icons/fa";

export default function StarRating() {
  return [
    <FaStar color="red" />,
    <FaStar color="red" />,
    <FaStar color="red" />,
    <FaStar color="grey" />,
    <FaStar color="grey" />
  ];
}

記述できましたら、index.jsに下記の様に追記してStarRatingファイルを読み込みます。
そしてroot.render以降に記述のあるAppをStarRatingに変更してください。

index.js
import StarRating from './StarRating';

スクリーンショット 2023-05-14 20.34.15.png
npm startすると下記のような星マークが出ているかと思います!

Stateを使い星の状態を管理する

それではまずこの星が動かせる様になる状態にします。

StarRating.js

StarRating.js
import React, {useState} from "react";
import { Star } from "./Star";

export default function StarRating({ totalStars = 5 }) {
  const [selectedStars, setSelectedStars] = useState(0);
  return (
    <>
      {[...Array(totalStars)].map((n, i) => (
        <Star
          key={i}
          selected={selectedStars > i}
          onSelect={() => setSelectedStars(i + 1)}
        />
      ))}
      <p>
        {selectedStars} of {totalStars} stars
      </p>
    </>
  );
}

Star.js

Star.js
import { FaStar } from "react-icons/fa";
import React from "react";

export const Star = ({selected = false, onSelect = f => f }) => {
  return(
    <FaStar color={selected ? "red" : "grey"} onClick={onSelect} />
  )
};

こちらを記載のファイルに記述していただければとりあえず簡易版星評価の完成です!
次で細かく何が行われているか見ていきます!

StarRatingファイル説明

このファイルが今回のメインになります。下記の部分がまずstateの定義です。
selectedStarsが変数になりこの変数で星の状態を管理します。setSelectedStarsはselectedStarsの関数になります。命名は自由ですが、基本的に宣言した変数にsetをつけたものになります。デフォルトで0を指定しています。

StarRating.js
const [selectedStars, setSelectedStars] = useState(0);

そして下記の部分ではtotalStars=5という引数を受け取っているのですがこの要素数に対して、[...Array(totalStars)].map((n, i)で一度展開し要素の数だけ星を生成しています。mapは中身を1つずつ取り出しながら繰り返し以降の処理を行なっていくものです。
その以降の処理として1つずつStarコンポーネントとして描画していきます。selected={selectedStars > i}は繰り返し処理でiより大きいかどうかの判断をします。
onSelect={() => setSelectedStars(i + 1)}ではクリックされた時にプラス1しています。

StarRating.js
{[...Array(totalStars)].map((n, i) => (
  <Star
    key={i}
    selected={selectedStars > i}
    onSelect={() => setSelectedStars(i + 1)}
  />
))}

Starファイル説明

このファイルでは先ほどのselectedとonSelectプロパティが追加されておりonClick={onSelect}はそのままクリックされたかどうかの判断部分になります。StarRatingで作成されるこのでここでpropsを受け取ります。selectedは受け取るとデフォルトは=falseになっています。そして color={selected ? "red" : "grey"}で判断されます。A ?  "" : ""はAがtrueの時に:の左側を実行し、falseであれば:の右側を実行するものです。ここで色を分岐させています。

Star.js
export const Star = ({selected = false, onSelect = f => f }) => (
  <FaStar color={selected ? "red" : "grey"} onClick={onSelect} />
);

この部分は=>以降のかっこに気をつけてください。何も考えずに進めていた私は{}を使っておりなんでreturnないんだと10分ほど悩んでおりました、、、笑

これで完成です!!
スクリーンショット 2023-05-14 21.24.46.png

おわりに

最後まで見ていただきありがとうございました。
まだまだ初学者なので至らない点あればご教授いただけると幸いです。
現在は下記著書で学んでおり、今回の学びを記事としアウトプットさせていただきました。
下記リンクよりよければご購入ください!!

Reactハンズオンラーニング第二版

1
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?