16
12

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.

過去にアイドルが居た場所に近づくとスマホが震えるアプリ

Last updated at Posted at 2017-10-12

#概要
都市の幽霊をコンセプトにしたアイドルのグループの・・・・・・・・・は、androidアプリ「都市の幽霊観測装置」をリリースしました。

ダウンロードはこちらから
都市の幽霊観測装置 - Google Play

このアプリをインストールして「過去にアイドルが居た場所」に近づくと、振動と通知で「そこにアイドルが居たよ」と教えてくれます。

コンセプトイメージ.001.jpeg

実際の通知画面はこちら。「・ちゃん」とは・・・・・・・・・メンバーの呼称です。

通知.png

#なぜこのアプリを作ったのか

・・・・・・・・・は「都市の幽霊」「東京そのものになる」というコンセプトを実現するために、これまでもビーコンや位置情報を使ったイベントを行ってきましたが、いずれも「モニターをガン見しながら」「こちらから探しにいく」プレイスタイルで、「都市に・ちゃんがぷかぷか幽霊のように浮いている」「アイドルをまとう」「日常に・ちゃんが混じる」といった世界観を、完全には表現できていませんでした。

LINE Beaconを持ったアイドルを探し回るイベント
http://qiita.com/a-r-i/items/8987a808f60a294e894b

アイドルが居る方角を指し示すコンパス
http://qiita.com/a-r-i/items/1389397ceeeff19b4e02

いっぽう都市の幽霊観測装置ではただ日常生活を送っているだけで、・ちゃんを観測できます
必要なのはアプリのインストールと起動だけ。
あとは受動的に、これまで通り生きる。
モニターを見る必要はありません。
土地に染み付いた震えで・ちゃんを感じる。
地表に「・ちゃんのGPSログという新しいレイヤー」が重なった世界で、いつもとかわらず過ごすだけ。

同じジオ・アプリのポケモンGO!と都市の幽霊観測装置を比較してみると、その特徴が際立ちます。

ユーザーの意識 プレイスタイル
ポケモンGO! モニター内 能動的にポケモンを探しに行く
都市の幽霊観測装置 現実世界 受動的に・ちゃんと出会うのを待つ

このアプリは「ポスト・ポケモンGO!時代のアイドル」を標榜してきた・・・・・・・・・の、ポスト・ポケモンGO!時代に向けたひとつの回答です。

“ポスト・ポケモンGO!”時代のアイドル・グループ!? 「・・・・・・・・・」デビュー直前初インタヴュー

僕たちはモニター抜きでアイドルと遠隔でつながることができるんじゃないかと考えていて。メディア環境の変化に応じてアイドルも変化してきた歴史があって、僕たちも動画サイトとSNS以降の対応すべき情報環境、つまりIoT時代や「ポケモンGO!」=都市の風景を拡張するAR時代の先に行きたいと思っています。言い方が難しいんですけど「今会えるアイドル」じゃなくて「常に纏える(まとえる)アイドル」を作りたいんです。

目に見えない都市の幽霊が街を徘徊し、気づかないうちに、彼女たちとすれ違っている。
東京に住むひとりひとりが、いつの間にか・ちゃんを身に纏っている。

名称未設定.001.jpeg

都市の幽霊観測装置で世界に・ちゃんをインストールしませんか。

Imgur

#仕組み
①GPSロガーで・ちゃんの位置情報を計測・記録
②ユーザーがスマートフォンで都市の幽霊観測装置を起動
③ユーザーが50m移動するごとに、・ちゃんのGPSログとユーザーの現在位置を比較・照合
④・ちゃんのGPSログのなかに、ユーザーの現在位置と近いもの(10m以内)があれば、通知を出し、スマホを振動させる。

ghostintecity.001.jpeg

##位置情報の記録
・ちゃんの位置情報ログはGPSロガーのHolux M-241を用いて取得しました。

##定期的にユーザーの位置情報を取得する
後述するように「日常生活でいつの間にかすれ違っている」ことを重視しているので、できるだけユーザーにアプリを意識させたくありませんでした。

androidのrequestLocationUpdatesメソッドは、アプリがバックグラウンド状態になっても、定期的に実行され、位置情報を取得します。
今回はデバイスが50m移動するごとに位置情報を取得する設定にしました。

##GPSログを検索する
ユーザーの位置情報を取得したら、「過去の・ちゃんの位置情報が記録されたデータベース」から「ユーザーの現在位置に近い・ちゃんのログがあるか」検索します。

具体的には、androidクライアントが、APIサーバーにユーザーの位置情報をPOSTします。
APIサーバーは、「ユーザーの現在位置±10mの範囲内にある、過去の・ちゃんの位置情報」をデータベースから検索し、その結果をクライアントに返します。

##通知を出す
ユーザーがいま居る場所から10m以内にある・ちゃんの位置情報が見つかったら、通知を出します。
その際、バイブレーションを作動させ、スマートフォンを10秒震わせます。

