49
42

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

React NativeAdvent Calendar 2019

Day 22

React Native で位置情報を取得して地図表示する

Last updated at Posted at 2019-12-22

地図アプリで目的地までの経路を検索したり、場所を活用したゲームや SNS など、位置情報を利用したアプリは生活の中で当たり前になりました。
本記事では、React Native で位置情報を取得 / 追跡してアプリに地図表示する方法を、サンプルアプリを作りつつ紹介したいと思います。
※今回 React Native 公式の Geolocation を用いるので、 Expo についての内容は含みません🙏

イントロダクション

Ract Native 位置情報に必要な概念

  • Geolocation
  • Location (Expoを使用する場合)

Geolocation とは

React Native の Geolocation は、Webアプリケーションで位置情報を取得する際に利用される Geolocation Web のアプリ拡張 API です。

Geolocation Web APIは、W3Cが仕様策定を進める規格であり、JavaScriptで位置情報を取得できるように標準化されています。
一般的なブラウザでサポートされており、 スマートフォンのようなGPS対応デバイス向けのウェブサイトだけではなく、 PC サイトなどでもユーザーの位置情報を利用したコンテンツを提供出来るようになっています。
Geolocation API において位置情報は、デバイスの GPS 機能の他に、IPアドレス、Wi-Fi などさまざまなソースを基に取得されます。また、Geolocation API から位置情報にアクセスする場合、ユーザーの許諾なしに位置情報を取得することは出来ません。

ちなみに、Geolocation API で取得できるのは緯度や経度といった情報であり、住所を取得することはできません。緯度/経度情報を住所に変換する場合には、Google Geocoding API などを利用します。

React Native の Geolocation で提供される機能としては主に以下の2つになります。今回はこの2つの機能について、サンプルアプリを作ってみたいと思います。

  • 現在位置情報の取得
  • 継続的に位置情報を追跡して取得

React Native Community への移管

Geolocation は 2019年はじめに発足した React Native の Lean Core 運動 の対象とされ、v0.60で React Native Community に機能が移管されました。
v0.60以降 Geolocation は react-native-geolocation から利用できます。

React Native から Geolocation が削除された commits

v0.61 時点では、公式ドキュメントが更新がされていないようなので注意です。(PR は出されていたようなのですが)

v0.59 以下の React Native では Geolocation を使用する場合には import が不要で、navigator.geolocation でグローバルにアクセスできます。
react-native-geolocation をインストールして使用することもできますが、その場合は 0.60 とは違って手動で react-native link する必要があります。手順は ドキュメントをご参照ください。
ですので、v0.59 以下で Geolocation を使った実装をしていた場合は、v0.60 以降 react-native-geolocation に移行作業をする必要があります。

v0.59未満での記述

navigator.geolocation.setRNConfiguration(config);

v0.60 以降での記述

import Geolocation from '@react-native-community/geolocation';

Geolocation.setRNConfiguration(config);

または root ファイル内で、以下のように修正

navigator.geolocation = require('@react-native-community/geolocation');

Android で Geolocation を使用する場合の注意点

React Native の Geolocation を Android アプリで使用する場合も注意が必要です。
Geolocation は android.location を使用しますが、この API は Google から正確性やパフォーマンスの観点から非推奨とされており、 Google Location Services API を使用することが推奨されています。

スクリーンショット 2019-12-20 7.59.54.png

React Native で Google Location Services API を使用する場合は以下のサードパーティを使用するのが良さそうです。

サンプルアプリを作ってみる

前提

では実際に react-native-geolocation を使って位置情報を取得し、地図表示するサンプルアプリを作ってみます。
今回作るサンプルアプリの package.json 抜粋です。

package.json
  "dependencies": {
    "react": "16.9.0",
    "react-native": "0.61.2",
    "@react-native-community/geolocation": "2.0.2",
    "react-native-maps": "0.26.1"
  },

地図表示には、React Native で地図を利用する場合によく使われる react-native-maps を使って実装したいと思います。
また、今回は位置情報に関する実装にフォーカスした内容のため、後述するコード内で使用している以下については、あまり触れていません🙏

  • react-native-maps
  • Hooks
  • TypeScript

