Android 多窗口模式(分屏模式)
一、多窗口功能概念
1: 分屏模式:
该模式可以在手机上使用, 该模式将屏幕一分为二, 同时显示两个应用界面
2: 画中画模式:
该模式主要用于TV, 在该模式下视频播放的窗口可以一直在最顶端显示。(视频小窗口)
3: Freeform模式:
该模式类似于常见的桌面操作系统, 应用界面的窗口可以自由的拖动和修改大小
二、 分屏模式的配置
1:Manifest 新增属性:
android:resizeableActivity="false"
这个属性可以放置于application和activity的标签下, 作用在对应的区间里面。这个属性表示application或者activity在分屏或者freeform的模式启动, true代表支持, false代表不支持。
android:supportsPictureInPicture= "false"
这个属性可以放置于activity的标签下, 这个属性表示activity是否支持画中画模式, true代表支持, false代表不支持。
2: layout 新增属性:
<activity android:name=".Main2Activity">
<layout android:defaultHeight="500dp"
android:defaultWidth="600dp"
android:gravity="top|end"
android:minHeight="450dp"
android:minWidth="300dp" />
</activity>
在freeform模式下定义的默认高度和宽度defaultWidth&&defaultHeight, freeform模式下的初始Gravity, freeform模式下最小高度和宽度minWidth&&defaultHeight。
3: 多窗口的API:
1: isInMultiWindowMode(): 查询是否处于多窗口模式
2: isInPictureInPictureMode(): 查询是否处于画中画模式
3:Activity.onMultiWindowModeChanged() 多窗口模式变化时进行通知(进入或退出多窗口)
4:onPictureInPictureModeChanged() 画中画模式变化时进行通知(进入或退出画中画模式)
5:enterPictureInPictureMode() 调用这个接口进入画中画模式,如果系统不支持,这个调用无效
6:setLaunchBounds() 在系统已经处于Freeform模式时,可以通过这个参数来控制新启动的Activity大小,如果系统不支持,这个调用无效。
4: 分屏模式的生命周期:
1: 在App由全屏进入分屏的模式(模拟器上测试)
2022-03-17 11:04:02.823 9834-9834/com.example.mutiscreen D/MainActivity: onCreate
2022-03-17 11:04:02.852 9834-9834/com.example.mutiscreen D/MainActivity: onStart
2022-03-17 11:04:02.908 9834-9834/com.example.mutiscreen D/MainActivity: onResume
2022-03-17 11:04:06.569 9834-9834/com.example.mutiscreen D/MainActivity: onPause
2022-03-17 11:04:07.060 9834-9834/com.example.mutiscreen D/MainActivity: onStop
2022-03-17 11:04:07.060 9834-9834/com.example.mutiscreen D/MainActivity: onSaveInstanceState //此时App进入后台, 准备在后台将应用启动为分屏模式。
2022-03-17 11:04:10.797 9834-9834/com.example.mutiscreen D/MainActivity: onDestory
2022-03-17 11:04:10.820 9834-9834/com.example.mutiscreen D/MainActivity: onCreate
2022-03-17 11:04:10.833 9834-9834/com.example.mutiscreen D/MainActivity: onStart
2022-03-17 11:04:10.834 9834-9834/com.example.mutiscreen D/MainActivity: onRestoreInstanceState
2022-03-17 11:04:10.843 9834-9834/com.example.mutiscreen D/MainActivity: onResume
2022-03-17 11:04:10.851 9834-9834/com.example.mutiscreen D/MainActivity: onMultiWindowModeChanged:true //回调onMultiWindowModeChanged
2022-03-17 11:04:10.852 9834-9834/com.example.mutiscreen D/MainActivity: isInMultiWindowMode:true
2022-03-17 11:04:11.022 9834-9834/com.example.mutiscreen D/MainActivity: onPause
很明显, 从上图的生命周期显示, 在后台将应用进入分屏模式的时候, 先执行destroy()的方法, 即在进入分屏之前, activity会被先销毁,再调用onRestoreInstanceState方法恢复数据, 回调onMultiWindowModeChanged()
我们的 APP 进入分屏模式时,在 onMultiWindowModeChanged 方法中如果有对 UI 等的操作,经过之后的自动重建就没有效果了。为了防止这种情况,需要在 AndroidManifest.xml 的 Activity 节点设置以下属性:android:configChanges="screenSize|smallestScreenSize|screenLayout|orientation"
设置了这个属性,在进入分屏模式时,Activity 就不会自动重建了。
2:在APP退出分屏到全屏模式:
2022-03-17 11:16:54.078 9942-9942/com.example.mutiscreen D/MainActivity: onPause
2022-03-17 11:16:54.079 9942-9942/com.example.mutiscreen D/MainActivity: onStop
2022-03-17 11:16:54.079 9942-9942/com.example.mutiscreen D/MainActivity: onSaveInstanceState
2022-03-17 11:16:54.080 9942-9942/com.example.mutiscreen D/MainActivity: onDestory
2022-03-17 11:16:54.114 9942-9942/com.example.mutiscreen D/MainActivity: onCreate
2022-03-17 11:16:54.129 9942-9942/com.example.mutiscreen D/MainActivity: onStart
2022-03-17 11:16:54.130 9942-9942/com.example.mutiscreen D/MainActivity: onRestoreInstanceState
2022-03-17 11:16:54.157 9942-9942/com.example.mutiscreen D/MainActivity: onResume
2022-03-17 11:16:54.241 9942-9942/com.example.mutiscreen D/MainActivity: onMultiWindowModeChanged:false
2022-03-17 11:16:54.243 9942-9942/com.example.mutiscreen D/MainActivity: isInMultiWindowMode:false
从上图显示, 一个配置变化销毁的过程, 然后是一个恢复的过程, 并且回调了onMultiWindowModeChanged方法, 此时的isInMultiWindowMode是false.
3: 在全屏模式下,activity在一个屏区间一次显示,如果希望在多屏模式下,一个应用的不同的activity显示在不同的屏区间,应该如何处理?
Intent intent = new Intent(MainActivity.this, Main2Activity.class);intent.setFlags(Intent.FLAG_ACTIVITY_LAUNCH_ADJACENT | Intent.FLAG_ACTIVITY_NEW_TASK);startActivity(intent);
如果我们的焦点从一个activity中切换到了和它同处于多窗口模式下的另外一个activity
2022-03-17 11:23:49.121 10708-10708/com.example.mutiscreen D/Lion: Main2Activity onCreate
2022-03-17 11:23:49.138 10708-10708/com.example.mutiscreen D/Lion: Main2Activity onStart
2022-03-17 11:23:49.139 10708-10708/com.example.mutiscreen D/Lion: Main2Activity onResume
2022-03-17 11:23:53.216 10708-10708/com.example.mutiscreen D/Lion: MainActivity onPause
如上图显示: 从第1个activity的焦点切换到第二个的时候, 执行了onPause()的方法,即当前的activity会暂时暂停,新获取焦点的activity回调onResume()。同时,我们可以指定新启动的应用activity的大小
Rect bounds = new Rect(500, 300, 100, 0); ActivityOptions options = ActivityOptions.makeBasic();options.setLaunchBounds(bounds); Intent intent = new Intent(MainActivity.this, Main2Activity.class); intent.setFlags(Intent.FLAG_ACTIVITY_LAUNCH_ADJACENT | Intent.FLAG_ACTIVITY_NEW_TASK); ActivityCompat.startActivity(MainActivity.this, intent, options.toBundle());
5 跨Activity对View的拖拽:
Android 4.0 开始支持activity内容的拖拽, 在多窗口的模式下,Android也实现了activity间 实现内容的拖拽。在activity间仅限于内容的拖拽, 对view的拖拽也是没办法实现的 。
DragAndDropPermissions
View.startDragAndDrop()
View.cancelDragAndDrop()
View.updateDragShadow()
Activity.requestDragAndDropPermissions()