android的ActivityManager会在logcat中打印出当前正在显示的app的包名以及类名,注,这是由ActivityManager打印,使用的日志TAG就是ActivityManager,所以我们在过虑信息时不能以自己程序的进程进行过滤,使用ActivityManager作为TAG过虑即可,当然ActivityManager还会显示其它信息,还可以使用Displayed来进行过虑,这样显示的信息就更少一些了,使用新版Logcat示例如下:
可以看到,显示的信息为:
Displayed org.appspot.loginactivity/.MainActivity: +253ms
Displayed说明了当前正在显示的应用,当前显示的应用包名为org.appspot.loginactivity,类名为.MainActivity,这是一个相对路径的类名,完整路径为org.appspot.loginactivity.MainActivity,这个界面启动的时间为:253ms
有了这些信息,我们就可以使用Intent来启动这个app了:
val intent = Intent()
intent.setClassName("org.appspot.loginactivity", "org.appspot.loginactivity.MainActivity")
intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK
startActivity(intent)
注:这里的类名必须要使用完整路径的类名org.appspot.loginactivity.MainActivity,不能使用.MainActivity这样的相对路径。
这有什么用呢?比如公司要我做一个自定义Launcher,只能显示公司的app和一些必要的系统自带app,自定义Launcher显示了需要的app,当点击app图标时,就需要打开该app,此动作由openApp函数实现,如下:
/** 打开指定包名的app */
fun openApp(packageName: String) {
val launcherActivityName = getLauncherActivityNameByPackageName(packageName)
if (launcherActivityName.isNullOrBlank()) {
val text = "无法启动应用:$packageName,因为获取不到该应用的启动类"
Toast.makeText(ContextHolder.getContext(), text, Toast.LENGTH_SHORT).apply { setGravity(Gravity.CENTER, 0, 0) }.show()
return
}
val intent = Intent(Intent.ACTION_MAIN).apply {
flags = Intent.FLAG_ACTIVITY_NEW_TASK
component = ComponentName(packageName, launcherActivityName)
}
ContextHolder.getContext().startActivity(intent)
}
private fun getLauncherActivityNameByPackageName(packageName: String): String? {
val resolveIntent = Intent(Intent.ACTION_MAIN).apply {
addCategory(Intent.CATEGORY_LAUNCHER) // 无图标的启动类没有这个Category
setPackage(packageName)
}
val resolveInfoList: List<ResolveInfo> = ContextHolder.getContext().packageManager.queryIntentActivities(resolveIntent, 0)
val iterator = resolveInfoList.iterator()
return if (iterator.hasNext()) iterator.next().activityInfo?.name else null
}
如上代码,只需要给openApp(packageName: String)函数传入一个应用的包名,即可实现打开该包名对应的app,但是有一个应用我们无法打开,就是“下载”app,这是一个系统应用,比如我们在浏览器下载了一个app,在“下载”app中可以列出浏览器中下载的文件,“下载”app如下:
由于通过前面的openApp函数无法打开这个应用,所以此时我们就可以通过查看该应用的包名和类名来启动它,它的启动信息如下:
Displayed com.android.documentsui/.files.FilesActivity
启动该app的代码如下:
val intent = Intent()
intent.setClassName("com.android.documentsui", "com.android.documentsui.files.FilesActivity")
intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK
startActivity(intent)