導入

今回は React Native v0.61 でサンプルアプリを作るので、react-native-geolocation をインストールします。

$ npm install @react-native-community/geolocation --save

設定

iOS

アプリの使用時に位置情報を有効にするには、Info.plistNSLocationWhenInUseUsageDescriptionNSLocationAlwaysAndWhenInUseUsageDescription キーを含める必要があります。

また、バックグラウンド状態で Geolocation を有効にするには、Info.plistNSLocationAlwaysUsageDescription キーを含め、Xcodeの 'Capabilities' タブからバックグラウンドモードとして Location を追加する必要があります。

スクリーンショット 2019-12-22 17.14.08.png

Android

Android で位置情報へのアクセスをリクエストするには、AndroidManifest.xml に次の行を追加する必要があります。

<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />

Android API >= 23 の場合、インストール時に許可されるものとは別にアプリ内でユーザーに許可を求める必要あります。ドキュメントによると、対応していない場合にアプリがクラッシュする可能性があるとのことなので PermissionsAndroid または react-native-permissions を使用して対応しましょう。

Android API >= 23 Requires an additional step to check for, and request the ACCESS_FINE_LOCATION permission using the PermissionsAndroid API. Failure to do so may result in a hard crash.

現在位置情報を取得する

早速、現在位置情報を取得してみましょう。
react-native-geolocation でユーザーの現在地を取得する場合は、getCurrentPosition() を使います。

以下のコードで一旦位置情報は取得できます。
compoonentDidMount() などで位置情報を取得し、setState() するなどの使い方が一般的かなと思います。

使用コード例

import Geolocation from '@react-native-community/geolocation';

...
  compoonentDidMount() {
    Geolocation.getCurrentPosition(
      position => this.setState({ position }),
      err => alert(err.message),
      { enableHighAccuracy: true, timeout: 10000, maximumAge: 10000 },
    );
  }
...

コード解説

getCurrentPosition() の第一引数には、成功時のコールバック関数を渡します。
コールバックの引数には、以下のようなオブジェクトが渡されます。

{
  "timestamp": 132329930456.35,
  "coords": {
    "accuracy": 5,
    "altitude": 0,
    "altitudeAccuracy": -1,
    "heading": 217,
    "latitude": 35.652832,
    "longitude": 139.839478,
    "speed": 5.77
  }
}

位置情報は coords プロパティから参照でき、以下の情報にアクセスが可能です。

プロパティ名 概要
latitude 緯度
longitude 経度
altitude 高度(m)
accuracy 緯度、経度の精度
altitudeAccuracy 高度の精度
heading 方角(度)
speed 速度(m/S)

その他、第二引数にはエラー時のコールバック関数を渡し、第三引数にはオプションを指定できます。

指定できるオプションのキー

  • timeout (ms): 位置情報取得にかける最大時間。
  • maximumAge (ms): 取得したキャッシュ値の最大キャッシュ時間。
  • enableHighAccuracy (bool): GPS などにアクセスし、より正確な位置情報を取得する。

現在位置を地図で表示する

取得した位置情報から地図にピンを立ててみましょう。

スクリーンショット 2019-12-22 16.07.21.png

以下がソースコードです。

import React, { useState, useEffect } from 'react';
import { SafeAreaView, View, Text, Dimensions, StyleSheet } from 'react-native';

import Geolocation, {
  GeolocationResponse,
} from '@react-native-community/geolocation';
import MapView, { Marker } from 'react-native-maps';

const { width, height } = Dimensions.get('window');
const ASPECT_RATIO = width / height;
const LATITUDE_DELTA = 0.01;
const LONGITUDE_DELTA = LATITUDE_DELTA * ASPECT_RATIO;

