1 /* 2 * Copyright (C) 2015 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy 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, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package android.content.om; 18 19 import android.annotation.IntDef; 20 import android.annotation.NonNull; 21 import android.annotation.Nullable; 22 import android.annotation.SystemApi; 23 import android.annotation.UserIdInt; 24 import android.compat.annotation.UnsupportedAppUsage; 25 import android.os.Parcel; 26 import android.os.Parcelable; 27 28 import java.lang.annotation.Retention; 29 import java.lang.annotation.RetentionPolicy; 30 import java.util.Objects; 31 32 /** 33 * Immutable overlay information about a package. All PackageInfos that 34 * represent an overlay package will have a corresponding OverlayInfo. 35 * 36 * @hide 37 */ 38 @SystemApi 39 public final class OverlayInfo implements Parcelable { 40 41 /** @hide */ 42 @IntDef(prefix = "STATE_", value = { 43 STATE_UNKNOWN, 44 STATE_MISSING_TARGET, 45 STATE_NO_IDMAP, 46 STATE_DISABLED, 47 STATE_ENABLED, 48 STATE_ENABLED_STATIC, 49 // @Deprecated STATE_TARGET_IS_BEING_REPLACED, 50 STATE_OVERLAY_IS_BEING_REPLACED, 51 }) 52 /** @hide */ 53 @Retention(RetentionPolicy.SOURCE) 54 public @interface State {} 55 56 /** 57 * An internal state used as the initial state of an overlay. OverlayInfo 58 * objects exposed outside the {@link 59 * com.android.server.om.OverlayManagerService} should never have this 60 * state. 61 * 62 * @hide 63 */ 64 public static final int STATE_UNKNOWN = -1; 65 66 /** 67 * The target package of the overlay is not installed. The overlay cannot be enabled. 68 * 69 * @hide 70 */ 71 public static final int STATE_MISSING_TARGET = 0; 72 73 /** 74 * Creation of idmap file failed (e.g. no matching resources). The overlay 75 * cannot be enabled. 76 * 77 * @hide 78 */ 79 public static final int STATE_NO_IDMAP = 1; 80 81 /** 82 * The overlay is currently disabled. It can be enabled. 83 * 84 * @see IOverlayManager#setEnabled 85 * @hide 86 */ 87 public static final int STATE_DISABLED = 2; 88 89 /** 90 * The overlay is currently enabled. It can be disabled. 91 * 92 * @see IOverlayManager#setEnabled 93 * @hide 94 */ 95 public static final int STATE_ENABLED = 3; 96 97 /** 98 * The target package is currently being upgraded or downgraded; the state 99 * will change once the package installation has finished. 100 * @hide 101 * 102 * @deprecated No longer used. Caused invalid transitions from enabled -> upgrading -> enabled, 103 * where an update is propagated when nothing has changed. Can occur during --dont-kill 104 * installs when code and resources are hot swapped and the Activity should not be relaunched. 105 * In all other cases, the process and therefore Activity is killed, so the state loop is 106 * irrelevant. 107 */ 108 @Deprecated 109 public static final int STATE_TARGET_IS_BEING_REPLACED = 4; 110 111 /** 112 * The overlay package is currently being upgraded or downgraded; the state 113 * will change once the package installation has finished. 114 * @hide 115 */ 116 public static final int STATE_OVERLAY_IS_BEING_REPLACED = 5; 117 118 /** 119 * The overlay package is currently enabled because it is marked as 120 * 'static'. It cannot be disabled but will change state if for instance 121 * its target is uninstalled. 122 * @hide 123 */ 124 public static final int STATE_ENABLED_STATIC = 6; 125 126 /** 127 * Overlay category: theme. 128 * <p> 129 * Change how Android (including the status bar, dialogs, ...) looks. 130 * 131 * @hide 132 */ 133 public static final String CATEGORY_THEME = "android.theme"; 134 135 /** 136 * Package name of the overlay package 137 * 138 * @hide 139 */ 140 public final String packageName; 141 142 /** 143 * Package name of the target package 144 * 145 * @hide 146 */ 147 public final String targetPackageName; 148 149 /** 150 * Name of the target overlayable declaration. 151 * 152 * @hide 153 */ 154 public final String targetOverlayableName; 155 156 /** 157 * Category of the overlay package 158 * 159 * @hide 160 */ 161 public final String category; 162 163 /** 164 * Full path to the base APK for this overlay package 165 * @hide 166 */ 167 public final String baseCodePath; 168 169 /** 170 * The state of this OverlayInfo as defined by the STATE_* constants in this class. 171 * @hide 172 */ 173 @UnsupportedAppUsage 174 public final @State int state; 175 176 /** 177 * User handle for which this overlay applies 178 * @hide 179 */ 180 public final int userId; 181 182 /** 183 * Priority as read from the manifest. Used if isStatic is true. Not 184 * intended to be exposed to 3rd party. 185 * 186 * @hide 187 */ 188 public final int priority; 189 190 /** 191 * isStatic as read from the manifest. If true, the overlay is 192 * unconditionally loaded and cannot be unloaded. Not intended to be 193 * exposed to 3rd party. 194 * 195 * @hide 196 */ 197 public final boolean isStatic; 198 199 /** 200 * Create a new OverlayInfo based on source with an updated state. 201 * 202 * @param source the source OverlayInfo to base the new instance on 203 * @param state the new state for the source OverlayInfo 204 * 205 * @hide 206 */ OverlayInfo(@onNull OverlayInfo source, @State int state)207 public OverlayInfo(@NonNull OverlayInfo source, @State int state) { 208 this(source.packageName, source.targetPackageName, source.targetOverlayableName, 209 source.category, source.baseCodePath, state, source.userId, source.priority, 210 source.isStatic); 211 } 212 213 /** @hide */ OverlayInfo(@onNull String packageName, @NonNull String targetPackageName, @Nullable String targetOverlayableName, @Nullable String category, @NonNull String baseCodePath, int state, int userId, int priority, boolean isStatic)214 public OverlayInfo(@NonNull String packageName, @NonNull String targetPackageName, 215 @Nullable String targetOverlayableName, @Nullable String category, 216 @NonNull String baseCodePath, int state, int userId, 217 int priority, boolean isStatic) { 218 this.packageName = packageName; 219 this.targetPackageName = targetPackageName; 220 this.targetOverlayableName = targetOverlayableName; 221 this.category = category; 222 this.baseCodePath = baseCodePath; 223 this.state = state; 224 this.userId = userId; 225 this.priority = priority; 226 this.isStatic = isStatic; 227 ensureValidState(); 228 } 229 230 /** @hide */ OverlayInfo(Parcel source)231 public OverlayInfo(Parcel source) { 232 packageName = source.readString(); 233 targetPackageName = source.readString(); 234 targetOverlayableName = source.readString(); 235 category = source.readString(); 236 baseCodePath = source.readString(); 237 state = source.readInt(); 238 userId = source.readInt(); 239 priority = source.readInt(); 240 isStatic = source.readBoolean(); 241 ensureValidState(); 242 } 243 244 /** 245 * Returns package name of the current overlay. 246 * @hide 247 */ 248 @SystemApi 249 @NonNull getPackageName()250 public String getPackageName() { 251 return packageName; 252 } 253 254 /** 255 * Returns the target package name of the current overlay. 256 * @hide 257 */ 258 @SystemApi 259 @NonNull getTargetPackageName()260 public String getTargetPackageName() { 261 return targetPackageName; 262 } 263 264 /** 265 * Returns the category of the current overlay. 266 * @hide\ 267 */ 268 @SystemApi 269 @Nullable getCategory()270 public String getCategory() { 271 return category; 272 } 273 274 /** 275 * Returns user handle for which this overlay applies to. 276 * @hide 277 */ 278 @SystemApi 279 @UserIdInt getUserId()280 public int getUserId() { 281 return userId; 282 } 283 284 /** 285 * Returns name of the target overlayable declaration. 286 * @hide 287 */ 288 @SystemApi 289 @Nullable getTargetOverlayableName()290 public String getTargetOverlayableName() { 291 return targetOverlayableName; 292 } 293 ensureValidState()294 private void ensureValidState() { 295 if (packageName == null) { 296 throw new IllegalArgumentException("packageName must not be null"); 297 } 298 if (targetPackageName == null) { 299 throw new IllegalArgumentException("targetPackageName must not be null"); 300 } 301 if (baseCodePath == null) { 302 throw new IllegalArgumentException("baseCodePath must not be null"); 303 } 304 switch (state) { 305 case STATE_UNKNOWN: 306 case STATE_MISSING_TARGET: 307 case STATE_NO_IDMAP: 308 case STATE_DISABLED: 309 case STATE_ENABLED: 310 case STATE_ENABLED_STATIC: 311 case STATE_TARGET_IS_BEING_REPLACED: 312 case STATE_OVERLAY_IS_BEING_REPLACED: 313 break; 314 default: 315 throw new IllegalArgumentException("State " + state + " is not a valid state"); 316 } 317 } 318 319 @Override describeContents()320 public int describeContents() { 321 return 0; 322 } 323 324 @Override writeToParcel(Parcel dest, int flags)325 public void writeToParcel(Parcel dest, int flags) { 326 dest.writeString(packageName); 327 dest.writeString(targetPackageName); 328 dest.writeString(targetOverlayableName); 329 dest.writeString(category); 330 dest.writeString(baseCodePath); 331 dest.writeInt(state); 332 dest.writeInt(userId); 333 dest.writeInt(priority); 334 dest.writeBoolean(isStatic); 335 } 336 337 public static final @android.annotation.NonNull Parcelable.Creator<OverlayInfo> CREATOR = 338 new Parcelable.Creator<OverlayInfo>() { 339 @Override 340 public OverlayInfo createFromParcel(Parcel source) { 341 return new OverlayInfo(source); 342 } 343 344 @Override 345 public OverlayInfo[] newArray(int size) { 346 return new OverlayInfo[size]; 347 } 348 }; 349 350 /** 351 * Return true if this overlay is enabled, i.e. should be used to overlay 352 * the resources in the target package. 353 * 354 * Disabled overlay packages are installed but are currently not in use. 355 * 356 * @return true if the overlay is enabled, else false. 357 * @hide 358 */ 359 @SystemApi isEnabled()360 public boolean isEnabled() { 361 switch (state) { 362 case STATE_ENABLED: 363 case STATE_ENABLED_STATIC: 364 return true; 365 default: 366 return false; 367 } 368 } 369 370 /** 371 * Translate a state to a human readable string. Only intended for 372 * debugging purposes. 373 * 374 * @return a human readable String representing the state. 375 * @hide 376 */ stateToString(@tate int state)377 public static String stateToString(@State int state) { 378 switch (state) { 379 case STATE_UNKNOWN: 380 return "STATE_UNKNOWN"; 381 case STATE_MISSING_TARGET: 382 return "STATE_MISSING_TARGET"; 383 case STATE_NO_IDMAP: 384 return "STATE_NO_IDMAP"; 385 case STATE_DISABLED: 386 return "STATE_DISABLED"; 387 case STATE_ENABLED: 388 return "STATE_ENABLED"; 389 case STATE_ENABLED_STATIC: 390 return "STATE_ENABLED_STATIC"; 391 case STATE_TARGET_IS_BEING_REPLACED: 392 return "STATE_TARGET_IS_BEING_REPLACED"; 393 case STATE_OVERLAY_IS_BEING_REPLACED: 394 return "STATE_OVERLAY_IS_BEING_REPLACED"; 395 default: 396 return "<unknown state>"; 397 } 398 } 399 400 @Override hashCode()401 public int hashCode() { 402 final int prime = 31; 403 int result = 1; 404 result = prime * result + userId; 405 result = prime * result + state; 406 result = prime * result + ((packageName == null) ? 0 : packageName.hashCode()); 407 result = prime * result + ((targetPackageName == null) ? 0 : targetPackageName.hashCode()); 408 result = prime * result + ((targetOverlayableName == null) ? 0 409 : targetOverlayableName.hashCode()); 410 result = prime * result + ((category == null) ? 0 : category.hashCode()); 411 result = prime * result + ((baseCodePath == null) ? 0 : baseCodePath.hashCode()); 412 return result; 413 } 414 415 @Override equals(@ullable Object obj)416 public boolean equals(@Nullable Object obj) { 417 if (this == obj) { 418 return true; 419 } 420 if (obj == null) { 421 return false; 422 } 423 if (getClass() != obj.getClass()) { 424 return false; 425 } 426 OverlayInfo other = (OverlayInfo) obj; 427 if (userId != other.userId) { 428 return false; 429 } 430 if (state != other.state) { 431 return false; 432 } 433 if (!packageName.equals(other.packageName)) { 434 return false; 435 } 436 if (!targetPackageName.equals(other.targetPackageName)) { 437 return false; 438 } 439 if (!Objects.equals(targetOverlayableName, other.targetOverlayableName)) { 440 return false; 441 } 442 if (!Objects.equals(category, other.category)) { 443 return false; 444 } 445 if (!baseCodePath.equals(other.baseCodePath)) { 446 return false; 447 } 448 return true; 449 } 450 451 @NonNull 452 @Override toString()453 public String toString() { 454 return "OverlayInfo { overlay=" + packageName + ", targetPackage=" + targetPackageName 455 + ((targetOverlayableName == null) ? "" 456 : ", targetOverlayable=" + targetOverlayableName) 457 + ", state=" + state + " (" + stateToString(state) + "), userId=" + userId + " }"; 458 } 459 } 460