一、iOS实现app_a跳转到app_b
首先设置app_b的URL Types,然后使用在另一个app_a进行跳转就可以了
然后可以在safari浏览器进行测试,地址栏输入bac://回车,如果手机中已经安装了app那么就可以打开了
app_a中执行以下代码以启动app_b
let url = URL.init(string: "com.open.ivygate://")
if UIApplication.shared.canOpenURL(url!) {
UIApplication.shared.openURL(url!)
}else{
print("尚未安装");
}
⚠️注意在iOS9之后需要在app_a的info中配置LSApplicationQueriesSchemes参数,否则会出现错误
-canOpenURL: failed for URL: "abc://" - error: "This app is not allowed to query for scheme abc"
二、Android实现app_a跳转到app_b
Android中实现此功能有很多种方式
1、包名,特定activity拉起
app_b需要在manifest文件对应Activity添加exported属性支持
在app_a中
Intent pIntent = new Intent(Intent.ACTION_MAIN);
ComponentName componentName = new ComponentName("com.example.jizhigang.crm_android_j","com.example.jizhigang.crm_android_j.mine.activity.TongjiActivity");
pIntent.setComponent(componentName);
pIntent.putExtra("","");
startActivity(pIntent);
这种通过包名的方法启动固定页面 会有多开的情况,而且会泄露包名,不建议使用
2、包名,启动页拉起
app_a中执行代码
Intent lIntent = getPackageManager().getLaunchIntentForPackage("com.example.jizhigang.crm_android_j");
if (lIntent != null){
lIntent.putExtra("type","110");
lIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(lIntent);
}else {
Log.d("通过包名启动失败","通过包名启动失败");
}
通过包名的方法启动launch 如果没有启动那么正常启动,如果运行在后台那么直接启动到前台,但是会暴露包名,不建议使用
3、通过url拉起
在app_a中执行
Intent uIntent = new Intent();
uIntent.setData(Uri.parse("csd://pull.csd.demo/cyn?type=110"));
uIntent.putExtra("","");
uIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(uIntent);
在app_b中对应的activity中配置,⚠️在原有intent-filter下方另外添加,不是在原先里面,两个同时存在⚠️
通过url拉起 可以启动到固定页面、不会出现app多开的情况,不会暴露包名,建议使用
4、检查app_b是否安装
/**
* 检查b应用是否安装
* @param packName b应用包名
* @return true已经安装 false没有安装
*/
private boolean checkPackInfo(String packName){
PackageInfo packageInfo = null;
try {
packageInfo = getPackageManager().getPackageInfo(packName,0);
}catch (PackageManager.NameNotFoundException e){
e.printStackTrace();
}
return packageInfo != null;
}
5、判断app_b应用是否在后台运行并直接打开
private Intent getAppOpenIntentByPackageName(Context context,String packageName){
//Activity完整名
String mainAct = null;
//根据包名寻找
PackageManager pkgMag = context.getPackageManager();
Intent intent = new Intent(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_LAUNCHER);
intent.setFlags(Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED|Intent.FLAG_ACTIVITY_NEW_TASK);
List list = pkgMag.queryIntentActivities(intent, PackageManager.GET_ACTIVITIES);
for (int i = 0; i < list.size(); i++) {
ResolveInfo info = list.get(i);
if (info.activityInfo.packageName.equals(packageName)) {
mainAct = info.activityInfo.name;
break;
}
}
if (TextUtils.isEmpty(mainAct)) {
return null;
}
intent.setComponent(new ComponentName(packageName, mainAct));
return intent;
}
private Context getPackageContext(Context context, String packageName) {
Context pkgContext = null;
if (context.getPackageName().equals(packageName)) {
pkgContext = context;
} else {
// 创建第三方应用的上下文环境
try {
pkgContext = context.createPackageContext(packageName,
Context.CONTEXT_IGNORE_SECURITY
| Context.CONTEXT_INCLUDE_CODE);
} catch (PackageManager.NameNotFoundException e) {
e.printStackTrace();
}
}
return pkgContext;
}
/**
* 启动app
* @param context
* @param packageName
* @return
*/
private boolean openPackage(Context context, String packageName) {
Context pkgContext = getPackageContext(context, packageName);
Intent intent = getAppOpenIntentByPackageName(context, packageName);
if (pkgContext != null && intent != null) {
pkgContext.startActivity(intent);
return true;
}
return false;
}
6、具体使用方法
//判断B应用是否安装
if (checkPackInfo("com.example.jizhigang.crm_android_j")) {//已经安装
//直接打开,模拟点击图标方式,不会出现多开,当然Activity的LaunchMode最好设为“singletop”
openPackage(this,"com.example.jizhigang.crm_android_j");
} else {//尚未安装
Toast.makeText(this, "没有安装" + "",Toast.LENGTH_LONG).show();
//TODO 下载操作
}
参考文章
https://www.jianshu.com/p/403d9e067b81
https://blog.csdn.net/dt235201314/article/details/80255143