一、前言
某海外应用A有需求如下:在A应用首页,提供B应用的入口。点击入口时,判断B应用是否已安装,未安装则跳转Google商店安装,已安装则直接进入B应用。当退回A应用时需回传一个参数。
A应用和B应用交互时序图如下:
根据需求和时序图,本文从如下三个方面展开来讲:A应用进入B应用、B应用退回A应用、A应用处理B应用退出。
二、A应用进入B应用方案
点击A应用入口时,判断B应用是否已安装,未安装则跳转Google商店安装,已安装则直接进入B应用;
2.1 判断应用是否已安装
/**
* 检测某个应用是否已安装
*
* @param context 上下文
* @param pkgName 包名
* @return 应用是否已安装
*/
public static boolean isAppInstalled(Context context, String pkgName) {
if (context == null || TextUtils.isEmpty(pkgName)) {
return false;
} else {
try {
return context.getPackageManager().getApplicationInfo(pkgName, 0).enabled;
} catch (PackageManager.NameNotFoundException var3) {
return false;
}
}
}
2.2 跳转 Google 商店安装 APK
/**
* 跳转 Google 商店安装 APK
*
* @param context 上下文
*/
public static void gotoGooglePlay(Context context) {
if (context == null) {
return;
}
try {
Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse("market://details?id=" + context.getPackageName()));
intent.setPackage("com.android.vending");
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);
} catch (Exception e) {
e.printStackTrace();
}
}
2.3 进入B应用
/**
* 进入B应用
*
* @param context 上下文
*/
public static void enterGameApp(Context context) {
if (context == null) {
return;
}
try {
Intent intent = new Intent("action_name_B");
intent.setPackage("package_name_B");
intent.putExtra("extra_tag_B", "value");
context.startActivity(intent);
} catch (Exception e) {
e.printStackTrace();
}
}
三、B应用退回A应用方案
注:参考Android应用间切换的机制,B应用应该是先主动调用Intent退回到A应用,然后销毁自身应用。
3.1 代码参考
Intent intent = new Intent("action_name_A");
intent.setPackage("package_name_A");
intent.putExtra("extra_tag_A", "value");
startActivity(intent)
3.2 manifest声明使用权限
<uses-permission android:name="app.permission.OPEN_MAIN_PAGE" />
四、A应用处理B应用退出方案
4.1 manifest 设计
对于不同应用间的数据传递需求,首先考虑到Intent携带参数的方式。从B应用退出到A应用时,A应用的返回处理逻辑放在主页是比较好的(和直接退出B应用返回上一个应用的UI交互是一致的)。
所以直接更改manifest的intent-filter如下:
<activity
android:name=".main.activity.MainActivity"
android:exported="true"
android:launchMode="singleTask"
android:theme="@style/ActivityTheme">
<intent-filter>
<action android:name="action_name_A" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</activity>
原来manifest中是这样的:
<activity
android:name=".main.activity.MainActivity"
android:launchMode="singleTask"
android:theme="@style/ActivityTheme" />
但考虑到对Android的组件增加了intent-filter或显示指定exported为true后,会出现安全问题,于是增加了permission控制,在manifest声明了如下permission:
<permission android:name="app.permission.OPEN_MAIN_PAGE" />
同时需要在activity声明中增加permission控制:
android:permission="app.permission.OPEN_MAIN_PAGE"
最终版manifest的声明如下:
<permission android:name="app.permission.OPEN_MAIN_PAGE" />
<application android:allowBackup="false">
<activity
android:name=".main.activity.MainActivity"
android:exported="true"
android:launchMode="singleTask"
android:permission="app.permission.OPEN_MAIN_PAGE"
android:theme="@style/ActivityTheme">
<intent-filter>
<action android:name="action_name_A" />
<category android:name="android.intent.category.DEFAULT" />
</intent-filter>
</activity>
</application>
4.2 代码设计
退出B应用时,需要使用Intent通过extra传递数据给A应用。A应用此处需要考虑的问题有:
- 在Activity的哪个回调方法中获取Intent的extra值?
- Activity在后台被销毁了,这种情况是否考虑?
- 直接退出B应用,是不需要传递数据给A应用的,是否考虑此种情况?
下面一起来分析一下:
首先,我们知道重复打开一个,启动模式为SingonTask的Activity,声明周期回调顺序如下:onNewIntent、onRestart、onStart、onResume,那么我们在onNewIntent回调中处理数据传递较好,接着讨论第二个问题;
其次,如果Activity在后台被销毁了,就不会走onNewIntent,会重新走onCreat、onStart、onResume,所以需要在onCreate做一个保障措施;
最后,因为正常退出B应用不需要传递数据(比如从Launcher启动B应用),但可能存在任务栈中A应用在B应用的下一级,那么退出B应用还是会回到A应用,此时不会调用onNewIntent方法,会调用onRestart、onStart、onResume。(此处主要想说明在onResume中去处理数据不太合理)。
所以,最后在A应用中处理B应用退出的代码如下:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mPresenter.handleIntent(getIntent());
}
@Override
protected void onNewIntent(Intent intent) {
super.onNewIntent(intent);
setIntent(intent);
mPresenter.handleIntent(intent);
}
@Override
public void handleIntent(Intent intent) {
String exerciseTime = intent.getStringExtra("EXTRA_ACTION");
Log.i(TAG, "handleIntent: exerciseTime = " + exerciseTime);
}
五、小结
走一段令人留恋的路,做一个不负自己的人。