345 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			345 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
package com.example.childguard;
 | 
						||
 | 
						||
import android.annotation.SuppressLint;
 | 
						||
import android.app.Activity;
 | 
						||
import android.app.NotificationChannel;
 | 
						||
import android.app.NotificationManager;
 | 
						||
import android.app.PendingIntent;
 | 
						||
import android.app.Service;
 | 
						||
import android.bluetooth.BluetoothDevice;
 | 
						||
import android.content.BroadcastReceiver;
 | 
						||
import android.content.Context;
 | 
						||
import android.content.Intent;
 | 
						||
import android.content.SharedPreferences;
 | 
						||
import android.content.IntentFilter;
 | 
						||
import android.content.pm.PackageManager;
 | 
						||
import android.graphics.Color;
 | 
						||
import android.os.Build;
 | 
						||
import android.os.Bundle;
 | 
						||
import android.os.Handler;
 | 
						||
import android.os.IBinder;
 | 
						||
import android.os.Looper;
 | 
						||
import android.os.Vibrator;
 | 
						||
import android.preference.PreferenceManager;
 | 
						||
import android.util.Log;
 | 
						||
import android.widget.Toast;
 | 
						||
 | 
						||
import androidx.annotation.Nullable;
 | 
						||
import androidx.core.app.ActivityCompat;
 | 
						||
import androidx.core.app.NotificationCompat;
 | 
						||
import androidx.recyclerview.widget.DividerItemDecoration;
 | 
						||
import androidx.recyclerview.widget.LinearLayoutManager;
 | 
						||
import androidx.recyclerview.widget.RecyclerView;
 | 
						||
 | 
						||
import com.google.firebase.firestore.DocumentReference;
 | 
						||
import com.google.firebase.firestore.FirebaseFirestore;
 | 
						||
 | 
						||
import java.text.DateFormat;
 | 
						||
import java.text.SimpleDateFormat;
 | 
						||
import java.util.ArrayList;
 | 
						||
import java.util.Arrays;
 | 
						||
import java.util.Date;
 | 
						||
import java.util.HashMap;
 | 
						||
import java.util.Map;
 | 
						||
 | 
						||
