TestServiceの全体的なリファクタリング #28
| @ -136,6 +136,8 @@ public class MainActivity extends AppCompatActivity { | |||||||
|         } |         } | ||||||
|         registerReceiver(receiver, intentFilter); |         registerReceiver(receiver, intentFilter); | ||||||
| 
 | 
 | ||||||
|  |         startForegroundService(new Intent(this, TestService.class)); | ||||||
|  | 
 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @ -334,8 +336,8 @@ public class MainActivity extends AppCompatActivity { | |||||||
|     @Override |     @Override | ||||||
|     public void onStop() {//アプリをバックグラウンドにした時のメソッド |     public void onStop() {//アプリをバックグラウンドにした時のメソッド | ||||||
|         super.onStop(); |         super.onStop(); | ||||||
|         Intent intent = new Intent(getApplication(), TestService.class); | //        Intent intent = new Intent(getApplication(), TestService.class); | ||||||
|         startService(intent);//TestServiceを起動 | //        startService(intent);//TestServiceを起動 | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public void NotificationBluetooth(Context context) {//実際に通知を行うメソッド |     public void NotificationBluetooth(Context context) {//実際に通知を行うメソッド | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| package com.example.childguard; | package com.example.childguard; | ||||||
| 
 | 
 | ||||||
| import android.annotation.SuppressLint; | import android.app.Notification; | ||||||
| import android.app.NotificationChannel; | import android.app.NotificationChannel; | ||||||
| import android.app.NotificationManager; | import android.app.NotificationManager; | ||||||
| import android.app.PendingIntent; | import android.app.PendingIntent; | ||||||
| @ -12,10 +12,9 @@ import android.content.Intent; | |||||||
| import android.content.IntentFilter; | import android.content.IntentFilter; | ||||||
| import android.content.SharedPreferences; | import android.content.SharedPreferences; | ||||||
| import android.content.pm.PackageManager; | import android.content.pm.PackageManager; | ||||||
| import android.graphics.Color; |  | ||||||
| import android.os.Build; |  | ||||||
| import android.os.Handler; | import android.os.Handler; | ||||||
| import android.os.IBinder; | import android.os.IBinder; | ||||||
|  | import android.os.VibrationEffect; | ||||||
| import android.os.Vibrator; | import android.os.Vibrator; | ||||||
| import android.preference.PreferenceManager; | import android.preference.PreferenceManager; | ||||||
| import android.util.Log; | import android.util.Log; | ||||||
| @ -23,272 +22,325 @@ import android.util.Log; | |||||||
| import androidx.annotation.Nullable; | import androidx.annotation.Nullable; | ||||||
| import androidx.core.app.ActivityCompat; | import androidx.core.app.ActivityCompat; | ||||||
| import androidx.core.app.NotificationCompat; | import androidx.core.app.NotificationCompat; | ||||||
|  | import androidx.core.app.NotificationManagerCompat; | ||||||
| 
 | 
 | ||||||
| import com.google.firebase.firestore.DocumentReference; | import com.google.firebase.firestore.DocumentReference; | ||||||
| import com.google.firebase.firestore.FirebaseFirestore; | import com.google.firebase.firestore.FirebaseFirestore; | ||||||
| 
 | 
 | ||||||
| import java.util.HashMap; |  | ||||||
| import java.util.Map; |  | ||||||
| 
 |  | ||||||
| public class TestService extends Service { | public class TestService extends Service { | ||||||
|     FirebaseFirestore db; | 
 | ||||||
|     DocumentReference mDocRef; |     private final Handler handler = new Handler(); | ||||||
|  |     private Runnable notificationRunnable; | ||||||
|  | 
 | ||||||
|  |     public static class NotificationContent { | ||||||
|  |         private final String title; | ||||||
|  |         private final String description; | ||||||
|  |         private final String channelId; | ||||||
|  |         private final int notificationId; | ||||||
|  | 
 | ||||||
|  |         public NotificationContent(String title, String description, String channelId, int notificationId) { | ||||||
|  |             this.title = title; | ||||||
|  |             this.description = description; | ||||||
|  |             this.channelId = channelId; | ||||||
|  |             this.notificationId = notificationId; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public String getTitle() { | ||||||
|  |             return title; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public String getDescription() { | ||||||
|  |             return description; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public String getChannelId() { | ||||||
|  |             return channelId; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public int getNotificationId() { | ||||||
|  |             return notificationId; | ||||||
|  |         } | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     public static final String TAG = "InspirationQuote"; |     public static final String TAG = "InspirationQuote"; | ||||||
|  |     private static final String BT_ALERT_CHANNEL_ID = "child_guard_bt_alert"; | ||||||
|  |     private static final String REPORTED_CHANNEL_ID = "child_guard_reported"; | ||||||
|  |     private static final String BACKGROUND_CHANNEL_ID = "child_guard_background"; | ||||||
|  |     private static final int REQUEST_CODE = 100; | ||||||
|  | //    private static final int NOTIFICATION_DELAY = 5 * 60 * 1000; // 5 minutes | ||||||
|  |     // DEBUG | ||||||
|  |     private static final int NOTIFICATION_DELAY = 5 * 1000; // 15 seconds | ||||||
|  |     private static final NotificationContent REPORTED_NOTIFICATION = | ||||||
|  |             new NotificationContent("子供の置き去りをしていませんか?", "第三者からの通報が行われました。", REPORTED_CHANNEL_ID, 2); | ||||||
|  |     private static final NotificationContent BLUETOOTH_NOTIFICATION = | ||||||
|  |             new NotificationContent("子供の置き去りをしていませんか?", "Bluetoothと車の切断から5分が経過しました", BT_ALERT_CHANNEL_ID, 3); | ||||||
| 
 | 
 | ||||||
|  |     private String userId = null; | ||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
|     public int onStartCommand(Intent intent, int flags, int startId) { |     public int onStartCommand(Intent intent, int flags, int startId) { | ||||||
| 
 |         this.userId = getSharedPreferences("app_situation", MODE_PRIVATE).getString("ID", null); | ||||||
| 
 |         if (this.userId == null) { | ||||||
|         //共有プリファレンス全体の準備 |  | ||||||
|         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."); |             Log.d("onResume", "ID not initialized."); | ||||||
|         } else { |             return flags; // IDが初期化されていない場合は何もしない | ||||||
|             mDocRef = FirebaseFirestore.getInstance().document("status/" + IdPref);//現在の位置を取得 |  | ||||||
|             initNotification(mDocRef);//現在の位置を引数に initNotification()を処理 |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         Bluetooth_status(); |         if (isNotificationChannelCreated(BACKGROUND_CHANNEL_ID)) { | ||||||
|         return flags; |             createRunningNotificationChannel(); | ||||||
|  |         } | ||||||
|  |         Intent notificationIntent = new Intent(this, MainActivity.class); | ||||||
|  |         PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, notificationIntent, PendingIntent.FLAG_IMMUTABLE); | ||||||
|  |         Notification notification = new NotificationCompat.Builder(this, BACKGROUND_CHANNEL_ID) | ||||||
|  |                 .setContentTitle("ChildGuardバックグラウンドサービス") | ||||||
|  |                 .setContentText("接続/通報監視サービスがバックグラウンドで実行されています") | ||||||
|  |                 .setSmallIcon(android.R.drawable.ic_menu_info_details) | ||||||
|  |                 .setContentIntent(pendingIntent) | ||||||
|  |                 .build(); | ||||||
|  |         startForeground(1, notification); | ||||||
| 
 | 
 | ||||||
|  |         setSnapshotListener(FirebaseFirestore.getInstance().document("status/" + this.userId)); | ||||||
| 
 | 
 | ||||||
|  |         if (isNotBluetoothGranted()) return flags; | ||||||
|  | 
 | ||||||
|  |         registerReceiver(receiver); | ||||||
|  |         return START_STICKY; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private void initNotification(DocumentReference mDocRef) {//サイト上で押されたボタンの管理 |     @Override | ||||||
|         // PeriodicTaskManagerのインスタンス化 |     public void onCreate() { | ||||||
| 
 |         super.onCreate(); | ||||||
|         // 共有プリファレンス全体の準備 |         if (isNotificationChannelCreated(BT_ALERT_CHANNEL_ID)) createAlertNotificationChannel(BT_ALERT_CHANNEL_ID); | ||||||
|         SharedPreferences sharedPreferences = getSharedPreferences("app_situation", MODE_PRIVATE); |         if (isNotificationChannelCreated(REPORTED_CHANNEL_ID)) createAlertNotificationChannel(REPORTED_CHANNEL_ID); | ||||||
|         //車の乗り降りを管理する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 {//isReportedがfalse=サイト上で降車状態のとき |  | ||||||
|                     ResetReported();//ResetReported();を処理→FireBaseのisReportedをfalseにする |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|         }); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public void ResetReported() {//FireBaseのisReportedをfalseに初期化するメソッド |     /** | ||||||
|         //共有プリファレンス全体の準備 |      * 通知チャネルが作成されているか確認 | ||||||
|         SharedPreferences sharedPreferences = getSharedPreferences("app_situation", MODE_PRIVATE); |      * @return 通知チャンネルの有無 true: 作成済み false: 未作成 | ||||||
|         String IdPref = sharedPreferences.getString("ID", null);//アプリに記録されているIDの取得 |      */ | ||||||
|         db = FirebaseFirestore.getInstance();//Firebaseとの紐づけ |     private boolean isNotificationChannelCreated(String channelId) { | ||||||
|         DocumentReference isReported = db.collection("status").document(IdPref);//更新するドキュメントとの紐づけ |         NotificationManager notificationManager = getSystemService(NotificationManager.class); | ||||||
|         Map<String, Boolean> DEFAULT_ITEM = new HashMap<>();//mapの宣言 |         return notificationManager.getNotificationChannel(channelId) == null; | ||||||
|         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() {//通知に関する設定の処理を行うメソッド |     /** | ||||||
|  |      * 通知チャネルの作成 | ||||||
|  |      */ | ||||||
|  |     private void createAlertNotificationChannel(String channelId) { | ||||||
|         int importance = NotificationManager.IMPORTANCE_DEFAULT; |         int importance = NotificationManager.IMPORTANCE_DEFAULT; | ||||||
|         //通知チャネルの実装 |         NotificationChannel channel = new NotificationChannel(channelId, "通知", importance); | ||||||
|         NotificationChannel channel = new NotificationChannel("CHANNEL_ID", "通知", importance); |  | ||||||
|         channel.setDescription("第三者により置き去りの通報が行われたときに通知します。"); |         channel.setDescription("第三者により置き去りの通報が行われたときに通知します。"); | ||||||
| 
 |  | ||||||
|         NotificationManager notificationManager = getSystemService(NotificationManager.class); |         NotificationManager notificationManager = getSystemService(NotificationManager.class); | ||||||
|         notificationManager.createNotificationChannel(channel); |         notificationManager.createNotificationChannel(channel); | ||||||
| 
 |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public void Notification(Context context) {//実際に通知を行うメソッド |     /** | ||||||
|         final String CHANNEL_ID = "my_channel_id"; |      * バックグラウンドで実行中の通知チャネルを作成 | ||||||
|         // 通知がクリックされたときに送信されるIntent |      */ | ||||||
|         Intent intent = new Intent(context, MainActivity.class); |     private void createRunningNotificationChannel() { | ||||||
|         intent.setAction("OPEN_ACTIVITY"); |         NotificationChannel serviceChannel = new NotificationChannel( | ||||||
|         // PendingIntentの作成 |                 BACKGROUND_CHANNEL_ID, | ||||||
|         int requestCode = 100; |                 "Foreground Service Channel", | ||||||
|         int flags = 0; |                 NotificationManager.IMPORTANCE_NONE | ||||||
|         PendingIntent pendingIntent = PendingIntent.getActivity(context, requestCode, intent, flags | PendingIntent.FLAG_IMMUTABLE); |         ); | ||||||
| 
 |         NotificationManager manager = getSystemService(NotificationManager.class); | ||||||
|         ((Vibrator) getSystemService(Context.VIBRATOR_SERVICE)).vibrate(2000);//バイブレーション |         if (manager != null) { | ||||||
| 
 |             manager.createNotificationChannel(serviceChannel); | ||||||
|         @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) {//実際に通知を行うメソッド |     /** | ||||||
|         final String CHANNEL_ID = "my_channel_id"; |      * 通知が許可がされているかどうかを確認 | ||||||
|         // 通知がクリックされたときに送信されるIntent |      * @return 通知の許可の有無 true: 許可されていない false: 許可されている | ||||||
|         Intent intent = new Intent(context, MainActivity.class); |      */ | ||||||
|         intent.setAction("OPEN_ACTIVITY"); |     private boolean isNotNotificationEnabled() { | ||||||
|         // PendingIntentの作成 |         NotificationManagerCompat notificationManagerCompat = NotificationManagerCompat.from(this); | ||||||
|         int requestCode = 100; |         if (!notificationManagerCompat.areNotificationsEnabled()) { | ||||||
|         int flags = 0; |             Log.d(TAG, "通知が許可されていません"); | ||||||
|         PendingIntent pendingIntent = PendingIntent.getActivity(context, requestCode, intent, flags | PendingIntent.FLAG_IMMUTABLE); |             return true; | ||||||
| 
 |         } else { | ||||||
|         ((Vibrator) getSystemService(Context.VIBRATOR_SERVICE)).vibrate(2000);//バイブレーション |             Log.d(TAG, "通知が許可されています"); | ||||||
| 
 |             return false; | ||||||
|         @SuppressLint("NotificationTrampoline") NotificationCompat.Builder builder = new NotificationCompat.Builder(context, "CHANNEL_ID") |  | ||||||
|                 .setSmallIcon(android.R.drawable.ic_menu_info_details) |  | ||||||
|                 .setContentTitle("子供の置き去りをしていませんか?")//通知のタイトル |  | ||||||
|                 .setContentText("Bluetoothと車の切断から5分が経過しました")//通知の本文 |  | ||||||
|                 .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());//通知の表示 |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Bluetoothの権限が許可されているかどうかを確認 | ||||||
|  |      * @return Bluetoothの権限の有無 true: 許可されていない false: 許可されている | ||||||
|  |      */ | ||||||
|  |     private boolean isNotBluetoothGranted() { | ||||||
|  |         if (ActivityCompat.checkSelfPermission(this, android.Manifest.permission.BLUETOOTH_CONNECT) != PackageManager.PERMISSION_GRANTED) { | ||||||
|  |             Log.d(TAG, "Bluetoothの権限が許可されていません"); | ||||||
|  |             return true; | ||||||
|  |         } else { | ||||||
|  |             Log.d(TAG, "Bluetoothの権限が許可されています"); | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     public void Bluetooth_status() { |     /** | ||||||
|  |      * ブロードキャストレシーバーを登録 | ||||||
|  |      * @param receiver ブロードキャストレシーバー | ||||||
|  |      */ | ||||||
|  |     private void registerReceiver(BroadcastReceiver receiver) { | ||||||
|         IntentFilter intentFilter = new IntentFilter(); |         IntentFilter intentFilter = new IntentFilter(); | ||||||
|         intentFilter.addAction(BluetoothDevice.ACTION_ACL_CONNECTED); |         intentFilter.addAction(BluetoothDevice.ACTION_ACL_CONNECTED); | ||||||
|         intentFilter.addAction(BluetoothDevice.ACTION_ACL_DISCONNECTED); |         intentFilter.addAction(BluetoothDevice.ACTION_ACL_DISCONNECTED); | ||||||
| 
 | 
 | ||||||
|         if (ActivityCompat.checkSelfPermission(this, android.Manifest.permission.BLUETOOTH_CONNECT) != PackageManager.PERMISSION_GRANTED) { |  | ||||||
|             Log.d("BT", "No permission to connect bluetooth devices"); |  | ||||||
|             return; |  | ||||||
|         } else { |  | ||||||
|             Log.d("BT", "Permission to connect bluetooth devices granted"); |  | ||||||
|         } |  | ||||||
|         registerReceiver(receiver, intentFilter); |         registerReceiver(receiver, intentFilter); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * Firestoreのスナップショットリスナーを設定 | ||||||
|  |      * @param mDocRef Firestoreのドキュメントリファレンス | ||||||
|  |      */ | ||||||
|  |     private void setSnapshotListener(DocumentReference mDocRef) { | ||||||
|  |         // Initialize the PeriodicTaskManager | ||||||
|  |         // (Assuming it's done elsewhere as it's not shown in the original code) | ||||||
|  | 
 | ||||||
|  |         // Prepare SharedPreferences | ||||||
|  |         SharedPreferences sharedPreferences = getSharedPreferences("app_situation", MODE_PRIVATE); | ||||||
|  | 
 | ||||||
|  |         // Add a snapshot listener to the document reference | ||||||
|  |         mDocRef.addSnapshotListener((documentSnapshot, e) -> { | ||||||
|  |             if (e != null) { | ||||||
|  |                 Log.w("nt", "Listen failed.", e); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (documentSnapshot != null && documentSnapshot.exists()) { | ||||||
|  |                 Log.d("nt", "イベント開始"); | ||||||
|  | 
 | ||||||
|  |                 // Handle document snapshot | ||||||
|  |                 SharedPreferences.Editor editor = sharedPreferences.edit(); | ||||||
|  |                 boolean isInCar = sharedPreferences.getBoolean("isInCarPref", false); | ||||||
|  |                 boolean newIsInCarState = Boolean.TRUE.equals(documentSnapshot.getBoolean("isInCar")); | ||||||
|  | 
 | ||||||
|  |                 editor.putBoolean("isInCarPref", newIsInCarState); | ||||||
|  |                 editor.apply(); | ||||||
|  | 
 | ||||||
|  |                 Log.d("nt", "レスポンスを検知しました1"); | ||||||
|  | 
 | ||||||
|  |                 if (isInCar) { | ||||||
|  |                     if (Boolean.TRUE.equals(documentSnapshot.getBoolean("isReported"))) { | ||||||
|  |                         resetReported(); | ||||||
|  |                         sendNotification(getApplicationContext(), REPORTED_NOTIFICATION); | ||||||
|  |                     } | ||||||
|  |                 } else { | ||||||
|  |                     resetReported(); | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 Log.d("nt", "Current data: null"); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 通報フラグをリセットする | ||||||
|  |      */ | ||||||
|  |     private void resetReported() { | ||||||
|  |         FirebaseFirestore db = FirebaseFirestore.getInstance();//Firebaseとの紐づけ | ||||||
|  |         DocumentReference isReported = db.collection("status").document(this.userId); | ||||||
|  |         //isReportedをfalseに更新 | ||||||
|  |         isReported.update("isReported", false).addOnSuccessListener(unused -> | ||||||
|  |                 Log.d(TAG, "DocumentSnapshot successfully updated!")).addOnFailureListener(e -> Log.w(TAG, "Error updating document", e)); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 通知をタップしたときにアプリを起動するPendingIntentを取得 | ||||||
|  |      * | ||||||
|  |      * @param context コンテキスト | ||||||
|  |      * @return PendingIntent | ||||||
|  |      */ | ||||||
|  |     private PendingIntent getPendingIntent(Context context) { | ||||||
|  |         Intent intent = new Intent(context, MainActivity.class); | ||||||
|  |         intent.setAction("OPEN_ACTIVITY"); | ||||||
|  |         return PendingIntent.getActivity(context, TestService.REQUEST_CODE, intent, PendingIntent.FLAG_IMMUTABLE); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * デバイスをバイブレーションさせる | ||||||
|  |      */ | ||||||
|  |     private void vibrateDevice() { | ||||||
|  |         Vibrator vibrator = (Vibrator) getSystemService(Context.VIBRATOR_SERVICE); | ||||||
|  |         if (vibrator.hasVibrator()) { | ||||||
|  |             vibrator.vibrate(VibrationEffect.createOneShot(2000, VibrationEffect.DEFAULT_AMPLITUDE)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * 通知を送信する | ||||||
|  |      * @param context コンテキスト | ||||||
|  |      * @param content NotificationContent 通知内容 | ||||||
|  |      */ | ||||||
|  |     private void sendNotification(Context context, NotificationContent content) {//通知を行うメソッド | ||||||
|  |         // 権限の保有を確認 | ||||||
|  |         if (isNotNotificationEnabled()) return; | ||||||
|  | 
 | ||||||
|  |         vibrateDevice(); | ||||||
|  | 
 | ||||||
|  |         NotificationCompat.Builder builder = new NotificationCompat.Builder(context, content.getChannelId()) | ||||||
|  |                 .setSmallIcon(android.R.drawable.ic_menu_info_details) | ||||||
|  |                 .setContentTitle(content.getTitle())//通知のタイトル | ||||||
|  |                 .setContentText(content.getDescription())//通知の内容 | ||||||
|  |                 .setContentIntent(getPendingIntent(context))//通知をタップするとActivityへ移動する | ||||||
|  |                 .setAutoCancel(true)//通知をタップすると削除する | ||||||
|  |                 .setPriority(NotificationCompat.PRIORITY_HIGH) // プライオリティを高く設定 | ||||||
|  |                 .setVisibility(NotificationCompat.VISIBILITY_PUBLIC); // ロック画面に表示する | ||||||
|  | 
 | ||||||
|  |         NotificationManager notificationManager = context.getSystemService(NotificationManager.class); | ||||||
|  | 
 | ||||||
|  |         notificationManager.notify(content.getNotificationId(), builder.build()); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     private final BroadcastReceiver receiver = new BroadcastReceiver() { |     private final BroadcastReceiver receiver = new BroadcastReceiver() { | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|         //PreferenceManager.getDefaultSharedPreferences("myPreferences",Context.MODE_PRIVATE); |  | ||||||
| 
 |  | ||||||
|         @Override |         @Override | ||||||
|         public void onReceive(Context context, Intent intent) { |         public void onReceive(Context context, Intent intent) { | ||||||
|             SharedPreferences pref = getSharedPreferences("Bluetooth_situation", MODE_PRIVATE); |             // 処理対象か確認 ---------------------------------------- | ||||||
|             SharedPreferences.Editor e = pref.edit(); |  | ||||||
|             String action = intent.getAction(); // may need to chain this to a recognizing function |  | ||||||
|             BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE); |             BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE); | ||||||
|             Boolean isInCar = pref.getBoolean("isInCarPref", false); |             if (device == null) { | ||||||
| 
 |                 Log.d("BT", "No device found"); | ||||||
| 
 |  | ||||||
|             if (ActivityCompat.checkSelfPermission(context, android.Manifest.permission.BLUETOOTH_CONNECT) != PackageManager.PERMISSION_GRANTED) { |  | ||||||
|                 Log.d("BT", "No permission to connect bluetooth devices"); |  | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|             String deviceHardwareAddress = device.getAddress(); // MAC address |             String deviceHardwareAddress = device.getAddress(); // MAC address | ||||||
|  |             if (deviceHardwareAddress == null) { | ||||||
|  |                 Log.d("BT", "No device address found"); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             String registeredId = PreferenceManager.getDefaultSharedPreferences(getApplicationContext()).getString("bluetooth_device_id", null); | ||||||
|  |             if (registeredId == null) { | ||||||
|  |                 Log.d("BT_Judge", "No registered device"); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             if (!registeredId.equals(deviceHardwareAddress)) { | ||||||
|  |                 Log.d("BT_Judge", "Not registered device"); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             boolean isInCar = getSharedPreferences("Bluetooth_situation", MODE_PRIVATE).getBoolean("isInCarPref", false); | ||||||
|  |             if (!isInCar) { | ||||||
|  |                 Log.d("BT_Judge", "Not in car"); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             // ----------------------------------------------------- | ||||||
| 
 | 
 | ||||||
|             String registeredId = PreferenceManager.getDefaultSharedPreferences(getApplicationContext()).getString("bluetooth_device_id", "none"); |             String action = intent.getAction(); // may need to chain this to a recognizing function | ||||||
| 
 |             if (BluetoothDevice.ACTION_ACL_DISCONNECTED.equals(action)) { | ||||||
|             if (BluetoothDevice.ACTION_ACL_CONNECTED.equals(action)) { |                 // bluetoothが切断されたときに乗車状態のとき | ||||||
|                 //Do something if connected |                 notificationRunnable = () -> { | ||||||
|                 //Bluetoothデバイスが接続されたときの処理 |                     // 5分経過した時点でも車に乗っていない場合 | ||||||
|                 Log.d("BT", "Device connected"); |                     sendNotification(context, BLUETOOTH_NOTIFICATION); | ||||||
| 
 |                 }; | ||||||
| 
 |                 handler.postDelayed(notificationRunnable, NOTIFICATION_DELAY); | ||||||
|                 Log.d("BT_Judge", "Registered: " + registeredId); |             } else if (BluetoothDevice.ACTION_ACL_CONNECTED.equals(action)) { | ||||||
| 
 |                 // 再接続された場合、通知をキャンセルする | ||||||
|                 if (deviceHardwareAddress.equals(registeredId)) { |                 if (notificationRunnable != null) { | ||||||
|                     //登録済みのデバイスだったときの処理 |                     handler.removeCallbacks(notificationRunnable); | ||||||
|                     Log.d("BT_Judge", "登録済み"); |                     notificationRunnable = null; | ||||||
|                     e.putBoolean("connection_status", true); |                     Log.d("BT", "Notification canceled due to reconnection"); | ||||||
| 
 |  | ||||||
|                 } else { |  | ||||||
|                     //登録していないデバイスだったときの処理 |  | ||||||
|                     Log.d("BT_Judge", "未登録"); |  | ||||||
|                     e.putBoolean("connection_status", false); |  | ||||||
|                 } |                 } | ||||||
|                 e.apply(); |  | ||||||
| 
 |  | ||||||
|             } else if (BluetoothDevice.ACTION_ACL_DISCONNECTED.equals(action) && !isInCar) {//bluetoothが切断されたときに乗車状態のとき |  | ||||||
| 
 |  | ||||||
|                 //Do something if disconnected |  | ||||||
|                 //デバイスが切断されたときの処理 |  | ||||||
|                 if (deviceHardwareAddress.equals(registeredId)) { |  | ||||||
|                     // 5分待機する |  | ||||||
|                     Handler handler = new Handler(); |  | ||||||
|                     handler.postDelayed(new Runnable() { |  | ||||||
|                         @Override |  | ||||||
|                         public void run() { |  | ||||||
|                             if (BluetoothDevice.ACTION_ACL_DISCONNECTED.equals(action) && !isInCar) {//その後bluetoothを再接続したり降車状態になったりしていない=置き去りが発生した可能性大 |  | ||||||
|                                 NotificationBluetooth(getApplicationContext());//通知を行うメソッド |  | ||||||
|                             } |  | ||||||
|                         } |  | ||||||
| 
 |  | ||||||
|                     }, 5 * 60 * 1000); // 5分をミリ秒に変換 |  | ||||||
|                 } |  | ||||||
|             } else { |  | ||||||
|                 Log.d("BT", "Device disconnected"); |  | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     @Nullable |     @Nullable | ||||||
|     @Override |     @Override | ||||||
|     public IBinder onBind(Intent intent) { |     public IBinder onBind(Intent intent) { | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user