1、概述
一些场合,要求界面是全屏的,然而在全屏界面下,弹出dialog 又会导致虚拟按键栏重新弹出来,也是比较难受的。(而且查了非常多方法都是不能完美的解决这个问题,要么是压根不能用,要么是会闪一下虚拟栏再隐藏,或者隐藏了再显示回来)
2、解决方式
重载dialog 的 show() 方法。设置为焦点失能,就不会弹出被隐藏过的 虚拟栏了
@Override public void show() { // Set the dialog to not focusable. getWindow().setFlags(WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE, WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE); // Show the dialog with NavBar hidden. super.show(); }
3、补充
设置全屏界面的方式
<style name="FullScreenTheme" parent="Theme.AppCompat.Light.NoActionBar"> <!--需要加上 android: 这个是命名空间, 自定义属性不需要--> <item name="android:windowFullscreen">true</item> <item name="android:windowContentOverlay">@null</item> </style>
4、不足
上文的解决方式确实可以让原本是全屏的界面继续保持全屏,然而实际使用中却遇到了问题(dialog确实是把虚拟按键隐藏了):
- 会有红色报错,提示
- dialog直接变成了点击outside区域不能diss dialog
- dialog变成是可穿透的,点击背景区域下的按钮等元素事件穿透了(大问题)
基于这些问题,其实该方案是不对的
5、正确姿势
经过使用测试,比较正确的使用方式是(工具类在 文章末尾)
5.1、先在activity 里做处理
- activity 设置上文的 FullScreenTheme 主题
- activity 的onCreate 方法里设置 显示navigation 时再隐藏
override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) NavigationBarUtil.hideNavigationBar(window) setContentView(R.layout.activity_main) }
5.2、普通Dialog 处理
fun overHeart(context: Context) { val overHeatDialogBuilder = AlertDialog.Builder(context, android.R.style.Theme_Material_Light_Dialog) overHeatDialogBuilder.setTitle(R.string.operation_result) overHeatDialogBuilder.setMessage(context.getString(R.string.overTemp)) overHeatDialogBuilder.setPositiveButton(context.getString(R.string.dialog_comfirm), { dialogInterface, i -> }) val dialog = overHeatDialogBuilder.create() //失能焦点 NavigationBarUtil.focusNotAle(dialog!!.window) dialog.show() //显示虚拟栏的时候 隐藏 NavigationBarUtil.hideNavigationBar(dialog!!.window) //再清理失能焦点 NavigationBarUtil.clearFocusNotAle(dialog!!.window) }
5.3、自定义 dialog 处理:
override fun show() { NavigationBarUtil.focusNotAle(window) super.show() NavigationBarUtil.hideNavigationBar(window) NavigationBarUtil.clearFocusNotAle(window) }
6、工具类
工具类用java写了,上文代码用了 kotlin
/** * Description:控制虚拟栏 * Created by leon on 2018/11/28. */ public class NavigationBarUtil { /** * 隐藏虚拟栏 ,显示的时候再隐藏掉 * @param window */ static public void hideNavigationBar(Window window) { window.getDecorView().setSystemUiVisibility(View.SYSTEM_UI_FLAG_HIDE_NAVIGATION); window.getDecorView().setOnSystemUiVisibilityChangeListener(new View.OnSystemUiVisibilityChangeListener() { @Override public void onSystemUiVisibilityChange(int visibility) { int uiOptions = View.SYSTEM_UI_FLAG_LAYOUT_STABLE | //布局位于状态栏下方 View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION | //全屏 View.SYSTEM_UI_FLAG_FULLSCREEN | //隐藏导航栏 View.SYSTEM_UI_FLAG_HIDE_NAVIGATION | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN; if (Build.VERSION.SDK_INT >= 19) { uiOptions |= View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY; } else { uiOptions |= View.SYSTEM_UI_FLAG_LOW_PROFILE; } window.getDecorView().setSystemUiVisibility(uiOptions); } }); } /** * dialog 需要全屏的时候用,和clearFocusNotAle() 成对出现 * 在show 前调用 focusNotAle show后调用clearFocusNotAle * @param window */ static public void focusNotAle(Window window) { window.setFlags(WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE, WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE); } /** * dialog 需要全屏的时候用,focusNotAle() 成对出现 * 在show 前调用 focusNotAle show后调用clearFocusNotAle * @param window */ static public void clearFocusNotAle(Window window) { window.clearFlags(WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE); } }