前言

对于双屏异显,就是说当应用在A屏幕打开时,却要在B屏幕上显示,或者应用在A屏幕显示的同时,要在B屏幕显示一些额外的页面。
正常的双屏异显的开发方案是使用Presentation,这是android专门针对副屏显示提供的类,其继承自 Dialog,所以创建和使用也和Dialog的用法接近。
本人刚开始是使用activity去尝试副屏显示,以至于遇到很多问题,所以在此记录一下,给大家排个雷。

注意:要进行副屏的开发,请使用android 8.0或android 9.0进行测试,因为后面的android版本更新了很多东西,可能低版本能通过的功能在上面需要增加额外的设置才有效果。


android模拟器开启副屏显示

这段网上教程很多,我简单说一下。在开发者选项里面,有个“模拟辅助显示设备”,随便选一个分辨率的就能开启一个新的屏幕了。

android 投屏 ipad android 投屏 副屏_xml


android 投屏 ipad android 投屏 副屏_android 投屏 ipad_02


android 投屏 ipad android 投屏 副屏_屏幕显示_03




如何拿到其他屏幕的信息?

使用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"/>

这其中有个悬浮窗的权限,是需要在应用的设置页面手动开启的,你可以使用动态申请的方式跳转到此页面,也可以等应用安装完后,打开此应用的权限设置页,允许悬浮窗功能。

android 投屏 ipad android 投屏 副屏_屏幕显示_04


android 投屏 ipad android 投屏 副屏_android studio_05


然后在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显示:应用不支持在辅助屏上显示

android 投屏 ipad android 投屏 副屏_屏幕显示_06


这就是使用activity进行辅助屏显示的大坑所在。此问题的原因网上几乎很难搜到,好在仍然有一个帖子对此现象提出了解决方案:

应用不支持在辅屏幕显示 在文章中,作者提到了需要在系统目录下创建android.software.activities_on_secondary_displays.xml文件,于是我又对此关键词进行了搜索,最终找到了android开发文档中的一段内容:辅助屏上显示android的要求

android 投屏 ipad android 投屏 副屏_xml_07


可以看到,第一条就讲述了需要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,这将显示你拥有的模拟器列表:

android 投屏 ipad android 投屏 副屏_xml_08


然后使用emulator -avd [模拟器名称] -writable-system来打开指定的模拟器,其中-writable-system参数就是开启系统目录的写功能。

android 投屏 ipad android 投屏 副屏_xml_09


然后就能对/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 投屏 ipad android 投屏 副屏_屏幕显示_10


通过我的亲生经历,可以看出,如果想在原生android系统中,用activity实现双屏异显的路子是走不通的,所以还是使用官方提供的Presentation类来进行多屏显示吧。