android: optimize notification removal on app exit, add app restart method

This commit is contained in:
ardocrat 2023-06-20 00:39:19 +03:00
parent c17990d0c5
commit 7dba27e6d1
2 changed files with 47 additions and 16 deletions

View file

@ -24,27 +24,32 @@ public class BackgroundService extends Service {
private final Runnable mUpdateSyncStatus = new Runnable() { private final Runnable mUpdateSyncStatus = new Runnable() {
@Override @Override
public void run() { public void run() {
if (mStopped) {
return;
}
// Update sync status at notification.
mNotificationBuilder.setContentText(getSyncStatusText()); mNotificationBuilder.setContentText(getSyncStatusText());
NotificationManager manager = getSystemService(NotificationManager.class); NotificationManager manager = getSystemService(NotificationManager.class);
manager.notify(SYNC_STATUS_NOTIFICATION_ID, mNotificationBuilder.build()); manager.notify(SYNC_STATUS_NOTIFICATION_ID, mNotificationBuilder.build());
// Send broadcast to MainActivity if app exit is needed after node stop.
if (exitAppAfterNodeStop()) { if (exitAppAfterNodeStop()) {
sendBroadcast(new Intent(MainActivity.FINISH_ACTIVITY_ACTION)); sendBroadcast(new Intent(MainActivity.FINISH_ACTIVITY_ACTION));
mStopped = true; mStopped = true;
} }
// Repeat notification update if service is not stopped.
if (!mStopped) { if (!mStopped) {
mHandler.postDelayed(this, 300); mHandler.postDelayed(this, 500);
} }
} }
}; };
@Override @Override
public void onCreate() { public void onCreate() {
// Prevent CPU to sleep at background.
PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE); PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE);
mWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, TAG); mWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, TAG);
mWakeLock.acquire(); mWakeLock.acquire();
// Create channel to show notifications.
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
NotificationChannel notificationChannel = new NotificationChannel( NotificationChannel notificationChannel = new NotificationChannel(
TAG, TAG, NotificationManager.IMPORTANCE_LOW TAG, TAG, NotificationManager.IMPORTANCE_LOW
@ -53,7 +58,7 @@ public class BackgroundService extends Service {
NotificationManager manager = getSystemService(NotificationManager.class); NotificationManager manager = getSystemService(NotificationManager.class);
manager.createNotificationChannel(notificationChannel); manager.createNotificationChannel(notificationChannel);
} }
// Show notification with sync status.
Intent i = getPackageManager().getLaunchIntentForPackage(this.getPackageName()); Intent i = getPackageManager().getLaunchIntentForPackage(this.getPackageName());
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, i, PendingIntent.FLAG_IMMUTABLE); PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, i, PendingIntent.FLAG_IMMUTABLE);
mNotificationBuilder = new NotificationCompat.Builder(this, TAG) mNotificationBuilder = new NotificationCompat.Builder(this, TAG)
@ -62,8 +67,9 @@ public class BackgroundService extends Service {
.setSmallIcon(R.drawable.ic_stat_name) .setSmallIcon(R.drawable.ic_stat_name)
.setContentIntent(pendingIntent); .setContentIntent(pendingIntent);
Notification notification = mNotificationBuilder.build(); Notification notification = mNotificationBuilder.build();
// Start service at foreground state to prevent killing by system.
startForeground(SYNC_STATUS_NOTIFICATION_ID, notification); startForeground(SYNC_STATUS_NOTIFICATION_ID, notification);
// Update sync status at notification.
mHandler.post(mUpdateSyncStatus); mHandler.post(mUpdateSyncStatus);
} }
@ -92,17 +98,24 @@ public class BackgroundService extends Service {
public void onStop() { public void onStop() {
mStopped = true; mStopped = true;
// Stop updating the notification.
mHandler.removeCallbacks(mUpdateSyncStatus);
// Remove service from foreground state.
stopForeground(Service.STOP_FOREGROUND_REMOVE); stopForeground(Service.STOP_FOREGROUND_REMOVE);
// Remove notification.
NotificationManager notificationManager = getSystemService(NotificationManager.class);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
notificationManager.deleteNotificationChannel(TAG);
}
notificationManager.cancel(SYNC_STATUS_NOTIFICATION_ID);
// Release wake lock to allow CPU to sleep at background.
if (mWakeLock.isHeld()) { if (mWakeLock.isHeld()) {
mWakeLock.release(); mWakeLock.release();
mWakeLock = null; mWakeLock = null;
} }
mHandler.removeCallbacks(mUpdateSyncStatus);
} }
// Start the service.
public static void start(Context context) { public static void start(Context context) {
if (!isServiceRunning(context)) { if (!isServiceRunning(context)) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
@ -113,10 +126,12 @@ public class BackgroundService extends Service {
} }
} }
// Stop the service.
public static void stop(Context context) { public static void stop(Context context) {
context.stopService(new Intent(context, BackgroundService.class)); context.stopService(new Intent(context, BackgroundService.class));
} }
// Check if service is running.
private static boolean isServiceRunning(Context context) { private static boolean isServiceRunning(Context context) {
ActivityManager activityManager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE); ActivityManager activityManager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
List<ActivityManager.RunningServiceInfo> services = activityManager.getRunningServices(Integer.MAX_VALUE); List<ActivityManager.RunningServiceInfo> services = activityManager.getRunningServices(Integer.MAX_VALUE);
@ -130,7 +145,10 @@ public class BackgroundService extends Service {
return false; return false;
} }
// Get sync status text for notification from native code.
private native String getSyncStatusText(); private native String getSyncStatusText();
// Get sync title text for notification from native code.
private native String getSyncTitle(); private native String getSyncTitle();
// Check if exit app is needed after node stop from native code.
private native boolean exitAppAfterNodeStop(); private native boolean exitAppAfterNodeStop();
} }

