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; |
| 30 | |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 31 | import android.util.Log; |
| 32 | import android.util.SparseArray; |
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, |
| 105 | AppOpsManager.OP_MONITOR_LOCATION }, |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 106 | new boolean[] { true, |
| 107 | true, |
| 108 | false, |
| 109 | false, |
Dianne Hackborn | 15ab775 | 2013-07-09 18:19:11 -0700 | [diff] [blame^] | 110 | false, |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 111 | false } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 112 | ); |
| 113 | |
| 114 | public static final OpsTemplate PERSONAL_TEMPLATE = new OpsTemplate( |
| 115 | new int[] { AppOpsManager.OP_READ_CONTACTS, |
| 116 | AppOpsManager.OP_WRITE_CONTACTS, |
| 117 | AppOpsManager.OP_READ_CALL_LOG, |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 118 | AppOpsManager.OP_WRITE_CALL_LOG, |
| 119 | AppOpsManager.OP_READ_CALENDAR, |
Dianne Hackborn | 782c3de | 2013-02-25 18:03:06 -0800 | [diff] [blame] | 120 | AppOpsManager.OP_WRITE_CALENDAR, |
| 121 | AppOpsManager.OP_READ_CLIPBOARD, |
| 122 | AppOpsManager.OP_WRITE_CLIPBOARD }, |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 123 | new boolean[] { true, |
| 124 | true, |
| 125 | true, |
| 126 | true, |
| 127 | true, |
Dianne Hackborn | 782c3de | 2013-02-25 18:03:06 -0800 | [diff] [blame] | 128 | true, |
| 129 | false, |
| 130 | false } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 131 | ); |
| 132 | |
Dianne Hackborn | 9833787 | 2013-02-04 18:27:40 -0800 | [diff] [blame] | 133 | public static final OpsTemplate MESSAGING_TEMPLATE = new OpsTemplate( |
| 134 | new int[] { AppOpsManager.OP_READ_SMS, |
| 135 | AppOpsManager.OP_RECEIVE_SMS, |
| 136 | AppOpsManager.OP_RECEIVE_EMERGECY_SMS, |
| 137 | AppOpsManager.OP_RECEIVE_MMS, |
| 138 | AppOpsManager.OP_RECEIVE_WAP_PUSH, |
| 139 | AppOpsManager.OP_WRITE_SMS, |
| 140 | AppOpsManager.OP_SEND_SMS, |
| 141 | AppOpsManager.OP_READ_ICC_SMS, |
| 142 | AppOpsManager.OP_WRITE_ICC_SMS }, |
| 143 | new boolean[] { true, |
| 144 | true, |
| 145 | true, |
| 146 | true, |
| 147 | true, |
| 148 | true, |
| 149 | true, |
| 150 | true, |
| 151 | true } |
| 152 | ); |
| 153 | |
Dianne Hackborn | 6e91bab | 2013-04-30 15:06:22 -0700 | [diff] [blame] | 154 | public static final OpsTemplate MEDIA_TEMPLATE = new OpsTemplate( |
Daniel Sandler | df3f6d6 | 2013-01-30 14:03:58 -0500 | [diff] [blame] | 155 | new int[] { AppOpsManager.OP_VIBRATE, |
Dianne Hackborn | 6e91bab | 2013-04-30 15:06:22 -0700 | [diff] [blame] | 156 | AppOpsManager.OP_CAMERA, |
| 157 | AppOpsManager.OP_RECORD_AUDIO, |
| 158 | AppOpsManager.OP_PLAY_AUDIO, |
| 159 | AppOpsManager.OP_TAKE_MEDIA_BUTTONS, |
| 160 | AppOpsManager.OP_TAKE_AUDIO_FOCUS, |
| 161 | AppOpsManager.OP_AUDIO_MASTER_VOLUME, |
| 162 | AppOpsManager.OP_AUDIO_VOICE_VOLUME, |
| 163 | AppOpsManager.OP_AUDIO_RING_VOLUME, |
| 164 | AppOpsManager.OP_AUDIO_MEDIA_VOLUME, |
| 165 | AppOpsManager.OP_AUDIO_ALARM_VOLUME, |
| 166 | AppOpsManager.OP_AUDIO_NOTIFICATION_VOLUME, |
| 167 | AppOpsManager.OP_AUDIO_BLUETOOTH_VOLUME, }, |
| 168 | new boolean[] { false, |
| 169 | true, |
| 170 | true, |
| 171 | false, |
| 172 | false, |
| 173 | false, |
| 174 | false, |
| 175 | false, |
| 176 | false, |
| 177 | false, |
| 178 | false, |
| 179 | false, |
| 180 | false } |
| 181 | ); |
| 182 | |
| 183 | public static final OpsTemplate DEVICE_TEMPLATE = new OpsTemplate( |
| 184 | new int[] { AppOpsManager.OP_POST_NOTIFICATION, |
Daniel Sandler | 328e2d2 | 2013-01-17 13:14:02 -0500 | [diff] [blame] | 185 | AppOpsManager.OP_ACCESS_NOTIFICATIONS, |
Dianne Hackborn | 3b13d2b | 2013-02-05 17:31:10 -0800 | [diff] [blame] | 186 | AppOpsManager.OP_CALL_PHONE, |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 187 | AppOpsManager.OP_WRITE_SETTINGS, |
Dianne Hackborn | ec74da6 | 2013-05-17 11:28:30 -0700 | [diff] [blame] | 188 | AppOpsManager.OP_SYSTEM_ALERT_WINDOW, |
| 189 | AppOpsManager.OP_WAKE_LOCK }, |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 190 | new boolean[] { false, |
Dianne Hackborn | 3b13d2b | 2013-02-05 17:31:10 -0800 | [diff] [blame] | 191 | true, |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 192 | true, |
Dianne Hackborn | 7ff56d3 | 2013-02-13 11:02:12 -0800 | [diff] [blame] | 193 | true, |
Dianne Hackborn | ec74da6 | 2013-05-17 11:28:30 -0700 | [diff] [blame] | 194 | true, |
| 195 | true, } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 196 | ); |
| 197 | |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 198 | public static final OpsTemplate[] ALL_TEMPLATES = new OpsTemplate[] { |
Dianne Hackborn | 6e91bab | 2013-04-30 15:06:22 -0700 | [diff] [blame] | 199 | LOCATION_TEMPLATE, PERSONAL_TEMPLATE, MESSAGING_TEMPLATE, |
| 200 | MEDIA_TEMPLATE, DEVICE_TEMPLATE |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 201 | }; |
| 202 | |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 203 | /** |
| 204 | * This class holds the per-item data in our Loader. |
| 205 | */ |
| 206 | public static class AppEntry { |
| 207 | private final AppOpsState mState; |
| 208 | private final ApplicationInfo mInfo; |
| 209 | private final File mApkFile; |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 210 | private final SparseArray<AppOpsManager.OpEntry> mOps |
| 211 | = new SparseArray<AppOpsManager.OpEntry>(); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 212 | private final SparseArray<AppOpEntry> mOpSwitches |
| 213 | = new SparseArray<AppOpEntry>(); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 214 | private String mLabel; |
| 215 | private Drawable mIcon; |
| 216 | private boolean mMounted; |
| 217 | |
| 218 | public AppEntry(AppOpsState state, ApplicationInfo info) { |
| 219 | mState = state; |
| 220 | mInfo = info; |
| 221 | mApkFile = new File(info.sourceDir); |
| 222 | } |
| 223 | |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 224 | public void addOp(AppOpEntry entry, AppOpsManager.OpEntry op) { |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 225 | mOps.put(op.getOp(), op); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 226 | mOpSwitches.put(AppOpsManager.opToSwitch(op.getOp()), entry); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 227 | } |
| 228 | |
| 229 | public boolean hasOp(int op) { |
| 230 | return mOps.indexOfKey(op) >= 0; |
| 231 | } |
| 232 | |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 233 | public AppOpEntry getOpSwitch(int op) { |
| 234 | return mOpSwitches.get(AppOpsManager.opToSwitch(op)); |
| 235 | } |
| 236 | |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 237 | public ApplicationInfo getApplicationInfo() { |
| 238 | return mInfo; |
| 239 | } |
| 240 | |
| 241 | public String getLabel() { |
| 242 | return mLabel; |
| 243 | } |
| 244 | |
| 245 | public Drawable getIcon() { |
| 246 | if (mIcon == null) { |
| 247 | if (mApkFile.exists()) { |
| 248 | mIcon = mInfo.loadIcon(mState.mPm); |
| 249 | return mIcon; |
| 250 | } else { |
| 251 | mMounted = false; |
| 252 | } |
| 253 | } else if (!mMounted) { |
| 254 | // If the app wasn't mounted but is now mounted, reload |
| 255 | // its icon. |
| 256 | if (mApkFile.exists()) { |
| 257 | mMounted = true; |
| 258 | mIcon = mInfo.loadIcon(mState.mPm); |
| 259 | return mIcon; |
| 260 | } |
| 261 | } else { |
| 262 | return mIcon; |
| 263 | } |
| 264 | |
| 265 | return mState.mContext.getResources().getDrawable( |
| 266 | android.R.drawable.sym_def_app_icon); |
| 267 | } |
| 268 | |
| 269 | @Override public String toString() { |
| 270 | return mLabel; |
| 271 | } |
| 272 | |
| 273 | void loadLabel(Context context) { |
| 274 | if (mLabel == null || !mMounted) { |
| 275 | if (!mApkFile.exists()) { |
| 276 | mMounted = false; |
| 277 | mLabel = mInfo.packageName; |
| 278 | } else { |
| 279 | mMounted = true; |
| 280 | CharSequence label = mInfo.loadLabel(context.getPackageManager()); |
| 281 | mLabel = label != null ? label.toString() : mInfo.packageName; |
| 282 | } |
| 283 | } |
| 284 | } |
| 285 | } |
| 286 | |
| 287 | /** |
| 288 | * This class holds the per-item data in our Loader. |
| 289 | */ |
| 290 | public static class AppOpEntry { |
| 291 | private final AppOpsManager.PackageOps mPkgOps; |
| 292 | private final ArrayList<AppOpsManager.OpEntry> mOps |
| 293 | = new ArrayList<AppOpsManager.OpEntry>(); |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 294 | private final ArrayList<AppOpsManager.OpEntry> mSwitchOps |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 295 | = new ArrayList<AppOpsManager.OpEntry>(); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 296 | private final AppEntry mApp; |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 297 | private final int mSwitchOrder; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 298 | |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 299 | public AppOpEntry(AppOpsManager.PackageOps pkg, AppOpsManager.OpEntry op, AppEntry app, |
| 300 | int switchOrder) { |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 301 | mPkgOps = pkg; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 302 | mApp = app; |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 303 | mSwitchOrder = switchOrder; |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 304 | mApp.addOp(this, op); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 305 | mOps.add(op); |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 306 | mSwitchOps.add(op); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 307 | } |
| 308 | |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 309 | private static void addOp(ArrayList<AppOpsManager.OpEntry> list, AppOpsManager.OpEntry op) { |
| 310 | for (int i=0; i<list.size(); i++) { |
| 311 | AppOpsManager.OpEntry pos = list.get(i); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 312 | if (pos.isRunning() != op.isRunning()) { |
| 313 | if (op.isRunning()) { |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 314 | list.add(i, op); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 315 | return; |
| 316 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 317 | continue; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 318 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 319 | if (pos.getTime() < op.getTime()) { |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 320 | list.add(i, op); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 321 | return; |
| 322 | } |
| 323 | } |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 324 | list.add(op); |
| 325 | } |
| 326 | |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 327 | public void addOp(AppOpsManager.OpEntry op) { |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 328 | mApp.addOp(this, op); |
| 329 | addOp(mOps, op); |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 330 | if (mApp.getOpSwitch(AppOpsManager.opToSwitch(op.getOp())) == null) { |
| 331 | addOp(mSwitchOps, op); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 332 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 333 | } |
| 334 | |
| 335 | public AppEntry getAppEntry() { |
| 336 | return mApp; |
| 337 | } |
| 338 | |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 339 | public int getSwitchOrder() { |
| 340 | return mSwitchOrder; |
| 341 | } |
| 342 | |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 343 | public AppOpsManager.PackageOps getPackageOps() { |
| 344 | return mPkgOps; |
| 345 | } |
| 346 | |
| 347 | public int getNumOpEntry() { |
| 348 | return mOps.size(); |
| 349 | } |
| 350 | |
| 351 | public AppOpsManager.OpEntry getOpEntry(int pos) { |
| 352 | return mOps.get(pos); |
| 353 | } |
| 354 | |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 355 | private CharSequence getCombinedText(ArrayList<AppOpsManager.OpEntry> ops, |
| 356 | CharSequence[] items) { |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 357 | if (ops.size() == 1) { |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 358 | return items[ops.get(0).getOp()]; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 359 | } else { |
| 360 | StringBuilder builder = new StringBuilder(); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 361 | for (int i=0; i<ops.size(); i++) { |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 362 | if (i > 0) { |
| 363 | builder.append(", "); |
| 364 | } |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 365 | builder.append(items[ops.get(i).getOp()]); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 366 | } |
| 367 | return builder.toString(); |
| 368 | } |
| 369 | } |
| 370 | |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 371 | public CharSequence getSummaryText(AppOpsState state) { |
| 372 | return getCombinedText(mOps, state.mOpSummaries); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 373 | } |
| 374 | |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 375 | public CharSequence getSwitchText(AppOpsState state) { |
| 376 | if (mSwitchOps.size() > 0) { |
| 377 | return getCombinedText(mSwitchOps, state.mOpLabels); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 378 | } else { |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 379 | return getCombinedText(mOps, state.mOpLabels); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 380 | } |
| 381 | } |
| 382 | |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 383 | public CharSequence getTimeText(Resources res, boolean showEmptyText) { |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 384 | if (isRunning()) { |
| 385 | return res.getText(R.string.app_ops_running); |
| 386 | } |
| 387 | if (getTime() > 0) { |
| 388 | return DateUtils.getRelativeTimeSpanString(getTime(), |
| 389 | System.currentTimeMillis(), |
| 390 | DateUtils.MINUTE_IN_MILLIS, |
| 391 | DateUtils.FORMAT_ABBREV_RELATIVE); |
| 392 | } |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 393 | return showEmptyText ? res.getText(R.string.app_ops_never_used) : ""; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 394 | } |
| 395 | |
| 396 | public boolean isRunning() { |
| 397 | return mOps.get(0).isRunning(); |
| 398 | } |
| 399 | |
| 400 | public long getTime() { |
| 401 | return mOps.get(0).getTime(); |
| 402 | } |
| 403 | |
| 404 | @Override public String toString() { |
| 405 | return mApp.getLabel(); |
| 406 | } |
| 407 | } |
| 408 | |
| 409 | /** |
| 410 | * Perform alphabetical comparison of application entry objects. |
| 411 | */ |
| 412 | public static final Comparator<AppOpEntry> APP_OP_COMPARATOR = new Comparator<AppOpEntry>() { |
| 413 | private final Collator sCollator = Collator.getInstance(); |
| 414 | @Override |
| 415 | public int compare(AppOpEntry object1, AppOpEntry object2) { |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 416 | if (object1.getSwitchOrder() != object2.getSwitchOrder()) { |
| 417 | return object1.getSwitchOrder() < object2.getSwitchOrder() ? -1 : 1; |
| 418 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 419 | if (object1.isRunning() != object2.isRunning()) { |
| 420 | // Currently running ops go first. |
| 421 | return object1.isRunning() ? -1 : 1; |
| 422 | } |
| 423 | if (object1.getTime() != object2.getTime()) { |
| 424 | // More recent times go first. |
| 425 | return object1.getTime() > object2.getTime() ? -1 : 1; |
| 426 | } |
| 427 | return sCollator.compare(object1.getAppEntry().getLabel(), |
| 428 | object2.getAppEntry().getLabel()); |
| 429 | } |
| 430 | }; |
| 431 | |
| 432 | private void addOp(List<AppOpEntry> entries, AppOpsManager.PackageOps pkgOps, |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 433 | AppEntry appEntry, AppOpsManager.OpEntry opEntry, boolean allowMerge, int switchOrder) { |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 434 | if (allowMerge && entries.size() > 0) { |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 435 | AppOpEntry last = entries.get(entries.size()-1); |
| 436 | if (last.getAppEntry() == appEntry) { |
| 437 | boolean lastExe = last.getTime() != 0; |
| 438 | boolean entryExe = opEntry.getTime() != 0; |
| 439 | if (lastExe == entryExe) { |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 440 | if (DEBUG) Log.d(TAG, "Add op " + opEntry.getOp() + " to package " |
| 441 | + pkgOps.getPackageName() + ": append to " + last); |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 442 | last.addOp(opEntry); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 443 | return; |
| 444 | } |
| 445 | } |
| 446 | } |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 447 | AppOpEntry entry = appEntry.getOpSwitch(opEntry.getOp()); |
| 448 | if (entry != null) { |
Dianne Hackborn | 8b0afc7 | 2013-02-01 17:29:15 -0800 | [diff] [blame] | 449 | entry.addOp(opEntry); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 450 | return; |
| 451 | } |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 452 | entry = new AppOpEntry(pkgOps, opEntry, appEntry, switchOrder); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 453 | if (DEBUG) Log.d(TAG, "Add op " + opEntry.getOp() + " to package " |
| 454 | + pkgOps.getPackageName() + ": making new " + entry); |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 455 | entries.add(entry); |
| 456 | } |
| 457 | |
| 458 | public List<AppOpEntry> buildState(OpsTemplate tpl) { |
| 459 | return buildState(tpl, 0, null); |
| 460 | } |
| 461 | |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 462 | private AppEntry getAppEntry(final Context context, final HashMap<String, AppEntry> appEntries, |
| 463 | final String packageName, ApplicationInfo appInfo) { |
| 464 | AppEntry appEntry = appEntries.get(packageName); |
| 465 | if (appEntry == null) { |
| 466 | if (appInfo == null) { |
| 467 | try { |
| 468 | appInfo = mPm.getApplicationInfo(packageName, |
| 469 | PackageManager.GET_DISABLED_COMPONENTS |
| 470 | | PackageManager.GET_UNINSTALLED_PACKAGES); |
| 471 | } catch (PackageManager.NameNotFoundException e) { |
| 472 | Log.w(TAG, "Unable to find info for package " + packageName); |
| 473 | return null; |
| 474 | } |
| 475 | } |
| 476 | appEntry = new AppEntry(this, appInfo); |
| 477 | appEntry.loadLabel(context); |
| 478 | appEntries.put(packageName, appEntry); |
| 479 | } |
| 480 | return appEntry; |
| 481 | } |
| 482 | |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 483 | public List<AppOpEntry> buildState(OpsTemplate tpl, int uid, String packageName) { |
| 484 | final Context context = mContext; |
| 485 | |
| 486 | final HashMap<String, AppEntry> appEntries = new HashMap<String, AppEntry>(); |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 487 | final List<AppOpEntry> entries = new ArrayList<AppOpEntry>(); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 488 | |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 489 | final ArrayList<String> perms = new ArrayList<String>(); |
| 490 | final ArrayList<Integer> permOps = new ArrayList<Integer>(); |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 491 | final int[] opToOrder = new int[AppOpsManager._NUM_OP]; |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 492 | for (int i=0; i<tpl.ops.length; i++) { |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 493 | if (tpl.showPerms[i]) { |
| 494 | String perm = AppOpsManager.opToPermission(tpl.ops[i]); |
| 495 | if (perm != null && !perms.contains(perm)) { |
| 496 | perms.add(perm); |
| 497 | permOps.add(tpl.ops[i]); |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 498 | opToOrder[tpl.ops[i]] = i; |
Dianne Hackborn | 27daaab | 2013-01-31 15:06:57 -0800 | [diff] [blame] | 499 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 500 | } |
| 501 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 502 | |
| 503 | List<AppOpsManager.PackageOps> pkgs; |
| 504 | if (packageName != null) { |
| 505 | pkgs = mAppOps.getOpsForPackage(uid, packageName, tpl.ops); |
| 506 | } else { |
| 507 | pkgs = mAppOps.getPackagesForOps(tpl.ops); |
| 508 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 509 | |
| 510 | if (pkgs != null) { |
| 511 | for (int i=0; i<pkgs.size(); i++) { |
| 512 | AppOpsManager.PackageOps pkgOps = pkgs.get(i); |
| 513 | AppEntry appEntry = getAppEntry(context, appEntries, pkgOps.getPackageName(), null); |
| 514 | if (appEntry == null) { |
| 515 | continue; |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 516 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 517 | for (int j=0; j<pkgOps.getOps().size(); j++) { |
| 518 | AppOpsManager.OpEntry opEntry = pkgOps.getOps().get(j); |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 519 | addOp(entries, pkgOps, appEntry, opEntry, packageName == null, |
| 520 | packageName == null ? 0 : opToOrder[opEntry.getOp()]); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 521 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 522 | } |
| 523 | } |
| 524 | |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 525 | List<PackageInfo> apps; |
| 526 | if (packageName != null) { |
| 527 | apps = new ArrayList<PackageInfo>(); |
| 528 | try { |
| 529 | PackageInfo pi = mPm.getPackageInfo(packageName, PackageManager.GET_PERMISSIONS); |
| 530 | apps.add(pi); |
| 531 | } catch (NameNotFoundException e) { |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 532 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 533 | } else { |
| 534 | String[] permsArray = new String[perms.size()]; |
| 535 | perms.toArray(permsArray); |
| 536 | apps = mPm.getPackagesHoldingPermissions(permsArray, 0); |
| 537 | } |
| 538 | for (int i=0; i<apps.size(); i++) { |
| 539 | PackageInfo appInfo = apps.get(i); |
| 540 | AppEntry appEntry = getAppEntry(context, appEntries, appInfo.packageName, |
| 541 | appInfo.applicationInfo); |
| 542 | if (appEntry == null) { |
| 543 | continue; |
| 544 | } |
| 545 | List<AppOpsManager.OpEntry> dummyOps = null; |
| 546 | AppOpsManager.PackageOps pkgOps = null; |
Daniel Sandler | 7d3a4c1 | 2013-01-31 15:22:51 -0500 | [diff] [blame] | 547 | if (appInfo.requestedPermissions != null) { |
| 548 | for (int j=0; j<appInfo.requestedPermissions.length; j++) { |
| 549 | if (appInfo.requestedPermissionsFlags != null) { |
| 550 | if ((appInfo.requestedPermissionsFlags[j] |
| 551 | & PackageInfo.REQUESTED_PERMISSION_GRANTED) == 0) { |
| 552 | if (DEBUG) Log.d(TAG, "Pkg " + appInfo.packageName + " perm " |
| 553 | + appInfo.requestedPermissions[j] + " not granted; skipping"); |
| 554 | break; |
| 555 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 556 | } |
Daniel Sandler | 7d3a4c1 | 2013-01-31 15:22:51 -0500 | [diff] [blame] | 557 | if (DEBUG) Log.d(TAG, "Pkg " + appInfo.packageName + ": requested perm " |
| 558 | + appInfo.requestedPermissions[j]); |
| 559 | for (int k=0; k<perms.size(); k++) { |
| 560 | if (!perms.get(k).equals(appInfo.requestedPermissions[j])) { |
| 561 | continue; |
| 562 | } |
| 563 | if (DEBUG) Log.d(TAG, "Pkg " + appInfo.packageName + " perm " + perms.get(k) |
| 564 | + " has op " + permOps.get(k) + ": " + appEntry.hasOp(permOps.get(k))); |
| 565 | if (appEntry.hasOp(permOps.get(k))) { |
| 566 | continue; |
| 567 | } |
| 568 | if (dummyOps == null) { |
| 569 | dummyOps = new ArrayList<AppOpsManager.OpEntry>(); |
| 570 | pkgOps = new AppOpsManager.PackageOps( |
| 571 | appInfo.packageName, appInfo.applicationInfo.uid, dummyOps); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 572 | |
Daniel Sandler | 7d3a4c1 | 2013-01-31 15:22:51 -0500 | [diff] [blame] | 573 | } |
| 574 | AppOpsManager.OpEntry opEntry = new AppOpsManager.OpEntry( |
| 575 | permOps.get(k), AppOpsManager.MODE_ALLOWED, 0, 0, 0); |
| 576 | dummyOps.add(opEntry); |
Dianne Hackborn | d69e8dc | 2013-02-07 00:01:10 -0800 | [diff] [blame] | 577 | addOp(entries, pkgOps, appEntry, opEntry, packageName == null, |
| 578 | packageName == null ? 0 : opToOrder[opEntry.getOp()]); |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 579 | } |
Dianne Hackborn | 0dd9902 | 2013-01-24 19:14:26 -0800 | [diff] [blame] | 580 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 581 | } |
| 582 | } |
| 583 | |
| 584 | // Sort the list. |
| 585 | Collections.sort(entries, APP_OP_COMPARATOR); |
| 586 | |
| 587 | // Done! |
| 588 | return entries; |
| 589 | } |
Dianne Hackborn | 18b64f4 | 2013-01-18 10:52:38 -0800 | [diff] [blame] | 590 | } |