android: targetSdkVersion升级中Only fullscreen activities can request orientation问题的解决方法

作者:Hanyu Liu 时间:2023-07-26 13:56:08 

前言

这几天同事跟我在升级Android target SDK和build tool版本的时候,碰到了一个非常搞笑的问题,基本可以算作是“坑”了!我在这里跟大家分享一下,希望对您有所帮助。

特征

当我们把targetSdkVersion升级到27,buildToolsVersion和相关的support library升级到27.0.1后,在Android 8.0(API level 26)上,部分Activity出现了一个莫名其妙的crash,异常信息如下:

java.lang.RuntimeException: Unable to start activity ComponentInfo{com.linkedin.android.XXXX.XXXX/com.linkedin.android.XXXX.XXXX.activity.LoginActivity}: java.lang.IllegalStateException: Only fullscreen activities can request orientation

当你在一个“translucent”的Activity里,试图执行setRequestedOrientation的时候就会触发这个异常。例如:

setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);

原因

这个问题貌似已经被广泛的讨论了,最终我们锁定了April 26的一个commit:

Prevent non-fullscreen activities from influencing orientation · aosp-mirror/platform_frameworks_base@3979159

这个改动中抛出异常有关的代码如下:


if (ActivityInfo.isFixedOrientation(requestedOrientation)
&& !fullscreen
&& appInfo.targetSdkVersion >= O) {
throw new IllegalStateException("Only fullscreen activities can request orientation");
}

基本的意思是说,“fullscreen”为否的activity是不能锁定orientation的,否则抛出异常。下面,我们在看一下“fullscreen”如何定义的。


public static boolean isTranslucentOrFloating(TypedArray attributes) {
final boolean isTranslucent = attributes.getBoolean(com.android.internal.R.styleable.Window_windowIsTranslucent, false);
final boolean isSwipeToDismiss = !attributes.hasValue( com.android.internal.R.styleable.Window_windowIsTranslucent)
         && attributes.getBoolean( com.android.internal.R.styleable.Window_windowSwipeToDismiss, false);
final boolean isFloating = attributes.getBoolean(com.android.internal.R.styleable.Window_windowIsFloating, false);
return isFloating || isTranslucent || isSwipeToDismiss;
}

根据上面的定义,如果一个Activity的Style符合下面三个条件之一,认为不是“fullscreen”:

  • “windowIsTranslucent”为true;

  • “windowIsTranslucent”为false,但“windowSwipeToDismiss”为true;

  • “windowIsFloating“为true;

综上可见,这个改动的目的是想阻止非全屏的Activity锁定屏幕旋转,因为当前Activity是透明的,浮动的或可滑动取消的,是否锁屏应该由全屏的Activity决定,而不是并没有全部占据屏幕的Activity决定。

修复

这个问题貌似在最新的SDK中已经修复,我们在API Level 27的设备上已经无法重现,但我们手头的API Level 26的设备还是能重现。而且根据上面的代码来看,如果想保留当前Activity的style,“isTranslucentOrFloating”的逻辑根本没法绕过,所以想绕开很难,目前能想到的大概两个方向:

  • 推迟SDK升级,等官方修复被大多数设备采用;

  • 升级SDK,但重构一下代码,看看已有的非“fullscreen” Activity是不是都是必要的,例如用Fragment实现周围半透明效果,能不能直接把Fragment加入到当前Activity(当然Detach Fragment是有重绘View的开销的)。

来源:https://zhuanlan.zhihu.com/p/32190223

标签:targetsdkversion,升级,android
0
投稿

猜你喜欢

  • .net实现序列化与反序列化实例解析

    2022-08-05 06:42:08
  • c#中@的3种作用

    2022-03-23 17:18:07
  • 带着新人看java虚拟机01(推荐)

    2023-05-28 04:38:16
  • Springboot内置的工具类之CollectionUtils示例讲解

    2023-08-23 20:47:13
  • C#数据类型转换(显式转型、隐式转型、强制转型)

    2021-11-24 13:44:25
  • Java MongoDB数据库连接方法梳理

    2023-11-25 01:01:20
  • Spring底层事务原理解析

    2021-11-25 05:40:29
  • 详解C#使用AD(Active Directory)验证内网用户名密码

    2023-03-03 23:17:47
  • 深入理解Java中的final关键字_动力节点Java学院整理

    2022-05-05 23:18:24
  • C语言文件操作大全

    2022-10-17 06:25:01
  • Android中系统自带锁WalkLock与KeyguardLock用法实例详解

    2023-11-26 01:50:49
  • Flutter 通过Clipper实现各种自定义形状的示例代码

    2023-06-19 14:25:11
  • Java应用/JVM宕机排查步骤操作

    2021-08-10 14:52:10
  • C#实现显示CPU使用率与内存使用率

    2022-09-26 15:03:58
  • android 修改launcher行数和列数的方法

    2021-10-02 14:19:37
  • Android批量插入数据到SQLite数据库的方法

    2022-09-28 06:18:06
  • JAVA swing布局管理器实例解析

    2022-01-11 05:05:15
  • C# DataTable中Compute方法用法集锦(数值/字符串/运算符/表等操作)

    2023-03-01 23:28:38
  • 全面解析java final关键字

    2023-03-16 03:04:55
  • AsyncTask官方文档教程整理

    2023-07-31 20:25:08
  • asp之家 软件编程 m.aspxhome.com