1、簡介
Android OverlayManagerService(OMS)是一個有意思的模塊,可以在運行時動態替換res,如圖片。簡單來說就是一個普通apk有若干res,另一個相關的overlay apk有相同的res,關閉overlay apk時,普通apk運行時加載的是自己本身的res,而打開overlay apk時,普通apk運行時加載的則是overlay apk中的res。
2、用法舉例
普通apk爲fruit.apk,package爲com.example.fruit,res爲/res/drawable/fruit.jpeg,這個圖片可以在xml或java中使用。目錄結構大致如下:
├── AndroidManifest.xml
├── Android.mk
├── java
│ └── com
│ └── example
│ └── fruit
│ ├── App.java
│ └── MainActivity.java
└── res
├── drawable
│ ├── fruit.jpeg
│ └── ic_launcher_background.xml
├── drawable-v24
│ └── ic_launcher_foreground.xml
├── layout
│ └── activity_main.xml
├── mipmap-anydpi-v26
│ ├── ic_launcher_round.xml
│ └── ic_launcher.xml
├── mipmap-hdpi
│ ├── ic_launcher.png
│ └── ic_launcher_round.png
├── mipmap-mdpi
│ ├── ic_launcher.png
│ └── ic_launcher_round.png
├── mipmap-xhdpi
│ ├── ic_launcher.png
│ └── ic_launcher_round.png
├── mipmap-xxhdpi
│ ├── ic_launcher.png
│ └── ic_launcher_round.png
├── mipmap-xxxhdpi
│ ├── ic_launcher.png
│ └── ic_launcher_round.png
└── values
├── colors.xml
├── strings.xml
└── styles.xml
Androd.mk使用LOCAL_CERTIFICATE := platform進行簽名,LOCAL_AAPT_FLAGS := --auto-add-overlay的目的是支持overlay apk的資源可以和普通apk的資源不同,代碼如下:
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_SRC_FILES := $(call all-java-files-under, java)
LOCAL_RESOURCE_DIR := $(LOCAL_PATH)/res
LOCAL_PACKAGE_NAME := myapp
LOCAL_AAPT_FLAGS := --auto-add-overlay
LOCAL_CERTIFICATE := platform
include $(BUILD_PACKAGE)
AndroidManifest.xm通過android:sharedUserId="android.uid.system"使用系統權限,並添加overlay相應的權限,代碼如下:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.fruit"
android:sharedUserId="android.uid.system">
<uses-permission android:name="android.permission.MODIFY_THEME_OVERLAY" />
<uses-permission android:name="android.permission.CHANGE_OVERLAY_PACKAGES" />
<uses-permission android:name="android.permission.INTERACT_ACROSS_USERS" />
<uses-permission android:name="android.permission.INTERACT_ACROSS_USERS_FULL" />
<application
android:name=".App"
android:allowBackup="true"
android:hardwareAccelerated="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
然後是對應的overlay apk,即fruitOverlay.apk,package爲com.example.fruit.overlay,有同樣的res即res/drawable/fruit.jpeg。目錄結構如下:
├── AndroidManifest.xml
├── Android.mk
└── res
└── drawable
└── fruit.jpeg
Android.mk中LOCAL_RRO_THEME = fruit指定fruitOverlay apk編譯後在vendor/overlay/fruit目錄下,include $(BUILD_RRO_PACKAGE)表示編譯overlay apk,RRO即Runtime Resource Overlay,對應的還有個SRO,表示Static Resource Overlay,SRO發生在編譯時,這裏不作介紹。
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_RESOURCE_DIR := $(LOCAL_PATH)/res
LOCAL_PACKAGE_NAME := fruitOverlay
LOCAL_RRO_THEME = fruit
include $(BUILD_RRO_PACKAGE)
AndroidManifest.xml的overlay標籤是個重要的內容,targetPackage表示作用的普通apk的package,priority表示優先級,isStatic爲true時表示overlay apk的狀態不可動態切換,代碼如下:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.fruit.overlay "
android:versionCode="1"
android:versionName="1.0">
<overlay android:targetPackage="com.example.fruit"
android:priority="1"
android:isStatic="false" />
</manifest>
最後,可以進行測試了,可以在命令行進行測試,也可以使用OverlayManagerService的接口。命令行“cmd overlay enable com.example.fruit.overlay”表示打開overlay, “cmd overlay disable com.example.fruit.overlay”表示關閉overlay,其它命令可通過“cmd overlay help”進行查看。OverlayManagerService的接口可參照“IOverlayManger.aidl”。
3、相關代碼
frameworks/base/core/java/android/content/om/
├── IOverlayManager.aidl
├── OverlayInfo.aidl
└── OverlayInfo.java
frameworks/base/services/core/java/com/android/server/om
├── IdmapManager.java
├── OverlayManagerServiceImpl.java
├── OverlayManagerService.java
├── OverlayManagerSettings.java
└── OverlayManagerShellCommand.java
IOverlayManager:aidl,提供了管理overlay package的接口。
OverlayInfo:封裝了ovelay package相關的所有信息。
IdmapManager:overlay package資源管理。
OverlayManagerService/OverlayManagerServiceImpl:OMS管理、實現類。
OverlayManagerSettings:專門用於設置overlay 的各個狀態。
OverlayManagerShellCommand:支持overlay命令行。
4、啓動流程
OMS啓動流程如下。
5、使用一個overlay
前面介紹了使用overlay的方法,overlay狀態發生變化時,對應的package的Activity會重啓,下面介紹下enable一個overlay的大致流程。