前回は、Firestore Database を使い、React でメッセージ送信機能を実装しました。
data:image/s3,"s3://crabby-images/f5b13/f5b13e85397fdc5039258a98556e946614ac0674" alt="firebase-react-firestore-create"
【Firebase】Firestore Databaseを使い、Reactでメッセージ送信機能を実装する
今回は、プロフィール編集画面を作成し、Firestore Storage にアバター画像を保存します。
まずは、ヘッダーのプロフィールをクリックすると、プロフィール画面へ遷移するようにします。
MUI でプロフィール画面を作りましょう。
tsx
import React, { useState } from "react";
import {
Paper,
Typography,
Box,
TextField,
Button,
Container,
} from "@mui/material";
const Profile = () => {
const [name, setName] = useState("");
const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {
console.log(e.target.files);
};
const handleSubmit = (event: React.FormEvent<HTMLFormElement>) => {
event.preventDefault();
};
return (
<Container maxWidth="sm">
<Paper sx={{ m: 4, p: 4 }}>
<Typography align="center">プロフィール編集</Typography>
<Box component="form" onSubmit={handleSubmit} noValidate sx={{ mt: 4 }}>
<input type="file" accept="image/*" onChange={handleChange} />
<TextField
margin="normal"
required
fullWidth
id="name"
label="ユーザー名"
name="name"
autoComplete="name"
autoFocus
value={name}
onChange={(e) => setName(e.target.value)}
/>
<Button
type="submit"
fullWidth
variant="contained"
sx={{ mt: 3, mb: 2 }}
>
保存
</Button>
</Box>
</Paper>
</Container>
);
};
export default Profile;
data:image/s3,"s3://crabby-images/48858/48858a9df3fa6d0aecfb3a0e8067f922dd66e4a7" alt="image2"
次に、react-router-domでプロフィール画面のパスを作成します。
App.tsx へ移動し、Routeを追加します。
tsx
<Route path="profile" element={<Profile />} />
Header.tsx へ移動し、リンクを設定しましょう。
tsx
<MenuItem onClick={handleClose}>
<Link href="profile" underline="none" color="inherit">
プロフィール
</Link>
</MenuItem>
プロフィール画面もヘッダーを表示させたいので、App.tsx でホーム画面とプロフィール画面のみヘッダーを表示するようにします。
react-router-domからOutletをインポートします。
tsx
import { BrowserRouter, Routes, Route, Outlet } from "react-router-dom";
Outletがchildrenみたいな役割を果たしてくれます。
Layout関数を作成します。
Layout関数の中にHeaderとOutletを設定します。
tsx
function Layout() {
return (
<>
<Header />
<Outlet />
</>
);
}
App関数のRoutesの中にLayoutを設定します。
tsx
function App() {
return (
<ThemeProvider theme={theme}>
<BrowserRouter>
<Routes>
<Route element={<Layout />}>
<Route path="/" element={<Home />} />
<Route path="profile" element={<Profile />} />
</Route>
<Route path="login" element={<Login />} />
<Route path="signup" element={<Signup />} />
<Route path="password-reset" element={<PasswordReset />} />
</Routes>
</BrowserRouter>
</ThemeProvider>
);
}
プロフィール画面を確認すると、
data:image/s3,"s3://crabby-images/4e111/4e111a8d4a76b244b3f799f7f13147119415418f" alt="image3"
ヘッダーが表示されました。
ちなみに、ログイン画面では、
data:image/s3,"s3://crabby-images/c34cc/c34ccf0a04e5efa0e68405f3329e706be566de88" alt="image4"
ヘッダーが表示されていません。
このままでは、ホーム画面でヘッダーが二重で表示されるので、Home.tsx のHeaderは削除しておきましょう。
『ファイルを選択』をクリックすると、画像を選択できるようになっています。
data:image/s3,"s3://crabby-images/92b80/92b806511f334625ffcbcac0e01668727726b30b" alt="image5"
HTML のレイアウトではなく、MUI のボタンを作成し、全体を統一します。
まずは、inputタグの下にボタンを作成します。
tsx
<Button variant="contained" color="primary" component="span">
画像を選択
</Button>
inputタグにidを追加します。
また、Buttonを label タグで囲みます。
labelタグにhtmlForを設定し、inputタグのidを指定します。
inputタグをdisplay:noneで非表示にします。
tsx
<input
id="image"
type="file"
accept="image/*"
onChange={handleChange}
style={{ display: "none" }}
/>
<label htmlFor="image">
<Button variant="contained" color="primary" component="span">
画像を選択
</Button>
</label>
では、動作確認してみます。
data:image/s3,"s3://crabby-images/101d6/101d6b4f2d81782b87199be24c1e6cfcd9454acd" alt="image6"
画像を追加し、Console を確認すると、
data:image/s3,"s3://crabby-images/f26d3/f26d3359c371b11c52850db4fc0636a67cd3f6a5" alt="image7"
画像のデータが表示されました。
このままでは、画像が選択されているか、画面では分からないので、画面に画像を表示させます。
useState で画像データの状態を管理しましょう。
tsx
const [image, setImage] = useState<File | null>();
tsx
const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {
if (e.target.files !== null) {
setImage(e.target.files[0]);
}
};
次に、MUI のAvatarを使い、画像を表示する場所を作成します。
Avatarの src には、imageがある場合、URL.createObjectURLでimageを指定します。
imageがない場合は、””としておきましょう。
tsx
<Box sx={{ display: "flex", justifyContent: "space-between" }}>
<Avatar src={image ? URL.createObjectURL(image) : ""} alt="" />
<div>
<input
id="image"
type="file"
accept="image/*"
onChange={handleChange}
style={{ display: "none" }}
/>
<label htmlFor="image">
<Button variant="contained" color="primary" component="span">
画像を選択
</Button>
</label>
</div>
</Box>
data:image/s3,"s3://crabby-images/ab4ec/ab4ec0ae72da7824504b1cc0714fc8348fd7263b" alt="image8"
では、画像を選択してみます。
data:image/s3,"s3://crabby-images/e7644/e7644ebd76faa4d4d9aa9dbe7f43084f59213ae0" alt="image9"
Avatar が設定されました。
画像が選択できたので、次は、Firebase の Storage に画像が保存できるようにします。
Firebase の Storage にアクセスします。
Rules タグをクリックします。
今のところ、ファイルの read や write が禁止されています。
data:image/s3,"s3://crabby-images/31c69/31c69a1e0e6ce9e1e3cff328626ec13c835e9816" alt="image10"
こちらを、認証されている場合は許可するようにします。
rules_version = '2';
service firebase.storage {
match /b/{bucket}/o {
match /{allPaths=**} {
allow read, write: if request.auth != null;
}
}
}
公開をクリックします。
data:image/s3,"s3://crabby-images/38a4e/38a4e3a06524d58ff6e95fe62e9f612fe0480eb8" alt="image11"
Profile.tsx へ戻ります。
以前 Firebase の初期設定した、Firebase フォルダの firebaseConfig からfirebaseAppをインポートします。
Firebase の設定は、こちらをご覧ください。
data:image/s3,"s3://crabby-images/d81a8/d81a81b2cbeb8d0b43980ba569fd4dc80047c984" alt="firebase-project-config-setting"
【Firebase】Firebase Project Configを設定する
data:image/s3,"s3://crabby-images/5d94f/5d94f6d1266a926dbc35e52af7a4ab958028710e" alt="firebase-firestore-database-gets"
【Firebase】Firestore Databeseのデータを、フロントエンドに表示する
data:image/s3,"s3://crabby-images/a13e5/a13e507673d05db8ab3315701934eb7f9e431fd2" alt="firebase-storage-get-preview"
【Firebase】Storageで保存した画像をブラウザに表示する
firebaseAppのfirestorageを使用します。
tsx
const firestorage = firebaseApp.firestorage;
handleSubmit 関数内で、try/catch を使います。
catch の場合は、エラーメッセージを表示するようにします。
tsx
const [error, setError] = useState(false);
tsx
const handleSubmit = (event: React.FormEvent<HTMLFormElement>) => {
event.preventDefault();
try {
} catch (err) {
console.log(err);
setError(true);
}
};
tsx
{
error && <Alert severity="error">送信できませんでした</Alert>;
}
firebase/storageからrefをインポートします。
tsx
import { ref } from "firebase/storage";
ref の第一引数に、先程設定したfirestorage、第二引数にファイル名としてimageオブジェクトのnameを指定します。
tsx
try {
if (image) {
const imageRef = ref(firestorage, image.name);
}
} catch (err) {
console.log(err);
setError(true);
}
firebase/storageからuploadBytesをインポートします。
tsx
import { ref, uploadBytes } from "firebase/storage";
uploadBytesの第一引数に imageRef、第二引数に image を指定します。
thenでConsoleに送信内容を表示するようにします。
tsx
try {
if (image) {
const imageRef = ref(firestorage, image.name);
uploadBytes(imageRef, image).then((snapshot) => {
console.log("Uploaded a file!", snapshot);
});
}
} catch (err) {
console.log(err);
setError(true);
}
では、動作確認してみます。
data:image/s3,"s3://crabby-images/e529a/e529a6966b3008a08d042f5a2c6eab71248a895d" alt="image12"
『保存』をクリックすると、
data:image/s3,"s3://crabby-images/96fde/96fde6c31f73091396ffd3b98226a4f20fc8310e" alt="image13"
画像が送信されたようです、
Firebase の Storage を確認してみましょう。
data:image/s3,"s3://crabby-images/2a489/2a489e4d3c05941f3dcb0b1da1466ba6f31289c3" alt="image14"
画像が保存されていました。
次回は、
data:image/s3,"s3://crabby-images/750c3/750c3659b255c8c5ee0e4583b824e0c17b58f6c9" alt="firebase-database-react-user"
【Firebase】プロフィール画面で作成したユーザー情報を、Firestore Databaseに保存する