第十章:キャンプ場どこでもドア - ロケーション検索とスムーズなフライト(FlyTo)の実装
地図上に10箇所のキャンプ場しかない場合、ユーザーはのんびりと地図をスクロールして探すことができます。 しかし、データベースに台湾全土の1,000箇所ものキャンプ場が登録されている場合、ユーザーが地図を拡大縮小しながら「武陵農場」がどの山に隠れているかを探すのは現実的ではありません。
彼らが必要とするのは強力な「どこでもドア」機能です:ページ上部の検索ボックス。ユーザーが文字を入力すると、自動的にサジェストメニュー(Autocomplete)が表示され、選択すると地図がドローンのようにその場所の上空までスムーズに移動してズームインします!
これは商業的に極めて価値の高い高度な機能で、あなたのシステムのユーザー体験をGoogle Mapsレベルに一気に引き上げます。
🎯 本章の目標
- Autocomplete(自動補完)付きのインテリジェント検索ボックスを実装
- Reactの状態管理とLeafletの
map.flyToアニメーション機能を連携 - 複数のコンポーネント(検索ボックス + 地図エンジン)を連動させるシステム設計の快感を体験
🔍 ステップ1:インテリジェント検索ボックス(SearchBar)の作成
ドロップダウンメニューのフィルタリングロジックやCSSレイアウトをゼロから作る必要はありません。このような煩雑なUIコンポーネントこそAIの出番です。
🔥【Vibe Prompt実践呪文】
Reactでキャンプ場検索ボックスコンポーネント(SearchBar.tsx)を作成してください。1. インターフェースデザイン:Tailwind CSSを使用してGoogle Mapsの検索ボックスのような質感に。影(shadow-lg)、角丸、左側に虫眼鏡アイコン(lucide-react)を含める。2. データ処理:配列campsData(id, name, lat, lngを含む)を受け取る。3. 自動補完ロジック:ユーザーがinputに入力すると、絶対配置(absolute z-50)のドロップダウンメニューが表示される。filterを使用して入力文字列を含むキャンプ場名をフィルタリング。4. イベントトリガー:ユーザーがドロップダウンメニューからキャンプ場を選択すると、callback propのonCampSelect(lat, lng)をトリガーし、入力ボックスをクリアしてドロップダウンを閉じる。
この呪文をAIに与えると、onChangeによるリアルタイムフィルタリング、ドロップダウンメニューのul/liレンダリングなど、すべてを美しくプロフェッショナルなスタイルで作成してくれます。
🛫 ステップ2:地図のフライト関数(FlyTo)との連携
完成したSearchBarコンポーネントを地図ページに配置し、onCampSelectコールバック関数を地図のフライト機能と連携させます。
React-Leafletで地図の視点を制御するには、まず地図の「インスタンス(Instance)」を取得する必要があります。これは通常<MapContainer>のrefを使用して行います。
メインページ(例:MapPage.tsx)を以下のように修正します:
import { useState, useRef } from 'react';
import { MapContainer, TileLayer } from 'react-leaflet';
import SearchBar from './SearchBar';
export default function MapPage() {
// 1. Leaflet地図の制御権を取得するためのRefを作成
const mapRef = useRef(null);
// 2. 検索ボックスでキャンプ場が選択された時の動作を定義
const handleCampSelect = (lat, lng) => {
const map = mapRef.current;
if (map) {
// キーマジック:目標のキャンプ場に瞬間移動し、レベル15までズーム!
// duration: 2はフライトプロセスに2秒かけることを意味し、スムーズな視覚効果を生み出す
map.flyTo([lat, lng], 15, { duration: 2 });
}
};
return (
<div className="relative h-screen w-full">
{/* 検索ボックスブロック:地図最上部に絶対配置 */}
<div className="absolute top-6 left-1/2 -translate-x-1/2 z-[1000] w-11/12 max-w-md">
<SearchBar campsData={mockData} onCampSelect={handleCampSelect} />
</div>
{/* 地図本体ブロック */}
<MapContainer
ref={mapRef} // mapRefをバインド
center={[23.5, 121.0]}
zoom={8}
className="h-full w-full z-0"
>
<TileLayer url="https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png" />
{/* ここにMarkerを配置... */}
</MapContainer>
</div>
);
}
保存後、テストしてみてください。検索ボックスに「武陵」と入力し、ドロップダウンメニューから「武陵農場」を選択すると、画面が台湾全体図から台中和平区の目的地まで、山々を飛び越えてスムーズに移動・ズームインします! この滑らかなインタラクション体験こそ、顧客が喜んで10万円を支払うキラー機能です。
✅ コース総まとめ(Course 2修了)
おめでとう!**Car Camping Map(フルスタック地図情報プラットフォーム)**実戦プロジェクトの全章を完了しました!
これはローカルで自己満足するだけの練習問題ではなく、企業レベルの機能を備えた実際の商業製品です:
- 空間データベース統合(Supabase API):データがハードコードされず、バックエンドから無限に拡張可能
- 大量データレンダリング最適化(Marker Clustering):1,000箇所のポイントでもラグなし
- モバイル版対策と適応(Gesture Handling & RWD):スクロールトラップを解決し、いつどこでも完璧に表示
- 地理的位置特定とインテリジェント検索(FlyTo & Autocomplete):Google Mapsレベルの究極のユーザー体験を提供
データベースの名前と緯度経度を変更するだけで、このシステムを「台湾全土のテスラ充電スタンドマップ」「台北深夜カフェマップ」または「不動産検索レーダー」に簡単に改造して商用化できます!
あなたはウェブページのコーディングエンジニアから、「大規模な地理情報システム(GIS)を開発できるアーキテクト」へと正式にレベルアップしました。 次の挑戦の準備はできていますか?次のプロジェクトでは、自動で収益を生むLineボットとバックエンドシステムの構築方法を教えます!
章のまとめ
- コアコンセプトと原理を理解
- 実装方法とテクニックを習得
- 一般的な問題と解決策に精通
- 実際のプロジェクトに適用可能
さらに読む
- 公式ドキュメントとAPIリファレンス
- GitHubのオープンソース例
- 技術書とオンラインコース
- コミュニティディスカッションと技術ブログ