Add AppListNotifications to Spa
Bug: 235727273 Test: Manual with Settings App Change-Id: I9d8f9ec74a3661d3d3784cbc7afe02a708035c7e
This commit is contained in:
@@ -0,0 +1,144 @@
|
||||
/*
|
||||
* Copyright (C) 2022 The Android Open Source Project
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package com.android.settings.spa.notification
|
||||
|
||||
import android.Manifest
|
||||
import android.annotation.IntRange
|
||||
import android.app.INotificationManager
|
||||
import android.app.NotificationChannel
|
||||
import android.app.NotificationManager.IMPORTANCE_NONE
|
||||
import android.app.NotificationManager.IMPORTANCE_UNSPECIFIED
|
||||
import android.app.usage.IUsageStatsManager
|
||||
import android.app.usage.UsageEvents
|
||||
import android.content.Context
|
||||
import android.content.pm.ApplicationInfo
|
||||
import android.os.Build
|
||||
import android.os.RemoteException
|
||||
import android.os.ServiceManager
|
||||
import android.util.Log
|
||||
import com.android.settingslib.spaprivileged.model.app.PackageManagers.hasRequestPermission
|
||||
import java.util.concurrent.TimeUnit
|
||||
import kotlin.math.max
|
||||
import kotlin.math.roundToInt
|
||||
import kotlinx.coroutines.flow.Flow
|
||||
import kotlinx.coroutines.flow.map
|
||||
|
||||
/**
|
||||
* This contains how often an app sends notifications and how recently it sent one.
|
||||
*/
|
||||
data class NotificationSentState(
|
||||
@IntRange(from = 0)
|
||||
var lastSent: Long = 0,
|
||||
|
||||
@IntRange(from = 0)
|
||||
var sentCount: Int = 0,
|
||||
)
|
||||
|
||||
class AppNotificationRepository(private val context: Context) {
|
||||
fun getAggregatedUsageEvents(userIdFlow: Flow<Int>): Flow<Map<String, NotificationSentState>> =
|
||||
userIdFlow.map { userId ->
|
||||
val aggregatedStats = mutableMapOf<String, NotificationSentState>()
|
||||
queryEventsForUser(userId)?.let { events ->
|
||||
val event = UsageEvents.Event()
|
||||
while (events.hasNextEvent()) {
|
||||
events.getNextEvent(event)
|
||||
if (event.eventType == UsageEvents.Event.NOTIFICATION_INTERRUPTION) {
|
||||
aggregatedStats.getOrPut(event.packageName, ::NotificationSentState)
|
||||
.apply {
|
||||
lastSent = max(lastSent, event.timeStamp)
|
||||
sentCount++
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
aggregatedStats
|
||||
}
|
||||
|
||||
private fun queryEventsForUser(userId: Int): UsageEvents? {
|
||||
val now = System.currentTimeMillis()
|
||||
val startTime = now - TimeUnit.DAYS.toMillis(DAYS_TO_CHECK)
|
||||
return try {
|
||||
usageStatsManager.queryEventsForUser(startTime, now, userId, context.packageName)
|
||||
} catch (e: RemoteException) {
|
||||
Log.e(TAG, "Failed IUsageStatsManager.queryEventsForUser(): ", e)
|
||||
null
|
||||
}
|
||||
}
|
||||
|
||||
fun isEnabled(app: ApplicationInfo): Boolean =
|
||||
notificationManager.areNotificationsEnabledForPackage(app.packageName, app.uid)
|
||||
|
||||
fun isChangeable(app: ApplicationInfo): Boolean {
|
||||
if (notificationManager.isImportanceLocked(app.packageName, app.uid)) {
|
||||
return false
|
||||
}
|
||||
|
||||
// If the app targets T but has not requested the permission, we cannot change the
|
||||
// permission state.
|
||||
return app.targetSdkVersion < Build.VERSION_CODES.TIRAMISU ||
|
||||
hasRequestPermission(app, Manifest.permission.POST_NOTIFICATIONS)
|
||||
}
|
||||
|
||||
fun setEnabled(app: ApplicationInfo, enabled: Boolean): Boolean {
|
||||
if (onlyHasDefaultChannel(app)) {
|
||||
getChannel(app, NotificationChannel.DEFAULT_CHANNEL_ID)?.let { channel ->
|
||||
channel.importance = if (enabled) IMPORTANCE_UNSPECIFIED else IMPORTANCE_NONE
|
||||
updateChannel(app, channel)
|
||||
}
|
||||
}
|
||||
return try {
|
||||
notificationManager.setNotificationsEnabledForPackage(app.packageName, app.uid, enabled)
|
||||
true
|
||||
} catch (e: Exception) {
|
||||
Log.w(TAG, "Error calling INotificationManager", e)
|
||||
false
|
||||
}
|
||||
}
|
||||
|
||||
private fun updateChannel(app: ApplicationInfo, channel: NotificationChannel) {
|
||||
notificationManager.updateNotificationChannelForPackage(app.packageName, app.uid, channel)
|
||||
}
|
||||
|
||||
private fun onlyHasDefaultChannel(app: ApplicationInfo): Boolean =
|
||||
notificationManager.onlyHasDefaultChannel(app.packageName, app.uid)
|
||||
|
||||
private fun getChannel(app: ApplicationInfo, channelId: String): NotificationChannel? =
|
||||
notificationManager.getNotificationChannelForPackage(
|
||||
app.packageName, app.uid, channelId, null, true
|
||||
)
|
||||
|
||||
companion object {
|
||||
private const val TAG = "AppNotificationsRepo"
|
||||
|
||||
const val DAYS_TO_CHECK = 7L
|
||||
|
||||
private val usageStatsManager by lazy {
|
||||
IUsageStatsManager.Stub.asInterface(
|
||||
ServiceManager.getService(Context.USAGE_STATS_SERVICE)
|
||||
)
|
||||
}
|
||||
|
||||
private val notificationManager by lazy {
|
||||
INotificationManager.Stub.asInterface(
|
||||
ServiceManager.getService(Context.NOTIFICATION_SERVICE)
|
||||
)
|
||||
}
|
||||
|
||||
fun calculateDailyFrequent(sentCount: Int): Int =
|
||||
(sentCount.toFloat() / DAYS_TO_CHECK).roundToInt()
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user