前言
对于双屏异显,就是说当应用在A屏幕打开时,却要在B屏幕上显示,或者应用在A屏幕显示的同时,要在B屏幕显示一些额外的页面。
正常的双屏异显的开发方案是使用Presentation
,这是android专门针对副屏显示提供的类,其继承自 Dialog
,所以创建和使用也和Dialog
的用法接近。
本人刚开始是使用activity去尝试副屏显示,以至于遇到很多问题,所以在此记录一下,给大家排个雷。
注意:要进行副屏的开发,请使用android 8.0或android 9.0进行测试,因为后面的android版本更新了很多东西,可能低版本能通过的功能在上面需要增加额外的设置才有效果。
android模拟器开启副屏显示
这段网上教程很多,我简单说一下。在开发者选项里面,有个“模拟辅助显示设备”,随便选一个分辨率的就能开启一个新的屏幕了。
如何拿到其他屏幕的信息?
使用android提供的接口:DisplayManager
,通过简单的几行代码,就可以拿到所有屏幕的信息。
DisplayManager displayManager = (DisplayManager) getSystemService(Context.DISPLAY_SERVICE);
Display[] displays = displayManager.getDisplays();
Log.d(TAG, "onCreate: displays length: " + displays.length);
增加权限和配置
AndroidManifest.xml
中增加如下权限:
<uses-permission android:name= "android.permission.SYSTEM_ALERT_WINDOW" />
<uses-permission android:name= "android.permission.SYSTEM_OVERLAY_WINDOW"/>
这其中有个悬浮窗的权限,是需要在应用的设置页面手动开启的,你可以使用动态申请的方式跳转到此页面,也可以等应用安装完后,打开此应用的权限设置页,允许悬浮窗功能。
然后在AndroidManifest.xml
中的activity中还需要增加android:resizeableActivity="true"
<activity
android:name=".SecondActivity"
android:exported="true"
android:launchMode="singleTask"
android:resizeableActivity="true" />
使用Activity在副屏展示一个画面
先展示一下如何在副屏打开一个activity:
DisplayManager displayManager = (DisplayManager) getSystemService(Context.DISPLAY_SERVICE);
Display[] displays = displayManager.getDisplays();
Log.d(TAG, "onCreate: displays length: " + displays.length);
//在副屏打开一个activity
if (displays.length > 1 && Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
Intent intent = new Intent();
intent.setClassName(getPackageName(), getPackageName() + ".SecondActivity");
intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_MULTIPLE_TASK);
ActivityOptions options = ActivityOptions.makeBasic();
options.setLaunchDisplayId(displays[1].getDisplayId());
startActivity(intent, options.toBundle());
}
上面代码中options.setLaunchDisplayId()
这个函数控制了新的activity在哪个屏幕显示(此api适用于8.0+),然后将此参数传入到startActivity()
中,就会在新屏幕打开activity了。当然,还有一个要注意的是:你要将intent的flag设置成FLAG_ACTIVITY_NEW_TASK
,因为副屏当前没有你的app的显示栈,所以你想要在副屏也创建一个activity,就需要新开一个显示栈。
应用不支持在辅助屏上显示
下面就是问题的关键了。如果你前面都顺利的话,那么你的activity依旧会无法在模拟器的副屏中显示,系统会弹出一个Toast显示:应用不支持在辅助屏上显示。
这就是使用activity进行辅助屏显示的大坑所在。此问题的原因网上几乎很难搜到,好在仍然有一个帖子对此现象提出了解决方案:
应用不支持在辅屏幕显示 在文章中,作者提到了需要在系统目录下创建android.software.activities_on_secondary_displays.xml
文件,于是我又对此关键词进行了搜索,最终找到了android开发文档中的一段内容:辅助屏上显示android的要求
可以看到,第一条就讲述了需要android.software.activities_on_secondary_displays.xml
这个文件标志存在,才能允许副屏中打开一个activity(同时,通过这段内容,我们也可以看出,想要在辅助屏中显示activity需要满足很多条件,因此当你们看到这里的时候就应该考虑放弃此方案了)
创建activities_on_secondary_displays.xml
首先,我们的模拟器默认是没有root权限的,即使使用adb root
或者使用su
命令也是无法获取root的,所以不能直接在system的permissions文件夹中创建新文件。好在android给我们提供了可行的方案。
找到Sdk下的emulator
文件夹(默认路径为:C:\Users\[你的用户名]\AppData\Local\Android\Sdk\emulator
),在emulator
文件夹中打开cmd,运行emulator -list-avds
,这将显示你拥有的模拟器列表:
然后使用emulator -avd [模拟器名称] -writable-system
来打开指定的模拟器,其中-writable-system
参数就是开启系统目录的写功能。
然后就能对/system/etc/permissions/
文件夹中的内容进行改写了。
结尾
通过创建activities_on_secondary_displays.xml
,我重新尝试在辅助屏创建activity,但依旧显示“应用不支持在辅助屏显示”的Toast,到这里我已无法解决。
但是还有一个方法可以验证我们创建的xml文件是否有效,在adb shell中使用:am start -W -n com.example.xxx/.SecondActivity --display 1
命令可以在指定屏幕打开我们的activity。
通过我的亲生经历,可以看出,如果想在原生android系统中,用activity实现双屏异显的路子是走不通的,所以还是使用官方提供的Presentation
类来进行多屏显示吧。