1
0

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.

NestとReactを使った簡易版twitterの作成-フロントエンド編

Last updated at Posted at 2023-02-25

NestとReactを使った簡易版twitterの作成-フロントエンド編

1.開発準備
2.ディレクトリ、ファイルの作成
3.App.tsx (ページ遷移制御)
4.Signup.tsx (サインアップ画面)
5.Login.tsx (ログイン画面)
6.Home.tsx (Tweet 投稿・表示画面)
6.最後に

開発準備

はじめに

ここではフロントエンド編の作成に取り組んでいきます。フロントエンドって聞いてぱっとしない方もいると思いますが要は見た目部分をぱぱっと作ってしまおう編です。

今回は 2 月に学んでもらった React を使ってフロントエンドを実装してもらう訳ですが、React を開発したのはかの有名な Facebook です。バックエンド編でザッカーバーグが出てきたのは実は伏線だったのかもしれませんね。

バックエンド同様質問は受け付けます。Slack で質問お願いします。

React アプリの起動

ターミナルを開いて、for-spring-training 上でコンテナ起動

docker-compose up

http://localhost:3000を開き以下の初期画面になっていたら正常です。
localhost.png

初期画面の変更

実際にフロントエンド開発をしていきましょう!

今回作るアプリの仕様

作るもの:簡易版 Twitter
今回のページ構成は以下の様になります

  1. ログイン
  2. ユーザー登録
  3. ツイート機能

ディレクトリファイルの作成

今回の frontend のディレクトリ構造は、このようになっています。

 src
 ├── pages
 │   ├── Signup.tsx
 │   ├── Login.tsx
 │   └── Home.tsx
 ├── App.tsx
 └── index.tsx

ページ遷移制御

frontend/src/App.tsx に下記のコードをコピーして下さい。

App.tsx
import { BrowserRouter, Routes, Route } from "react-router-dom"
import Home from "./pages/Home";
import Login from "./pages/Login";
import Signup from "./pages/Signup";

function App() {
  return (
    <div className="App">
      <BrowserRouter>
        <Routes>
          <Route path='/' element={<Home/>}/>
          <Route path='/login' element={<Login/>}/>
          <Route path='/signup' element={<Signup/>}/>
        </Routes>
      </BrowserRouter>
    </div>
  );
}

export default App;

解説

App.tsx では、React Router を用いて、ページの遷移を設定しています。 今回は、Homeが tweet 実行、表示画面、Login がログイン画面Signupがサインアップ画面です。

サインアップ画面

Signup 画面を作成していきます。
まずは、frontend/src/pages/Signup.tsx に下記のコードをコピーして下さい。

Signup.tsx
import React, { useState } from "react";
import { useNavigate } from "react-router-dom";
import {
  MDBBtn,
  MDBContainer,
  MDBCard,
  MDBCardBody,
  MDBCardImage,
  MDBRow,
  MDBCol,
  MDBInput,
  MDBCheckbox,
} from "mdb-react-ui-kit";
import axios from "axios";

// サインアップ処理
function SignUp() {
  const [name, setName] = useState("");
  const [email, setEmail] = useState("");
  const [password, setPassword] = useState("");

  const navigate = useNavigate();
  const handleSignUp = async (event: React.FormEvent<HTMLFormElement>) => {
    event.preventDefault();
    //axiosでpostを送る部分
    try {
      const response = await axios.post("http://localhost:3002/users/create", {
        name: name,
        email: email,
        password: password,
      });
      sessionStorage.setItem("id", response.data.id);
      sessionStorage.setItem("name", response.data.name);
      navigate("/");
    } catch (error) {
      console.error(error);
    }
  };
  return (
    <MDBContainer className="my-5">
      <MDBCard>
        <MDBRow className="g-0 d-flex align-items-center">
          <MDBCol md="4">
            <MDBCardImage
              src="https://mdbootstrap.com/img/new/ecommerce/vertical/004.jpg"
              alt="phone"
              className="rounded-t-5 rounded-tr-lg-0"
              fluid
            />
          </MDBCol>

          <MDBCol md="8">
            <MDBCardBody>
              <form onSubmit={handleSignUp}>
                <MDBInput
                  wrapperClass="mb-4"
                  label="name"
                  id="form1"
                  type="name"
                  value={name}
                  onChange={(e) => setName(e.target.value)}
                />
                <MDBInput
                  wrapperClass="mb-4"
                  label="Email address"
                  id="form1"
                  type="email"
                  value={email}
                  onChange={(e) => setEmail(e.target.value)}
                />
                <MDBInput
                  wrapperClass="mb-4"
                  label="Password"
                  id="form2"
                  type="password"
                  value={password}
                  onChange={(e) => setPassword(e.target.value)}
                />

                <MDBBtn className="mb-4 w-100">Sign up</MDBBtn>
                <div className="d-flex justify-content-between mx-4 mb-4">
                  <a href="/login">Already have an account?</a>
                </div>
              </form>
            </MDBCardBody>
          </MDBCol>
        </MDBRow>
      </MDBCard>
    </MDBContainer>
  );
}

