如何从framework层面跳过app开屏广告(这里的app是自己写的demo,比较简单,这里仅记录刚刚学到的思路)。
App的组成
这个app demo仅有两个activity,一个是开屏广告页SplashActivity,另一个是内容显示页MainActivity,SplashActivity这个广告页模拟现在的app的开屏广告,右上角有一个倒计时10s按钮,必须等到倒计时完成或者点击按钮才能进入MainActivity。
以下是AndroidManifest.xml组成:
xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools">
<application
android:allowBackup="true"
android:dataExtractionRules="@xml/data_extraction_rules"
android:fullBackupContent="@xml/backup_rules"
android:icon="@mipmap/ic_launcher"
android:label="开屏广告"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/Theme.MyApplication">
<!-- 开屏页 -->
<activity
android:name=".SplashActivity"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<!-- 主界面 -->
<activity
android:name=".MainActivity"
android:exported="true"> <!-- 这里exported="true"很重要 -->
</activity>
</application>
</manifest>
跳开屏广告思路
就是修改framework/base的Activity.java的startActivity方法,因为在桌面点击app启动的时候,就会调用这个方法。这个思路也有个坏处,如果app的广告activity中,进行了一些资源的设置或者后续逻辑的铺垫,就会影响后面打开的activity。
如何确认当前页面的Activity
目的是确定广告页和实际内容页的activity,方便我们进行跳过广告进入实际的页面,命令是adb shell dumpsys activity activities | grep -i "ResumedActivity"。
bash
$ dumpsys activity activities | grep -i "ResumedActivity"
topResumedActivity=ActivityRecord{2cd1b30 u0 com.example.myapplication/.MainActivity t45}
ResumedActivity: ActivityRecord{2cd1b30 u0 com.example.myapplication/.MainActivity t45}
Activity.java的startActivity修改
java
framework/base/core/java/android/app/Activity.java
@Override
public void startActivity(Intent intent, @Nullable Bundle options) {
android.util.Log.d("test1", "startActivity");
ComponentName srcCom =
new ComponentName("com.example.myapplication",
"com.example.myapplication.SplashActivity"); // 广告页
ComponentName destCom =
new ComponentName("com.example.myapplication",
"com.example.myapplication.MainActivity"); // 内容页
if(intent.getComponent() != null ) {
android.util.Log.d("test1", "[startActivity] intent.getComponent() != null == "+intent.getComponent());
if(intent.getComponent().equals(srcCom)) { // 如果当前要跳转的是广告页,重定向到内容页
android.util.Log.d("test1", "[startActivity] intent.getComponent().equals(srcCom)");
intent.setComponent(destCom);
}
}else {
android.util.Log.d("test1", "[startActivity] intent.getComponent() == null");
}
getAutofillClientController().onStartActivity(intent, mIntent);
if (options != null) {
startActivityForResult(intent, -1, options);
} else {
// Note we want to go through this call for compatibility with
// applications that may have overridden the method.
startActivityForResult(intent, -1);
}
}
关于AndroidManifest.xml中exported="true"简单说明
在一开始的AndroidManifest.xml中,MainActivity的exported="false",就导致死活跳转不了,而且连app都打不开了,直接弹窗说"未安装该应用"。log报错如下:
log
java.lang.SecurityException: Permission Denial: starting Intent ...
cmp=com.example.myapplication/.MainActivity
... not exported from uid 10163
原因是Launcher 想启动 MainActivity,但这个 Activity exported=false,系统禁止跨进程启动。
系统规则是:
| exported | 含义 |
|---|---|
| true | 允许其他应用启动 |
| false | 只能本应用启动 |
因为这里是自己的app demo,所以exported属性咱们自己可以改,真实的app肯定改不了,所以这里提供的仅仅是一个思路。