am c6578818
: Merge "Make recent location app list package based" into klp-dev
* commit 'c6578818d8984a4bb5fcd2d41844fc9e8e7d7a62': Make recent location app list package based
This commit is contained in:
@@ -37,7 +37,6 @@ import android.widget.CompoundButton;
|
|||||||
import android.widget.Switch;
|
import android.widget.Switch;
|
||||||
|
|
||||||
import com.android.settings.R;
|
import com.android.settings.R;
|
||||||
import com.android.settings.fuelgauge.BatteryStatsHelper;
|
|
||||||
|
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.Comparator;
|
import java.util.Comparator;
|
||||||
@@ -62,7 +61,6 @@ public class LocationSettings extends LocationSettingsBase
|
|||||||
private boolean mValidListener;
|
private boolean mValidListener;
|
||||||
private Preference mLocationMode;
|
private Preference mLocationMode;
|
||||||
private PreferenceCategory mCategoryRecentLocationRequests;
|
private PreferenceCategory mCategoryRecentLocationRequests;
|
||||||
private BatteryStatsHelper mStatsHelper;
|
|
||||||
/** Receives UPDATE_INTENT */
|
/** Receives UPDATE_INTENT */
|
||||||
private BroadcastReceiver mReceiver;
|
private BroadcastReceiver mReceiver;
|
||||||
|
|
||||||
@@ -76,18 +74,6 @@ public class LocationSettings extends LocationSettingsBase
|
|||||||
createPreferenceHierarchy();
|
createPreferenceHierarchy();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onAttach(Activity activity) {
|
|
||||||
super.onAttach(activity);
|
|
||||||
mStatsHelper = new BatteryStatsHelper(activity, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onCreate(Bundle icicle) {
|
|
||||||
super.onCreate(icicle);
|
|
||||||
mStatsHelper.create(icicle);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onResume() {
|
public void onResume() {
|
||||||
super.onResume();
|
super.onResume();
|
||||||
@@ -107,13 +93,6 @@ public class LocationSettings extends LocationSettingsBase
|
|||||||
super.onPause();
|
super.onPause();
|
||||||
mValidListener = false;
|
mValidListener = false;
|
||||||
mSwitch.setOnCheckedChangeListener(null);
|
mSwitch.setOnCheckedChangeListener(null);
|
||||||
mStatsHelper.pause();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onDestroy() {
|
|
||||||
super.onDestroy();
|
|
||||||
mStatsHelper.destroy();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void addPreferencesSorted(List<Preference> prefs, PreferenceGroup container) {
|
private void addPreferencesSorted(List<Preference> prefs, PreferenceGroup container) {
|
||||||
@@ -153,7 +132,7 @@ public class LocationSettings extends LocationSettingsBase
|
|||||||
|
|
||||||
mCategoryRecentLocationRequests =
|
mCategoryRecentLocationRequests =
|
||||||
(PreferenceCategory) root.findPreference(KEY_RECENT_LOCATION_REQUESTS);
|
(PreferenceCategory) root.findPreference(KEY_RECENT_LOCATION_REQUESTS);
|
||||||
RecentLocationApps recentApps = new RecentLocationApps(activity, mStatsHelper);
|
RecentLocationApps recentApps = new RecentLocationApps(activity);
|
||||||
List<Preference> recentLocationRequests = recentApps.getAppList();
|
List<Preference> recentLocationRequests = recentApps.getAppList();
|
||||||
if (recentLocationRequests.size() > 0) {
|
if (recentLocationRequests.size() > 0) {
|
||||||
addPreferencesSorted(recentLocationRequests, mCategoryRecentLocationRequests);
|
addPreferencesSorted(recentLocationRequests, mCategoryRecentLocationRequests);
|
||||||
|
@@ -20,7 +20,6 @@ import android.app.ActivityManager;
|
|||||||
import android.app.AppOpsManager;
|
import android.app.AppOpsManager;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.pm.ApplicationInfo;
|
import android.content.pm.ApplicationInfo;
|
||||||
import android.content.pm.PackageInfo;
|
|
||||||
import android.content.pm.PackageManager;
|
import android.content.pm.PackageManager;
|
||||||
import android.graphics.drawable.Drawable;
|
import android.graphics.drawable.Drawable;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
@@ -32,11 +31,8 @@ import android.util.Log;
|
|||||||
|
|
||||||
import com.android.settings.R;
|
import com.android.settings.R;
|
||||||
import com.android.settings.applications.InstalledAppDetails;
|
import com.android.settings.applications.InstalledAppDetails;
|
||||||
import com.android.settings.fuelgauge.BatterySipper;
|
|
||||||
import com.android.settings.fuelgauge.BatteryStatsHelper;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -45,51 +41,15 @@ import java.util.List;
|
|||||||
public class RecentLocationApps {
|
public class RecentLocationApps {
|
||||||
private static final String TAG = RecentLocationApps.class.getSimpleName();
|
private static final String TAG = RecentLocationApps.class.getSimpleName();
|
||||||
private static final String ANDROID_SYSTEM_PACKAGE_NAME = "android";
|
private static final String ANDROID_SYSTEM_PACKAGE_NAME = "android";
|
||||||
private static final String GOOGLE_SERVICES_SHARED_UID = "com.google.uid.shared";
|
|
||||||
private static final String GCORE_PACKAGE_NAME = "com.google.android.gms";
|
|
||||||
|
|
||||||
private static final int RECENT_TIME_INTERVAL_MILLIS = 15 * 60 * 1000;
|
private static final int RECENT_TIME_INTERVAL_MILLIS = 15 * 60 * 1000;
|
||||||
|
|
||||||
private final PreferenceActivity mActivity;
|
private final PreferenceActivity mActivity;
|
||||||
private final BatteryStatsHelper mStatsHelper;
|
|
||||||
private final PackageManager mPackageManager;
|
private final PackageManager mPackageManager;
|
||||||
private final Drawable mGCoreIcon;
|
|
||||||
|
|
||||||
// Stores all the packages that requested location within the designated interval
|
public RecentLocationApps(PreferenceActivity activity) {
|
||||||
// key - package name of the app
|
|
||||||
// value - whether the app has requested high power location
|
|
||||||
|
|
||||||
public RecentLocationApps(PreferenceActivity activity, BatteryStatsHelper sipperUtil) {
|
|
||||||
mActivity = activity;
|
mActivity = activity;
|
||||||
mPackageManager = activity.getPackageManager();
|
mPackageManager = activity.getPackageManager();
|
||||||
mStatsHelper = sipperUtil;
|
|
||||||
Drawable icon = null;
|
|
||||||
try {
|
|
||||||
ApplicationInfo appInfo = mPackageManager.getApplicationInfo(
|
|
||||||
GCORE_PACKAGE_NAME, PackageManager.GET_META_DATA);
|
|
||||||
icon = mPackageManager.getApplicationIcon(appInfo);
|
|
||||||
} catch (PackageManager.NameNotFoundException e) {
|
|
||||||
if (Log.isLoggable(TAG, Log.INFO)) {
|
|
||||||
Log.i(TAG, "GCore not installed");
|
|
||||||
}
|
|
||||||
icon = null;
|
|
||||||
}
|
|
||||||
mGCoreIcon = icon;
|
|
||||||
}
|
|
||||||
|
|
||||||
private class UidEntryClickedListener
|
|
||||||
implements Preference.OnPreferenceClickListener {
|
|
||||||
private BatterySipper mSipper;
|
|
||||||
|
|
||||||
public UidEntryClickedListener(BatterySipper sipper) {
|
|
||||||
mSipper = sipper;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean onPreferenceClick(Preference preference) {
|
|
||||||
mStatsHelper.startBatteryDetailPage(mActivity, mSipper, false);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private class PackageEntryClickedListener
|
private class PackageEntryClickedListener
|
||||||
@@ -128,50 +88,11 @@ public class RecentLocationApps {
|
|||||||
return pref;
|
return pref;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Stores a BatterySipper object and records whether the sipper has been used.
|
|
||||||
*/
|
|
||||||
private static final class BatterySipperWrapper {
|
|
||||||
private BatterySipper mSipper;
|
|
||||||
private boolean mUsed;
|
|
||||||
|
|
||||||
public BatterySipperWrapper(BatterySipper sipper) {
|
|
||||||
mSipper = sipper;
|
|
||||||
mUsed = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public BatterySipper batterySipper() {
|
|
||||||
return mSipper;
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean used() {
|
|
||||||
return mUsed;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setUsed() {
|
|
||||||
mUsed = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fills a list of applications which queried location recently within
|
* Fills a list of applications which queried location recently within
|
||||||
* specified time.
|
* specified time.
|
||||||
*/
|
*/
|
||||||
public List<Preference> getAppList() {
|
public List<Preference> getAppList() {
|
||||||
// Retrieve Uid-based battery blaming info and generate a package to BatterySipper HashMap
|
|
||||||
// for later faster looking up.
|
|
||||||
mStatsHelper.refreshStats(true);
|
|
||||||
List<BatterySipper> usageList = mStatsHelper.getUsageList();
|
|
||||||
// Key: package Uid. Value: BatterySipperWrapper.
|
|
||||||
HashMap<Integer, BatterySipperWrapper> sipperMap =
|
|
||||||
new HashMap<Integer, BatterySipperWrapper>(usageList.size());
|
|
||||||
for (BatterySipper sipper: usageList) {
|
|
||||||
int uid = sipper.getUid();
|
|
||||||
if (uid != 0) {
|
|
||||||
sipperMap.put(uid, new BatterySipperWrapper(sipper));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Retrieve a location usage list from AppOps
|
// Retrieve a location usage list from AppOps
|
||||||
AppOpsManager aoManager =
|
AppOpsManager aoManager =
|
||||||
(AppOpsManager) mActivity.getSystemService(Context.APP_OPS_SERVICE);
|
(AppOpsManager) mActivity.getSystemService(Context.APP_OPS_SERVICE);
|
||||||
@@ -191,8 +112,7 @@ public class RecentLocationApps {
|
|||||||
boolean isAndroidOs = (uid == Process.SYSTEM_UID)
|
boolean isAndroidOs = (uid == Process.SYSTEM_UID)
|
||||||
&& ANDROID_SYSTEM_PACKAGE_NAME.equals(ops.getPackageName());
|
&& ANDROID_SYSTEM_PACKAGE_NAME.equals(ops.getPackageName());
|
||||||
if (!isAndroidOs && ActivityManager.getCurrentUser() == UserHandle.getUserId(uid)) {
|
if (!isAndroidOs && ActivityManager.getCurrentUser() == UserHandle.getUserId(uid)) {
|
||||||
BatterySipperWrapper wrapper = sipperMap.get(uid);
|
Preference pref = getPreferenceFromOps(now, ops);
|
||||||
Preference pref = getPreferenceFromOps(now, ops, wrapper);
|
|
||||||
if (pref != null) {
|
if (pref != null) {
|
||||||
prefs.add(pref);
|
prefs.add(pref);
|
||||||
}
|
}
|
||||||
@@ -202,57 +122,15 @@ public class RecentLocationApps {
|
|||||||
return prefs;
|
return prefs;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves the icon for given BatterySipper object.
|
|
||||||
*
|
|
||||||
* The icons on location blaming page are actually Uid-based rather than package based. For
|
|
||||||
* those packages that share the same Uid, BatteryStatsHelper picks the one with the most CPU
|
|
||||||
* usage. Both "Contact Sync" and GCore belong to "Google Services" and they share the same Uid.
|
|
||||||
* As a result, sometimes Contact icon may be chosen to represent "Google Services" by
|
|
||||||
* BatteryStatsHelper.
|
|
||||||
*
|
|
||||||
* In order to avoid displaying Contact icon for "Google Services", we hack this method to
|
|
||||||
* always return Puzzle icon for all packages that share the Uid of "Google Services".
|
|
||||||
*/
|
|
||||||
private Drawable getIcon(BatterySipper sipper, AppOpsManager.PackageOps ops) {
|
|
||||||
Drawable icon = null;
|
|
||||||
if (mGCoreIcon != null) {
|
|
||||||
try {
|
|
||||||
PackageInfo info = mPackageManager.getPackageInfo(
|
|
||||||
ops.getPackageName(), PackageManager.GET_META_DATA);
|
|
||||||
if (info != null && GOOGLE_SERVICES_SHARED_UID.equals(info.sharedUserId)) {
|
|
||||||
if (Log.isLoggable(TAG, Log.VERBOSE)) {
|
|
||||||
Log.v(TAG, "shareUserId matches GCore, force using puzzle icon");
|
|
||||||
}
|
|
||||||
icon = mGCoreIcon;
|
|
||||||
}
|
|
||||||
} catch (PackageManager.NameNotFoundException e) {
|
|
||||||
if (Log.isLoggable(TAG, Log.VERBOSE)) {
|
|
||||||
Log.v(TAG, e.toString());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (icon == null) {
|
|
||||||
icon = sipper.getIcon();
|
|
||||||
}
|
|
||||||
return icon;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a Preference entry for the given PackageOps.
|
* Creates a Preference entry for the given PackageOps.
|
||||||
*
|
*
|
||||||
* This method examines the time interval of the PackageOps first. If the PackageOps is older
|
* This method examines the time interval of the PackageOps first. If the PackageOps is older
|
||||||
* than the designated interval, this method ignores the PackageOps object and returns null.
|
* than the designated interval, this method ignores the PackageOps object and returns null.
|
||||||
*
|
* When the PackageOps is fresh enough, this method returns a Preference pointing to the App
|
||||||
* When the PackageOps is fresh enough, if the package has a corresponding battery blaming entry
|
* Info page for that package.
|
||||||
* in the Uid-based battery sipper list, this method returns a Preference pointing to the Uid
|
|
||||||
* battery blaming page. If the package doesn't have a battery sipper entry (typically shouldn't
|
|
||||||
* happen), this method returns a Preference pointing to the App Info page for that package.
|
|
||||||
*/
|
*/
|
||||||
private Preference getPreferenceFromOps(
|
private Preference getPreferenceFromOps(long now, AppOpsManager.PackageOps ops) {
|
||||||
long now,
|
|
||||||
AppOpsManager.PackageOps ops,
|
|
||||||
BatterySipperWrapper wrapper) {
|
|
||||||
String packageName = ops.getPackageName();
|
String packageName = ops.getPackageName();
|
||||||
List<AppOpsManager.OpEntry> entries = ops.getOps();
|
List<AppOpsManager.OpEntry> entries = ops.getOps();
|
||||||
boolean highBattery = false;
|
boolean highBattery = false;
|
||||||
@@ -284,48 +162,27 @@ public class RecentLocationApps {
|
|||||||
// The package is fresh enough, continue.
|
// The package is fresh enough, continue.
|
||||||
|
|
||||||
Preference pref = null;
|
Preference pref = null;
|
||||||
if (wrapper != null) {
|
try {
|
||||||
// Contains sipper. Link to Battery Blaming page.
|
ApplicationInfo appInfo = mPackageManager.getApplicationInfo(
|
||||||
|
packageName, PackageManager.GET_META_DATA);
|
||||||
// We're listing by UID rather than package. Check whether the entry has been used
|
// Multiple users can install the same package. Each user gets a different Uid for
|
||||||
// before to prevent the same UID from showing up twice.
|
// the same package.
|
||||||
if (!wrapper.used()) {
|
//
|
||||||
BatterySipper sipper = wrapper.batterySipper();
|
// Here we retrieve the Uid with package name, that will be the Uid for that package
|
||||||
sipper.loadNameAndIcon();
|
// associated with the current active user. If the Uid differs from the Uid in ops,
|
||||||
pref = createRecentLocationEntry(
|
// that means this entry belongs to another inactive user and we should ignore that.
|
||||||
getIcon(sipper, ops),
|
if (appInfo.uid == ops.getUid()) {
|
||||||
sipper.getLabel(),
|
pref = createRecentLocationEntry(
|
||||||
highBattery,
|
mPackageManager.getApplicationIcon(appInfo),
|
||||||
new UidEntryClickedListener(sipper));
|
mPackageManager.getApplicationLabel(appInfo),
|
||||||
wrapper.setUsed();
|
highBattery,
|
||||||
}
|
new PackageEntryClickedListener(packageName));
|
||||||
} else {
|
} else if (Log.isLoggable(TAG, Log.VERBOSE)) {
|
||||||
// No corresponding sipper. Link to App Info page.
|
Log.v(TAG, "package " + packageName + " with Uid " + ops.getUid() +
|
||||||
|
" belongs to another inactive account, ignored.");
|
||||||
// This is grouped by package rather than UID, but that's OK because this branch
|
}
|
||||||
// shouldn't happen in practice.
|
} catch (PackageManager.NameNotFoundException e) {
|
||||||
try {
|
Log.wtf(TAG, "Package not found: " + packageName, e);
|
||||||
ApplicationInfo appInfo = mPackageManager.getApplicationInfo(
|
|
||||||
packageName, PackageManager.GET_META_DATA);
|
|
||||||
// Multiple users can install the same package. Each user gets a different Uid for
|
|
||||||
// the same package.
|
|
||||||
//
|
|
||||||
// Here we retrieve the Uid with package name, that will be the Uid for that package
|
|
||||||
// associated with the current active user. If the Uid differs from the Uid in ops,
|
|
||||||
// that means this entry belongs to another inactive user and we should ignore that.
|
|
||||||
if (appInfo.uid == ops.getUid()) {
|
|
||||||
pref = createRecentLocationEntry(
|
|
||||||
mPackageManager.getApplicationIcon(appInfo),
|
|
||||||
mPackageManager.getApplicationLabel(appInfo),
|
|
||||||
highBattery,
|
|
||||||
new PackageEntryClickedListener(packageName));
|
|
||||||
} else if (Log.isLoggable(TAG, Log.VERBOSE)) {
|
|
||||||
Log.v(TAG, "package " + packageName + " with Uid " + ops.getUid() +
|
|
||||||
" belongs to another inactive account, ignored.");
|
|
||||||
}
|
|
||||||
} catch (PackageManager.NameNotFoundException e) {
|
|
||||||
Log.wtf(TAG, "Package not found: " + packageName, e);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return pref;
|
return pref;
|
||||||
|
Reference in New Issue
Block a user