export default SignUp;

解説

次に、Signup.tsx のコードを説明していきます。

const handleSignUp = async (event: React.FormEvent<HTMLFormElement>) => {
  event.preventDefault();
  try {
    //axiosでpost
    const response = await axios.post("http://localhost:3002/users/create", {
      name: name,
      email: email,
      password: password,
    });
    //sesionStrageにidとnameを保存
    sessionStorage.setItem("id", response.data.id);
    sessionStorage.setItem("name", response.data.name);
    //ページ遷移
    navigate("/");
  } catch (error) {
    console.error(error);
    s;
  }
};

この handleSignUp 関数が実行されると、axios というライブラリを使用して非同期処理による HTTP 通信を行ないます。その後取得したユーザーの id と name を sessionStorage を使用してクライアント側に保存しています。

そして処理が成功した場合には、 navigate('/'); で、処理が終わった後に画面の遷移を行なっています。

Axios は、JavaScript/TypeScript で非同期 API 呼び出しを容易にするライブラリのことです。

Signup.tsx の form のコードを説明していきます。

<form onSubmit={handleSignUp}>
  <MDBInput
    wrapperClass="mb-4"
    label="name"
    id="form1"
    type="name"
    value={name}
    onChange={(e) => setName(e.target.value)}
  />
  <MDBInput
    wrapperClass="mb-4"
    label="Email address"
    id="form1"
    type="email"
    value={email}
    onChange={(e) => setEmail(e.target.value)}
  />
  <MDBInput
    wrapperClass="mb-4"
    label="Password"
    id="form2"
    type="password"
    value={password}
    onChange={(e) => setPassword(e.target.value)}
  />

  <MDBBtn className="mb-4 w-100">Sign up</MDBBtn>
  <div className="d-flex justify-content-between mx-4 mb-4">
    <a href="/login">Already have an account?</a>
  </div>
</form>

form onSubmit={handleSignUp} の中の onSubmit は、form の中の button(今回では、Sign up)が押された際に動くものです。従ってボタンが押された際に、上記で説明した、handleSignUp 関数が動くことになります。

ログイン画面

まずは、frontend/src/pages/Login.tsx に下記のコードをコピーして下さい。

Login.tsx
import React, { useState } from "react";
import { useNavigate } from "react-router-dom";
import {
  MDBBtn,
  MDBContainer,
  MDBCard,
  MDBCardBody,
  MDBCardImage,
  MDBRow,
  MDBCol,
  MDBInput,
} from "mdb-react-ui-kit";
import axios from "axios";

