11
8

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

待ち合わせ専用!距離で色が変わるアプリをNetlifyとFirebaseで作ってみた!

Last updated at Posted at 2024-08-22

こんにちは、推しと一緒に住んでいる元幼稚園教諭のりゆです。
推しとはのことなのですが、最近特に夫との待ち合わせが至福のひと時なのです。
ただ現実で待ち合わせている時、あることに気づいてしまったのです。

推しが近づいてくるところが見られないぞ!?

かといって世に出ているGPSアプリは、それぞれの個人情報を大切にしたいのでやりたくないのですね。
私自身、現在地を把握したいという欲求はありませんでした。
ただ推しが近づいてくるドキドキ感を味わいたいのです!!

現状世の中を調べてみましたが、私が求めているサービスはありませんでした・・・😅
なので!ないなら作るしかないですね!!
大まかな距離で居場所がわかり、数字ではなく直感的に近づいてきていることが認識できて待ち合わせ時間が今までより楽しい体験になるものを作ろうと考えました!

距離に応じて色が光るWEBアプリ爆誕!!

色の変化をわかりやすくするため、動画にしました!ご覧ください↓
※動画編集上人揺れが起きていますが、アプリ内では起こりません!

距離500mごとに応じて色が変わっていきます。(最大3500mまで)
色は赤〜青色の中でグラデーションになっています。

どうやって使うの?

推しとの待ち合わせに使います!

・実際に会うまでに、アプリを開きます。

推しの真ん中にあるハートの部分の色が距離に応じて色が変化していきます。

・2人の距離が、近いと赤色、遠いと青色になっています。

 ・実際に推しが来るまで、どんどん近づいていくことが色でわかり、ときめく体験ができます!!

実際に作ったシステム構成図

制作環境

→ウェブアプリ開発プラットフォームです!!

Netlify

→静的サイトをホスティングできる

Firebase

→動的コンテンツの提供
→リアルタイムにデータを同期・保存できるものも提供されている。

Netlify でGPS座標を取得するアプリを作る

・Netlifyでログインし、アプリを作成する

アプリのコード

<html>
  <head>
    <meta charset="utf-8">
    <title>GPS</title>
    <meta name="viewport" content="width=device-width" />
    <meta name="description" content="Displays the straight line distance to Station." />
    <script src="https://geographiclib.sourceforge.io/scripts/geographiclib-geodesic.min.js"></script>
    <style>
      * {
        margin: 0;
        padding: 0;
      }

      body {
        display: flex;
        align-items: center;
        justify-content: center;
        color: #FAFAFA;
        font-size: 24px;
        font-weight: 900;
        background: #FFC0CB; /* 背景色はピンク色 */
      }

      .display {
        display: flex;
        flex-direction: column;
        align-items: center;
      }

      #distanceText {
        font-size: 24px;
        margin-bottom: 10px;
      }

      #m {
        font-size: 40px;
      }

      #timestamp {
        position: fixed;
        bottom: 8px; right: 8px;
        color: #0D47A1;
        font-size: 12px;
      }
    </style>
  </head>
  <body>
    <div>
      <div class="display">
        <p id="distanceText">2人の距離</p> <!-- 距離表示の上にテキストを追加 -->
        <p id="m"></p>
      </div>
      <p id="timestamp"></p>
    </div>
    <script>
      const goalLatLong = [,]; // 目標地点の緯度,経度

      navigator.geolocation.watchPosition(handleWatchPosition, null, {
        enableHighAccuracy: true,
        timeout: 60000,
        maximumAge: 0
      });

      function handleWatchPosition(evt) {
        const latitude = evt.coords.latitude;
        const longitude = evt.coords.longitude;
        const accuracy = evt.coords.accuracy;
        const altitude = evt.coords.altitude;
        const altitudeAccuracy = evt.coords.altitudeAccuracy;
        const heading = evt.coords.heading || 0;
        const speed = evt.coords.speed;
        const timestamp = evt.timestamp;

        document.getElementById('timestamp').textContent = timestamp;

        if (latitude && longitude) {
          const geod = window.geodesic.Geodesic.WGS84;
          const r = geod.Inverse(
            latitude, longitude,
            ...goalLatLong
          );

          document.getElementById('m').textContent = `${ r.s12.toFixed(2) }m`;
        }
      }
    </script>
  </body>
</html>

↓実際の画面

・距離に応じて色の変化を緩やかなグラデーションにした画面
(コードはChatGPTに生成してもらった)

・人の形をimageとしてコードに挿入
→人の画像はprocreateで描きました!
 ハートの形は色が変わるところが見たいため、部分的に透過して画像を作っています。

FirebaseでGPSの座標を保存する(index.html)

最終コード

