Test: SummarizationGlobalPreferenceControllerTest Test: SummarizationPreferenceControllerTest Flag: android.app.nm_summarization Bug: 390415383 Change-Id: I773d69c707ae5f902860196fb44821ede15261f5
816 lines
29 KiB
Java
816 lines
29 KiB
Java
/*
|
|
* Copyright (C) 2015 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.notification;
|
|
|
|
import static android.app.NotificationManager.IMPORTANCE_NONE;
|
|
import static android.app.NotificationManager.IMPORTANCE_UNSPECIFIED;
|
|
import static android.content.pm.LauncherApps.ShortcutQuery.FLAG_MATCH_CACHED;
|
|
import static android.content.pm.LauncherApps.ShortcutQuery.FLAG_MATCH_DYNAMIC;
|
|
import static android.content.pm.LauncherApps.ShortcutQuery.FLAG_MATCH_PINNED_BY_ANY_LAUNCHER;
|
|
|
|
import static com.android.server.notification.Flags.notificationHideUnusedChannels;
|
|
|
|
import android.app.Flags;
|
|
import android.app.INotificationManager;
|
|
import android.app.NotificationChannel;
|
|
import android.app.NotificationChannelGroup;
|
|
import android.app.NotificationHistory;
|
|
import android.app.NotificationManager;
|
|
import android.app.usage.IUsageStatsManager;
|
|
import android.app.usage.UsageEvents;
|
|
import android.companion.ICompanionDeviceManager;
|
|
import android.content.ComponentName;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.content.pm.ApplicationInfo;
|
|
import android.content.pm.LauncherApps;
|
|
import android.content.pm.PackageInfo;
|
|
import android.content.pm.PackageManager;
|
|
import android.content.pm.ParceledListSlice;
|
|
import android.content.pm.ShortcutInfo;
|
|
import android.content.pm.ShortcutManager;
|
|
import android.graphics.drawable.Drawable;
|
|
import android.os.Build;
|
|
import android.os.RemoteException;
|
|
import android.os.ServiceManager;
|
|
import android.os.UserHandle;
|
|
import android.service.notification.Adjustment;
|
|
import android.service.notification.ConversationChannelWrapper;
|
|
import android.service.notification.NotificationListenerFilter;
|
|
import android.text.format.DateUtils;
|
|
import android.util.IconDrawableFactory;
|
|
import android.util.Log;
|
|
|
|
import androidx.annotation.VisibleForTesting;
|
|
|
|
import com.android.internal.util.CollectionUtils;
|
|
import com.android.settings.R;
|
|
import com.android.settingslib.bluetooth.CachedBluetoothDevice;
|
|
import com.android.settingslib.bluetooth.LocalBluetoothManager;
|
|
import com.android.settingslib.notification.ConversationIconFactory;
|
|
import com.android.settingslib.utils.StringUtil;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.Arrays;
|
|
import java.util.Collection;
|
|
import java.util.HashMap;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import java.util.Objects;
|
|
|
|
public class NotificationBackend {
|
|
private static final String TAG = "NotificationBackend";
|
|
|
|
static IUsageStatsManager sUsageStatsManager = IUsageStatsManager.Stub.asInterface(
|
|
ServiceManager.getService(Context.USAGE_STATS_SERVICE));
|
|
private static final int DAYS_TO_CHECK = 7;
|
|
static INotificationManager sINM = INotificationManager.Stub.asInterface(
|
|
ServiceManager.getService(Context.NOTIFICATION_SERVICE));
|
|
|
|
public AppRow loadAppRow(Context context, PackageManager pm, ApplicationInfo app) {
|
|
final AppRow row = new AppRow();
|
|
if (notificationHideUnusedChannels()) {
|
|
row.showAllChannels = false;
|
|
}
|
|
row.pkg = app.packageName;
|
|
row.uid = app.uid;
|
|
try {
|
|
row.label = app.loadLabel(pm);
|
|
} catch (Throwable t) {
|
|
Log.e(TAG, "Error loading application label for " + row.pkg, t);
|
|
row.label = row.pkg;
|
|
}
|
|
row.icon = IconDrawableFactory.newInstance(context).getBadgedIcon(app);
|
|
row.banned = getNotificationsBanned(row.pkg, row.uid);
|
|
row.showBadge = canShowBadge(row.pkg, row.uid);
|
|
row.bubblePreference = getBubblePreference(row.pkg, row.uid);
|
|
row.userId = UserHandle.getUserId(row.uid);
|
|
row.blockedChannelCount = getBlockedChannelCount(row.pkg, row.uid);
|
|
row.channelCount = getChannelCount(row.pkg, row.uid);
|
|
recordAggregatedUsageEvents(context, row);
|
|
if (Flags.uiRichOngoing()) {
|
|
row.canBePromoted = canBePromoted(row.pkg, row.uid);
|
|
}
|
|
return row;
|
|
}
|
|
|
|
public AppRow loadAppRow(Context context, PackageManager pm, PackageInfo app) {
|
|
final AppRow row = loadAppRow(context, pm, app.applicationInfo);
|
|
recordCanBeBlocked(app, row);
|
|
return row;
|
|
}
|
|
|
|
void recordCanBeBlocked(PackageInfo app, AppRow row) {
|
|
try {
|
|
row.systemApp = row.lockedImportance =
|
|
sINM.isImportanceLocked(app.packageName, app.applicationInfo.uid);
|
|
} catch (RemoteException e) {
|
|
Log.w(TAG, "Error calling NMS", e);
|
|
}
|
|
|
|
// if the app targets T but has not requested the permission, we cannot change the
|
|
// permission state
|
|
if (app.applicationInfo.targetSdkVersion > Build.VERSION_CODES.S_V2) {
|
|
if (app.requestedPermissions == null || Arrays.stream(app.requestedPermissions)
|
|
.noneMatch(p -> p.equals(android.Manifest.permission.POST_NOTIFICATIONS))) {
|
|
row.lockedImportance = true;
|
|
row.permissionStateLocked = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
static public CharSequence getDeviceList(ICompanionDeviceManager cdm, LocalBluetoothManager lbm,
|
|
String pkg, int userId) {
|
|
if (cdm == null) {
|
|
return "";
|
|
}
|
|
boolean multiple = false;
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
try {
|
|
List<String> associatedMacAddrs = CollectionUtils.mapNotNull(
|
|
cdm.getAssociations(pkg, userId),
|
|
a -> a.isSelfManaged() ? null : a.getDeviceMacAddress().toString());
|
|
if (associatedMacAddrs != null) {
|
|
for (String assocMac : associatedMacAddrs) {
|
|
final Collection<CachedBluetoothDevice> cachedDevices =
|
|
lbm.getCachedDeviceManager().getCachedDevicesCopy();
|
|
for (CachedBluetoothDevice cachedBluetoothDevice : cachedDevices) {
|
|
if (Objects.equals(assocMac, cachedBluetoothDevice.getAddress())) {
|
|
if (multiple) {
|
|
sb.append(", ");
|
|
} else {
|
|
multiple = true;
|
|
}
|
|
sb.append(cachedBluetoothDevice.getName());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} catch (RemoteException e) {
|
|
Log.w(TAG, "Error calling CDM", e);
|
|
}
|
|
return sb.toString();
|
|
}
|
|
|
|
public boolean enableSwitch(Context context, ApplicationInfo app) {
|
|
try {
|
|
PackageInfo info = context.getPackageManager().getPackageInfo(
|
|
app.packageName, PackageManager.GET_PERMISSIONS);
|
|
final AppRow row = new AppRow();
|
|
recordCanBeBlocked(info, row);
|
|
boolean systemBlockable = !row.systemApp || (row.systemApp && row.banned);
|
|
return systemBlockable && !row.lockedImportance;
|
|
} catch (PackageManager.NameNotFoundException e) {
|
|
e.printStackTrace();
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public boolean getNotificationsBanned(String pkg, int uid) {
|
|
try {
|
|
final boolean enabled = sINM.areNotificationsEnabledForPackage(pkg, uid);
|
|
return !enabled;
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public boolean setNotificationsEnabledForPackage(String pkg, int uid, boolean enabled) {
|
|
try {
|
|
if (onlyHasDefaultChannel(pkg, uid)) {
|
|
NotificationChannel defaultChannel =
|
|
getChannel(pkg, uid, NotificationChannel.DEFAULT_CHANNEL_ID, null);
|
|
defaultChannel.setImportance(enabled ? IMPORTANCE_UNSPECIFIED : IMPORTANCE_NONE);
|
|
updateChannel(pkg, uid, defaultChannel);
|
|
}
|
|
sINM.setNotificationsEnabledForPackage(pkg, uid, enabled);
|
|
return true;
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public boolean canShowBadge(String pkg, int uid) {
|
|
try {
|
|
return sINM.canShowBadge(pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public boolean setShowBadge(String pkg, int uid, boolean showBadge) {
|
|
try {
|
|
sINM.setShowBadge(pkg, uid, showBadge);
|
|
return true;
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public int getBubblePreference(String pkg, int uid) {
|
|
try {
|
|
return sINM.getBubblePreferenceForPackage(pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
public boolean setAllowBubbles(String pkg, int uid, int preference) {
|
|
try {
|
|
sINM.setBubblesAllowed(pkg, uid, preference);
|
|
return true;
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public NotificationChannel getChannel(String pkg, int uid, String channelId) {
|
|
return getChannel(pkg, uid, channelId, null);
|
|
}
|
|
|
|
public NotificationChannel getChannel(String pkg, int uid, String channelId,
|
|
String conversationId) {
|
|
if (channelId == null) {
|
|
return null;
|
|
}
|
|
try {
|
|
return sINM.getNotificationChannelForPackage(pkg, uid, channelId, conversationId, true);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public NotificationChannelGroup getGroup(String pkg, int uid, String groupId) {
|
|
if (groupId == null) {
|
|
return null;
|
|
}
|
|
try {
|
|
return sINM.getNotificationChannelGroupForPackage(groupId, pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public ParceledListSlice<NotificationChannelGroup> getGroups(String pkg, int uid) {
|
|
try {
|
|
return sINM.getNotificationChannelGroupsForPackage(pkg, uid, false);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return ParceledListSlice.emptyList();
|
|
}
|
|
}
|
|
|
|
public ParceledListSlice<NotificationChannelGroup> getGroupsWithRecentBlockedFilter(String pkg,
|
|
int uid) {
|
|
try {
|
|
return sINM.getRecentBlockedNotificationChannelGroupsForPackage(pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return ParceledListSlice.emptyList();
|
|
}
|
|
}
|
|
|
|
public ParceledListSlice<ConversationChannelWrapper> getConversations(String pkg, int uid) {
|
|
try {
|
|
return sINM.getConversationsForPackage(pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return ParceledListSlice.emptyList();
|
|
}
|
|
}
|
|
|
|
public ParceledListSlice<ConversationChannelWrapper> getConversations(boolean onlyImportant) {
|
|
try {
|
|
return sINM.getConversations(onlyImportant);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return ParceledListSlice.emptyList();
|
|
}
|
|
}
|
|
|
|
public boolean hasSentValidMsg(String pkg, int uid) {
|
|
try {
|
|
return sINM.hasSentValidMsg(pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public boolean isInInvalidMsgState(String pkg, int uid) {
|
|
try {
|
|
return sINM.isInInvalidMsgState(pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public boolean hasUserDemotedInvalidMsgApp(String pkg, int uid) {
|
|
try {
|
|
return sINM.hasUserDemotedInvalidMsgApp(pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public void setInvalidMsgAppDemoted(String pkg, int uid, boolean isDemoted) {
|
|
try {
|
|
sINM.setInvalidMsgAppDemoted(pkg, uid, isDemoted);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
}
|
|
|
|
public boolean hasSentValidBubble(String pkg, int uid) {
|
|
try {
|
|
return sINM.hasSentValidBubble(pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns all notification channels associated with the package and uid that will bypass DND
|
|
*/
|
|
public ParceledListSlice<NotificationChannel> getNotificationChannelsBypassingDnd(String pkg,
|
|
int uid) {
|
|
try {
|
|
return sINM.getNotificationChannelsBypassingDnd(pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return ParceledListSlice.emptyList();
|
|
}
|
|
}
|
|
|
|
public void updateChannel(String pkg, int uid, NotificationChannel channel) {
|
|
try {
|
|
sINM.updateNotificationChannelForPackage(pkg, uid, channel);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
}
|
|
|
|
public void updateChannelGroup(String pkg, int uid, NotificationChannelGroup group) {
|
|
try {
|
|
sINM.updateNotificationChannelGroupForPackage(pkg, uid, group);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
}
|
|
|
|
public int getDeletedChannelCount(String pkg, int uid) {
|
|
try {
|
|
return sINM.getDeletedChannelCount(pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
public int getBlockedChannelCount(String pkg, int uid) {
|
|
try {
|
|
return sINM.getBlockedChannelCount(pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
public boolean onlyHasDefaultChannel(String pkg, int uid) {
|
|
try {
|
|
return sINM.onlyHasDefaultChannel(pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public int getChannelCount(String pkg, int uid) {
|
|
try {
|
|
return sINM.getNumNotificationChannelsForPackage(pkg, uid, false);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
public boolean shouldHideSilentStatusBarIcons(Context context) {
|
|
try {
|
|
return sINM.shouldHideSilentStatusIcons(context.getPackageName());
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public void setHideSilentStatusIcons(boolean hide) {
|
|
try {
|
|
sINM.setHideSilentStatusIcons(hide);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
}
|
|
|
|
public List<String> getAssistantAdjustments(String pkg) {
|
|
try {
|
|
return sINM.getAllowedAssistantAdjustments(pkg);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
return new ArrayList<>();
|
|
}
|
|
|
|
public boolean showSilentInStatusBar(String pkg) {
|
|
try {
|
|
return !sINM.shouldHideSilentStatusIcons(pkg);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public NotificationHistory getNotificationHistory(String pkg, String attributionTag) {
|
|
try {
|
|
return sINM.getNotificationHistory(pkg, attributionTag);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
return new NotificationHistory();
|
|
}
|
|
|
|
protected void recordAggregatedUsageEvents(Context context, AppRow appRow) {
|
|
long now = System.currentTimeMillis();
|
|
long startTime = now - (DateUtils.DAY_IN_MILLIS * DAYS_TO_CHECK);
|
|
UsageEvents events = null;
|
|
try {
|
|
events = sUsageStatsManager.queryEventsForPackageForUser(
|
|
startTime, now, appRow.userId, appRow.pkg, context.getPackageName());
|
|
} catch (RemoteException e) {
|
|
e.printStackTrace();
|
|
}
|
|
recordAggregatedUsageEvents(events, appRow);
|
|
}
|
|
|
|
protected void recordAggregatedUsageEvents(UsageEvents events, AppRow appRow) {
|
|
appRow.sentByChannel = new HashMap<>();
|
|
appRow.sentByApp = new NotificationsSentState();
|
|
if (events != null) {
|
|
UsageEvents.Event event = new UsageEvents.Event();
|
|
while (events.hasNextEvent()) {
|
|
events.getNextEvent(event);
|
|
|
|
if (event.getEventType() == UsageEvents.Event.NOTIFICATION_INTERRUPTION) {
|
|
String channelId = event.mNotificationChannelId;
|
|
if (channelId != null) {
|
|
NotificationsSentState stats = appRow.sentByChannel.get(channelId);
|
|
if (stats == null) {
|
|
stats = new NotificationsSentState();
|
|
appRow.sentByChannel.put(channelId, stats);
|
|
}
|
|
if (event.getTimeStamp() > stats.lastSent) {
|
|
stats.lastSent = event.getTimeStamp();
|
|
appRow.sentByApp.lastSent = event.getTimeStamp();
|
|
}
|
|
stats.sentCount++;
|
|
appRow.sentByApp.sentCount++;
|
|
calculateAvgSentCounts(stats);
|
|
}
|
|
}
|
|
|
|
}
|
|
calculateAvgSentCounts(appRow.sentByApp);
|
|
}
|
|
}
|
|
|
|
public static CharSequence getSentSummary(Context context, NotificationsSentState state,
|
|
boolean sortByRecency) {
|
|
if (state == null) {
|
|
return null;
|
|
}
|
|
if (sortByRecency) {
|
|
if (state.lastSent == 0) {
|
|
return context.getString(R.string.notifications_sent_never);
|
|
}
|
|
return StringUtil.formatRelativeTime(
|
|
context, System.currentTimeMillis() - state.lastSent, true);
|
|
} else {
|
|
if (state.avgSentDaily > 0) {
|
|
return StringUtil.getIcuPluralsString(context, state.avgSentDaily,
|
|
R.string.notifications_sent_daily);
|
|
}
|
|
return StringUtil.getIcuPluralsString(context, state.avgSentWeekly,
|
|
R.string.notifications_sent_weekly);
|
|
}
|
|
}
|
|
|
|
private void calculateAvgSentCounts(NotificationsSentState stats) {
|
|
if (stats != null) {
|
|
stats.avgSentDaily = Math.round((float) stats.sentCount / DAYS_TO_CHECK);
|
|
if (stats.sentCount < DAYS_TO_CHECK) {
|
|
stats.avgSentWeekly = stats.sentCount;
|
|
}
|
|
}
|
|
}
|
|
|
|
public ComponentName getAllowedNotificationAssistant() {
|
|
try {
|
|
return sINM.getAllowedNotificationAssistant();
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public ComponentName getDefaultNotificationAssistant() {
|
|
try {
|
|
return sINM.getDefaultNotificationAssistant();
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public void setNASMigrationDoneAndResetDefault(int userId, boolean loadFromConfig) {
|
|
try {
|
|
sINM.setNASMigrationDoneAndResetDefault(userId, loadFromConfig);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
}
|
|
|
|
public boolean setNotificationAssistantGranted(ComponentName cn) {
|
|
try {
|
|
sINM.setNotificationAssistantAccessGranted(cn, true);
|
|
if (cn == null) {
|
|
return sINM.getAllowedNotificationAssistant() == null;
|
|
} else {
|
|
return cn.equals(sINM.getAllowedNotificationAssistant());
|
|
}
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public void createConversationNotificationChannel(String pkg, int uid,
|
|
NotificationChannel parent, String conversationId) {
|
|
try {
|
|
sINM.createConversationNotificationChannelForPackage(pkg, uid, parent, conversationId);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
}
|
|
|
|
public ShortcutInfo getConversationInfo(Context context, String pkg, int uid, String id) {
|
|
LauncherApps la = context.getSystemService(LauncherApps.class);
|
|
|
|
LauncherApps.ShortcutQuery query = new LauncherApps.ShortcutQuery()
|
|
.setPackage(pkg)
|
|
.setQueryFlags(FLAG_MATCH_DYNAMIC
|
|
| FLAG_MATCH_PINNED_BY_ANY_LAUNCHER | FLAG_MATCH_CACHED)
|
|
.setShortcutIds(Arrays.asList(id));
|
|
List<ShortcutInfo> shortcuts = la.getShortcuts(
|
|
query, UserHandle.of(UserHandle.getUserId(uid)));
|
|
if (shortcuts != null && !shortcuts.isEmpty()) {
|
|
return shortcuts.get(0);
|
|
}
|
|
return null;
|
|
}
|
|
|
|
public Drawable getConversationDrawable(Context context, ShortcutInfo info, String pkg,
|
|
int uid, boolean important) {
|
|
if (info == null) {
|
|
return null;
|
|
}
|
|
ConversationIconFactory iconFactory = new ConversationIconFactory(context,
|
|
context.getSystemService(LauncherApps.class),
|
|
context.getPackageManager(),
|
|
IconDrawableFactory.newInstance(context, false),
|
|
context.getResources().getDimensionPixelSize(
|
|
R.dimen.conversation_icon_size));
|
|
return iconFactory.getConversationDrawable(info, pkg, uid, important);
|
|
}
|
|
|
|
public void requestPinShortcut(Context context, ShortcutInfo shortcutInfo) {
|
|
ShortcutManager sm = context.getSystemService(ShortcutManager.class);
|
|
sm.requestPinShortcut(shortcutInfo, null);
|
|
}
|
|
|
|
public void resetNotificationImportance() {
|
|
try {
|
|
sINM.unlockAllNotificationChannels();
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
}
|
|
|
|
public NotificationListenerFilter getListenerFilter(ComponentName cn, int userId) {
|
|
NotificationListenerFilter nlf = null;
|
|
try {
|
|
nlf = sINM.getListenerFilter(cn, userId);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
return nlf != null ? nlf : new NotificationListenerFilter();
|
|
}
|
|
|
|
public void setListenerFilter(ComponentName cn, int userId, NotificationListenerFilter nlf) {
|
|
try {
|
|
sINM.setListenerFilter(cn, userId, nlf);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
}
|
|
|
|
public boolean isNotificationListenerAccessGranted(ComponentName cn) {
|
|
try {
|
|
return sINM.isNotificationListenerAccessGranted(cn);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public boolean isNotificationBundlingSupported() {
|
|
try {
|
|
return !sINM.getUnsupportedAdjustmentTypes().contains(Adjustment.KEY_TYPE);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public boolean isNotificationBundlingEnabled(Context context) {
|
|
try {
|
|
return sINM.getAllowedAssistantAdjustments(context.getPackageName())
|
|
.contains(Adjustment.KEY_TYPE);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public void setNotificationBundlingEnabled(boolean enabled) {
|
|
try {
|
|
if (enabled) {
|
|
sINM.allowAssistantAdjustment(Adjustment.KEY_TYPE);
|
|
} else {
|
|
sINM.disallowAssistantAdjustment(Adjustment.KEY_TYPE);
|
|
}
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
}
|
|
|
|
public boolean isNotificationSummarizationSupported() {
|
|
try {
|
|
return !sINM.getUnsupportedAdjustmentTypes().contains(Adjustment.KEY_SUMMARIZATION);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public boolean isNotificationSummarizationEnabled(Context context) {
|
|
try {
|
|
return sINM.getAllowedAssistantAdjustments(context.getPackageName())
|
|
.contains(Adjustment.KEY_SUMMARIZATION);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public void setNotificationSummarizationEnabled(boolean enabled) {
|
|
try {
|
|
if (enabled) {
|
|
sINM.allowAssistantAdjustment(Adjustment.KEY_SUMMARIZATION);
|
|
} else {
|
|
sINM.disallowAssistantAdjustment(Adjustment.KEY_SUMMARIZATION);
|
|
}
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
}
|
|
|
|
public boolean isBundleTypeApproved(@Adjustment.Types int type) {
|
|
try {
|
|
int[] approved = sINM.getAllowedAdjustmentKeyTypes();
|
|
for (int approvedType : approved) {
|
|
if (type == approvedType) {
|
|
return true;
|
|
}
|
|
}
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public void setBundleTypeState(@Adjustment.Types int type, boolean enabled) {
|
|
try {
|
|
sINM.setAssistantAdjustmentKeyTypeState(type, enabled);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Retrieves whether the app with given package and uid is permitted to post promoted
|
|
* notifications.
|
|
*/
|
|
public boolean canBePromoted(String pkg, int uid) {
|
|
try {
|
|
return sINM.appCanBePromoted(pkg, uid);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Sets whether the app with given package and uid is permitted to post promoted notifications.
|
|
*/
|
|
public void setCanBePromoted(String pkg, int uid, boolean allowed) {
|
|
// We shouldn't get here with the flag off, but just in case, do nothing.
|
|
if (!Flags.uiRichOngoing()) {
|
|
Log.wtf(TAG, "tried to setCanBePromoted without flag on");
|
|
return;
|
|
}
|
|
try {
|
|
sINM.setCanBePromoted(pkg, uid, allowed, /* fromUser= */ true);
|
|
} catch (Exception e) {
|
|
Log.w(TAG, "Error calling NoMan", e);
|
|
}
|
|
}
|
|
|
|
@VisibleForTesting
|
|
void setNm(INotificationManager inm) {
|
|
sINM = inm;
|
|
}
|
|
|
|
/**
|
|
* NotificationsSentState contains how often an app sends notifications and how recently it sent
|
|
* one.
|
|
*/
|
|
public static class NotificationsSentState {
|
|
public int avgSentDaily = 0;
|
|
public int avgSentWeekly = 0;
|
|
public long lastSent = 0;
|
|
public int sentCount = 0;
|
|
}
|
|
|
|
static class Row {
|
|
public String section;
|
|
}
|
|
|
|
public static class AppRow extends Row {
|
|
public String pkg;
|
|
public int uid;
|
|
public Drawable icon;
|
|
public CharSequence label;
|
|
public Intent settingsIntent;
|
|
public boolean banned;
|
|
public boolean first; // first app in section
|
|
public boolean systemApp;
|
|
public boolean lockedImportance;
|
|
public boolean showBadge;
|
|
// For apps target T but have not but has not requested the permission
|
|
// we cannot change the permission state
|
|
public boolean permissionStateLocked;
|
|
public int bubblePreference = NotificationManager.BUBBLE_PREFERENCE_NONE;
|
|
public int userId;
|
|
public int blockedChannelCount;
|
|
public int channelCount;
|
|
public Map<String, NotificationsSentState> sentByChannel;
|
|
public NotificationsSentState sentByApp;
|
|
public boolean showAllChannels = true;
|
|
public boolean canBePromoted;
|
|
}
|
|
}
|