function Login() {
  const navigate = useNavigate();
  const [email, setEmail] = useState("");
  const [password, setPassword] = useState("");

  const handleSignIn = async (event: React.FormEvent<HTMLFormElement>) => {
    event.preventDefault();
    try {
      //axiosでpost
      const response = await axios.post("http://localhost:3002/users/login", {
        email: email,
        password: password,
      });
      //sessionStrageにidとnameを保存
      sessionStorage.setItem("id", response.data.id);
      sessionStorage.setItem("name", response.data.name);
      navigate("/");
    } catch (error) {
      console.error(error);
    }
  };
  return (
    <MDBContainer className="my-5">
      <MDBCard>
        <MDBRow className="g-0 d-flex align-items-center">
          <MDBCol md="4">
            <MDBCardImage
              src="https://mdbootstrap.com/img/new/ecommerce/vertical/004.jpg"
              alt="phone"
              className="rounded-t-5 rounded-tr-lg-0"
              fluid
            />
          </MDBCol>

          <MDBCol md="8">
            <MDBCardBody>
              <form onSubmit={handleSignIn}>
                <MDBInput
                  wrapperClass="mb-4"
                  label="Email address"
                  id="form1"
                  type="email"
                  value={email}
                  onChange={(e) => setEmail(e.target.value)}
                />
                <MDBInput
                  wrapperClass="mb-4"
                  label="Password"
                  id="form2"
                  type="password"
                  value={password}
                  onChange={(e) => setPassword(e.target.value)}
                />

                <MDBBtn className="mb-4 w-100">Sign in</MDBBtn>

                <div className="d-flex justify-content-between mx-4 mb-4">
                  <a href="/signup">Create a new account?</a>
                </div>
              </form>
            </MDBCardBody>
          </MDBCol>
        </MDBRow>
      </MDBCard>
    </MDBContainer>
  );
}
export default Login;

解説

次に、Login.tsx のコードを説明していきます。
handleSignIn 関数の説明していきます。

const handleSignIn = async (event: React.FormEvent<HTMLFormElement>) => {
  event.preventDefault();
  try {
    //axiosでpost
    const response = await axios.post("http://localhost:3002/users/login", {
      email: email,
      password: password,
    });
    // レスポンスの中身を確認
    sessionStorage.setItem("id", response.data.id);
    sessionStorage.setItem("name", response.data.name);
    navigate("/");
  } catch (error) {
    console.error(error);
  }
};

handleSignIn 関数は実行されると、Signup の時と同様に、axios というライブラリを使用して非同期処理による HTTP 通信を行います。

Login.tsx の form のコードを説明していきます。

<form onSubmit={handleSignIn}>
  <MDBInput
    wrapperClass="mb-4"
    label="Email address"
    id="form1"
    type="email"
    value={email}
    onChange={(e) => setEmail(e.target.value)}
  />
  <MDBInput
    wrapperClass="mb-4"
    label="Password"
    id="form2"
    type="password"
    value={password}
    onChange={(e) => setPassword(e.target.value)}
  />

  <MDBBtn className="mb-4 w-100">Sign in</MDBBtn>

  <div className="d-flex justify-content-between mx-4 mb-4">
    <a href="/signup">Create a new account?</a>
  </div>
</form>

form で囲われているため、ボタンが押された際に(Sign in)、Signup.tsx で説明したのと同様に、handleSignUp 関数が動くことになります。

tweet 投稿・表示画面

次に、Tweet 投稿・表示画面画面を作成していきます。 まずは、frontend/src/pages/Home.tsx に下記のコードをコピーして下さい。

Home.tsx
import React, { useEffect, useState } from "react";
import {
  MDBCard,
  MDBCardBody,
  MDBCardImage,
  MDBCol,
  MDBContainer,
  MDBInput,
  MDBRow,
  MDBTextArea,
  MDBBtn,
} from "mdb-react-ui-kit";
import { useNavigate } from "react-router-dom";
import axios from "axios";

type Tweets = {
  userName: string;
  content: string;
  createdAt: string;
};

