押下したらモーダルを表示するカード型コンポーネントを作りたい
設定画面などで四角いカードを配置し、押下されたときにモーダル表示するようなコンポーネントを作りたくなりました。 以下みたいなやつです。この記事では、まず上記のようなカード型コンポーネントを作り、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>
)
}
参考