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.

【React】押下したらモーダルを表示するカード型コンポーネントを作る【MaterialUI】

Last updated at Posted at 2021-11-16

押下したらモーダルを表示するカード型コンポーネントを作りたい

設定画面などで四角いカードを配置し、押下されたときにモーダル表示するようなコンポーネントを作りたくなりました。 以下みたいなやつです。

image.png

この記事では、まず上記のようなカード型コンポーネントを作り、MaterialUIのModalを使って、クリックしたときにModalを表示させるカード型のコンポーネントを実装していきます。

インストールするもの

今回使うのはMaterialUIの以下を使用します。 ・Modal ・button ・icon

なので、初めて使うときはこちらからインストールしてください。

あと、カードを作る際に今回はstyled-componentを採用しています。
styled-componentを使用する際は下記からインストールしてください。

実装方法

モーダル本体

モーダル本体をimportしたら、そのままカード型で表示されるようにしていきます。

部品として使いまわすために、iconや表示するテキストなどの可変の部分は引数として受け取るようにします。
引数は以下の5つです。
・Icon(カードに表示するアイコン)
・text(カードに表示する文字)
・onClick(YESを押されたときに発火するfunction)
・modalTilte(モーダルで表示するタイトル)
・modalText(モーダルで表示する文章)※不要であれば、""を送ればOK

また今回は、YES/NOボタンを分離せずそのまま乗っけちゃいます。こちらも可変にしておいたほうが便利かもです。

modal.js
import React, { Fragment, useEffect, useState } from 'react';
import styled from "styled-components";
import { makeStyles } from '@material-ui/core/styles';
import Modal from '@material-ui/core/Modal';
import { MaterialUICommonButton } from "../MaterialUICommonButton";

const OKButtomWrapper = styled.div`
  margin-left:45%;
  float:left;
`;

const NGButtomWrapper = styled.div``;


const RectangleWrapper = styled.h2`
  border: solid;
  border-color: #F0F0F0 ;
  color:black;
  border-width:2px;
  display: flex;
  padding-left:1%;
  padding-top:1%;
  padding-bottom:1%;
  height:100%;
  width:100%;
`;

const RectangleIcon = styled.div`
  padding-top:1%;
  padding-right:1%;
`;

const RectangleText = styled.div`
  padding-top:1%;
`;


function rand() {
  return Math.round(Math.random() * 20) - 10;
}

function getModalStyle() {
  const top = 50 + rand();
  const left = 50 + rand();

  return {
    top: `${top}%`,
    left: `${left}%`,
    transform: `translate(-${top}%, -${left}%)`,
  };
}

const useStyles = makeStyles((theme) => ({
  paper: {
    position: 'absolute',
    width: 400,
    backgroundColor: theme.palette.background.paper,
    border: '2px solid #000',
    boxShadow: theme.shadows[5],
    padding: theme.spacing(2, 4, 3),
  },
}));

export default function CardModal({ Icon, text, onClick, modalTilte, modalText }) {
  const classes = useStyles();
  // getModalStyle is not a pure function, we roll the style only on the first render
  const [modalStyle] = useState(getModalStyle);
  const [open, setOpen] = useState(false);

  const handleOpen = () => {
    setOpen(true);
  };

  const handleClose = () => {
    setOpen(false);
  };

  const handleOK = () => {
    onClick()
    handleClose()
  }

  const body = (
    <div style={modalStyle} className={classes.paper}>
      <h2 id="simple-modal-title">{modalTilte}</h2>
      <p id="simple-modal-description">
        {modalText}
      </p>
      <OKButtomWrapper>
        <MaterialUICommonButton onClick={() => handleOK()} btnLabel={"OK"}></MaterialUICommonButton>
      </OKButtomWrapper>
      <NGButtomWrapper>
        <MaterialUICommonButton onClick={() => handleClose()} btnLabel={"キャンセル"}></MaterialUICommonButton>
      </NGButtomWrapper>
    </div>
  );

  return (
    <Fragment>
      <RectangleWrapper onClick={() => { handleOpen() }}>
        <RectangleIcon>
          <Icon fontSize='large' />
        </RectangleIcon>
        <RectangleText>
          {text}
        </RectangleText>
      </RectangleWrapper>
      <Modal
        open={open}
        onClose={handleClose}
        aria-labelledby="simple-modal-title"
        aria-describedby="simple-modal-description"
      >
        {body}
      </Modal>
    </Fragment>
  );
}

呼び出し側

以下の引数で呼び出して、表示させます。 引数に使うためのアイコンを、importしています。
APP.js
import CardModal from "../CardModal";
import VerifiedUserIcon from '@mui/icons-material/VerifiedUser';
import MeetingRoomIcon from '@mui/icons-material/MeetingRoom';
import DeleteForeverIcon from '@mui/icons-material/DeleteForever';

const Wrapper = styled.div`
  margin-left: 20%;
  margin-right: 20%;
  padding-top:5%;
`

export const APP = () => {

  return(
    <Wrapper>
      <CardModal
        Icon={VerifiedUserIcon}
        text={"ログアウト"}
        onClick={() => console.log("変更画面へ")}
        modalTilte={"本人確認"}
        modalText={"本人確認のため、現在登録されているメールアドレスとパスワードを確認します"}
      />
      <CardModal
        Icon={MeetingRoomIcon}
        text={"ログアウト"}
        onClick={() => console.log("ログアウト成功!")}
        modalTilte={"ログアウト"}
        modalText={"ログアウトしますか?"}
      />
      <CardModal
        Icon={MeetingRoomIcon}
        text={"アカウント削除"}
        onClick={() => console.log("アカウント削除!")}
        modalTilte={"アカウント削除"}
        modalText={"本当に削除してもよろしいですか?"}
      />
    </Wrapper>
  )

}

image.png
押下すると、モーダルが表示されます。
image.png

参考

1
0
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
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?