export default function Home() {
  // 遷移するための関数
  const navigate = useNavigate();

  // ーーーーーツイート取得----------
  // ツイートの一覧を管理するstate
  const [tweetList, setTweetList] = useState<Tweets[]>([]);
  // 全件のツイートを取得する処理
  const getTweetList = async () => {
    const response = await axios.get("http://localhost:3002/tweets");
    setTweetList(response.data.tweetList);
  };
  useEffect(() => {
    // 全件のツイートを取得する処理を呼ぶ関数
    getTweetList();
  }, []);

  // ーーーーーツイート投稿----------
  // 入力欄のツイートの内容を管理するstate
  const [newTweet, setNewTweet] = useState("");
  // ツイートの内容を変更する処理
  const handleTweetChange = (event: React.ChangeEvent<HTMLTextAreaElement>) => {
    setNewTweet(event.target.value);
  };
  // ツイートを投稿する処理
  const handleTweetSubmit = async () => {
    try {
      if (newTweet !== "") {
        await axios.post("http://localhost:3002/tweets", {
          id: Number(sessionStorage.getItem("id")),
          content: newTweet,
        });
        //  ツイートを投稿したら全件のツイートを再取得する処理を呼ぶ関数
        getTweetList();
      }
      //  ツイートを投稿したらツイートの内容を空にする処理
      setNewTweet("");
    } catch (error) {
      console.error(error);
    }
  };

  // ーーーーー検索の処理----------
  // 検索欄の内容を管理するstate
  const [search, setSearch] = useState("");
  //  検索欄の内容を変更をする処理
  const handleSearch = (e: any) => {
    setSearch(e.target.value);
  };
  // 検索ボタンを押した時の処理
  const handleSearchPost = async (e: any) => {
    e.preventDefault();
    // 検索欄の内容が空の場合は全件のツイートを取得する処理を呼ぶ関数
    // 検索欄の内容が空でない場合は検索結果のツイートを取得する処理を呼ぶ関数
    search === "" ? getTweetList() : getSearchTweetList();
  };
  // 検索結果のツイートを取得する処理
  const getSearchTweetList = async () => {
    const res = await axios.post("http://localhost:3002/tweets/search", {
      text: search,
    });
    setTweetList(res.data.tweetList);
  };

  // サインアウトの処理
  const handleSignOut = () => {
    sessionStorage.clear();
    navigate("/login");
  };
  return (
    <MDBContainer className="mt-5  w-screen">
      <MDBBtn
        color="danger"
        rounded
        className="float-start w-2"
        onClick={handleSignOut}
      >
        sign-out
      </MDBBtn>
      <MDBRow className="justify-content-center w-100">
        <MDBCol md="8" lg="6">
          <MDBCard
            className="shadow-0 border"
            style={{ backgroundColor: "#f0f2f5" }}
          >
            <MDBCardBody>
              {/* 検索部分 */}
              <form className="mb-6" onSubmit={handleSearchPost}>
                <MDBInput
                  wrapperClass="mb-2"
                  placeholder={"ツイートを検索"}
                  value={search}
                  label="Search"
                  onChange={handleSearch}
                />
                <MDBBtn className="float-end" type="submit">
                  検索
                </MDBBtn>
              </form>
              <div
                className="mb-4"
                style={{ maxHeight: "400px", overflow: "scroll" }}
              >
                {/* tweet表示部分 */}
                {tweetList.map((data: Tweets, index: number) => (
                  <MDBCard className="mb-4" key={index}>
                    <MDBCardBody>
                      <p className="float-end">{data.createdAt}</p>
                      <div className="d-flex justify-content-between mb-3">
                        <div className="d-flex flex-row align-items-center pl">
                          <MDBCardImage
                            src="https://cdn.w3.org/thumbnails/200/avatars/7mtpjeh4in8kw04ksso8ss4ocsksswo.webp"
                            alt="avatar"
                            width="25"
                            height="25"
                          />
                          <p className="small mb-3 ms-2">{data.userName}</p>
                        </div>
                      </div>
                      <p className="">{data.content}</p>
                    </MDBCardBody>
                  </MDBCard>
                ))}
              </div>
              <p className="border-top pt-2">message</p>
              <MDBTextArea
                style={{ backgroundColor: "#fff" }}
                className="text-dark"
                color="primary"
                contrast
                id="textAreaExample"
                label="message"
                value={newTweet}
                rows={4}
                onChange={handleTweetChange}
              ></MDBTextArea>
              <MDBBtn
                color="primary"
                rounded
                className="float-end mt-2"
                onClick={handleTweetSubmit}
              >
                Send
              </MDBBtn>
            </MDBCardBody>
          </MDBCard>
        </MDBCol>
      </MDBRow>
    </MDBContainer>
  );
}

最後に

お疲れ様でした。以上で春休み課題は終了です!正常に動作するか確認してみてください!
何かわからないことがあればいつでも質問お待ちしております!

1
0
1

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
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?