Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 1 | /** |
| 2 | * Copyright (C) 2013 The Android Open Source Project |
| 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); you may not |
| 5 | * use this file except in compliance with the License. You may obtain a copy |
| 6 | * of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| 12 | * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| 13 | * License for the specific language governing permissions and limitations |
| 14 | * under the License. |
| 15 | */ |
| 16 | |
| 17 | package com.android.settings.applications; |
| 18 | |
| 19 | import android.app.AppOpsManager; |
| 20 | import android.content.Context; |
| 21 | import android.content.pm.ApplicationInfo; |
| 22 | import android.content.pm.PackageInfo; |
| 23 | import android.content.pm.PackageManager; |
| 24 | import android.content.pm.PackageManager.NameNotFoundException; |
| 25 | import android.content.res.Resources; |
| 26 | import android.graphics.drawable.Drawable; |
| 27 | import android.os.Parcel; |
| 28 | import android.os.Parcelable; |
| 29 | import android.text.format.DateUtils; |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 30 | import android.util.Log; |
| 31 | import android.util.SparseArray; |
Jason Monk | 39b4674 | 2015-09-10 15:52:51 -0400 | [diff] [blame] | 32 | |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 33 | import com.android.settings.R; |
| 34 | |
| 35 | import java.io.File; |
| 36 | import java.text.Collator; |
| 37 | import java.util.ArrayList; |
| 38 | import java.util.Collections; |
| 39 | import java.util.Comparator; |
| 40 | import java.util.HashMap; |
| 41 | import java.util.List; |
| 42 | |
| 43 | public class AppOpsState { |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 44 | static final String TAG = "AppOpsState"; |
| 45 | static final boolean DEBUG = false; |
| 46 | |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 47 | final Context mContext; |
| 48 | final AppOpsManager mAppOps; |
| 49 | final PackageManager mPm; |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 50 | final CharSequence[] mOpSummaries; |
| 51 | final CharSequence[] mOpLabels; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 52 | |
| 53 | List<AppOpEntry> mApps; |
| 54 | |
| 55 | public AppOpsState(Context context) { |
| 56 | mContext = context; |
| 57 | mAppOps = (AppOpsManager)context.getSystemService(Context.APP_OPS_SERVICE); |
| 58 | mPm = context.getPackageManager(); |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 59 | mOpSummaries = context.getResources().getTextArray(R.array.app_ops_summaries); |
| 60 | mOpLabels = context.getResources().getTextArray(R.array.app_ops_labels); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 61 | } |
| 62 | |
| 63 | public static class OpsTemplate implements Parcelable { |
| 64 | public final int[] ops; |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 65 | public final boolean[] showPerms; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 66 | |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 67 | public OpsTemplate(int[] _ops, boolean[] _showPerms) { |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 68 | ops = _ops; |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 69 | showPerms = _showPerms; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 70 | } |
| 71 | |
| 72 | OpsTemplate(Parcel src) { |
| 73 | ops = src.createIntArray(); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 74 | showPerms = src.createBooleanArray(); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 75 | } |
| 76 | |
| 77 | @Override |
| 78 | public int describeContents() { |
| 79 | return 0; |
| 80 | } |
| 81 | |
| 82 | @Override |
| 83 | public void writeToParcel(Parcel dest, int flags) { |
| 84 | dest.writeIntArray(ops); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 85 | dest.writeBooleanArray(showPerms); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 86 | } |
| 87 | |
| 88 | public static final Creator<OpsTemplate> CREATOR = new Creator<OpsTemplate>() { |
| 89 | @Override public OpsTemplate createFromParcel(Parcel source) { |
| 90 | return new OpsTemplate(source); |
| 91 | } |
| 92 | |
| 93 | @Override public OpsTemplate[] newArray(int size) { |
| 94 | return new OpsTemplate[size]; |
| 95 | } |
| 96 | }; |
| 97 | } |
| 98 | |
| 99 | public static final OpsTemplate LOCATION_TEMPLATE = new OpsTemplate( |
| 100 | new int[] { AppOpsManager.OP_COARSE_LOCATION, |
| 101 | AppOpsManager.OP_FINE_LOCATION, |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 102 | AppOpsManager.OP_GPS, |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 103 | AppOpsManager.OP_WIFI_SCAN, |
Dianne Hackborn | 15ab775 | 2013-07-09 18:19:11 -0700 | [diff] [blame] | 104 | AppOpsManager.OP_NEIGHBORING_CELLS, |
David Christie | 4a06e4c | 2013-07-29 16:01:21 -0700 | [diff] [blame] | 105 | AppOpsManager.OP_MONITOR_LOCATION, |
Michael Wright | bdf28e4 | 2014-07-08 18:54:55 -0700 | [diff] [blame] | 106 | AppOpsManager.OP_MONITOR_HIGH_POWER_LOCATION }, |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 107 | new boolean[] { true, |
| 108 | true, |
| 109 | false, |
| 110 | false, |
Dianne Hackborn | 15ab775 | 2013-07-09 18:19:11 -0700 | [diff] [blame] | 111 | false, |
David Christie | 4a06e4c | 2013-07-29 16:01:21 -0700 | [diff] [blame] | 112 | false, |
Michael Wright | bdf28e4 | 2014-07-08 18:54:55 -0700 | [diff] [blame] | 113 | false } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 114 | ); |
| 115 | |
| 116 | public static final OpsTemplate PERSONAL_TEMPLATE = new OpsTemplate( |
| 117 | new int[] { AppOpsManager.OP_READ_CONTACTS, |
| 118 | AppOpsManager.OP_WRITE_CONTACTS, |
| 119 | AppOpsManager.OP_READ_CALL_LOG, |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 120 | AppOpsManager.OP_WRITE_CALL_LOG, |
| 121 | AppOpsManager.OP_READ_CALENDAR, |
Dianne Hackborn | 782c3de | 2013-02-25 18:03:06 -0800 | [diff] [blame] | 122 | AppOpsManager.OP_WRITE_CALENDAR, |
| 123 | AppOpsManager.OP_READ_CLIPBOARD, |
| 124 | AppOpsManager.OP_WRITE_CLIPBOARD }, |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 125 | new boolean[] { true, |
| 126 | true, |
| 127 | true, |
| 128 | true, |
| 129 | true, |
Dianne Hackborn | 782c3de | 2013-02-25 18:03:06 -0800 | [diff] [blame] | 130 | true, |
| 131 | false, |
| 132 | false } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 133 | ); |
| 134 | |
Dianne Hackborn | 9833787 | 2013-02-04 18:27:40 -0800 | [diff] [blame] | 135 | public static final OpsTemplate MESSAGING_TEMPLATE = new OpsTemplate( |
| 136 | new int[] { AppOpsManager.OP_READ_SMS, |
| 137 | AppOpsManager.OP_RECEIVE_SMS, |
| 138 | AppOpsManager.OP_RECEIVE_EMERGECY_SMS, |
| 139 | AppOpsManager.OP_RECEIVE_MMS, |
| 140 | AppOpsManager.OP_RECEIVE_WAP_PUSH, |
| 141 | AppOpsManager.OP_WRITE_SMS, |
| 142 | AppOpsManager.OP_SEND_SMS, |
| 143 | AppOpsManager.OP_READ_ICC_SMS, |
| 144 | AppOpsManager.OP_WRITE_ICC_SMS }, |
| 145 | new boolean[] { true, |
| 146 | true, |
| 147 | true, |
| 148 | true, |
| 149 | true, |
| 150 | true, |
| 151 | true, |
| 152 | true, |
| 153 | true } |
| 154 | ); |
| 155 | |
Dianne Hackborn | 6e91bab | 2013-04-30 15:06:22 -0700 | [diff] [blame] | 156 | public static final OpsTemplate MEDIA_TEMPLATE = new OpsTemplate( |
Daniel Sandler | df3f6d6 | 2013-01-30 14:03:58 -0500 | [diff] [blame] | 157 | new int[] { AppOpsManager.OP_VIBRATE, |
Dianne Hackborn | 6e91bab | 2013-04-30 15:06:22 -0700 | [diff] [blame] | 158 | AppOpsManager.OP_CAMERA, |
| 159 | AppOpsManager.OP_RECORD_AUDIO, |
| 160 | AppOpsManager.OP_PLAY_AUDIO, |
| 161 | AppOpsManager.OP_TAKE_MEDIA_BUTTONS, |
| 162 | AppOpsManager.OP_TAKE_AUDIO_FOCUS, |
| 163 | AppOpsManager.OP_AUDIO_MASTER_VOLUME, |
| 164 | AppOpsManager.OP_AUDIO_VOICE_VOLUME, |
| 165 | AppOpsManager.OP_AUDIO_RING_VOLUME, |
| 166 | AppOpsManager.OP_AUDIO_MEDIA_VOLUME, |
| 167 | AppOpsManager.OP_AUDIO_ALARM_VOLUME, |
| 168 | AppOpsManager.OP_AUDIO_NOTIFICATION_VOLUME, |
Emily Bernier | 4241d0b | 2014-05-28 15:51:58 -0400 | [diff] [blame] | 169 | AppOpsManager.OP_AUDIO_BLUETOOTH_VOLUME, |
Jean-Michel Trivi | 2792b10 | 2016-11-11 09:43:01 -0800 | [diff] [blame^] | 170 | AppOpsManager.OP_AUDIO_ACCESSIBILITY_VOLUME, |
Emily Bernier | 4241d0b | 2014-05-28 15:51:58 -0400 | [diff] [blame] | 171 | AppOpsManager.OP_MUTE_MICROPHONE}, |
Dianne Hackborn | 6e91bab | 2013-04-30 15:06:22 -0700 | [diff] [blame] | 172 | new boolean[] { false, |
| 173 | true, |
| 174 | true, |
| 175 | false, |
| 176 | false, |
| 177 | false, |
| 178 | false, |
| 179 | false, |
| 180 | false, |
| 181 | false, |
| 182 | false, |
| 183 | false, |
Dianne Hackborn | f467c0a | 2015-12-07 16:36:09 -0800 | [diff] [blame] | 184 | false, |
Dianne Hackborn | 6e91bab | 2013-04-30 15:06:22 -0700 | [diff] [blame] | 185 | false } |
| 186 | ); |
| 187 | |
| 188 | public static final OpsTemplate DEVICE_TEMPLATE = new OpsTemplate( |
| 189 | new int[] { AppOpsManager.OP_POST_NOTIFICATION, |
Daniel Sandler | 328e2d2 | 2013-01-17 13:14:02 -0500 | [diff] [blame] | 190 | AppOpsManager.OP_ACCESS_NOTIFICATIONS, |
Dianne Hackborn | 3b13d2b | 2013-02-05 17:31:10 -0800 | [diff] [blame] | 191 | AppOpsManager.OP_CALL_PHONE, |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 192 | AppOpsManager.OP_WRITE_SETTINGS, |
Dianne Hackborn | ec74da6 | 2013-05-17 11:28:30 -0700 | [diff] [blame] | 193 | AppOpsManager.OP_SYSTEM_ALERT_WINDOW, |
Michael Wright | bdf28e4 | 2014-07-08 18:54:55 -0700 | [diff] [blame] | 194 | AppOpsManager.OP_WAKE_LOCK, |
Jeff Davidson | 90b3b21 | 2014-08-12 15:59:01 -0700 | [diff] [blame] | 195 | AppOpsManager.OP_PROJECT_MEDIA, |
Dianne Hackborn | 7781ef6 | 2015-04-04 17:48:18 -0700 | [diff] [blame] | 196 | AppOpsManager.OP_ACTIVATE_VPN, |
| 197 | AppOpsManager.OP_ASSIST_STRUCTURE, |
| 198 | AppOpsManager.OP_ASSIST_SCREENSHOT}, |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 199 | new boolean[] { false, |
Dianne Hackborn | 3b13d2b | 2013-02-05 17:31:10 -0800 | [diff] [blame] | 200 | true, |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 201 | true, |
Dianne Hackborn | 7ff56d3 | 2013-02-13 11:02:12 -0800 | [diff] [blame] | 202 | true, |
Dianne Hackborn | ec74da6 | 2013-05-17 11:28:30 -0700 | [diff] [blame] | 203 | true, |
Michael Wright | bdf28e4 | 2014-07-08 18:54:55 -0700 | [diff] [blame] | 204 | true, |
Jeff Davidson | 90b3b21 | 2014-08-12 15:59:01 -0700 | [diff] [blame] | 205 | false, |
Dianne Hackborn | 7781ef6 | 2015-04-04 17:48:18 -0700 | [diff] [blame] | 206 | false, |
| 207 | false, |
| 208 | false } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 209 | ); |
| 210 | |
Dianne Hackborn | f467c0a | 2015-12-07 16:36:09 -0800 | [diff] [blame] | 211 | public static final OpsTemplate RUN_IN_BACKGROUND_TEMPLATE = new OpsTemplate( |
| 212 | new int[] { AppOpsManager.OP_RUN_IN_BACKGROUND }, |
| 213 | new boolean[] { false } |
| 214 | ); |
| 215 | |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 216 | public static final OpsTemplate[] ALL_TEMPLATES = new OpsTemplate[] { |
Dianne Hackborn | 6e91bab | 2013-04-30 15:06:22 -0700 | [diff] [blame] | 217 | LOCATION_TEMPLATE, PERSONAL_TEMPLATE, MESSAGING_TEMPLATE, |
Dianne Hackborn | f467c0a | 2015-12-07 16:36:09 -0800 | [diff] [blame] | 218 | MEDIA_TEMPLATE, DEVICE_TEMPLATE, RUN_IN_BACKGROUND_TEMPLATE |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 219 | }; |
| 220 | |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 221 | /** |
| 222 | * This class holds the per-item data in our Loader. |
| 223 | */ |
| 224 | public static class AppEntry { |
| 225 | private final AppOpsState mState; |
| 226 | private final ApplicationInfo mInfo; |
| 227 | private final File mApkFile; |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 228 | private final SparseArray<AppOpsManager.OpEntry> mOps |
| 229 | = new SparseArray<AppOpsManager.OpEntry>(); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 230 | private final SparseArray<AppOpEntry> mOpSwitches |
| 231 | = new SparseArray<AppOpEntry>(); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 232 | private String mLabel; |
| 233 | private Drawable mIcon; |
| 234 | private boolean mMounted; |
| 235 | |
| 236 | public AppEntry(AppOpsState state, ApplicationInfo info) { |
| 237 | mState = state; |
| 238 | mInfo = info; |
| 239 | mApkFile = new File(info.sourceDir); |
| 240 | } |
| 241 | |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 242 | public void addOp(AppOpEntry entry, AppOpsManager.OpEntry op) { |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 243 | mOps.put(op.getOp(), op); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 244 | mOpSwitches.put(AppOpsManager.opToSwitch(op.getOp()), entry); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 245 | } |
| 246 | |
| 247 | public boolean hasOp(int op) { |
| 248 | return mOps.indexOfKey(op) >= 0; |
| 249 | } |
| 250 | |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 251 | public AppOpEntry getOpSwitch(int op) { |
| 252 | return mOpSwitches.get(AppOpsManager.opToSwitch(op)); |
| 253 | } |
| 254 | |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 255 | public ApplicationInfo getApplicationInfo() { |
| 256 | return mInfo; |
| 257 | } |
| 258 | |
| 259 | public String getLabel() { |
| 260 | return mLabel; |
| 261 | } |
| 262 | |
| 263 | public Drawable getIcon() { |
| 264 | if (mIcon == null) { |
| 265 | if (mApkFile.exists()) { |
| 266 | mIcon = mInfo.loadIcon(mState.mPm); |
| 267 | return mIcon; |
| 268 | } else { |
| 269 | mMounted = false; |
| 270 | } |
| 271 | } else if (!mMounted) { |
| 272 | // If the app wasn't mounted but is now mounted, reload |
| 273 | // its icon. |
| 274 | if (mApkFile.exists()) { |
| 275 | mMounted = true; |
| 276 | mIcon = mInfo.loadIcon(mState.mPm); |
| 277 | return mIcon; |
| 278 | } |
| 279 | } else { |
| 280 | return mIcon; |
| 281 | } |
| 282 | |
Alan Viverette | 0ba89bd | 2014-10-10 10:58:58 -0700 | [diff] [blame] | 283 | return mState.mContext.getDrawable( |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 284 | android.R.drawable.sym_def_app_icon); |
| 285 | } |
| 286 | |
| 287 | @Override public String toString() { |
| 288 | return mLabel; |
| 289 | } |
| 290 | |
| 291 | void loadLabel(Context context) { |
| 292 | if (mLabel == null || !mMounted) { |
| 293 | if (!mApkFile.exists()) { |
| 294 | mMounted = false; |
| 295 | mLabel = mInfo.packageName; |
| 296 | } else { |
| 297 | mMounted = true; |
| 298 | CharSequence label = mInfo.loadLabel(context.getPackageManager()); |
| 299 | mLabel = label != null ? label.toString() : mInfo.packageName; |
| 300 | } |
| 301 | } |
| 302 | } |
| 303 | } |
| 304 | |
| 305 | /** |
| 306 | * This class holds the per-item data in our Loader. |
| 307 | */ |
| 308 | public static class AppOpEntry { |
| 309 | private final AppOpsManager.PackageOps mPkgOps; |
| 310 | private final ArrayList<AppOpsManager.OpEntry> mOps |
| 311 | = new ArrayList<AppOpsManager.OpEntry>(); |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 312 | private final ArrayList<AppOpsManager.OpEntry> mSwitchOps |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 313 | = new ArrayList<AppOpsManager.OpEntry>(); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 314 | private final AppEntry mApp; |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 315 | private final int mSwitchOrder; |
Dianne Hackborn | f467c0a | 2015-12-07 16:36:09 -0800 | [diff] [blame] | 316 | private int mOverriddenPrimaryMode = -1; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 317 | |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 318 | public AppOpEntry(AppOpsManager.PackageOps pkg, AppOpsManager.OpEntry op, AppEntry app, |
| 319 | int switchOrder) { |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 320 | mPkgOps = pkg; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 321 | mApp = app; |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 322 | mSwitchOrder = switchOrder; |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 323 | mApp.addOp(this, op); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 324 | mOps.add(op); |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 325 | mSwitchOps.add(op); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 326 | } |
| 327 | |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 328 | private static void addOp(ArrayList<AppOpsManager.OpEntry> list, AppOpsManager.OpEntry op) { |
| 329 | for (int i=0; i<list.size(); i++) { |
| 330 | AppOpsManager.OpEntry pos = list.get(i); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 331 | if (pos.isRunning() != op.isRunning()) { |
| 332 | if (op.isRunning()) { |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 333 | list.add(i, op); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 334 | return; |
| 335 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 336 | continue; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 337 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 338 | if (pos.getTime() < op.getTime()) { |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 339 | list.add(i, op); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 340 | return; |
| 341 | } |
| 342 | } |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 343 | list.add(op); |
| 344 | } |
| 345 | |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 346 | public void addOp(AppOpsManager.OpEntry op) { |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 347 | mApp.addOp(this, op); |
| 348 | addOp(mOps, op); |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 349 | if (mApp.getOpSwitch(AppOpsManager.opToSwitch(op.getOp())) == null) { |
| 350 | addOp(mSwitchOps, op); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 351 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 352 | } |
| 353 | |
| 354 | public AppEntry getAppEntry() { |
| 355 | return mApp; |
| 356 | } |
| 357 | |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 358 | public int getSwitchOrder() { |
| 359 | return mSwitchOrder; |
| 360 | } |
| 361 | |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 362 | public AppOpsManager.PackageOps getPackageOps() { |
| 363 | return mPkgOps; |
| 364 | } |
| 365 | |
| 366 | public int getNumOpEntry() { |
| 367 | return mOps.size(); |
| 368 | } |
| 369 | |
| 370 | public AppOpsManager.OpEntry getOpEntry(int pos) { |
| 371 | return mOps.get(pos); |
| 372 | } |
| 373 | |
Dianne Hackborn | f467c0a | 2015-12-07 16:36:09 -0800 | [diff] [blame] | 374 | public int getPrimaryOpMode() { |
| 375 | return mOverriddenPrimaryMode >= 0 ? mOverriddenPrimaryMode : mOps.get(0).getMode(); |
| 376 | } |
| 377 | |
| 378 | public void overridePrimaryOpMode(int mode) { |
| 379 | mOverriddenPrimaryMode = mode; |
| 380 | } |
| 381 | |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 382 | private CharSequence getCombinedText(ArrayList<AppOpsManager.OpEntry> ops, |
| 383 | CharSequence[] items) { |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 384 | if (ops.size() == 1) { |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 385 | return items[ops.get(0).getOp()]; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 386 | } else { |
| 387 | StringBuilder builder = new StringBuilder(); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 388 | for (int i=0; i<ops.size(); i++) { |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 389 | if (i > 0) { |
| 390 | builder.append(", "); |
| 391 | } |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 392 | builder.append(items[ops.get(i).getOp()]); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 393 | } |
| 394 | return builder.toString(); |
| 395 | } |
| 396 | } |
| 397 | |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 398 | public CharSequence getSummaryText(AppOpsState state) { |
| 399 | return getCombinedText(mOps, state.mOpSummaries); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 400 | } |
| 401 | |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 402 | public CharSequence getSwitchText(AppOpsState state) { |
| 403 | if (mSwitchOps.size() > 0) { |
| 404 | return getCombinedText(mSwitchOps, state.mOpLabels); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 405 | } else { |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 406 | return getCombinedText(mOps, state.mOpLabels); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 407 | } |
| 408 | } |
| 409 | |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 410 | public CharSequence getTimeText(Resources res, boolean showEmptyText) { |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 411 | if (isRunning()) { |
| 412 | return res.getText(R.string.app_ops_running); |
| 413 | } |
| 414 | if (getTime() > 0) { |
| 415 | return DateUtils.getRelativeTimeSpanString(getTime(), |
| 416 | System.currentTimeMillis(), |
| 417 | DateUtils.MINUTE_IN_MILLIS, |
| 418 | DateUtils.FORMAT_ABBREV_RELATIVE); |
| 419 | } |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 420 | return showEmptyText ? res.getText(R.string.app_ops_never_used) : ""; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 421 | } |
| 422 | |
| 423 | public boolean isRunning() { |
| 424 | return mOps.get(0).isRunning(); |
| 425 | } |
| 426 | |
| 427 | public long getTime() { |
| 428 | return mOps.get(0).getTime(); |
| 429 | } |
| 430 | |
| 431 | @Override public String toString() { |
| 432 | return mApp.getLabel(); |
| 433 | } |
| 434 | } |
| 435 | |
| 436 | /** |
Dianne Hackborn | f467c0a | 2015-12-07 16:36:09 -0800 | [diff] [blame] | 437 | * Perform app op state comparison of application entry objects. |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 438 | */ |
Dianne Hackborn | f467c0a | 2015-12-07 16:36:09 -0800 | [diff] [blame] | 439 | public static final Comparator<AppOpEntry> RECENCY_COMPARATOR = new Comparator<AppOpEntry>() { |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 440 | private final Collator sCollator = Collator.getInstance(); |
| 441 | @Override |
| 442 | public int compare(AppOpEntry object1, AppOpEntry object2) { |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 443 | if (object1.getSwitchOrder() != object2.getSwitchOrder()) { |
| 444 | return object1.getSwitchOrder() < object2.getSwitchOrder() ? -1 : 1; |
| 445 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 446 | if (object1.isRunning() != object2.isRunning()) { |
| 447 | // Currently running ops go first. |
| 448 | return object1.isRunning() ? -1 : 1; |
| 449 | } |
| 450 | if (object1.getTime() != object2.getTime()) { |
| 451 | // More recent times go first. |
| 452 | return object1.getTime() > object2.getTime() ? -1 : 1; |
| 453 | } |
| 454 | return sCollator.compare(object1.getAppEntry().getLabel(), |
| 455 | object2.getAppEntry().getLabel()); |
| 456 | } |
| 457 | }; |
| 458 | |
Dianne Hackborn | f467c0a | 2015-12-07 16:36:09 -0800 | [diff] [blame] | 459 | /** |
| 460 | * Perform alphabetical comparison of application entry objects. |
| 461 | */ |
| 462 | public static final Comparator<AppOpEntry> LABEL_COMPARATOR = new Comparator<AppOpEntry>() { |
| 463 | private final Collator sCollator = Collator.getInstance(); |
| 464 | @Override |
| 465 | public int compare(AppOpEntry object1, AppOpEntry object2) { |
| 466 | return sCollator.compare(object1.getAppEntry().getLabel(), |
| 467 | object2.getAppEntry().getLabel()); |
| 468 | } |
| 469 | }; |
| 470 | |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 471 | private void addOp(List<AppOpEntry> entries, AppOpsManager.PackageOps pkgOps, |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 472 | AppEntry appEntry, AppOpsManager.OpEntry opEntry, boolean allowMerge, int switchOrder) { |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 473 | if (allowMerge && entries.size() > 0) { |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 474 | AppOpEntry last = entries.get(entries.size()-1); |
| 475 | if (last.getAppEntry() == appEntry) { |
| 476 | boolean lastExe = last.getTime() != 0; |
| 477 | boolean entryExe = opEntry.getTime() != 0; |
| 478 | if (lastExe == entryExe) { |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 479 | if (DEBUG) Log.d(TAG, "Add op " + opEntry.getOp() + " to package " |
| 480 | + pkgOps.getPackageName() + ": append to " + last); |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 481 | last.addOp(opEntry); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 482 | return; |
| 483 | } |
| 484 | } |
| 485 | } |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 486 | AppOpEntry entry = appEntry.getOpSwitch(opEntry.getOp()); |
| 487 | if (entry != null) { |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 488 | entry.addOp(opEntry); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 489 | return; |
| 490 | } |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 491 | entry = new AppOpEntry(pkgOps, opEntry, appEntry, switchOrder); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 492 | if (DEBUG) Log.d(TAG, "Add op " + opEntry.getOp() + " to package " |
| 493 | + pkgOps.getPackageName() + ": making new " + entry); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 494 | entries.add(entry); |
| 495 | } |
| 496 | |
Dianne Hackborn | f467c0a | 2015-12-07 16:36:09 -0800 | [diff] [blame] | 497 | public AppOpsManager getAppOpsManager() { |
| 498 | return mAppOps; |
| 499 | } |
| 500 | |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 501 | public List<AppOpEntry> buildState(OpsTemplate tpl) { |
Dianne Hackborn | f467c0a | 2015-12-07 16:36:09 -0800 | [diff] [blame] | 502 | return buildState(tpl, 0, null, RECENCY_COMPARATOR); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 503 | } |
| 504 | |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 505 | private AppEntry getAppEntry(final Context context, final HashMap<String, AppEntry> appEntries, |
| 506 | final String packageName, ApplicationInfo appInfo) { |
| 507 | AppEntry appEntry = appEntries.get(packageName); |
| 508 | if (appEntry == null) { |
| 509 | if (appInfo == null) { |
| 510 | try { |
| 511 | appInfo = mPm.getApplicationInfo(packageName, |
| 512 | PackageManager.GET_DISABLED_COMPONENTS |
| 513 | | PackageManager.GET_UNINSTALLED_PACKAGES); |
| 514 | } catch (PackageManager.NameNotFoundException e) { |
| 515 | Log.w(TAG, "Unable to find info for package " + packageName); |
| 516 | return null; |
| 517 | } |
| 518 | } |
| 519 | appEntry = new AppEntry(this, appInfo); |
| 520 | appEntry.loadLabel(context); |
| 521 | appEntries.put(packageName, appEntry); |
| 522 | } |
| 523 | return appEntry; |
| 524 | } |
| 525 | |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 526 | public List<AppOpEntry> buildState(OpsTemplate tpl, int uid, String packageName) { |
Dianne Hackborn | f467c0a | 2015-12-07 16:36:09 -0800 | [diff] [blame] | 527 | return buildState(tpl, uid, packageName, RECENCY_COMPARATOR); |
| 528 | } |
| 529 | |
| 530 | public List<AppOpEntry> buildState(OpsTemplate tpl, int uid, String packageName, |
| 531 | Comparator<AppOpEntry> comparator) { |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 532 | final Context context = mContext; |
| 533 | |
| 534 | final HashMap<String, AppEntry> appEntries = new HashMap<String, AppEntry>(); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 535 | final List<AppOpEntry> entries = new ArrayList<AppOpEntry>(); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 536 | |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 537 | final ArrayList<String> perms = new ArrayList<String>(); |
| 538 | final ArrayList<Integer> permOps = new ArrayList<Integer>(); |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 539 | final int[] opToOrder = new int[AppOpsManager._NUM_OP]; |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 540 | for (int i=0; i<tpl.ops.length; i++) { |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 541 | if (tpl.showPerms[i]) { |
| 542 | String perm = AppOpsManager.opToPermission(tpl.ops[i]); |
| 543 | if (perm != null && !perms.contains(perm)) { |
| 544 | perms.add(perm); |
| 545 | permOps.add(tpl.ops[i]); |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 546 | opToOrder[tpl.ops[i]] = i; |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 547 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 548 | } |
| 549 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 550 | |
| 551 | List<AppOpsManager.PackageOps> pkgs; |
| 552 | if (packageName != null) { |
| 553 | pkgs = mAppOps.getOpsForPackage(uid, packageName, tpl.ops); |
| 554 | } else { |
| 555 | pkgs = mAppOps.getPackagesForOps(tpl.ops); |
| 556 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 557 | |
| 558 | if (pkgs != null) { |
| 559 | for (int i=0; i<pkgs.size(); i++) { |
| 560 | AppOpsManager.PackageOps pkgOps = pkgs.get(i); |
| 561 | AppEntry appEntry = getAppEntry(context, appEntries, pkgOps.getPackageName(), null); |
| 562 | if (appEntry == null) { |
| 563 | continue; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 564 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 565 | for (int j=0; j<pkgOps.getOps().size(); j++) { |
| 566 | AppOpsManager.OpEntry opEntry = pkgOps.getOps().get(j); |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 567 | addOp(entries, pkgOps, appEntry, opEntry, packageName == null, |
| 568 | packageName == null ? 0 : opToOrder[opEntry.getOp()]); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 569 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 570 | } |
| 571 | } |
| 572 | |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 573 | List<PackageInfo> apps; |
| 574 | if (packageName != null) { |
| 575 | apps = new ArrayList<PackageInfo>(); |
| 576 | try { |
| 577 | PackageInfo pi = mPm.getPackageInfo(packageName, PackageManager.GET_PERMISSIONS); |
| 578 | apps.add(pi); |
| 579 | } catch (NameNotFoundException e) { |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 580 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 581 | } else { |
| 582 | String[] permsArray = new String[perms.size()]; |
| 583 | perms.toArray(permsArray); |
| 584 | apps = mPm.getPackagesHoldingPermissions(permsArray, 0); |
| 585 | } |
| 586 | for (int i=0; i<apps.size(); i++) { |
| 587 | PackageInfo appInfo = apps.get(i); |
| 588 | AppEntry appEntry = getAppEntry(context, appEntries, appInfo.packageName, |
| 589 | appInfo.applicationInfo); |
| 590 | if (appEntry == null) { |
| 591 | continue; |
| 592 | } |
| 593 | List<AppOpsManager.OpEntry> dummyOps = null; |
| 594 | AppOpsManager.PackageOps pkgOps = null; |
Daniel Sandler | 7d3a4c1 | 2013-01-31 15:22:51 -0500 | [diff] [blame] | 595 | if (appInfo.requestedPermissions != null) { |
| 596 | for (int j=0; j<appInfo.requestedPermissions.length; j++) { |
| 597 | if (appInfo.requestedPermissionsFlags != null) { |
| 598 | if ((appInfo.requestedPermissionsFlags[j] |
| 599 | & PackageInfo.REQUESTED_PERMISSION_GRANTED) == 0) { |
| 600 | if (DEBUG) Log.d(TAG, "Pkg " + appInfo.packageName + " perm " |
| 601 | + appInfo.requestedPermissions[j] + " not granted; skipping"); |
Danny Baumann | 67e72d9 | 2013-08-05 14:28:58 +0200 | [diff] [blame] | 602 | continue; |
Daniel Sandler | 7d3a4c1 | 2013-01-31 15:22:51 -0500 | [diff] [blame] | 603 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 604 | } |
Daniel Sandler | 7d3a4c1 | 2013-01-31 15:22:51 -0500 | [diff] [blame] | 605 | if (DEBUG) Log.d(TAG, "Pkg " + appInfo.packageName + ": requested perm " |
| 606 | + appInfo.requestedPermissions[j]); |
| 607 | for (int k=0; k<perms.size(); k++) { |
| 608 | if (!perms.get(k).equals(appInfo.requestedPermissions[j])) { |
| 609 | continue; |
| 610 | } |
| 611 | if (DEBUG) Log.d(TAG, "Pkg " + appInfo.packageName + " perm " + perms.get(k) |
| 612 | + " has op " + permOps.get(k) + ": " + appEntry.hasOp(permOps.get(k))); |
| 613 | if (appEntry.hasOp(permOps.get(k))) { |
| 614 | continue; |
| 615 | } |
| 616 | if (dummyOps == null) { |
| 617 | dummyOps = new ArrayList<AppOpsManager.OpEntry>(); |
| 618 | pkgOps = new AppOpsManager.PackageOps( |
| 619 | appInfo.packageName, appInfo.applicationInfo.uid, dummyOps); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 620 | |
Daniel Sandler | 7d3a4c1 | 2013-01-31 15:22:51 -0500 | [diff] [blame] | 621 | } |
| 622 | AppOpsManager.OpEntry opEntry = new AppOpsManager.OpEntry( |
Svetoslav | 5b45c3c | 2015-07-01 18:33:32 -0700 | [diff] [blame] | 623 | permOps.get(k), AppOpsManager.MODE_ALLOWED, 0, 0, 0, -1, null); |
Daniel Sandler | 7d3a4c1 | 2013-01-31 15:22:51 -0500 | [diff] [blame] | 624 | dummyOps.add(opEntry); |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 625 | addOp(entries, pkgOps, appEntry, opEntry, packageName == null, |
| 626 | packageName == null ? 0 : opToOrder[opEntry.getOp()]); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 627 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 628 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 629 | } |
| 630 | } |
| 631 | |
| 632 | // Sort the list. |
Dianne Hackborn | f467c0a | 2015-12-07 16:36:09 -0800 | [diff] [blame] | 633 | Collections.sort(entries, comparator); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 634 | |
| 635 | // Done! |
| 636 | return entries; |
| 637 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 638 | } |