前回は、Prisma を使って、ログイン機能を追加しました。
data:image/s3,"s3://crabby-images/0612e/0612e7756de908bc05bb9bf8c78bd212f0211802" alt="graphql-prisma-login"
【GraphQL】Prismaを使って、ログイン機能を追加する
今回は、バックエンドで作成した GraphQL を、フロントエンドで接続します。
バックエンドのコードは、こちらです。
フロントエンドのテンプレートは、React と MUI で作成しました。
GraphQL を接続するために、React に GraphQL と Apollo Client をインストールします。
ターミナルで、npm install --save graphql @apollo/clientを実行しましょう。
GraphQL と Apollo Client がインストール完了した後、index.tsx に ApolloClient と ApolloProvider、InMemoryCache をインポートします。
ts
import { ApolloClient, ApolloProvider, InMemoryCache } from "@apollo/client";
ApolloClientで、バックエンドの URI と、キャッシュを設定します。
バックエンドの URI は http://localhost:4000/graphql、キャッシュは先程インポートした、InMemoryCacheを指定します。
tsx
const client = new ApolloClient({
uri: "http://localhost:4000/graphql",
cache: new InMemoryCache(),
});
url ではなく、uri なので、注意しましょう。
<App />をApolloProviderでラップします。
client を指定しなければいけないので、先程作成したclientを指定します。
tsx
ReactDOM.render(
<React.StrictMode>
<BrowserRouter>
<ApolloProvider client={client}>
<App />
</ApolloProvider>
</BrowserRouter>
</React.StrictMode>,
document.getElementById("root")
);
これで、GraphQL を接続する準備が完了したので、実際にデータを取得してみます。
pages フォルダの Book.tsx を開きます。
Apollo Client から gql と useQuery をインポートします。
tsx
import { gql } from "@apollo/client";
gql で、取得したいクエリの雛形を作ります。
gql の中身は、バッククォートで囲みます。
tsx
const booksData = gql``;
バックエンドの graphql に移動して、books で一度データを取得してみましょう。
data:image/s3,"s3://crabby-images/04587/0458741ea4dd3d4f1b45ef025839dd7353b7e004" alt="image2"
isRead が波線になっているので、バックエンドの schema.ts のtype BookにisReadを追加します。
ts
type Book {
id: Int!
title: String!
author: String!
createdAt: String!
category: Category!
isRead: Boolean!
}
では、GraphQL に戻って、allBooks ボタンをクリックすると、
data:image/s3,"s3://crabby-images/72632/726329c18307483fd17a901c26a5154a726d12e1" alt="image3"
本のデータを取得することができます。
今作成した、GraphQL をコピーして、フロントエンドのgql内に貼り付けます。
tsx
const booksData = gql`
query allBooks {
books {
id
title
author
category {
name
}
isRead
createdAt
}
}
`;
次に、useQueryをインポートします。
tsx
import { gql, useQuery } from "@apollo/client";
useQuery を使って、先程作成したbooksDataからdataを取り出します。
tsx
const { data } = useQuery(booksData);
アドレスは http://localhost:3000/book を指定して、console.log()で確認すると、
data:image/s3,"s3://crabby-images/ee212/ee212b1b5494930cca3fb03339410ce19dde0058" alt="image4"
GraphQL のデータを取得できているのがわかります。
data を使って、books のデータをブラウザに表示させてみます。
data から books を取り出します。
tsx
const { books } = data;
books を map で展開します。
tsx
<TableBody>
{books.map((book) => (
<TableRow key={book.id}>
<TableCell>{book.id}</TableCell>
<TableCell>{book.title}</TableCell>
<TableCell>{book.author}</TableCell>
<TableCell>{book.category.name}</TableCell>
<TableCell>{book.isRead}</TableCell>
<TableCell>{book.createdAt}</TableCell>
</TableRow>
))}
</TableBody>
book の型を指定します。
tsx
type TypeBook = {
id: number;
title: string;
author: string;
category: {
name: string;
};
isRead: boolean;
createdAt: string;
};
tsx
<TableBody>
{books.map((book: TypeBook) => (
<TableRow key={book.id}>
<TableCell>{book.id}</TableCell>
<TableCell>{book.title}</TableCell>
<TableCell>{book.author}</TableCell>
<TableCell>{book.category.name}</TableCell>
<TableCell>{book.isRead}</TableCell>
<TableCell>{book.createdAt}</TableCell>
</TableRow>
))}
</TableBody>
一度、ブラウザで確認すると、
data:image/s3,"s3://crabby-images/35f5f/35f5f4eb8510b6e678365fa4ae37ee5bd0062b7c" alt="image5"
エラーが発生しました。
これは、GraphQL のデータを取得するより、React のレンダリングする方が早いので、エラーになりました。
エラーが発生しないように、データを取得するまでローディングします。
useQueryにloadingを追加します。
tsx
const { data, loading } = useQuery(booksData);
loading が発生しているときは、メッセージを表示するようにします。
tsx
if (loading) return <div>Loading...</div>;
では、サイド確認すると、
data:image/s3,"s3://crabby-images/ff642/ff642dc15a344df8e2095793a36e9af701988b19" alt="image6"
data:image/s3,"s3://crabby-images/38a13/38a13e12d7bda998c2efb3d61b80c8c897dbcccd" alt="image7"
ローディングが発生した後、books のデータを表示することができました。
createDataは、必要ないので、削除します。
今のところ、読了欄には、何も表示されていません。
isRead が true の場合『済』、false の場合『未』を表示します。
tsx
<TableBody>
{books.map((book: TypeBook) => (
<TableRow key={book.id}>
<TableCell>{book.id}</TableCell>
<TableCell>{book.title}</TableCell>
<TableCell>{book.author}</TableCell>
<TableCell>{book.category.name}</TableCell>
{book.isRead ? <TableCell>済</TableCell> : <TableCell>未</TableCell>}
<TableCell>{book.createdAt}</TableCell>
</TableRow>
))}
</TableBody>
data:image/s3,"s3://crabby-images/fa138/fa1384d8942184222281f88501c59d00d3df3596" alt="image8"
無事、表示されました。
また、作成日も数字の羅列なので、修正します。
tsx
<TableBody>
{books.map((book: TypeBook) => (
<TableRow key={book.id}>
<TableCell>{book.id}</TableCell>
<TableCell>{book.title}</TableCell>
<TableCell>{book.author}</TableCell>
<TableCell>{book.category.name}</TableCell>
{book.isRead ? <TableCell>済</TableCell> : <TableCell>未</TableCell>}
<TableCell>
{`${new Date(Number(book.createdAt))}`
.split(" ")
.splice(1, 3)
.join(" ")}
</TableCell>
</TableRow>
))}
</TableBody>
data:image/s3,"s3://crabby-images/d99b0/d99b0dffbdf913554bc00cbd26901b8fb50b43d9" alt="image9"
GraphQL が誤っている場合、エラーメッセージを表示するようにします。
useQueryで、errorを取得します。
tsx
const { data, error, loading } = useQuery(booksData);
loadingと同様に、errorがある場合の表示を設定しましょう。
tsx
if (error) return <div>データを取得することができませんでした</div>;
試しに、gqlのidをiddにしてみると、
data:image/s3,"s3://crabby-images/881a0/881a07e12e47fd516332d3d910ecfb9c13e54e7b" alt="image10"
エラーメッセージを表示することができました。
全文は、以下の通りです。
フロントエンド:
バックエンド:
次回は、GraphQL で接続したデータ一覧から、詳細画面へ遷移します。
data:image/s3,"s3://crabby-images/07b76/07b766ce849a4049e1ace14b544e90d51803769b" alt="graphql-client-detail"
【GraphQL】フロントエンドで詳細画面へ遷移する