package com.example.childguard; import android.annotation.SuppressLint; import android.app.NotificationChannel; import android.app.NotificationManager; import android.app.PendingIntent; import android.app.Service; import android.content.Context; import android.content.Intent; import android.content.SharedPreferences; import android.content.pm.PackageManager; import android.graphics.Color; import android.os.Build; import android.os.Handler; import android.os.IBinder; import android.os.Looper; import android.os.Vibrator; import android.util.Log; import androidx.annotation.Nullable; import androidx.core.app.ActivityCompat; import androidx.core.app.NotificationCompat; import com.google.firebase.firestore.DocumentReference; import com.google.firebase.firestore.FirebaseFirestore; 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; @Override public int onStartCommand(Intent intent, int flags, int startId) { //共有プリファレンス全体の準備 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 (documentSnapshot.getBoolean("isReported") == true && isInCar) {//isReportedがtrueかつisInCarがtrueのとき=サイト上で第三者ボタンが押されたときに乗車状態のとき ResetReported();// ResetReported();を処理→FireBaseのisReportedをfalseにする int importance = NotificationManager.IMPORTANCE_DEFAULT; NotificationChannel channel = new NotificationChannel("CHANNEL_ID", "通報通知", importance); channel.setDescription("第3者からの通報を検知しました"); NotificationManager notificationManager = getSystemService(NotificationManager.class); notificationManager.createNotificationChannel(channel); Log.d("nt", "レスポンスを検知しました2"); NotificationSetting();//通知に関する設定のメソッド Notification(getApplicationContext());//通知を行うメソッド } else if(isInCar){//Bluetoothの切断後5分以上乗車状態のままのとき→QRコード読み取りを忘れているとき→置き去り発生 ResetReported();//ResetReported();を処理→FireBaseのisReportedをfalseにする periodicTaskManager.startPeriodicTask();//通知のループをストップする }else { ResetReported();//ResetReported();を処理→FireBaseのisReportedをfalseにする periodicTaskManager.stopPeriodicTask();//5分毎に通知を行う } } }); } 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 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("Bluetootと車の切断から"+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 *60* 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) {//Bluetoot切断からの時間経過(5分刻み) E.putInt("time",5); E.apply();; }else { E.putInt("time",time*2); 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() { // 定期的な処理の停止 handler.removeCallbacks(periodicTask); } } @Nullable @Override public IBinder onBind(Intent intent) { return null; } }