关于作者:CSDN内容合伙人、技术专家, 从零开始做日活千万级APP。
专注于分享各领域原创系列文章 ,擅长java后端、移动开发、商业变现、人工智能等,希望大家多多支持。
未经允许不得转载
目录
一、导读二、概览三、分析四、 推荐阅读
一、导读
我们继续总结学习基础知识,温故知新。
本文记录一次bug解决的过程,
Only fullscreen opaque activities can request orientation
二、概览
今天将 targetSdkVersion 的版升级到了29,出现了一些奇怪的报错,日志如下
# main(1)java.lang.IllegalStateExceptionOnly fullscreen opaque activities can request orientation解析原始java.lang.RuntimeException:Unable to start activity ComponentInfo{com.xxx/com.xyz.QActivity}: java.lang.IllegalStateException: Only fullscreen opaque activities can request orientationandroid.app.ActivityThread.performLaunchActivity(ActivityThread.java:3303)......Caused by:java.lang.IllegalStateException:Only fullscreen opaque activities can request orientationandroid.app.Activity.onCreate(Activity.java:10811)androidx.core.app.ComponentActivity.void onCreate(android.os.Bundle)(ComponentActivity.java:861)androidx.activity.ComponentActivity.void onCreate(android.os.Bundle)(ComponentActivity.java:3231)androidx.fragment.app.FragmentActivity.void onCreate(android.os.Bundle)(FragmentActivity.java:2731)com.xyz.QActivity.void onCreate(android.os.Bundle)(QActivity.java:518)android.app.Activity.performCreate(Activity.java:7383)android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:11218)android.app.ActivityThread.performLaunchActivity(ActivityThread.java:31256)android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:31411)android.app.ActivityThread.-wrap12(Unknown Source:0)android.app.ActivityThread$H.handleMessage(ActivityThread.java:19194)android.os.Handler.dispatchMessage(Handler.java:1018)android.os.Looper.loop(Looper.java:166)android.app.ActivityThread.main(ActivityThread.java:75129)java.lang.reflect.Method.invoke(Native Method)com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:2145)com.android.internal.os.ZygoteInit.main(ZygoteInit.java:9211)
这个问题只有8。0的系统才会出现,下面我们一起来分析下原因。
三、分析
报的错误是只有全屏不透明的activity可以设置屏幕方向,那我们先看看activit的设置
<activity android:name="com.xyz.QActivity" android:launchMode="singleTop" android:screenOrientation="portrait" 设置方向 android:theme="@style/TransparentTheme" /> 透明主题
以上我们设置了两个属性,我们再来看看源码,为什么8.0会报错
protected void onCreate(@Nullable Bundle savedInstanceState) { // 在 onCreate() 方法中会进行屏幕配置检查 // 如果固定屏幕方向,且设置了透明色或者悬浮,则会抛出异常 if (getApplicationInfo().targetSdkVersion > O && mActivityInfo.isFixedOrientation()) { if (isTranslucentOrFloating) { throw new IllegalStateException( "Only fullscreen opaque activities can request orientation"); } } //....} //ActivityInfo.java/** * Determines whether the {@link Activity} is considered translucent or floating. * @hide*/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;}
知道了原因,我们来看看怎么解决问题,其思路是这样的,
检查是否透明色或者悬浮如果有的话就设置屏幕不固定说大白话就是进onCreate的时候,判断是透明窗口风格,把屏幕朝向改为未指定类型SCREEN_ORIENTATION_UNSPECIFIED,
因为Activity是透明的,所以其方向依赖于父Activity,所以这个改动对结果不会产生任何影响。
那我们怎么判断是否透明呢?答案是反射ActivityInfo,获取isTranslucentOrFloating的返回值,
private boolean isTranslucentOrFloating() { boolean isTranslucentOrFloating = false; try { int[] styleableRes = (int[]) Class.forName("com.android.internal.R$styleable").getField("Window").get(null); final TypedArray ta = obtainStyledAttributes(styleableRes); Method m = ActivityInfo.class.getMethod("isTranslucentOrFloating", TypedArray.class); m.setAccessible(true); isTranslucentOrFloating = (boolean) m.invoke(null, ta); ta.recycle(); m.setAccessible(false); } catch (Exception e) { } return isTranslucentOrFloating; }
那怎么设置方向呢?,答案还是反射
/** * 反射 **/ private static void fixOrientation(Activity activity) { try { Class activityClass = Activity.class; Field mActivityInfoField = activityClass.getDeclaredField("mActivityInfo"); mActivityInfoField.setAccessible(true); ActivityInfo activityInfo = (ActivityInfo) mActivityInfoField.get(activity); // 获取activity 方向是否是固定的 Method isFixedOrientation = ActivityInfo.class.getMethod("isFixedOrientation"); isFixedOrientation.setAccessible(true); // 获取方法的值,即 源码中隐藏的方法 isFixedOrientation() boolean hasFixedOri = (boolean) isFixedOrientation.invoke(activityInfo); isFixedOrientation.setAccessible(false);// 如果是固定方向 if (hasFixedOri) { // 当然我们也可以把判断过程去掉 //设置屏幕不固定 activityInfo.screenOrientation = ActivityInfo.SCREEN_ORIENTATION_BEHIND; } else{ // do nothing} mActivityInfoField.setAccessible(false); } catch (Exception e) { } try { Class activityClass = Activity.class; Field mActivityInfoField = activityClass.getDeclaredField("mActivityInfo"); mActivityInfoField.setAccessible(true); ActivityInfo activityInfo = (ActivityInfo) mActivityInfoField.get(activity); //设置屏幕不固定 activityInfo.screenOrientation = ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED; } catch (Exception e) { } }下面这个是源码 /** * Returns true if the activity's orientation is fixed. * @hide */ public boolean isFixedOrientation() { return isFixedOrientationLandscape() || isFixedOrientationPortrait() || screenOrientation == SCREEN_ORIENTATION_LOCKED; }
当然,我们还需要进行一次版本的判断
@Overrideprotected void onCreate(@Nullable Bundle savedInstanceState) { //切记:在父类oncreate()方法调用前调用该方法修改配置 if (Build.VERSION.SDK_INT == 26 && isTranslucentOrFloating()) { fixOrientation(this); } super.onCreate(savedInstanceState);}@Overrideprotected void setRequestedOrientation(int requestedOrientation) { // 这里也最好改下,最好是不要让主动进行设置 if (Build.VERSION.SDK_INT == 26 && isTranslucentOrFloating()) { return; }}
另外, Orientation 新增了已个behind属性,表示当前的Activity和栈中在它下方的Activity使用相同的方向。建议这里也改一下。
<activity android:name="com.xyz.QActivity" android:launchMode="singleTop" android:screenOrientation="behind" 设置方向 android:theme="@style/TransparentTheme" /> 透明主题
四、 推荐阅读
Java 专栏
SQL 专栏
数据结构与算法
Android学习专栏
未经允许不得转载