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)