はじめに
こんにちは!かほです♪🐥
現在、本業では技術広報業務やエンジニア業務などに従事しています。
今回の記事では、Supabase×Next.jsを用いたアプリ開発における全文検索機能の実装方法について説明します。
※全文検索・・・一応念のため、今回のタイトルにある「全文検索」の意味はRDBMS
側の用語から来ています。下記で行なっている内容としては、SQLのLIKE句で%keyword%
していることと変わらないですが、LIKE句の場合はインデックスが効かない問題があるということです。RDBMS
のFull Text Searchを使うことで、事前にインデックスを貼って活かすことができます。今回の「全文検索」というキーワードはRDBMS
における「全文検索」です。
下記の3記事の内容はすでに実装済です🙇♀️
まだ未実装の方は、ご参考までに♪🐥
この記事の読者対象
・Supabase×Next.jsで全文検索機能を含んだアプリを今後作ってみたい方
・Next.jsを使ったリアルタイム検索機能の実装方法を知りたい方
・Supabaseを今後使ってみたい方
・Next.jsを使って高速でアプリを作りたい方
開発環境
"dependencies": {
"@supabase/supabase-js": "^2.2.1",
"eslint": "8.28.0",
"eslint-config-next": "13.0.6",
"next": "13.0.6",
"react": "18.2.0",
"react-dom": "18.2.0",
},
"devDependencies": {
"@types/node": "^18.11.13",
"@types/react": "18.0.26",
}
リアルタイム検索機能を実装しよう
下記がsearch.tsx
の全体像です。
import Head from "next/head";
import styles from "../styles/Home.module.css";
import { supabase } from "../utils/supabase";
import { useState, useEffect } from "react";
export default function Search() {
const [posts, setPosts] = useState([]);
const [keyword, setKeyword] = useState("");
useEffect(() => {
fetchPosts();
}, []);
async function fetchPosts() {
const { data } = await supabase.from("posts").select("*");
setPosts(data);
}
const search = async (value: string) => {
if (value !== "") {
const { data: posts, error } = await supabase
.from("posts")
.select()
.textSearch("title", `${value}`);//どんな検索方法を行いたいかによって、書き方が変わります。
if (error) throw error;
setPosts(posts);
return;
}else{
await fetchPosts();
}
};
const handleChange = async (e: React.ChangeEvent<HTMLInputElement>) => {
setKeyword(e.target.value);
search(e.target.value);
};
return (
<>
<div className={styles.container}>
<Head>
<title>検索機能</title>
<link rel="icon" href="/favicon.ico" />
</Head>
<main className={styles.main}>
<div>
<h1>検索機能</h1>
<div>
<input
type="text"
value={keyword}
className="my-8 rounded border border-black"
placeholder="search"
onChange={(e) => handleChange(e)}
/>
<table>
<thead>
<tr>
<th>投稿日</th>
<th>タイトル</th>
<th>内容</th>
</tr>
</thead>
<tbody>
{posts.map((post) => (
<tr key={post.id}>
<td>{post.created_at.substr(0, 10)}</td>
<td>{post.title}</td>
<td>{post.content}</td>
</tr>
))}
</tbody>
</table>
</div>
</div>
</main>
<footer className={styles.footer}></footer>
</div>
</>
);
}
コードを上から順に説明します。
fetchPosts()
メソッドによって、投稿の一覧表示を行っています。
こちらのメソッドの説明については、下記の記事「話題のSupabaseでサクッと投稿機能をつくってみた!」の「一覧機能を実装しよう」の段落と内容が同じですので、ぜひご参考までに🙌
また、検索フォームに入れる値の状態管理を行っています。
const [posts, setPosts] = useState([]);
const [keyword, setKeyword] = useState("");
useEffect(() => {
fetchPosts();
}, []);
async function fetchPosts() {
const { data } = await supabase.from("posts").select("*");
setPosts(data);
}
次にsupabase.tsを呼び出し、検索部分の実装を行います。
今回はtextSearch()
を使用し、postsテーブルのデータを検索します。
from()
の引数にテーブル名を入れ、textSearch()
の第1引数にテーブルの対象カラム、第2引数に検索したいキーワード等を入れます。
textSearch()
を使用する方法は、下記のsupabase公式ドキュメントにも記載されていますので、ご興味のある方はご参照ください🙌
今回は、if文で検索フォームに値がある場合とない場合のロジックを分けて書いています。
検索フォームに値がある場合は、上記の検索方法を用いて、データをsetPosts(posts)
により状態管理します。
検索フォームに値がない場合は、投稿が一覧表示されるために、fetchPosts()
メソッドを記述します。
const search = async (value: string) => {
if (value !== "") {
const { data: posts, error } = await supabase
.from("posts")
.select()
.textSearch("title", `${value}`);//どんな検索方法を行いたいかによって、書き方が変わります。
if (error) throw error;
setPosts(posts);
return;
}else{
await fetchPosts();
}
};
最後に、検索フォームの値の状態管理を行います。
今回は検索フォームの値を入れたと同時に、検索結果の出力も行う想定です。
そのため、検索フォームの値とsearchメソッドによって出力される検索結果の値にonChange
イベントの設定を行います。
const handleChange = async (e: React.ChangeEvent<HTMLInputElement>) => {
setKeyword(e.target.value);
search(e.target.value);
};
その他の検索方法も試してみよう!
上記で説明した検索方法は、完全一致検索ですが、その他にも様々な検索方法を行うことができます。
そのうちの数個を軽く紹介します♪🐥
- 完全一致検索
上記で説明した検索方法です。完全に一致した値のみを検索します。
![](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.ap-northeast-1.amazonaws.com%2F0%2F1547587%2Fa8349b08-c791-f3e0-5aa0-e889704f65c5.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=b09e6d2212ad8190927c0bf440348d5c)
const { data: posts, error } = await supabase
.from("posts")
.select()
.textSearch("title", `${value}`)
- OR検索
OR(いずれかのキーワードを含む)検索は|
でキーワードを連結します。
![](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.ap-northeast-1.amazonaws.com%2F0%2F1547587%2Fe22bb23c-0e56-61b6-6436-1cc437d5590c.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=c7283611c87079cdb6ba5426a79eb7e4)
const { data: posts, error } = await supabase
.from("posts")
.select()
.textSearch("title", `'bbb' | 'ccc'`);
- AND検索
AND(すべてのキーワードを含む)検索は&
でキーワードを連結します。
![](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.ap-northeast-1.amazonaws.com%2F0%2F1547587%2F5c6631b9-8ce8-833f-f827-22bec91037e8.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=bce0c2039f816eefb6ba9793241903ae)
const { data: posts, error } = await supabase
.from("posts")
.select()
.textSearch("title", `'bbb' & 'ccc'`);
追記
上記で説明した検索機能の実装方法の他にも、PGroongaと呼ばれる拡張機能(Extensions)を使った実装方法もあります!ご興味がある方は下記の公式ドキュメントをチェックしてみてください♪🐥
最後に
これで実装はおしまいです!
今回は、Supabase×Next.jsを用いたアプリ開発における全文検索機能の実装方法について説明しました。今後は主にフロントエンド周辺、コミュニティ運営、Tech PR(技術広報)の記事を中心に書いていく予定ですので、気になる方はぜひフォローをよろしくお願いします♪ではでは〜🐥
Twitterアカウント:https://twitter.com/kaho_eng
参考資料