const CurrentUserMapView = () => {
  const [position, setPosition] = useState<GeolocationResponse['coords']>({
    latitude: 0,
    longitude: 0,
    accuracy: 0,
    altitude: 0,
    altitudeAccuracy: 0,
    heading: 0,
    speed: 0,
  });

  useEffect(() => {
    Geolocation.getCurrentPosition(
      position => setPosition(position.coords),
      err => alert(err.message),
      { enableHighAccuracy: true, timeout: 10000 },
    );
  }, []);

  return (
    <SafeAreaView style={styles.container}>
      {position && (
        <MapView
          style={styles.map}
          initialRegion={{
            latitude: position.latitude,
            longitude: position.longitude,
            latitudeDelta: LATITUDE_DELTA,
            longitudeDelta: LONGITUDE_DELTA,
          }}>
          <Marker
            coordinate={{
              latitude: position.latitude,
              longitude: position.longitude,
            }}
          />
          {/* Debug 用に coords オブジェクトを表示 */}
          <View style={styles.debugContainer}>
            <Text>{`coords: {`}</Text>
            {Object.keys(position).map(key => {
              return <Text key={key}>{`  ${key} : ${position[key]}`}</Text>;
            })}
            <Text>{`}`}</Text>
          </View>
        </MapView>
      )}
    </SafeAreaView>
  );
};

const styles = StyleSheet.create({
  container: {
    width: '100%',
    height: '100%',
  },
  map: {
    flex: 1,
  },
  debugContainer: {
    backgroundColor: '#fff',
    opacity: 0.8,
  },
});

export default CurrentUserMapView;

コード解説

getCurrentPosition() で取得した緯度/経度情報 を react-native-maps の MapView, Maker コンポーネントに渡すだけで、地図に表示することができました。
今回は iOS のデフォルト地図アプリで表示していますが、Google Map で表示することも可能です。

追跡した位置情報を取得する

続いて、watchPosition() を使用することで、ユーザーを追跡して位置情報を取得することができます。

使用コード例

import Geolocation from '@react-native-community/geolocation';

...
  componentDidMount() {
    this._watchID = Geolocation.watchPosition(
      position => this.setState({ position });
      error => alert(error.message),
      { enableHighAccuracy: true, timeout: 10000, distanceFilter: 10 },
    );
  }

  componentWillUnmount() {
    Geolocation.clearWatch(this._watchID);
  }
...

コード解説

watchPosition()getCurrentPosition() とインターフェイスとしては同じように扱えますが、以下のような違いがあります。

  • オプションとして以下を指定できる。
    • distanceFilter (m): 新たな位置情報を返すユーザーの最短移動距離。
    • useSignificantChanges (bool): バッテリー効率の高いネイティブの API を使用する。
  • 返り値である watchId を取得して、アンマウント時などに clearWatch() or stopObserving() する。

追跡した位置情報を地図で表示する

現在位置を地図で表示した時と同じように、ユーザーの追跡を地図で表示してみましょう。

Dec-22-2019 11-10-11.gif

以下がソースコードです。

import React, { useState, useEffect } from 'react';
import { SafeAreaView, View, Text, Dimensions, StyleSheet } from 'react-native';

import Geolocation, {
  GeolocationResponse,
} from '@react-native-community/geolocation';

import MapView, { Marker } from 'react-native-maps';

const { width, height } = Dimensions.get('window');
const ASPECT_RATIO = width / height;
const LATITUDE_DELTA = 0.01;
const LONGITUDE_DELTA = LATITUDE_DELTA * ASPECT_RATIO;