public class TestService extends Service {
 | 
						||
    FirebaseFirestore db;
 | 
						||
    DocumentReference mDocRef;
 | 
						||
 | 
						||
    public static final String TAG = "InspirationQuote";
 | 
						||
 | 
						||
    public PeriodicTaskManager periodicTaskManager;
 | 
						||
 | 
						||
    //時間を取得するやつ↓
 | 
						||
    public static String getNowDate() {
 | 
						||
        @SuppressLint("SimpleDateFormat") final DateFormat df = new SimpleDateFormat("yyy/MM/dd HH:mm:ss");
 | 
						||
        final Date date = new Date(System.currentTimeMillis());
 | 
						||
        return df.format(date);
 | 
						||
    }
 | 
						||
 | 
						||
   @SuppressLint("NotifyDataSetChanged")
 | 
						||
   public void NotifityRecycle(){
 | 
						||
 | 
						||
       RecyclerView recyclerView = recyclerView().findViewById(R.id.recyclerView1);
 | 
						||
 | 
						||
        //RecyclerViewのサイズを固定
 | 
						||
        recyclerView.setHasFixedSize(true);
 | 
						||
 | 
						||
        //RecyclerViewに区切り線を入れる
 | 
						||
        RecyclerView.ItemDecoration itemDecoration =
 | 
						||
                new DividerItemDecoration(this, DividerItemDecoration.VERTICAL);
 | 
						||
        recyclerView.addItemDecoration(itemDecoration);
 | 
						||
 | 
						||
        //レイアウトマネージャを設定
 | 
						||
        RecyclerView.LayoutManager layoutManager = new LinearLayoutManager(this);
 | 
						||
        recyclerView.setLayoutManager(layoutManager);
 | 
						||
 | 
						||
        //①リスト構造(String型の可変長の配列)を宣言
 | 
						||
        ArrayList<String> arrayList = new ArrayList<>();
 | 
						||
 | 
						||
        //③Adapterとリスト構造を結び付け
 | 
						||
        //RecyclerAdapterクラスを呼び出す
 | 
						||
        RecyclerAdapter2 adapter = new RecyclerAdapter2(arrayList, null);
 | 
						||
 | 
						||
        //④RecyclerViewとAdapterの結び付け
 | 
						||
        recyclerView.setAdapter(adapter);
 | 
						||
 | 
						||
        SharedPreferences pref = PreferenceManager.getDefaultSharedPreferences(this);
 | 
						||
 | 
						||
       arrayList.add(0,"通知を検知しました: " + getNowDate());
 | 
						||
 | 
						||
       //一番下に値が追加されたことをAdapterが画面に通知
 | 
						||
       adapter.notifyItemInserted(arrayList.size());
 | 
						||
 | 
						||
       //共有プリファレンス 書き込みの準備
 | 
						||
       SharedPreferences.Editor e = pref.edit();
 | 
						||
 | 
						||
       //リストを,区切りで結合する
 | 
						||
       String str = String.join(",", arrayList);
 | 
						||
 | 
						||
       //変数名currentに、値の代入
 | 
						||
       e.putString("current", str.toString());
 | 
						||
 | 
						||
       //確定処理
 | 
						||
       e.apply();
 | 
						||
 | 
						||
       String strTo = pref.getString("current", null);
 | 
						||
       if (strTo == null) return;
 | 
						||
 | 
						||
       String[] list = strTo.split(",");
 | 
						||
       arrayList.clear();
 | 
						||
       arrayList.addAll(Arrays.asList(list));
 | 
						||
       adapter.notifyDataSetChanged();
 | 
						||
    }
 | 
						||
 | 
						||
    private Activity recyclerView() {
 | 
						||
        return null;
 | 
						||
    }
 | 
						||
 | 
						||
    @Override
 | 
						||
    public int onStartCommand(Intent intent, int flags, int startId) {
 | 
						||
 | 
						||
       NotifityRecycle();
 | 
						||
        //共有プリファレンス全体の準備
 | 
						||
        SharedPreferences sharedPreferences = getSharedPreferences("app_situation", MODE_PRIVATE);
 | 
						||
        String IdPref = sharedPreferences.getString("ID", null);//アプリに記録されているIDの取得
 | 
						||
        if (IdPref == null) {//FireBaseのIDがアプリに登録されているとき
 | 
						||
            Log.d("onResume", "ID not initialized.");
 | 
						||
        } else {
 | 
						||
            mDocRef = FirebaseFirestore.getInstance().document("status/" + IdPref);//現在の位置を取得
 | 
						||
            initNotification(mDocRef);//現在の位置を引数に initNotification()を処理
 | 
						||
        }
 | 
						||
        return flags;
 | 
						||
 | 
						||
 | 
						||
    }
 | 
						||
 | 
						||
    private void initNotification(DocumentReference mDocRef) {//サイト上で押されたボタンの管理
 | 
						||
        // PeriodicTaskManagerのインスタンス化
 | 
						||
        periodicTaskManager = new PeriodicTaskManager();
 | 
						||
        // 共有プリファレンス全体の準備
 | 
						||
        SharedPreferences sharedPreferences = getSharedPreferences("app_situation", MODE_PRIVATE);
 | 
						||
        //車の乗り降りを管理するtrue=乗車、false=降車
 | 
						||
        //exists()でdocumentSnapshotの中のファイルの存在の確認
 | 
						||
        mDocRef.addSnapshotListener((documentSnapshot, e) -> {
 | 
						||
 | 
						||
            Log.d("nt", "イベント開始");
 | 
						||
            //共有プリファレンス 書き込みの準備
 | 
						||
            SharedPreferences.Editor E = sharedPreferences.edit();
 | 
						||
            //車の乗り降りを管理するtrue=乗車、false=降車
 | 
						||
            if (documentSnapshot.exists()) {//exists()でdocumentSnapshotの中のファイルの存在の確認
 | 
						||
                Boolean isInCar = sharedPreferences.getBoolean("isInCarPref", false);//現在の乗降状態を保存する共有プリファレンス
 | 
						||
                E.putBoolean("isInCarPref", documentSnapshot.getBoolean("isInCar"));//乗降状態の判定
 | 
						||
                E.apply();//確定処理
 | 
						||
                Log.d("nt", "レスポンスを検知しました1");
 | 
						||
                if (isInCar) {//isReportedがtrue=サイト上で乗車状態のとき
 | 
						||
                    if (documentSnapshot.getBoolean("isReported")) {
 | 
						||
                        ResetReported();// ResetReported();を処理→FireBaseのisReportedをfalseにする
 | 
						||
                        NotificationSetting();//通知に関する設定のメソッド
 | 
						||
                        Notification(getApplicationContext());//通知を行うメソッド
 | 
						||
                    }
 | 
						||
                } else if(!isInCar){//isReportedがfalse=サイト上で降車状態のとき
 | 
						||
                    ResetReported();//ResetReported();を処理→FireBaseのisReportedをfalseにする
 | 
						||
                }else {
 | 
						||
                    ResetReported();//ResetReported();を処理→FireBaseのisReportedをfalseにする
 | 
						||
                }
 | 
						||
            }
 | 
						||
 | 
						||
        });
 | 
						||
    }
 | 
						||
 | 
						||
    public void ResetReported() {//FireBaseのisReportedをfalseに初期化するメソッド
 | 
						||
        //共有プリファレンス全体の準備
 | 
						||
        SharedPreferences sharedPreferences = getSharedPreferences("app_situation", MODE_PRIVATE);
 | 
						||
        String IdPref = sharedPreferences.getString("ID", null);//アプリに記録されているIDの取得
 | 
						||
        db = FirebaseFirestore.getInstance();//Firebaseとの紐づけ
 | 
						||
        DocumentReference isReported = db.collection("status").document(IdPref);//更新するドキュメントとの紐づけ
 | 
						||
        Map<String, Boolean> DEFAULT_ITEM = new HashMap<>();//mapの宣言
 | 
						||
        DEFAULT_ITEM.put("isReported", false);
 | 
						||
        //isReportedをfalseに更新
 | 
						||
        isReported.update("isReported", false).addOnSuccessListener(unused -> Log.d(TAG, "DocumentSnapshot successfully updated!")).addOnFailureListener(e -> Log.w(TAG, "Error updating document", e));
 | 
						||
    }
 | 
						||
 | 
						||
    public void NotificationSetting() {//通知に関する設定の処理を行うメソッド
 | 
						||
            int importance = NotificationManager.IMPORTANCE_DEFAULT;
 | 
						||
            //通知チャネルの実装
 | 
						||
            NotificationChannel channel = new NotificationChannel("CHANNEL_ID", "通知", importance);
 | 
						||
            channel.setDescription("第三者により置き去りの通報が行われたときに通知します。");
 | 
						||
 | 
						||
            NotificationManager notificationManager = getSystemService(NotificationManager.class);
 | 
						||
            notificationManager.createNotificationChannel(channel);
 | 
						||
 | 
						||
    }
 | 
						||
 | 
						||
    public void Notification(Context context) {//実際に通知を行うメソッド
 | 
						||
         final String CHANNEL_ID = "my_channel_id";
 | 
						||
        // 通知がクリックされたときに送信されるIntent
 | 
						||
        Intent intent = new Intent(context, MainActivity.class);
 | 
						||
        intent.setAction("OPEN_ACTIVITY");
 | 
						||
        // PendingIntentの作成
 | 
						||
        int requestCode = 100;
 | 
						||
        int flags = 0;
 | 
						||
        PendingIntent pendingIntent = PendingIntent.getActivity(context, requestCode, intent, flags | PendingIntent.FLAG_IMMUTABLE);
 | 
						||
 | 
						||
        ((Vibrator) getSystemService(Context.VIBRATOR_SERVICE)).vibrate(2000);//バイブレーション
 | 
						||
 | 
						||
        @SuppressLint("NotificationTrampoline") NotificationCompat.Builder builder = new NotificationCompat.Builder(context, "CHANNEL_ID")
 | 
						||
                .setSmallIcon(android.R.drawable.ic_menu_info_details)
 | 
						||
                .setContentTitle("子供の置き去りをしていませんか?")//通知のタイトル
 | 
						||
                .setContentText("第三者からの通報が行われました。")//通知の本文
 | 
						||
                .setContentIntent(pendingIntent)//通知をタップするとActivityへ移動する
 | 
						||
                .setAutoCancel(true)//通知をタップすると削除する
 | 
						||
                .setPriority(NotificationCompat.PRIORITY_HIGH) // プライオリティを高く設定
 | 
						||
                .setVisibility(NotificationCompat.VISIBILITY_PUBLIC); // ロック画面に表示する
 | 
						||
 | 
						||
        // NotificationChannelの作成(Android 8.0以降)
 | 
						||
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
 | 
						||
            NotificationManager notificationManager = context.getSystemService(NotificationManager.class);
 | 
						||
            if (notificationManager != null) {
 | 
						||
                NotificationChannel channel = new NotificationChannel(
 | 
						||
                        CHANNEL_ID,
 | 
						||
                        "Channel Name",
 | 
						||
                        NotificationManager.IMPORTANCE_HIGH
 | 
						||
                );
 | 
						||
 | 
						||
                channel.setDescription("Channel Description");
 | 
						||
                channel.enableLights(true);
 | 
						||
                channel.setLightColor(Color.RED);
 | 
						||
                channel.enableVibration(true);
 | 
						||
                notificationManager.createNotificationChannel(channel);
 | 
						||
            }
 | 
						||
        }
 | 
						||
 | 
						||
 | 
						||
        NotificationManager notificationManager = (NotificationManager)context.getSystemService(context.NOTIFICATION_SERVICE);
 | 
						||
        if (ActivityCompat.checkSelfPermission(this, android.Manifest.permission.POST_NOTIFICATIONS) != PackageManager.PERMISSION_GRANTED) {
 | 
						||
            return;
 | 
						||
        }
 | 
						||
        notificationManager.notify(R.string.app_name, builder.build());//通知の表示
 | 
						||
    }
 | 
						||
    public void NotificationBluetooth(Context context, int time) {//実際に通知を行うメソッド
 | 
						||
        final String CHANNEL_ID = "my_channel_id";
 | 
						||
        // 通知がクリックされたときに送信されるIntent
 | 
						||
        Intent intent = new Intent(context, MainActivity.class);
 | 
						||
        intent.setAction("OPEN_ACTIVITY");
 | 
						||
        // PendingIntentの作成
 | 
						||
        int requestCode = 100;
 | 
						||
        int flags = 0;
 | 
						||
        PendingIntent pendingIntent = PendingIntent.getActivity(context, requestCode, intent, flags | PendingIntent.FLAG_IMMUTABLE);
 | 
						||
 | 
						||
        ((Vibrator) getSystemService(Context.VIBRATOR_SERVICE)).vibrate(2000);//バイブレーション
 | 
						||
 | 
						||
        @SuppressLint("NotificationTrampoline") NotificationCompat.Builder builder = new NotificationCompat.Builder(context, "CHANNEL_ID")
 | 
						||
                .setSmallIcon(android.R.drawable.ic_menu_info_details)
 | 
						||
                .setContentTitle("子供の置き去りをしていませんか?")//通知のタイトル
 | 
						||
                .setContentText("Bluetoothと車の切断から"+time+"分が経過しました")//通知の本文
 | 
						||
                .setContentIntent(pendingIntent)//通知をタップするとActivityへ移動する
 | 
						||
                .setAutoCancel(true)//通知をタップすると削除する
 | 
						||
                .setPriority(NotificationCompat.PRIORITY_HIGH) // プライオリティを高く設定
 | 
						||
                .setVisibility(NotificationCompat.VISIBILITY_PUBLIC); // ロック画面に表示する
 | 
						||
 | 
						||
        // NotificationChannelの作成(Android 8.0以降)
 | 
						||
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
 | 
						||
            NotificationManager notificationManager = context.getSystemService(NotificationManager.class);
 | 
						||
            if (notificationManager != null) {
 | 
						||
                NotificationChannel channel = new NotificationChannel(
 | 
						||
                        CHANNEL_ID,
 | 
						||
                        "Channel Name",
 | 
						||
                        NotificationManager.IMPORTANCE_HIGH
 | 
						||
                );
 | 
						||
 | 
						||
                channel.setDescription("Channel Description");
 | 
						||
                channel.enableLights(true);
 | 
						||
                channel.setLightColor(Color.RED);
 | 
						||
                channel.enableVibration(true);
 | 
						||
                notificationManager.createNotificationChannel(channel);
 | 
						||
            }
 | 
						||
        }
 | 
						||
 | 
						||
 | 
						||
        NotificationManager notificationManager = (NotificationManager)context.getSystemService(context.NOTIFICATION_SERVICE);
 | 
						||
        if (ActivityCompat.checkSelfPermission(this, android.Manifest.permission.POST_NOTIFICATIONS) != PackageManager.PERMISSION_GRANTED) {
 | 
						||
            return;
 | 
						||
        }
 | 
						||
        notificationManager.notify(R.string.app_name, builder.build());//通知の表示
 | 
						||
    }
 | 
						||
    public class PeriodicTaskManager {//Bluetoothの切断後に乗車状態にならなかった場合に5分毎に通知を送るメソッド
 | 
						||
 | 
						||
        private static final long INTERVAL = 5 * 1000; //300秒
 | 
						||
        private final Handler handler;
 | 
						||
        private final Runnable periodicTask;
 | 
						||
 | 
						||
        public PeriodicTaskManager() {
 | 
						||
 | 
						||
            handler = new Handler(Looper.getMainLooper());
 | 
						||
            periodicTask = new Runnable() {
 | 
						||
 | 
						||
                public void run() {
 | 
						||
 | 
						||
                    //共有プリファレンス全体の準備
 | 
						||
                    SharedPreferences sharedPreferences = getSharedPreferences("app_situation", MODE_PRIVATE);
 | 
						||
 | 
						||
                    int time=sharedPreferences.getInt("time",0);
 | 
						||
                    //共有プリファレンス 書き込みの準備
 | 
						||
                    SharedPreferences.Editor E = sharedPreferences.edit();
 | 
						||
                    if(time==0) {//Bluetooth切断からの時間経過(5分刻み)
 | 
						||
                        E.putInt("time",5);
 | 
						||
                        E.apply();;
 | 
						||
                    }else {
 | 
						||
                        E.putInt("time",time+5);
 | 
						||
                        E.apply();
 | 
						||
                    }
 | 
						||
 | 
						||
                    // 5分毎に実行される処理
 | 
						||
                    NotificationBluetooth(getApplicationContext(),time);
 | 
						||
                    Log.d("PeriodicTask", "5分後に処理を実行します");
 | 
						||
 | 
						||
                    handler.postDelayed(this, INTERVAL);
 | 
						||
                }
 | 
						||
            };
 | 
						||
        }
 | 
						||
 | 
						||
        public void startPeriodicTask() {
 | 
						||
            // 最初の実行
 | 
						||
            handler.postDelayed(periodicTask,INTERVAL);//一回目は5分後に行う
 | 
						||
        }
 | 
						||
 | 
						||
        public void stopPeriodicTask() {
 | 
						||
            //共有プリファレンス全体の準備
 | 
						||
            SharedPreferences sharedPreferences = getSharedPreferences("app_situation", MODE_PRIVATE);
 | 
						||
 | 
						||
            int time=sharedPreferences.getInt("time",0);
 | 
						||
            //共有プリファレンス 書き込みの準備
 | 
						||
            SharedPreferences.Editor E = sharedPreferences.edit();
 | 
						||
            E.putInt("time",0);
 | 
						||
            E.apply();
 | 
						||
            // 定期的な処理の停止
 | 
						||
            handler.removeCallbacks(periodicTask);
 | 
						||
        }
 | 
						||
    }
 | 
						||
    @Nullable
 | 
						||
    @Override
 | 
						||
    public IBinder onBind(Intent intent) {
 | 
						||
        return null;
 | 
						||
    }
 | 
						||
} |