Add manage notifications screen
Test: make -j RunSettingsRoboTests Bug: 74318867 Change-Id: I1c872c976677ee38f7d9d9523d18fc8ca03fa547 Add manage notifications screen Test: make -j RunSettingsRoboTests Bug: 74318867 Change-Id: Ia3626e9f69e8b91b1a2bba9ef549c775972e749a
This commit is contained in:
@@ -2661,6 +2661,23 @@
|
|||||||
android:resource="@string/sound_dashboard_summary"/>
|
android:resource="@string/sound_dashboard_summary"/>
|
||||||
</activity>
|
</activity>
|
||||||
|
|
||||||
|
<!-- Show apps for which application-level notification settings are applicable -->
|
||||||
|
<activity android:name="Settings$NotificationAppListActivity"
|
||||||
|
android:label="@string/app_notifications_title"
|
||||||
|
android:icon="@drawable/ic_notifications"
|
||||||
|
android:exported="true"
|
||||||
|
android:taskAffinity="">
|
||||||
|
<intent-filter android:priority="1">
|
||||||
|
<action android:name="android.settings.ALL_APPS_NOTIFICATION_SETTINGS" />
|
||||||
|
<category android:name="android.intent.category.DEFAULT" />
|
||||||
|
</intent-filter>
|
||||||
|
<intent-filter android:priority="150">
|
||||||
|
<action android:name="com.android.settings.action.SETTINGS" />
|
||||||
|
</intent-filter>
|
||||||
|
<meta-data android:name="com.android.settings.FRAGMENT_CLASS"
|
||||||
|
android:value="com.android.settings.applications.manageapplications.ManageApplications" />
|
||||||
|
</activity>
|
||||||
|
|
||||||
<!-- Show application-level notification settings (app passed in as extras) -->
|
<!-- Show application-level notification settings (app passed in as extras) -->
|
||||||
<activity android:name="Settings$AppNotificationSettingsActivity"
|
<activity android:name="Settings$AppNotificationSettingsActivity"
|
||||||
android:exported="true">
|
android:exported="true">
|
||||||
|
@@ -37,6 +37,14 @@
|
|||||||
android:id="@+id/sort_order_size"
|
android:id="@+id/sort_order_size"
|
||||||
android:title="@string/sort_order_size"
|
android:title="@string/sort_order_size"
|
||||||
android:showAsAction="never" />
|
android:showAsAction="never" />
|
||||||
|
<item
|
||||||
|
android:id="@+id/sort_order_recent_notification"
|
||||||
|
android:title="@string/sort_order_recent_notification"
|
||||||
|
android:showAsAction="never" />
|
||||||
|
<item
|
||||||
|
android:id="@+id/sort_order_frequent_notification"
|
||||||
|
android:title="@string/sort_order_frequent_notification"
|
||||||
|
android:showAsAction="never" />
|
||||||
<item
|
<item
|
||||||
android:id="@+id/reset_app_preferences"
|
android:id="@+id/reset_app_preferences"
|
||||||
android:title="@string/reset_app_preferences"
|
android:title="@string/reset_app_preferences"
|
||||||
|
@@ -3818,6 +3818,10 @@
|
|||||||
<string name="sort_order_alpha">Sort by name</string>
|
<string name="sort_order_alpha">Sort by name</string>
|
||||||
<!-- Manage applications screen, menu item. Sorts all of the apps in the list based on their file size. This is used to uninstall when space is getting low. -->
|
<!-- Manage applications screen, menu item. Sorts all of the apps in the list based on their file size. This is used to uninstall when space is getting low. -->
|
||||||
<string name="sort_order_size">Sort by size</string>
|
<string name="sort_order_size">Sort by size</string>
|
||||||
|
<!-- Manage applications screen, menu item. Sorts all of the apps in the list based on how recently they sent notifications.-->
|
||||||
|
<string name="sort_order_recent_notification">Most recent</string>
|
||||||
|
<!-- Manage applications screen, menu item. Sorts all of the apps in the list based on how often they send notifications. -->
|
||||||
|
<string name="sort_order_frequent_notification">Most frequent</string>
|
||||||
<!-- [CHAR LIMIT=25] Manage applications screen, menu item. Show running services. -->
|
<!-- [CHAR LIMIT=25] Manage applications screen, menu item. Show running services. -->
|
||||||
<string name="show_running_services">Show running services</string>
|
<string name="show_running_services">Show running services</string>
|
||||||
<!-- [CHAR LIMIT=25] Manage applications screen, menu item. Show background cached processes. -->
|
<!-- [CHAR LIMIT=25] Manage applications screen, menu item. Show background cached processes. -->
|
||||||
@@ -7308,6 +7312,13 @@
|
|||||||
<!-- Default Apps > Default notification assistant -->
|
<!-- Default Apps > Default notification assistant -->
|
||||||
<string name="default_notification_assistant">Notification assistant</string>
|
<string name="default_notification_assistant">Notification assistant</string>
|
||||||
|
|
||||||
|
<!-- app summary of notification app list screen [CHAR LIMIT=100] -->
|
||||||
|
<string name="notifications_sent_daily">~<xliff:g id="number">%1$s</xliff:g> sent daily</string>
|
||||||
|
<!-- app summary of notification app list screen [CHAR LIMIT=100] -->
|
||||||
|
<string name="notifications_sent_weekly">~<xliff:g id="number">%1$s</xliff:g> sent weekly</string>
|
||||||
|
<!-- app summary of notification app list screen [CHAR LIMIT=100] -->
|
||||||
|
<string name="notifications_sent_never">Never</string>
|
||||||
|
|
||||||
<!-- Sound & notification > Advanced section: Title for managing notification listeners option. [CHAR LIMIT=30] -->
|
<!-- Sound & notification > Advanced section: Title for managing notification listeners option. [CHAR LIMIT=30] -->
|
||||||
<string name="manage_notification_access_title">Notification access</string>
|
<string name="manage_notification_access_title">Notification access</string>
|
||||||
|
|
||||||
|
@@ -69,14 +69,13 @@
|
|||||||
|
|
||||||
<!-- See all apps button -->
|
<!-- See all apps button -->
|
||||||
<Preference
|
<Preference
|
||||||
android:title="@string/notifications_title"
|
|
||||||
android:key="all_notifications"
|
android:key="all_notifications"
|
||||||
|
android:title="@string/notifications_title"
|
||||||
|
android:fragment="com.android.settings.applications.manageapplications.ManageApplications"
|
||||||
android:order="22">
|
android:order="22">
|
||||||
<intent
|
<extra
|
||||||
android:action="android.intent.action.MAIN"
|
android:name="classname"
|
||||||
android:targetPackage="com.android.settings"
|
android:value="com.android.settings.Settings$NotificationAppListActivity" />
|
||||||
android:targetClass="com.android.settings.Settings$ManageApplicationsActivity">
|
|
||||||
</intent>
|
|
||||||
</Preference>
|
</Preference>
|
||||||
</PreferenceCategory>
|
</PreferenceCategory>
|
||||||
|
|
||||||
|
@@ -106,6 +106,7 @@ public class Settings extends SettingsActivity {
|
|||||||
public static class ZenModeEventRuleSettingsActivity extends SettingsActivity { /* empty */ }
|
public static class ZenModeEventRuleSettingsActivity extends SettingsActivity { /* empty */ }
|
||||||
public static class SoundSettingsActivity extends SettingsActivity { /* empty */ }
|
public static class SoundSettingsActivity extends SettingsActivity { /* empty */ }
|
||||||
public static class ConfigureNotificationSettingsActivity extends SettingsActivity { /* empty */ }
|
public static class ConfigureNotificationSettingsActivity extends SettingsActivity { /* empty */ }
|
||||||
|
public static class NotificationAppListActivity extends SettingsActivity { /* empty */ }
|
||||||
public static class AppNotificationSettingsActivity extends SettingsActivity { /* empty */ }
|
public static class AppNotificationSettingsActivity extends SettingsActivity { /* empty */ }
|
||||||
public static class ChannelNotificationSettingsActivity extends SettingsActivity { /* empty */ }
|
public static class ChannelNotificationSettingsActivity extends SettingsActivity { /* empty */ }
|
||||||
public static class ChannelGroupNotificationSettingsActivity extends SettingsActivity { /* empty */ }
|
public static class ChannelGroupNotificationSettingsActivity extends SettingsActivity { /* empty */ }
|
||||||
|
@@ -15,70 +15,191 @@
|
|||||||
*/
|
*/
|
||||||
package com.android.settings.applications;
|
package com.android.settings.applications;
|
||||||
|
|
||||||
import android.app.Notification;
|
import android.app.usage.UsageEvents;
|
||||||
import android.app.NotificationManager;
|
import android.app.usage.UsageStatsManager;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.pm.PackageManager;
|
import android.text.format.DateUtils;
|
||||||
import android.os.UserHandle;
|
import android.util.ArrayMap;
|
||||||
import android.service.notification.NotificationListenerService;
|
|
||||||
|
|
||||||
import com.android.internal.widget.LockPatternUtils;
|
import com.android.settings.R;
|
||||||
import com.android.settings.notification.NotificationBackend;
|
|
||||||
import com.android.settings.notification.NotificationBackend.AppRow;
|
|
||||||
import com.android.settingslib.applications.ApplicationsState;
|
import com.android.settingslib.applications.ApplicationsState;
|
||||||
import com.android.settingslib.applications.ApplicationsState.AppEntry;
|
import com.android.settingslib.applications.ApplicationsState.AppEntry;
|
||||||
import com.android.settingslib.applications.ApplicationsState.AppFilter;
|
import com.android.settingslib.applications.ApplicationsState.AppFilter;
|
||||||
|
import com.android.settingslib.utils.StringUtil;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
import java.util.Comparator;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Connects the info provided by ApplicationsState and the NotificationBackend.
|
* Connects the info provided by ApplicationsState and UsageStatsManager.
|
||||||
* Also provides app filters that can use the notification data.
|
* Also provides app filters that can use the notification data.
|
||||||
*/
|
*/
|
||||||
public class AppStateNotificationBridge extends AppStateBaseBridge {
|
public class AppStateNotificationBridge extends AppStateBaseBridge {
|
||||||
|
|
||||||
private final NotificationBackend mNotifBackend;
|
private UsageStatsManager mUsageStatsManager;
|
||||||
private final PackageManager mPm;
|
private static final int DAYS_TO_CHECK = 7;
|
||||||
private final Context mContext;
|
|
||||||
|
|
||||||
public AppStateNotificationBridge(Context context, ApplicationsState appState,
|
public AppStateNotificationBridge(ApplicationsState appState,
|
||||||
Callback callback, NotificationBackend notifBackend) {
|
Callback callback, UsageStatsManager usageStatsManager) {
|
||||||
super(appState, callback);
|
super(appState, callback);
|
||||||
mContext = context;
|
mUsageStatsManager = usageStatsManager;
|
||||||
mPm = mContext.getPackageManager();
|
|
||||||
mNotifBackend = notifBackend;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void loadAllExtraInfo() {
|
protected void loadAllExtraInfo() {
|
||||||
ArrayList<AppEntry> apps = mAppSession.getAllApps();
|
ArrayList<AppEntry> apps = mAppSession.getAllApps();
|
||||||
final int N = apps.size();
|
if (apps == null) return;
|
||||||
for (int i = 0; i < N; i++) {
|
|
||||||
AppEntry app = apps.get(i);
|
final Map<String, NotificationsSentState> map = getAggregatedUsageEvents();
|
||||||
app.extraInfo = mNotifBackend.loadAppRow(mContext, mPm, app.info);
|
for (AppEntry entry : apps) {
|
||||||
|
NotificationsSentState stats = map.get(entry.info.packageName);
|
||||||
|
calculateAvgSentCounts(stats);
|
||||||
|
entry.extraInfo = stats;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void updateExtraInfo(AppEntry app, String pkg, int uid) {
|
protected void updateExtraInfo(AppEntry entry, String pkg, int uid) {
|
||||||
app.extraInfo = mNotifBackend.loadAppRow(mContext, mPm, app.info);
|
Map<String, NotificationsSentState> map = getAggregatedUsageEvents();
|
||||||
|
NotificationsSentState stats = map.get(entry.info.packageName);
|
||||||
|
calculateAvgSentCounts(stats);
|
||||||
|
entry.extraInfo = stats;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static final AppFilter FILTER_APP_NOTIFICATION_BLOCKED = new AppFilter() {
|
public static CharSequence getSummary(Context context, NotificationsSentState state,
|
||||||
|
boolean sortByRecency) {
|
||||||
|
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.avgSentWeekly > 0) {
|
||||||
|
return context.getString(R.string.notifications_sent_weekly, state.avgSentWeekly);
|
||||||
|
}
|
||||||
|
return context.getString(R.string.notifications_sent_daily, state.avgSentDaily);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected Map<String, NotificationsSentState> getAggregatedUsageEvents() {
|
||||||
|
ArrayMap<String, NotificationsSentState> aggregatedStats = new ArrayMap<>();
|
||||||
|
|
||||||
|
long now = System.currentTimeMillis();
|
||||||
|
long startTime = now - (DateUtils.DAY_IN_MILLIS * DAYS_TO_CHECK);
|
||||||
|
UsageEvents events = mUsageStatsManager.queryEvents(startTime, now);
|
||||||
|
if (events != null) {
|
||||||
|
UsageEvents.Event event = new UsageEvents.Event();
|
||||||
|
while (events.hasNextEvent()) {
|
||||||
|
events.getNextEvent(event);
|
||||||
|
NotificationsSentState stats = aggregatedStats.get(event.getPackageName());
|
||||||
|
if (stats == null) {
|
||||||
|
stats = new NotificationsSentState();
|
||||||
|
aggregatedStats.put(event.getPackageName(), stats);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (event.getEventType() == UsageEvents.Event.NOTIFICATION_INTERRUPTION) {
|
||||||
|
if (event.getTimeStamp() > stats.lastSent) {
|
||||||
|
stats.lastSent = event.getTimeStamp();
|
||||||
|
}
|
||||||
|
stats.sentCount++;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return aggregatedStats;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static NotificationsSentState getNotificationsSentState(AppEntry entry) {
|
||||||
|
if (entry == null || entry.extraInfo == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
if (entry.extraInfo instanceof NotificationsSentState) {
|
||||||
|
return (NotificationsSentState) entry.extraInfo;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final AppFilter FILTER_APP_NOTIFICATION_RECENCY = new AppFilter() {
|
||||||
@Override
|
@Override
|
||||||
public void init() {
|
public void init() {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean filterApp(AppEntry info) {
|
public boolean filterApp(AppEntry info) {
|
||||||
if (info == null || info.extraInfo == null) {
|
NotificationsSentState state = getNotificationsSentState(info);
|
||||||
return false;
|
if (state != null) {
|
||||||
}
|
return state.lastSent != 0;
|
||||||
if (info.extraInfo instanceof AppRow) {
|
|
||||||
AppRow row = (AppRow) info.extraInfo;
|
|
||||||
return row.banned;
|
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
public static final AppFilter FILTER_APP_NOTIFICATION_FREQUENCY = new AppFilter() {
|
||||||
|
@Override
|
||||||
|
public void init() {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean filterApp(AppEntry info) {
|
||||||
|
NotificationsSentState state = getNotificationsSentState(info);
|
||||||
|
if (state != null) {
|
||||||
|
return state.sentCount != 0;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
public static final Comparator<AppEntry> RECENT_NOTIFICATION_COMPARATOR
|
||||||
|
= new Comparator<AppEntry>() {
|
||||||
|
@Override
|
||||||
|
public int compare(AppEntry object1, AppEntry object2) {
|
||||||
|
NotificationsSentState state1 = getNotificationsSentState(object1);
|
||||||
|
NotificationsSentState state2 = getNotificationsSentState(object2);
|
||||||
|
if (state1 == null && state2 != null) return -1;
|
||||||
|
if (state1 != null && state2 == null) return 1;
|
||||||
|
if (state1 != null && state2 != null) {
|
||||||
|
if (state1.lastSent < state2.lastSent) return 1;
|
||||||
|
if (state1.lastSent > state2.lastSent) return -1;
|
||||||
|
}
|
||||||
|
return ApplicationsState.ALPHA_COMPARATOR.compare(object1, object2);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
public static final Comparator<AppEntry> FREQUENCY_NOTIFICATION_COMPARATOR
|
||||||
|
= new Comparator<AppEntry>() {
|
||||||
|
@Override
|
||||||
|
public int compare(AppEntry object1, AppEntry object2) {
|
||||||
|
NotificationsSentState state1 = getNotificationsSentState(object1);
|
||||||
|
NotificationsSentState state2 = getNotificationsSentState(object2);
|
||||||
|
if (state1 == null && state2 != null) return -1;
|
||||||
|
if (state1 != null && state2 == null) return 1;
|
||||||
|
if (state1 != null && state2 != null) {
|
||||||
|
if (state1.sentCount < state2.sentCount) return 1;
|
||||||
|
if (state1.sentCount > state2.sentCount) return -1;
|
||||||
|
}
|
||||||
|
return ApplicationsState.ALPHA_COMPARATOR.compare(object1, object2);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 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;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -41,7 +41,8 @@ public class AppFilterRegistry {
|
|||||||
FILTER_APPS_ENABLED,
|
FILTER_APPS_ENABLED,
|
||||||
FILTER_APPS_INSTANT,
|
FILTER_APPS_INSTANT,
|
||||||
FILTER_APPS_DISABLED,
|
FILTER_APPS_DISABLED,
|
||||||
FILTER_APPS_BLOCKED,
|
FILTER_APPS_RECENT,
|
||||||
|
FILTER_APPS_FREQUENT,
|
||||||
FILTER_APPS_PERSONAL,
|
FILTER_APPS_PERSONAL,
|
||||||
FILTER_APPS_WORK,
|
FILTER_APPS_WORK,
|
||||||
FILTER_APPS_USAGE_ACCESS,
|
FILTER_APPS_USAGE_ACCESS,
|
||||||
@@ -60,23 +61,24 @@ public class AppFilterRegistry {
|
|||||||
public static final int FILTER_APPS_ENABLED = 3;
|
public static final int FILTER_APPS_ENABLED = 3;
|
||||||
public static final int FILTER_APPS_INSTANT = 4;
|
public static final int FILTER_APPS_INSTANT = 4;
|
||||||
public static final int FILTER_APPS_DISABLED = 5;
|
public static final int FILTER_APPS_DISABLED = 5;
|
||||||
public static final int FILTER_APPS_BLOCKED = 6;
|
public static final int FILTER_APPS_RECENT = 6;
|
||||||
public static final int FILTER_APPS_PERSONAL = 7;
|
public static final int FILTER_APPS_FREQUENT = 7;
|
||||||
public static final int FILTER_APPS_WORK = 8;
|
public static final int FILTER_APPS_PERSONAL = 8;
|
||||||
public static final int FILTER_APPS_USAGE_ACCESS = 9;
|
public static final int FILTER_APPS_WORK = 9;
|
||||||
public static final int FILTER_APPS_WITH_OVERLAY = 10;
|
public static final int FILTER_APPS_USAGE_ACCESS = 10;
|
||||||
public static final int FILTER_APPS_WRITE_SETTINGS = 11;
|
public static final int FILTER_APPS_WITH_OVERLAY = 11;
|
||||||
public static final int FILTER_APPS_INSTALL_SOURCES = 12;
|
public static final int FILTER_APPS_WRITE_SETTINGS = 12;
|
||||||
public static final int FILTER_APP_HAS_DIRECTORY_ACCESS = 13;
|
public static final int FILTER_APPS_INSTALL_SOURCES = 13;
|
||||||
public static final int FILTER_APP_CAN_CHANGE_WIFI_STATE = 14;
|
public static final int FILTER_APP_HAS_DIRECTORY_ACCESS = 14;
|
||||||
// Next id: 15
|
public static final int FILTER_APP_CAN_CHANGE_WIFI_STATE = 15;
|
||||||
|
// Next id: 16
|
||||||
|
|
||||||
private static AppFilterRegistry sRegistry;
|
private static AppFilterRegistry sRegistry;
|
||||||
|
|
||||||
private final AppFilterItem[] mFilters;
|
private final AppFilterItem[] mFilters;
|
||||||
|
|
||||||
private AppFilterRegistry() {
|
private AppFilterRegistry() {
|
||||||
mFilters = new AppFilterItem[15];
|
mFilters = new AppFilterItem[16];
|
||||||
|
|
||||||
// High power whitelist, on
|
// High power whitelist, on
|
||||||
mFilters[FILTER_APPS_POWER_WHITELIST] = new AppFilterItem(
|
mFilters[FILTER_APPS_POWER_WHITELIST] = new AppFilterItem(
|
||||||
@@ -118,11 +120,17 @@ public class AppFilterRegistry {
|
|||||||
FILTER_APPS_INSTANT,
|
FILTER_APPS_INSTANT,
|
||||||
R.string.filter_instant_apps);
|
R.string.filter_instant_apps);
|
||||||
|
|
||||||
// Blocked Notifications
|
// Recent Notifications
|
||||||
mFilters[FILTER_APPS_BLOCKED] = new AppFilterItem(
|
mFilters[FILTER_APPS_RECENT] = new AppFilterItem(
|
||||||
AppStateNotificationBridge.FILTER_APP_NOTIFICATION_BLOCKED,
|
AppStateNotificationBridge.FILTER_APP_NOTIFICATION_RECENCY,
|
||||||
FILTER_APPS_BLOCKED,
|
FILTER_APPS_RECENT,
|
||||||
R.string.filter_notif_blocked_apps);
|
R.string.sort_order_recent_notification);
|
||||||
|
|
||||||
|
// Frequent Notifications
|
||||||
|
mFilters[FILTER_APPS_FREQUENT] = new AppFilterItem(
|
||||||
|
AppStateNotificationBridge.FILTER_APP_NOTIFICATION_FREQUENCY,
|
||||||
|
FILTER_APPS_FREQUENT,
|
||||||
|
R.string.sort_order_frequent_notification);
|
||||||
|
|
||||||
// Personal
|
// Personal
|
||||||
mFilters[FILTER_APPS_PERSONAL] = new AppFilterItem(
|
mFilters[FILTER_APPS_PERSONAL] = new AppFilterItem(
|
||||||
@@ -196,6 +204,8 @@ public class AppFilterRegistry {
|
|||||||
return FILTER_APP_HAS_DIRECTORY_ACCESS;
|
return FILTER_APP_HAS_DIRECTORY_ACCESS;
|
||||||
case ManageApplications.LIST_TYPE_WIFI_ACCESS:
|
case ManageApplications.LIST_TYPE_WIFI_ACCESS:
|
||||||
return FILTER_APP_CAN_CHANGE_WIFI_STATE;
|
return FILTER_APP_CAN_CHANGE_WIFI_STATE;
|
||||||
|
case ManageApplications.LIST_TYPE_NOTIFICATION:
|
||||||
|
return FILTER_APPS_RECENT;
|
||||||
default:
|
default:
|
||||||
return FILTER_APPS_ALL;
|
return FILTER_APPS_ALL;
|
||||||
}
|
}
|
||||||
|
@@ -17,11 +17,13 @@
|
|||||||
package com.android.settings.applications.manageapplications;
|
package com.android.settings.applications.manageapplications;
|
||||||
|
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
||||||
.FILTER_APPS_BLOCKED;
|
.FILTER_APPS_ALL;
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
||||||
.FILTER_APPS_DISABLED;
|
.FILTER_APPS_DISABLED;
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
||||||
.FILTER_APPS_ENABLED;
|
.FILTER_APPS_ENABLED;
|
||||||
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
||||||
|
.FILTER_APPS_FREQUENT;
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
||||||
.FILTER_APPS_INSTANT;
|
.FILTER_APPS_INSTANT;
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
||||||
@@ -30,12 +32,15 @@ import static com.android.settings.applications.manageapplications.AppFilterRegi
|
|||||||
.FILTER_APPS_POWER_WHITELIST;
|
.FILTER_APPS_POWER_WHITELIST;
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
||||||
.FILTER_APPS_POWER_WHITELIST_ALL;
|
.FILTER_APPS_POWER_WHITELIST_ALL;
|
||||||
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
||||||
|
.FILTER_APPS_RECENT;
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
||||||
.FILTER_APPS_WORK;
|
.FILTER_APPS_WORK;
|
||||||
|
|
||||||
import android.annotation.Nullable;
|
import android.annotation.Nullable;
|
||||||
import android.annotation.StringRes;
|
import android.annotation.StringRes;
|
||||||
import android.app.Activity;
|
import android.app.Activity;
|
||||||
|
import android.app.usage.UsageStatsManager;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.pm.ApplicationInfo;
|
import android.content.pm.ApplicationInfo;
|
||||||
@@ -81,6 +86,7 @@ import com.android.settings.applications.AppStateBaseBridge;
|
|||||||
import com.android.settings.applications.AppStateDirectoryAccessBridge;
|
import com.android.settings.applications.AppStateDirectoryAccessBridge;
|
||||||
import com.android.settings.applications.AppStateInstallAppsBridge;
|
import com.android.settings.applications.AppStateInstallAppsBridge;
|
||||||
import com.android.settings.applications.AppStateNotificationBridge;
|
import com.android.settings.applications.AppStateNotificationBridge;
|
||||||
|
import com.android.settings.applications.AppStateNotificationBridge.NotificationsSentState;
|
||||||
import com.android.settings.applications.AppStateOverlayBridge;
|
import com.android.settings.applications.AppStateOverlayBridge;
|
||||||
import com.android.settings.applications.AppStatePowerBridge;
|
import com.android.settings.applications.AppStatePowerBridge;
|
||||||
import com.android.settings.applications.AppStateUsageBridge;
|
import com.android.settings.applications.AppStateUsageBridge;
|
||||||
@@ -92,7 +98,6 @@ import com.android.settings.applications.DirectoryAccessDetails;
|
|||||||
import com.android.settings.applications.InstalledAppCounter;
|
import com.android.settings.applications.InstalledAppCounter;
|
||||||
import com.android.settings.applications.UsageAccessDetails;
|
import com.android.settings.applications.UsageAccessDetails;
|
||||||
import com.android.settings.applications.appinfo.AppInfoDashboardFragment;
|
import com.android.settings.applications.appinfo.AppInfoDashboardFragment;
|
||||||
import com.android.settings.applications.appinfo.AppNotificationPreferenceController;
|
|
||||||
import com.android.settings.applications.appinfo.DrawOverlayDetails;
|
import com.android.settings.applications.appinfo.DrawOverlayDetails;
|
||||||
import com.android.settings.applications.appinfo.ExternalSourcesDetails;
|
import com.android.settings.applications.appinfo.ExternalSourcesDetails;
|
||||||
import com.android.settings.applications.appinfo.WriteSettingsDetails;
|
import com.android.settings.applications.appinfo.WriteSettingsDetails;
|
||||||
@@ -102,8 +107,6 @@ import com.android.settings.dashboard.SummaryLoader;
|
|||||||
import com.android.settings.fuelgauge.HighPowerDetail;
|
import com.android.settings.fuelgauge.HighPowerDetail;
|
||||||
import com.android.settings.notification.AppNotificationSettings;
|
import com.android.settings.notification.AppNotificationSettings;
|
||||||
import com.android.settings.notification.ConfigureNotificationSettings;
|
import com.android.settings.notification.ConfigureNotificationSettings;
|
||||||
import com.android.settings.notification.NotificationBackend;
|
|
||||||
import com.android.settings.notification.NotificationBackend.AppRow;
|
|
||||||
import com.android.settings.widget.LoadingViewController;
|
import com.android.settings.widget.LoadingViewController;
|
||||||
import com.android.settings.wifi.AppStateChangeWifiStateBridge;
|
import com.android.settings.wifi.AppStateChangeWifiStateBridge;
|
||||||
import com.android.settings.wifi.ChangeWifiStateDetails;
|
import com.android.settings.wifi.ChangeWifiStateDetails;
|
||||||
@@ -217,7 +220,7 @@ public class ManageApplications extends InstrumentedFragment
|
|||||||
private View mSpinnerHeader;
|
private View mSpinnerHeader;
|
||||||
private Spinner mFilterSpinner;
|
private Spinner mFilterSpinner;
|
||||||
private FilterSpinnerAdapter mFilterAdapter;
|
private FilterSpinnerAdapter mFilterAdapter;
|
||||||
private NotificationBackend mNotifBackend;
|
private UsageStatsManager mUsageStatsManager;
|
||||||
private ResetAppsHelper mResetAppsHelper;
|
private ResetAppsHelper mResetAppsHelper;
|
||||||
private String mVolumeUuid;
|
private String mVolumeUuid;
|
||||||
private int mStorageType;
|
private int mStorageType;
|
||||||
@@ -283,6 +286,12 @@ public class ManageApplications extends InstrumentedFragment
|
|||||||
} else if (className.equals(Settings.ChangeWifiStateActivity.class.getName())) {
|
} else if (className.equals(Settings.ChangeWifiStateActivity.class.getName())) {
|
||||||
mListType = LIST_TYPE_WIFI_ACCESS;
|
mListType = LIST_TYPE_WIFI_ACCESS;
|
||||||
screenTitle = R.string.change_wifi_state_title;
|
screenTitle = R.string.change_wifi_state_title;
|
||||||
|
} else if (className.equals(Settings.NotificationAppListActivity.class.getName())) {
|
||||||
|
mListType = LIST_TYPE_NOTIFICATION;
|
||||||
|
mUsageStatsManager =
|
||||||
|
(UsageStatsManager) getContext().getSystemService(Context.USAGE_STATS_SERVICE);
|
||||||
|
mSortOrder = R.id.sort_order_recent_notification;
|
||||||
|
screenTitle = R.string.app_notifications_title;
|
||||||
} else {
|
} else {
|
||||||
if (screenTitle == -1) {
|
if (screenTitle == -1) {
|
||||||
screenTitle = R.string.application_info_label;
|
screenTitle = R.string.application_info_label;
|
||||||
@@ -383,7 +392,9 @@ public class ManageApplications extends InstrumentedFragment
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (mListType == LIST_TYPE_NOTIFICATION) {
|
if (mListType == LIST_TYPE_NOTIFICATION) {
|
||||||
mFilterAdapter.enableFilter(FILTER_APPS_BLOCKED);
|
mFilterAdapter.enableFilter(FILTER_APPS_RECENT);
|
||||||
|
mFilterAdapter.enableFilter(FILTER_APPS_FREQUENT);
|
||||||
|
mFilterAdapter.disableFilter(FILTER_APPS_ALL);
|
||||||
}
|
}
|
||||||
if (mListType == LIST_TYPE_HIGH_POWER) {
|
if (mListType == LIST_TYPE_HIGH_POWER) {
|
||||||
mFilterAdapter.enableFilter(FILTER_APPS_POWER_WHITELIST_ALL);
|
mFilterAdapter.enableFilter(FILTER_APPS_POWER_WHITELIST_ALL);
|
||||||
@@ -579,6 +590,7 @@ public class ManageApplications extends InstrumentedFragment
|
|||||||
HelpUtils.prepareHelpMenuItem(activity, menu, getHelpResource(), getClass().getName());
|
HelpUtils.prepareHelpMenuItem(activity, menu, getHelpResource(), getClass().getName());
|
||||||
mOptionsMenu = menu;
|
mOptionsMenu = menu;
|
||||||
inflater.inflate(R.menu.manage_apps, menu);
|
inflater.inflate(R.menu.manage_apps, menu);
|
||||||
|
|
||||||
updateOptionsMenu();
|
updateOptionsMenu();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -620,6 +632,10 @@ public class ManageApplications extends InstrumentedFragment
|
|||||||
&& mListType != LIST_TYPE_HIGH_POWER);
|
&& mListType != LIST_TYPE_HIGH_POWER);
|
||||||
|
|
||||||
mOptionsMenu.findItem(R.id.reset_app_preferences).setVisible(mListType == LIST_TYPE_MAIN);
|
mOptionsMenu.findItem(R.id.reset_app_preferences).setVisible(mListType == LIST_TYPE_MAIN);
|
||||||
|
|
||||||
|
// Hide notification menu items, because sorting happens when filtering
|
||||||
|
mOptionsMenu.findItem(R.id.sort_order_recent_notification).setVisible(false);
|
||||||
|
mOptionsMenu.findItem(R.id.sort_order_frequent_notification).setVisible(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -846,8 +862,8 @@ public class ManageApplications extends InstrumentedFragment
|
|||||||
mContext = manageApplications.getActivity();
|
mContext = manageApplications.getActivity();
|
||||||
mAppFilter = appFilter;
|
mAppFilter = appFilter;
|
||||||
if (mManageApplications.mListType == LIST_TYPE_NOTIFICATION) {
|
if (mManageApplications.mListType == LIST_TYPE_NOTIFICATION) {
|
||||||
mExtraInfoBridge = new AppStateNotificationBridge(mContext, mState, this,
|
mExtraInfoBridge = new AppStateNotificationBridge(mState, this,
|
||||||
manageApplications.mNotifBackend);
|
manageApplications.mUsageStatsManager);
|
||||||
} else if (mManageApplications.mListType == LIST_TYPE_USAGE_ACCESS) {
|
} else if (mManageApplications.mListType == LIST_TYPE_USAGE_ACCESS) {
|
||||||
mExtraInfoBridge = new AppStateUsageBridge(mContext, mState, this);
|
mExtraInfoBridge = new AppStateUsageBridge(mContext, mState, this);
|
||||||
} else if (mManageApplications.mListType == LIST_TYPE_HIGH_POWER) {
|
} else if (mManageApplications.mListType == LIST_TYPE_HIGH_POWER) {
|
||||||
@@ -877,8 +893,16 @@ public class ManageApplications extends InstrumentedFragment
|
|||||||
|
|
||||||
public void setFilter(AppFilterItem appFilter) {
|
public void setFilter(AppFilterItem appFilter) {
|
||||||
mAppFilter = appFilter;
|
mAppFilter = appFilter;
|
||||||
|
|
||||||
|
// Notification filters require resorting the list
|
||||||
|
if (FILTER_APPS_FREQUENT == appFilter.getFilterType()) {
|
||||||
|
rebuild(R.id.sort_order_frequent_notification);
|
||||||
|
} else if (FILTER_APPS_RECENT == appFilter.getFilterType()) {
|
||||||
|
rebuild(R.id.sort_order_recent_notification);
|
||||||
|
} else {
|
||||||
rebuild();
|
rebuild();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public void setExtraViewController(FileViewHolderController extraViewController) {
|
public void setExtraViewController(FileViewHolderController extraViewController) {
|
||||||
mExtraViewController = extraViewController;
|
mExtraViewController = extraViewController;
|
||||||
@@ -995,6 +1019,12 @@ public class ManageApplications extends InstrumentedFragment
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
case R.id.sort_order_recent_notification:
|
||||||
|
comparatorObj = AppStateNotificationBridge.RECENT_NOTIFICATION_COMPARATOR;
|
||||||
|
break;
|
||||||
|
case R.id.sort_order_frequent_notification:
|
||||||
|
comparatorObj = AppStateNotificationBridge.FREQUENCY_NOTIFICATION_COMPARATOR;
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
comparatorObj = ApplicationsState.ALPHA_COMPARATOR;
|
comparatorObj = ApplicationsState.ALPHA_COMPARATOR;
|
||||||
break;
|
break;
|
||||||
@@ -1235,9 +1265,9 @@ public class ManageApplications extends InstrumentedFragment
|
|||||||
switch (mManageApplications.mListType) {
|
switch (mManageApplications.mListType) {
|
||||||
case LIST_TYPE_NOTIFICATION:
|
case LIST_TYPE_NOTIFICATION:
|
||||||
if (entry.extraInfo != null) {
|
if (entry.extraInfo != null) {
|
||||||
holder.setSummary(
|
holder.setSummary(AppStateNotificationBridge.getSummary(mContext,
|
||||||
AppNotificationPreferenceController.getNotificationSummary(
|
(NotificationsSentState) entry.extraInfo,
|
||||||
(AppRow) entry.extraInfo, mContext));
|
(mLastSortMode == R.id.sort_order_recent_notification)));
|
||||||
} else {
|
} else {
|
||||||
holder.setSummary(null);
|
holder.setSummary(null);
|
||||||
}
|
}
|
||||||
|
@@ -0,0 +1,413 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2018 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.applications;
|
||||||
|
|
||||||
|
import static android.text.format.DateUtils.DAY_IN_MILLIS;
|
||||||
|
|
||||||
|
import static com.android.settings.applications.AppStateNotificationBridge
|
||||||
|
.FILTER_APP_NOTIFICATION_FREQUENCY;
|
||||||
|
import static com.android.settings.applications.AppStateNotificationBridge
|
||||||
|
.FILTER_APP_NOTIFICATION_RECENCY;
|
||||||
|
import static com.android.settings.applications.AppStateNotificationBridge
|
||||||
|
.FREQUENCY_NOTIFICATION_COMPARATOR;
|
||||||
|
import static com.android.settings.applications.AppStateNotificationBridge
|
||||||
|
.RECENT_NOTIFICATION_COMPARATOR;
|
||||||
|
|
||||||
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
|
|
||||||
|
import static junit.framework.Assert.assertFalse;
|
||||||
|
import static junit.framework.Assert.assertTrue;
|
||||||
|
|
||||||
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
|
import static org.mockito.ArgumentMatchers.anyLong;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
import android.app.usage.UsageEvents;
|
||||||
|
import android.app.usage.UsageEvents.Event;
|
||||||
|
import android.app.usage.UsageStatsManager;
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.pm.ApplicationInfo;
|
||||||
|
import android.os.Looper;
|
||||||
|
import android.os.Parcel;
|
||||||
|
|
||||||
|
import com.android.settings.R;
|
||||||
|
import com.android.settings.applications.AppStateNotificationBridge.NotificationsSentState;
|
||||||
|
import com.android.settings.testutils.SettingsRobolectricTestRunner;
|
||||||
|
import com.android.settingslib.applications.ApplicationsState;
|
||||||
|
import com.android.settingslib.applications.ApplicationsState.AppEntry;
|
||||||
|
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.mockito.Mock;
|
||||||
|
import org.mockito.MockitoAnnotations;
|
||||||
|
import org.robolectric.RuntimeEnvironment;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
@RunWith(SettingsRobolectricTestRunner.class)
|
||||||
|
public class AppStateNotificationBridgeTest {
|
||||||
|
|
||||||
|
private static String PKG1 = "pkg1";
|
||||||
|
private static String PKG2 = "pkg2";
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
private ApplicationsState.Session mSession;
|
||||||
|
@Mock
|
||||||
|
private ApplicationsState mState;
|
||||||
|
@Mock
|
||||||
|
private UsageStatsManager mUsageStats;
|
||||||
|
private Context mContext;
|
||||||
|
private AppStateNotificationBridge mBridge;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() {
|
||||||
|
MockitoAnnotations.initMocks(this);
|
||||||
|
when(mState.newSession(any())).thenReturn(mSession);
|
||||||
|
when(mState.getBackgroundLooper()).thenReturn(mock(Looper.class));
|
||||||
|
mContext = RuntimeEnvironment.application.getApplicationContext();
|
||||||
|
|
||||||
|
mBridge = new AppStateNotificationBridge(mState,
|
||||||
|
mock(AppStateBaseBridge.Callback.class), mUsageStats);
|
||||||
|
}
|
||||||
|
|
||||||
|
private AppEntry getMockAppEntry(String pkg) {
|
||||||
|
AppEntry entry = mock(AppEntry.class);
|
||||||
|
entry.info = mock(ApplicationInfo.class);
|
||||||
|
entry.info.packageName = pkg;
|
||||||
|
return entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
private UsageEvents getUsageEvents(List<Event> events) {
|
||||||
|
UsageEvents usageEvents = new UsageEvents(events, new String[] {PKG1, PKG2});
|
||||||
|
Parcel parcel = Parcel.obtain();
|
||||||
|
parcel.setDataPosition(0);
|
||||||
|
usageEvents.writeToParcel(parcel, 0);
|
||||||
|
parcel.setDataPosition(0);
|
||||||
|
return UsageEvents.CREATOR.createFromParcel(parcel);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetAggregatedUsageEvents_noEvents() {
|
||||||
|
when(mUsageStats.queryEvents(anyLong(), anyLong())).thenReturn(mock(UsageEvents.class));
|
||||||
|
|
||||||
|
assertThat(mBridge.getAggregatedUsageEvents()).isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetAggregatedUsageEvents_onlyNotificationEvents() {
|
||||||
|
List<Event> events = new ArrayList<>();
|
||||||
|
Event good = new Event();
|
||||||
|
good.mEventType = Event.NOTIFICATION_INTERRUPTION;
|
||||||
|
good.mPackage = PKG1;
|
||||||
|
good.mTimeStamp = 1;
|
||||||
|
events.add(good);
|
||||||
|
Event bad = new Event();
|
||||||
|
bad.mEventType = Event.CHOOSER_ACTION;
|
||||||
|
bad.mPackage = PKG1;
|
||||||
|
bad.mTimeStamp = 2;
|
||||||
|
events.add(bad);
|
||||||
|
|
||||||
|
UsageEvents usageEvents = getUsageEvents(events);
|
||||||
|
when(mUsageStats.queryEvents(anyLong(), anyLong())).thenReturn(usageEvents);
|
||||||
|
|
||||||
|
Map<String, NotificationsSentState> map = mBridge.getAggregatedUsageEvents();
|
||||||
|
assertThat(map.get(PKG1).sentCount).isEqualTo(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetAggregatedUsageEvents_multipleEventsAgg() {
|
||||||
|
List<Event> events = new ArrayList<>();
|
||||||
|
Event good = new Event();
|
||||||
|
good.mEventType = Event.NOTIFICATION_INTERRUPTION;
|
||||||
|
good.mPackage = PKG1;
|
||||||
|
good.mTimeStamp = 6;
|
||||||
|
events.add(good);
|
||||||
|
Event good1 = new Event();
|
||||||
|
good1.mEventType = Event.NOTIFICATION_INTERRUPTION;
|
||||||
|
good1.mPackage = PKG1;
|
||||||
|
good1.mTimeStamp = 1;
|
||||||
|
events.add(good1);
|
||||||
|
|
||||||
|
UsageEvents usageEvents = getUsageEvents(events);
|
||||||
|
when(mUsageStats.queryEvents(anyLong(), anyLong())).thenReturn(usageEvents);
|
||||||
|
|
||||||
|
Map<String, NotificationsSentState> map = mBridge.getAggregatedUsageEvents();
|
||||||
|
assertThat(map.get(PKG1).sentCount).isEqualTo(2);
|
||||||
|
assertThat(map.get(PKG1).lastSent).isEqualTo(6);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetAggregatedUsageEvents_multiplePkgs() {
|
||||||
|
List<Event> events = new ArrayList<>();
|
||||||
|
Event good = new Event();
|
||||||
|
good.mEventType = Event.NOTIFICATION_INTERRUPTION;
|
||||||
|
good.mPackage = PKG1;
|
||||||
|
good.mTimeStamp = 6;
|
||||||
|
events.add(good);
|
||||||
|
Event good1 = new Event();
|
||||||
|
good1.mEventType = Event.NOTIFICATION_INTERRUPTION;
|
||||||
|
good1.mPackage = PKG2;
|
||||||
|
good1.mTimeStamp = 1;
|
||||||
|
events.add(good1);
|
||||||
|
|
||||||
|
UsageEvents usageEvents = getUsageEvents(events);
|
||||||
|
when(mUsageStats.queryEvents(anyLong(), anyLong())).thenReturn(usageEvents);
|
||||||
|
|
||||||
|
Map<String, NotificationsSentState> map
|
||||||
|
= mBridge.getAggregatedUsageEvents();
|
||||||
|
assertThat(map.get(PKG1).sentCount).isEqualTo(1);
|
||||||
|
assertThat(map.get(PKG2).sentCount).isEqualTo(1);
|
||||||
|
assertThat(map.get(PKG1).lastSent).isEqualTo(6);
|
||||||
|
assertThat(map.get(PKG2).lastSent).isEqualTo(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testLoadAllExtraInfo_noEvents() {
|
||||||
|
when(mUsageStats.queryEvents(anyLong(), anyLong())).thenReturn(mock(UsageEvents.class));
|
||||||
|
ArrayList<AppEntry> apps = new ArrayList<>();
|
||||||
|
apps.add(getMockAppEntry(PKG1));
|
||||||
|
when(mSession.getAllApps()).thenReturn(apps);
|
||||||
|
|
||||||
|
mBridge.loadAllExtraInfo();
|
||||||
|
assertThat(apps.get(0).extraInfo).isNull();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testLoadAllExtraInfo_multipleEventsAgg() {
|
||||||
|
List<Event> events = new ArrayList<>();
|
||||||
|
for (int i = 0; i < 7; i++) {
|
||||||
|
Event good = new Event();
|
||||||
|
good.mEventType = Event.NOTIFICATION_INTERRUPTION;
|
||||||
|
good.mPackage = PKG1;
|
||||||
|
good.mTimeStamp = i;
|
||||||
|
events.add(good);
|
||||||
|
}
|
||||||
|
|
||||||
|
UsageEvents usageEvents = getUsageEvents(events);
|
||||||
|
when(mUsageStats.queryEvents(anyLong(), anyLong())).thenReturn(usageEvents);
|
||||||
|
|
||||||
|
ArrayList<AppEntry> apps = new ArrayList<>();
|
||||||
|
apps.add(getMockAppEntry(PKG1));
|
||||||
|
when(mSession.getAllApps()).thenReturn(apps);
|
||||||
|
|
||||||
|
mBridge.loadAllExtraInfo();
|
||||||
|
assertThat(((NotificationsSentState) apps.get(0).extraInfo).sentCount).isEqualTo(7);
|
||||||
|
assertThat(((NotificationsSentState) apps.get(0).extraInfo).lastSent).isEqualTo(6);
|
||||||
|
assertThat(((NotificationsSentState) apps.get(0).extraInfo).avgSentDaily).isEqualTo(1);
|
||||||
|
assertThat(((NotificationsSentState) apps.get(0).extraInfo).avgSentWeekly).isEqualTo(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testLoadAllExtraInfo_multiplePkgs() {
|
||||||
|
List<Event> events = new ArrayList<>();
|
||||||
|
for (int i = 0; i < 8; i++) {
|
||||||
|
Event good = new Event();
|
||||||
|
good.mEventType = Event.NOTIFICATION_INTERRUPTION;
|
||||||
|
good.mPackage = PKG1;
|
||||||
|
good.mTimeStamp = i;
|
||||||
|
events.add(good);
|
||||||
|
}
|
||||||
|
Event good1 = new Event();
|
||||||
|
good1.mEventType = Event.NOTIFICATION_INTERRUPTION;
|
||||||
|
good1.mPackage = PKG2;
|
||||||
|
good1.mTimeStamp = 1;
|
||||||
|
events.add(good1);
|
||||||
|
|
||||||
|
UsageEvents usageEvents = getUsageEvents(events);
|
||||||
|
when(mUsageStats.queryEvents(anyLong(), anyLong())).thenReturn(usageEvents);
|
||||||
|
|
||||||
|
ArrayList<AppEntry> apps = new ArrayList<>();
|
||||||
|
apps.add(getMockAppEntry(PKG1));
|
||||||
|
apps.add(getMockAppEntry(PKG2));
|
||||||
|
when(mSession.getAllApps()).thenReturn(apps);
|
||||||
|
|
||||||
|
mBridge.loadAllExtraInfo();
|
||||||
|
assertThat(((NotificationsSentState) apps.get(0).extraInfo).sentCount).isEqualTo(8);
|
||||||
|
assertThat(((NotificationsSentState) apps.get(0).extraInfo).lastSent).isEqualTo(7);
|
||||||
|
assertThat(((NotificationsSentState) apps.get(0).extraInfo).avgSentWeekly).isEqualTo(0);
|
||||||
|
assertThat(((NotificationsSentState) apps.get(0).extraInfo).avgSentDaily).isEqualTo(1);
|
||||||
|
|
||||||
|
assertThat(((NotificationsSentState) apps.get(1).extraInfo).sentCount).isEqualTo(1);
|
||||||
|
assertThat(((NotificationsSentState) apps.get(1).extraInfo).lastSent).isEqualTo(1);
|
||||||
|
assertThat(((NotificationsSentState) apps.get(1).extraInfo).avgSentWeekly).isEqualTo(1);
|
||||||
|
assertThat(((NotificationsSentState) apps.get(1).extraInfo).avgSentDaily).isEqualTo(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testUpdateExtraInfo_noEvents() {
|
||||||
|
when(mUsageStats.queryEvents(anyLong(), anyLong())).thenReturn(mock(UsageEvents.class));
|
||||||
|
AppEntry entry = getMockAppEntry(PKG1);
|
||||||
|
|
||||||
|
mBridge.updateExtraInfo(entry, "", 0);
|
||||||
|
assertThat(entry.extraInfo).isNull();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testUpdateExtraInfo_multipleEventsAgg() {
|
||||||
|
List<Event> events = new ArrayList<>();
|
||||||
|
for (int i = 0; i < 13; i++) {
|
||||||
|
Event good = new Event();
|
||||||
|
good.mEventType = Event.NOTIFICATION_INTERRUPTION;
|
||||||
|
good.mPackage = PKG1;
|
||||||
|
good.mTimeStamp = i;
|
||||||
|
events.add(good);
|
||||||
|
}
|
||||||
|
|
||||||
|
UsageEvents usageEvents = getUsageEvents(events);
|
||||||
|
when(mUsageStats.queryEvents(anyLong(), anyLong())).thenReturn(usageEvents);
|
||||||
|
|
||||||
|
AppEntry entry = getMockAppEntry(PKG1);
|
||||||
|
mBridge.updateExtraInfo(entry, "", 0);
|
||||||
|
|
||||||
|
assertThat(((NotificationsSentState) entry.extraInfo).sentCount).isEqualTo(13);
|
||||||
|
assertThat(((NotificationsSentState) entry.extraInfo).lastSent).isEqualTo(12);
|
||||||
|
assertThat(((NotificationsSentState) entry.extraInfo).avgSentDaily).isEqualTo(2);
|
||||||
|
assertThat(((NotificationsSentState) entry.extraInfo).avgSentWeekly).isEqualTo(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSummary_recency() {
|
||||||
|
NotificationsSentState neverSent = new NotificationsSentState();
|
||||||
|
NotificationsSentState sent = new NotificationsSentState();
|
||||||
|
sent.lastSent = System.currentTimeMillis() - (2 * DAY_IN_MILLIS);
|
||||||
|
|
||||||
|
assertThat(AppStateNotificationBridge.getSummary(mContext, neverSent, true)).isEqualTo(
|
||||||
|
mContext.getString(R.string.notifications_sent_never));
|
||||||
|
assertThat(AppStateNotificationBridge.getSummary(mContext, sent, true).toString())
|
||||||
|
.contains("2");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSummary_frequency() {
|
||||||
|
NotificationsSentState sentRarely = new NotificationsSentState();
|
||||||
|
sentRarely.avgSentWeekly = 1;
|
||||||
|
NotificationsSentState sentOften = new NotificationsSentState();
|
||||||
|
sentOften.avgSentDaily = 8;
|
||||||
|
|
||||||
|
assertThat(AppStateNotificationBridge.getSummary(mContext, sentRarely, false).toString())
|
||||||
|
.contains("1");
|
||||||
|
assertThat(AppStateNotificationBridge.getSummary(mContext, sentOften, false).toString())
|
||||||
|
.contains("8");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testFilterRecency() {
|
||||||
|
NotificationsSentState allowState = new NotificationsSentState();
|
||||||
|
allowState.lastSent = 1;
|
||||||
|
AppEntry allow = mock(AppEntry.class);
|
||||||
|
allow.extraInfo = allowState;
|
||||||
|
|
||||||
|
assertTrue(FILTER_APP_NOTIFICATION_RECENCY.filterApp(allow));
|
||||||
|
|
||||||
|
NotificationsSentState denyState = new NotificationsSentState();
|
||||||
|
denyState.lastSent = 0;
|
||||||
|
AppEntry deny = mock(AppEntry.class);
|
||||||
|
deny.extraInfo = denyState;
|
||||||
|
|
||||||
|
assertFalse(FILTER_APP_NOTIFICATION_RECENCY.filterApp(deny));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testFilterFrequency() {
|
||||||
|
NotificationsSentState allowState = new NotificationsSentState();
|
||||||
|
allowState.sentCount = 1;
|
||||||
|
AppEntry allow = mock(AppEntry.class);
|
||||||
|
allow.extraInfo = allowState;
|
||||||
|
|
||||||
|
assertTrue(FILTER_APP_NOTIFICATION_FREQUENCY.filterApp(allow));
|
||||||
|
|
||||||
|
NotificationsSentState denyState = new NotificationsSentState();
|
||||||
|
denyState.sentCount = 0;
|
||||||
|
AppEntry deny = mock(AppEntry.class);
|
||||||
|
deny.extraInfo = denyState;
|
||||||
|
|
||||||
|
assertFalse(FILTER_APP_NOTIFICATION_FREQUENCY.filterApp(deny));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testComparators_nullsNoCrash() {
|
||||||
|
List<AppEntry> entries = new ArrayList<>();
|
||||||
|
AppEntry a = mock(AppEntry.class);
|
||||||
|
a.label = "1";
|
||||||
|
AppEntry b = mock(AppEntry.class);
|
||||||
|
b.label = "2";
|
||||||
|
entries.add(a);
|
||||||
|
entries.add(b);
|
||||||
|
|
||||||
|
entries.sort(RECENT_NOTIFICATION_COMPARATOR);
|
||||||
|
entries.sort(FREQUENCY_NOTIFICATION_COMPARATOR);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testRecencyComparator() {
|
||||||
|
List<AppEntry> entries = new ArrayList<>();
|
||||||
|
|
||||||
|
NotificationsSentState earlier = new NotificationsSentState();
|
||||||
|
earlier.lastSent = 1;
|
||||||
|
AppEntry earlyEntry = mock(AppEntry.class);
|
||||||
|
earlyEntry.extraInfo = earlier;
|
||||||
|
entries.add(earlyEntry);
|
||||||
|
|
||||||
|
NotificationsSentState later = new NotificationsSentState();
|
||||||
|
later.lastSent = 8;
|
||||||
|
AppEntry lateEntry = mock(AppEntry.class);
|
||||||
|
lateEntry.extraInfo = later;
|
||||||
|
entries.add(lateEntry);
|
||||||
|
|
||||||
|
entries.sort(RECENT_NOTIFICATION_COMPARATOR);
|
||||||
|
|
||||||
|
assertThat(entries).containsExactly(lateEntry, earlyEntry);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testFrequencyComparator() {
|
||||||
|
List<AppEntry> entries = new ArrayList<>();
|
||||||
|
|
||||||
|
NotificationsSentState notFrequentWeekly = new NotificationsSentState();
|
||||||
|
notFrequentWeekly.sentCount = 2;
|
||||||
|
AppEntry notFrequentWeeklyEntry = mock(AppEntry.class);
|
||||||
|
notFrequentWeeklyEntry.extraInfo = notFrequentWeekly;
|
||||||
|
entries.add(notFrequentWeeklyEntry);
|
||||||
|
|
||||||
|
NotificationsSentState notFrequentDaily = new NotificationsSentState();
|
||||||
|
notFrequentDaily.sentCount = 7;
|
||||||
|
AppEntry notFrequentDailyEntry = mock(AppEntry.class);
|
||||||
|
notFrequentDailyEntry.extraInfo = notFrequentDaily;
|
||||||
|
entries.add(notFrequentDailyEntry);
|
||||||
|
|
||||||
|
NotificationsSentState veryFrequentWeekly = new NotificationsSentState();
|
||||||
|
veryFrequentWeekly.sentCount = 6;
|
||||||
|
AppEntry veryFrequentWeeklyEntry = mock(AppEntry.class);
|
||||||
|
veryFrequentWeeklyEntry.extraInfo = veryFrequentWeekly;
|
||||||
|
entries.add(veryFrequentWeeklyEntry);
|
||||||
|
|
||||||
|
NotificationsSentState veryFrequentDaily = new NotificationsSentState();
|
||||||
|
veryFrequentDaily.sentCount = 19;
|
||||||
|
AppEntry veryFrequentDailyEntry = mock(AppEntry.class);
|
||||||
|
veryFrequentDailyEntry.extraInfo = veryFrequentDaily;
|
||||||
|
entries.add(veryFrequentDailyEntry);
|
||||||
|
|
||||||
|
entries.sort(FREQUENCY_NOTIFICATION_COMPARATOR);
|
||||||
|
|
||||||
|
assertThat(entries).containsExactly(veryFrequentDailyEntry, notFrequentDailyEntry,
|
||||||
|
veryFrequentWeeklyEntry, notFrequentWeeklyEntry);
|
||||||
|
}
|
||||||
|
}
|
@@ -312,7 +312,7 @@ public class RecentAppsPreferenceControllerTest {
|
|||||||
|
|
||||||
mController.displayPreference(mScreen);
|
mController.displayPreference(mScreen);
|
||||||
|
|
||||||
verify(mCategory).addPreference(argThat(summaryMatches("0 min. ago")));
|
verify(mCategory).addPreference(argThat(summaryMatches("0 minutes ago")));
|
||||||
}
|
}
|
||||||
|
|
||||||
private static ArgumentMatcher<Preference> summaryMatches(String expected) {
|
private static ArgumentMatcher<Preference> summaryMatches(String expected) {
|
||||||
|
@@ -16,7 +16,6 @@
|
|||||||
|
|
||||||
package com.android.settings.applications.manageapplications;
|
package com.android.settings.applications.manageapplications;
|
||||||
|
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_BLOCKED;
|
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_ENABLED;
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_ENABLED;
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_USAGE_ACCESS;
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_USAGE_ACCESS;
|
||||||
import static com.google.common.truth.Truth.assertThat;
|
import static com.google.common.truth.Truth.assertThat;
|
||||||
@@ -80,7 +79,7 @@ public class AppFilterItemTest {
|
|||||||
@Test
|
@Test
|
||||||
public void hash_differentItem_differentHash() {
|
public void hash_differentItem_differentHash() {
|
||||||
final AppFilterItem item = AppFilterRegistry.getInstance().get(FILTER_APPS_USAGE_ACCESS);
|
final AppFilterItem item = AppFilterRegistry.getInstance().get(FILTER_APPS_USAGE_ACCESS);
|
||||||
final AppFilterItem item2 = AppFilterRegistry.getInstance().get(FILTER_APPS_BLOCKED);
|
final AppFilterItem item2 = AppFilterRegistry.getInstance().get(FILTER_APPS_ENABLED);
|
||||||
|
|
||||||
assertThat(item.hashCode()).isNotEqualTo(item2.hashCode());
|
assertThat(item.hashCode()).isNotEqualTo(item2.hashCode());
|
||||||
}
|
}
|
||||||
|
@@ -19,6 +19,9 @@ package com.android.settings.applications.manageapplications;
|
|||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_ALL;
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_ALL;
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_INSTALL_SOURCES;
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_INSTALL_SOURCES;
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_POWER_WHITELIST;
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_POWER_WHITELIST;
|
||||||
|
|
||||||
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry
|
||||||
|
.FILTER_APPS_RECENT;
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_USAGE_ACCESS;
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_USAGE_ACCESS;
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_WITH_OVERLAY;
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_WITH_OVERLAY;
|
||||||
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_WRITE_SETTINGS;
|
import static com.android.settings.applications.manageapplications.AppFilterRegistry.FILTER_APPS_WRITE_SETTINGS;
|
||||||
@@ -58,7 +61,8 @@ public class AppFilterRegistryTest {
|
|||||||
.isEqualTo(FILTER_APPS_INSTALL_SOURCES);
|
.isEqualTo(FILTER_APPS_INSTALL_SOURCES);
|
||||||
|
|
||||||
assertThat(registry.getDefaultFilterType(LIST_TYPE_MAIN)).isEqualTo(FILTER_APPS_ALL);
|
assertThat(registry.getDefaultFilterType(LIST_TYPE_MAIN)).isEqualTo(FILTER_APPS_ALL);
|
||||||
assertThat(registry.getDefaultFilterType(LIST_TYPE_NOTIFICATION)).isEqualTo(FILTER_APPS_ALL);
|
assertThat(registry.getDefaultFilterType(LIST_TYPE_NOTIFICATION))
|
||||||
|
.isEqualTo(FILTER_APPS_RECENT);
|
||||||
assertThat(registry.getDefaultFilterType(LIST_TYPE_STORAGE)).isEqualTo(FILTER_APPS_ALL);
|
assertThat(registry.getDefaultFilterType(LIST_TYPE_STORAGE)).isEqualTo(FILTER_APPS_ALL);
|
||||||
|
|
||||||
assertThat(registry.getDefaultFilterType(LIST_TYPE_GAMES)).isEqualTo(FILTER_APPS_ALL);
|
assertThat(registry.getDefaultFilterType(LIST_TYPE_GAMES)).isEqualTo(FILTER_APPS_ALL);
|
||||||
|
@@ -67,12 +67,16 @@ public class ManageApplicationsTest {
|
|||||||
@Mock
|
@Mock
|
||||||
private Menu mMenu;
|
private Menu mMenu;
|
||||||
private MenuItem mAppReset;
|
private MenuItem mAppReset;
|
||||||
|
private MenuItem mSortRecent;
|
||||||
|
private MenuItem mSortFrequent;
|
||||||
private ManageApplications mFragment;
|
private ManageApplications mFragment;
|
||||||
|
|
||||||
@Before
|
@Before
|
||||||
public void setUp() {
|
public void setUp() {
|
||||||
MockitoAnnotations.initMocks(this);
|
MockitoAnnotations.initMocks(this);
|
||||||
mAppReset = new RoboMenuItem(R.id.reset_app_preferences);
|
mAppReset = new RoboMenuItem(R.id.reset_app_preferences);
|
||||||
|
mSortRecent = new RoboMenuItem(R.id.sort_order_recent_notification);
|
||||||
|
mSortFrequent = new RoboMenuItem(R.id.sort_order_frequent_notification);
|
||||||
ReflectionHelpers.setStaticField(ApplicationsState.class, "sInstance", mState);
|
ReflectionHelpers.setStaticField(ApplicationsState.class, "sInstance", mState);
|
||||||
when(mState.newSession(any())).thenReturn(mSession);
|
when(mState.newSession(any())).thenReturn(mSession);
|
||||||
when(mState.getBackgroundLooper()).thenReturn(Looper.myLooper());
|
when(mState.getBackgroundLooper()).thenReturn(Looper.myLooper());
|
||||||
@@ -100,6 +104,18 @@ public class ManageApplicationsTest {
|
|||||||
assertThat(mMenu.findItem(R.id.reset_app_preferences).isVisible()).isFalse();
|
assertThat(mMenu.findItem(R.id.reset_app_preferences).isVisible()).isFalse();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void updateMenu_hideNotificationOptions() {
|
||||||
|
setUpOptionMenus();
|
||||||
|
ReflectionHelpers.setField(mFragment, "mListType", LIST_TYPE_NOTIFICATION);
|
||||||
|
ReflectionHelpers.setField(mFragment, "mOptionsMenu", mMenu);
|
||||||
|
|
||||||
|
mFragment.updateOptionsMenu();
|
||||||
|
assertThat(mMenu.findItem(R.id.sort_order_recent_notification).isVisible()).isFalse();
|
||||||
|
assertThat(mMenu.findItem(R.id.sort_order_frequent_notification).isVisible()).isFalse();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void onCreateView_shouldNotShowLoadingContainer() {
|
public void onCreateView_shouldNotShowLoadingContainer() {
|
||||||
final ManageApplications fragment = spy(new ManageApplications());
|
final ManageApplications fragment = spy(new ManageApplications());
|
||||||
@@ -220,6 +236,12 @@ public class ManageApplicationsTest {
|
|||||||
if (id == mAppReset.getItemId()) {
|
if (id == mAppReset.getItemId()) {
|
||||||
return mAppReset;
|
return mAppReset;
|
||||||
}
|
}
|
||||||
|
if (id == mSortFrequent.getItemId()) {
|
||||||
|
return mSortFrequent;
|
||||||
|
}
|
||||||
|
if (id == mSortRecent.getItemId()) {
|
||||||
|
return mSortRecent;
|
||||||
|
}
|
||||||
return new RoboMenuItem(id);
|
return new RoboMenuItem(id);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@@ -360,7 +360,7 @@ public class PowerUsageSummaryLegacyTest {
|
|||||||
|
|
||||||
mFragment.updateLastFullChargePreference(TIME_SINCE_LAST_FULL_CHARGE_MS);
|
mFragment.updateLastFullChargePreference(TIME_SINCE_LAST_FULL_CHARGE_MS);
|
||||||
|
|
||||||
assertThat(mLastFullChargePref.getSubtitle()).isEqualTo("2 hr. ago");
|
assertThat(mLastFullChargePref.getSubtitle()).isEqualTo("2 hours ago");
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@@ -196,7 +196,7 @@ public class PowerUsageSummaryTest {
|
|||||||
mFragment.updateLastFullChargePreference();
|
mFragment.updateLastFullChargePreference();
|
||||||
|
|
||||||
assertThat(mLastFullChargePref.getTitle()).isEqualTo("Last full charge");
|
assertThat(mLastFullChargePref.getTitle()).isEqualTo("Last full charge");
|
||||||
assertThat(mLastFullChargePref.getSubtitle()).isEqualTo("2 hr. ago");
|
assertThat(mLastFullChargePref.getSubtitle()).isEqualTo("2 hours ago");
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@@ -290,7 +290,7 @@ public class RecentNotifyingAppsPreferenceControllerTest {
|
|||||||
|
|
||||||
mController.displayPreference(mScreen);
|
mController.displayPreference(mScreen);
|
||||||
|
|
||||||
verify(mCategory).addPreference(argThat(summaryMatches("0 min. ago")));
|
verify(mCategory).addPreference(argThat(summaryMatches("0 minutes ago")));
|
||||||
}
|
}
|
||||||
|
|
||||||
private static ArgumentMatcher<Preference> summaryMatches(String expected) {
|
private static ArgumentMatcher<Preference> summaryMatches(String expected) {
|
||||||
|
Reference in New Issue
Block a user