View file

@ -1,9 +1,7 @@
package mw.gri.android; package mw.gri.android;
import android.content.BroadcastReceiver; import android.content.*;
import android.content.Context; import android.content.pm.PackageManager;
import android.content.Intent;
import android.content.IntentFilter;
import android.hardware.SensorManager; import android.hardware.SensorManager;
import android.os.Bundle; import android.os.Bundle;
import android.os.Process; import android.os.Process;
@ -27,6 +25,7 @@ public class MainActivity extends GameActivity {
@Override @Override
public void onReceive(Context ctx, Intent i) { public void onReceive(Context ctx, Intent i) {
if (i.getAction().equals(FINISH_ACTIVITY_ACTION)) { if (i.getAction().equals(FINISH_ACTIVITY_ACTION)) {
unregisterReceiver(this);
onExit(); onExit();
} }
} }
@ -86,11 +85,11 @@ public class MainActivity extends GameActivity {
@Override @Override
protected void onDestroy() { protected void onDestroy() {
unregisterReceiver(mBroadcastReceiver);
if (!mManualExit) { if (!mManualExit) {
unregisterReceiver(mBroadcastReceiver);
onTermination(); onTermination();
} }
// Temp fix: kill process after 3 seconds to prevent app hanging at next launch. // Temp fix: kill process after 3 seconds to prevent app hanging at next launch.
new Thread(() -> { new Thread(() -> {
try { try {
@ -117,5 +116,19 @@ public class MainActivity extends GameActivity {
finish(); finish();
} }
// Called from native code to restart the app.
public void onAppRestart() {
BackgroundService.stop(this);
// Restart Activity.
Intent intent = getPackageManager().getLaunchIntentForPackage(getPackageName());
ComponentName componentName = intent.getComponent();
Intent mainIntent = Intent.makeRestartActivityTask(componentName);
startActivity(mainIntent);
// Kill old process.
Process.killProcess(Process.myPid());
}
public native void onTermination(); public native void onTermination();
} }