const TrackUserMapView = () => {
  const [position, setPosition] = useState<GeolocationResponse['coords']>({
    latitude: 0,
    longitude: 0,
    accuracy: 0,
    altitude: 0,
    altitudeAccuracy: 0,
    heading: 0,
    speed: 0,
  });

  useEffect(() => {
    const watchId = Geolocation.watchPosition(
      position => setPosition(position.coords),
      err => alert(err.message),
      { enableHighAccuracy: true, timeout: 10000, distanceFilter: 1 },
    );
    return () => Geolocation.clearWatch(watchId);
  }, []);

  return (
    <SafeAreaView style={styles.container}>
      {position && (
        <MapView
          style={styles.map}
          initialRegion={{
            latitude: position.latitude,
            longitude: position.longitude,
            latitudeDelta: LATITUDE_DELTA,
            longitudeDelta: LONGITUDE_DELTA,
          }}>
          <Marker
            image={require('./walking_man.png')}
            coordinate={{
              latitude: position.latitude,
              longitude: position.longitude,
            }}
          />
          {/* Debug 用に coords オブジェクトを表示 */}
          <View style={styles.debugContainer}>
            <Text>{`coords: {`}</Text>
            {Object.keys(position).map(key => {
              return <Text key={key}>{`  ${key} : ${position[key]}`}</Text>;
            })}
            <Text>{`}`}</Text>
          </View>
        </MapView>
      )}
    </SafeAreaView>
  );
};

const styles = StyleSheet.create({
  container: {
    width: '100%',
    height: '100%',
  },
  map: {
    flex: 1,
  },
  debugContainer: {
    backgroundColor: '#fff',
    opacity: 0.8,
  },
});

export default TrackUserMapView;

コード解説

useEffect() 内の処理以外は、現在位置を地図で表示した場合と基本的に同じコードです。
今回は人間が走る速度でも、ある程度の動きが欲しかったので、オプション distanceFilter には 1 を指定しています。

また、ピンが動くのも味気なかったので、オープンソースである Twemoji を使って、マーカーを表示しました。
<Marker />props.image に画像リソースを渡すだけでピンを好きなものに変更できます。

Simulator でのデバック

Simulator では、位置情報のデバック用に設定があり、メニューの [Debug] > [Location] から、シーンや緯度/経度の指定ができます。
上記のキャプチャでは City Run でデバックしています。

項目 概要
None 位置指定なし
Custom Location 緯度/経度を指定
City Run 走る速度で移動
City Bicycle Ride 自転車の速度で移動
Freeway Drive 車の速度で移動
Apple Apple 本社の座標 (default)

実際に他のモードも試してみましたが、角を曲がるときに速度が落ちたり、信号待ちで結構な時間動かなかったり (最初バグかと思う) 忠実に現実の動きを再現しているようです。
どのモードでも Apple 本社の周りからどこかに向かって移動しています。

Dec-22-2019 11-15-05.gifDec-22-2019 11-12-26.gif

ちなみに上記 Freeway Drive の場合では、マーカーがすぐに地図外へ移動してしまうので、常に地図の中心に来るように MapViewregion を動的に変更しています。

jsx
        <MapView
          style={styles.map}
-          initialRegion={{
+          region={{
            latitude: position.latitude,
            longitude: position.longitude,
            latitudeDelta: LATITUDE_DELTA,
            longitudeDelta: LONGITUDE_DELTA,
          }}>

おわりに

以上、簡単な React Native サンプルアプリで位置情報を地図表示する方法の紹介でした。

実際にユーザーに使ってもらうアプリとなると、位置情報の取得処理がユーザーデバイスの環境に依存してしまうことや、パーミッション周りの体験設計、iOS/Android それぞれの規約を順守することなど、大変なことは出てきそうです。
位置情報は個人情報としても重要なものと一般的に解釈されており、先日リリースされた iOS13 でも位置情報周りのセキュリティが強化され、パーミッション設定にもアップデートもありました。Android でも今年に Android Q でパーミッション設定に似たようなアップデートがあったばかりなので、この辺りはこれからも変化が続きそうです。

参考:

セキュリティ、バックグラウンド時のバッテリー消費の観点など、アプリの位置情報取得についてはまだまだ課題があるようです。個人的にこの先の流れにも興味があるので、今後も追っていく中で何か情報発信ができればと思っています。

最後まで読んでいただきありがとうございました。
本記事の内容で誤り or ここはこうできるよなどあれば、コメントいただけますと大変嬉しいです!

明日の React Native Advent Calendar 2019 は 「PIXTA」の @naoshihoshi さんによる記事です。よろしくお願いします!

49
42
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
49
42

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?