<!DOCTYPE html>
<html lang="ja">
  <head>
    <meta charset="UTF-8">
    <title>GPS-based Color Gradient</title>
    <meta name="viewport" content="width=device-width" />
    <meta name="description" content="Displays the straight line distance to the current location." />
    <script src="https://cdn.jsdelivr.net/npm/geographiclib@1.52.2/geographiclib.min.js"></script>
    <script defer src="path/to/firebase-app-compat.js"></script>
    <script defer src="path/to/firebase-firestore-compat.js"></script>
    <script defer src="path/to/init.js"></script>
    <style>
      * {
        margin: 0;
        padding: 0;
      }

      body {
        display: flex;
        align-items: center;
        justify-content: center;
        color: #000000;
        font-size: 24px;
        font-weight: 900;
        background: hsl(0, 70%, 90%);
        transition: background 0.5s ease;
        height: 100vh;
        text-align: center;
      }

      .display {
        display: none;
        flex-direction: column;
        align-items: center;
        justify-content: center;
      }

      #distanceText {
        font-size: 24px;
        margin-bottom: 10px;
      }

      #m {
        font-size: 40px;
      }

      #timestamp {
        position: fixed;
        bottom: 8px;
        right: 8px;
        color: #000000;
        font-size: 12px;
        z-index: 1;
      }

      img {
        position: absolute;
        top: 50%;
        left: 50%;
        transform: translate(-50%, -50%);
        height: 100vh;
        width: 100vw;
        object-fit: cover;
        z-index: 0;
      }
    </style>
  </head>
  <body>
    <img src="path/to/image.png" alt="Character Image">
    <div class="display">
      <p id="distanceText">2人の距離</p>
      <p id="m"></p>
    </div>
    <p id="timestamp"></p>
    <script type="module">
      const app = firebase.app();
      const db = firebase.firestore();

      (async () => {
        const updateRef = db.doc('path/to/document');
        const now = new Date();
        await updateRef.set({
          value: `${now.toLocaleString()}`
        });
        const { value } = (await updateRef.get()).data();

        console.log(value);
      })();

      const goalLatLong = [,]; // Replace with target location's latitude, longitude
      navigator.geolocation.watchPosition(handleWatchPosition, null, {
        enableHighAccuracy: true,
        timeout: 60000,
        maximumAge: 0
      });

      function handleWatchPosition(evt) {
        const latitude = evt.coords.latitude;
        const longitude = evt.coords.longitude;
        const timestamp = evt.timestamp;
        document.getElementById('timestamp').textContent = new Date(timestamp).toLocaleString();
        if (latitude && longitude) {
          const geod = window.GeographicLib.Geodesic.WGS84;
          const r = geod.Inverse(latitude, longitude, ...goalLatLong);
          const distance = r.s12;
          document.getElementById('m').textContent = `${distance.toFixed(2)}m`;

          const maxDistance = 4000;
          const hueIntervals = [0, 30, 60, 90, 120, 180, 210, 270];
          let hue;
          if (distance <= 500) {
            hue = interpolateHue(distance, 0, 500, hueIntervals[0], hueIntervals[1]);
          } else if (distance <= 1000) {
            hue = interpolateHue(distance, 500, 1000, hueIntervals[1], hueIntervals[2]);
          } else if (distance <= 1500) {
            hue = interpolateHue(distance, 1000, 1500, hueIntervals[2], hueIntervals[3]);
          } else if (distance <= 2000) {
            hue = interpolateHue(distance, 1500, 2000, hueIntervals[3], hueIntervals[4]);
          } else if (distance <= 2500) {
            hue = interpolateHue(distance, 2000, 2500, hueIntervals[4], hueIntervals[5]);
          } else if (distance <= 3000) {
            hue = interpolateHue(distance, 2500, 3000, hueIntervals[5], hueIntervals[6]);
          } else if (distance <= 3500) {
            hue = interpolateHue(distance, 3000, 3500, hueIntervals[6], hueIntervals[7]);
          } else if (distance <= 4000) {
            hue = interpolateHue(distance, 3500, 4000, hueIntervals[7], hueIntervals[7]);
          } else {
            hue = hueIntervals[7];
          }
          const saturation = 70;
          const lightness = 90;
          document.body.style.backgroundColor = `hsl(${hue}, ${saturation}%, ${lightness}%)`;
        }
      }

      function interpolateHue(distance, minDist, maxDist, startHue, endHue) {
        const ratio = (distance - minDist) / (maxDist - minDist);
        return startHue + ratio * (endHue - startHue);
      }
    </script>
  </body>
</html>

ターミナルで「firebase.deploy」をして値を取る
Cloud Firestoreの中にGPSの値を取ることができる

完成したので使ってもらった

・家族2名に使ってもらって感想を聞いてみました!
 とてもよいFBをもらえたと思います。

WEBアプリの価値について
・日常にない体験ができるね!
・アイデアが今まで見たことがなくて、面白そう。
・好きな人が近づいてくることがわかると、ワクワクするね!
・具体的な居場所が分からないので、個人情報も守られて良い。

WEBアプリの機能・UIについて
・顔が似てるって特別感があって良いね
・500mという距離の感覚と、色合いもちょうどいいよ

WEBアプリの機能・UIについて
・常にアプリを開いている状態だったら、めちゃめちゃ困る。
 タブを後ろに持っていくなど、画面の切り替えができたらいいな。
・GPSをとることでバッテリーがたくさん使われる、その点が気になる

FBをうけて、私はどう思った?

・日常にない体験で新しいアイデアだという言葉がとても嬉しかった!

→距離で色が変わっていく体験はどのように受け入れられるのか不安に思っていました・・・
なので実際に今の日常がよりよく変わるだろうと感じている人がいるがわかり安心しました。
これからも自信を持って検証していきたいと感じました。

・GPSアプリを常に開いている状態になってしまうのは私も気になっていて、これから他に手段はないか模索していきたいと考えている。

その他まだまだ検証していくことがたくさんありそうだ!!

11
8
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
11
8

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?