観測されると、・ちゃんがピョコッと跳び出てくるイメージです

通知.png

##クライアントサイドのコード

MainActivity.java
    @Override
    public void onLocationChanged(Location location) {

     //ユーザーの位置情報
        final Double observerLatitude = location.getLatitude();
        final Double observerLongitude = location.getLongitude();

        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    JSONObject paramJSON = new JSONObject();
                    paramJSON.put("latitude", observerLatitude);
                    paramJSON.put("longitude", observerLongitude);

                    final String USERNAME   = "";
                    final String PASSWORD   = "";
                    final String userPassword = USERNAME + ":" + PASSWORD;
                    final String encodeAuthorization = Base64.encodeToString(userPassword.getBytes(), Base64.NO_WRAP);

                    try {
                        String buffer = "";
                        HttpURLConnection con = null;
                        URL url = new URL(""); //APIのURL
                        con = (HttpURLConnection) url.openConnection();
                        con.setRequestMethod("POST");
                        con.setInstanceFollowRedirects(false);
                        con.setRequestProperty("Accept-Language", "jp");
                        con.setDoOutput(true);
                        con.setRequestProperty("Authorization", "Basic " + encodeAuthorization);
                        OutputStream os = con.getOutputStream();
                        PrintStream ps = new PrintStream(os);
                        ps.print(paramJSON); //ユーザーの位置情報を、APIにPOST
                        ps.close();

                        BufferedReader reader =
                                new BufferedReader(new InputStreamReader(con.getInputStream(), "UTF-8"));
                        buffer = reader.readLine();

                        JSONObject jsonObject = new JSONObject(buffer);

                        //APIからtrue(・ちゃんがこの近くに居た)と帰ってくれば、通知処理に進む。
                        if(jsonObject.getBoolean("result")){

                            Log.d("結果", "・ちゃんを観測した");

                            String ghostDate = jsonObject.getString("timestamp");

                            //通知処理
                            Intent notificationIntent = new Intent(context, MainActivity.class);
                            Notification.Builder builder = new Notification.Builder(context);
                            NotificationManager manager= (NotificationManager)getSystemService(Context.NOTIFICATION_SERVICE);
                            builder.setSmallIcon(R.mipmap.ic_launcher);
                            builder.setContentTitle("・ちゃんを観測");
                            builder.setContentText(ghostDate + "の・ちゃんを観測");
                            builder.setDefaults(Notification.PRIORITY_DEFAULT);
                            manager.notify(1,builder.build());

                            //バイブレーションを10秒震わせる
                            vib = (Vibrator)getSystemService(VIBRATOR_SERVICE);
                            vib.vibrate(10000);

                        }else{
                            Log.d("結果", "・ちゃんが観測できない");
                        }
                        con.disconnect();
                    } catch (MalformedURLException e) {
                        e.printStackTrace();
                    } catch (ProtocolException e) {
                        e.printStackTrace();
                    } catch (IOException e) {
                        e.printStackTrace();
                    } catch (Exception e) {
                        e.printStackTrace();
                    }

                } catch(Exception ex) {
                    System.out.println(ex);
                }
            }
        }).start();
    }

##サーバーサイドのコード

search_controller.rb
def search_location
    observer_location_json_key_array = []

    params.each_key{|key|
      observer_location_json_key_array.push(key)
    }

    observer_location_raw = observer_location_json_key_array[0]
    observer_latitude = observer_location_raw.split(",")[0].split(":")[1].to_f
    observer_longitude = observer_location_raw.split(",")[1].split(":")[1].delete("}").to_f

    #観測通知を出す範囲。ユーザーが、過去の・ちゃんの位置に、どれくらい近ければ通知を出すか。約10m以内に設定。
    range = 0.00005
    min_latitude = observer_latitude - range
    max_latitude = observer_latitude + range
    min_longitude = observer_longitude - range
    max_longitude = observer_longitude + range

    #「ユーザーの位置情報+調整レンジ」の範囲内にある位置情報が存在するか、検索。
    result = Location.where(latitude: min_latitude..max_latitude,longitude:min_longitude..max_longitude)

    #存在すれば、trueとその位置情報が記録された日時をjsonに格納。
    if result[0] == nil
      result_json = { result: false}
    else
      result_json = { result:true, timestamp: result[0].timestamp}
    end

    #jsonを返す。
    render json: result_json
end

#参考文献
GPSロギング
GPSロガー「HOLUX M-241」をMacで利用する

位置情報の取得
[Android] GPSで位置情報を取得するアプリを作る

POSTリクエスト
https://qiita.com/kaikusakari/items/b1dd02803bb16956010a
http://wicachi.com/archives/151021【6.0動作確認済】手軽にAndroidでJSON形式POSTする方法(非推奨API有り).html

Basic認証
http://www.programing-style.com/android/android-api/android-basic-authentication/

通知
Androidアプリでステータスバーに通知を出すやりかた。

バイブレーション
Androidでバイブレーター動作

16
12
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
16
12

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?