全面屏虚拟键适配方案

写之前先吐槽下自己,工作了这么多年,终于能静下心来写博客了

最近公司有个需求,要实现类似于抖音的小视频全屏播放的样式,当虚拟键盘展示的时候,竖屏视频就撑满整个屏幕,当虚拟键盘隐藏的时候,就需要让视频底部距离屏幕底部有个虚拟键高度的黑边,总结起来就是要保持视频的原始比例,效果如下图:

虚拟键显示

虚拟键隐藏

所以,我们要做的很简单,就是监听NavigationBar的显示和隐藏。

方案一:监听一个全屏 View的高度

之前看到一个思路是,使用addOnGlobalLayoutListener监听一个全屏 View 的高度,然后不停的去检测当前是否展示了 NavigationBar,个人不太喜欢这个方案,有兴趣可以自行查找。

方案二:监听数据库System表字段变化

该方案通过监控settings数据库System表中navigationbar_is_min的变化,来判断当前是否显示虚拟键盘。经过测试,部分手机onChange方法并不会触发。

经过多番查证,问题有两个:

1.原来android5.0之后增加了多用户的特性,虚拟键盘的navigationbar_is_min字段从Settings.db的System表格移到了Global表。

2.不同手机品牌使用的注册字段也不一样。

优化后的方法如下:

  private void initDeviceInfo() {
        String brand = Build.BRAND;
        if (brand.equalsIgnoreCase("HUAWEI")) {
            mDeviceInfo = "navigationbar_is_min";
        } else if (brand.equalsIgnoreCase("XIAOMI")) {
            mDeviceInfo = "force_fsg_nav_bar";
        } else if (brand.equalsIgnoreCase("VIVO")) {
            mDeviceInfo = "navigation_gesture_on";
        } else if (brand.equalsIgnoreCase("OPPO")) {
            mDeviceInfo = "navigation_gesture_on";
        } else {
            mDeviceInfo = "navigationbar_is_min";
        }
    }
     /**
     * 注册监听实时监控虚拟键
     */
    private void registerNavigationBarObserver() {
        if (null == mActivity || !checkDeviceHasNavigationBar()) {
            return;
        }

        if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
            mActivity.getContentResolver().registerContentObserver(Settings.System.getUriFor
                    (mDeviceInfo), true, mNavigationBarObserver);
        } else {
            mActivity.getContentResolver().registerContentObserver(Settings.Global.getUriFor
                    (mDeviceInfo), true, mNavigationBarObserver);
        }
    }

  private ContentObserver mNavigationBarObserver = new ContentObserver(new Handler()) {
        @Override
        public void onChange(boolean selfChange) {
            if (null == mActivity) {
                return;
            }
            resetVideoHeightByNavigation(checkNavigation());
        }
    };

当然,我们注册也不是说所有手机都注册,非全面屏的手机不用注册,所以需要判断一下,但是google 官方提供的检测手机是否有NavigationBar 的方法需要在9.0之后才能用(不是很明白为什么设置 NavigationBar颜色的方法早就有了,但是检测的方法要现在才出╮(╯_╰)╭),所以我们只能另辟蹊径了。
我这里是用检测手机是否存在物理按键的方式来反向判断是否存在虚拟键的,因为全面屏的定义就是去除物理按键,替换为虚拟键。经检测,目前在红米6 pro 上检测不准确,尴尬。。。代码如下:

    /**
     * 检查设备是否有虚拟键
     *
     * @return
     */
    public boolean checkDeviceHasNavigationBar() {
        //通过判断设备是否有返回键、菜单键(不是虚拟键,是手机屏幕外的按键)来确定是否有navigation bar
        boolean hasMenuKey = ViewConfiguration.get(mActivity)
                .hasPermanentMenuKey();
        boolean hasBackKey = KeyCharacterMap
                .deviceHasKey(KeyEvent.KEYCODE_BACK);

        if (!hasMenuKey && !hasBackKey) {
            // 做任何你需要做的,这个设备有一个导航栏
            return true;
        }
        return false;
    }

注册完成之后,当触发 onChange 回调时,我们就可以根据当前是否展示了虚拟键盘来做对应的处理了。原理同样是检查settings数据库中字段的变化,但是当你去之前的表中检查时你会发现,在 VIVO 和 OPPO 的手机上永远返回0!!!


呵呵

最后终于查到,是 VIVO 和 OPPO 又移到 Secure 表中!!!什么?你问我那为什么注册的时候能成功?我只能说,不造啊╮(╯_╰)╭


vivo oppo

获取当前虚拟键是否展示的方法:

    /**
     * 是否展示了 navigationbar
     *
     * @return
     */
    private boolean checkNavigation() {
        int navigationBarIsMin = 0;
        if (Build.VERSION.SDK_INT < Build.VERSION_CODES.LOLLIPOP) {
            navigationBarIsMin = Settings.System.getInt(mActivity.getContentResolver(),
                    mDeviceInfo, 0);
        } else {
            if (Build.BRAND.equalsIgnoreCase("VIVO") || Build.BRAND.equalsIgnoreCase("OPPO")) {
                navigationBarIsMin = Settings.Secure.getInt(mActivity.getContentResolver(),
                        mDeviceInfo, 0);
            } else {
                navigationBarIsMin = Settings.Global.getInt(mActivity.getContentResolver(),
                        mDeviceInfo, 0);
            }
        }
        return navigationBarIsMin != 1;
    }

最后附上获取虚拟键高度的方法:

    /**
     * 获取虚拟键的高度
     *
     * @return
     */
    public int getNavigationBarHeight() {
        Resources resources = mActivity.getResources();
        int resourceId = resources.getIdentifier("navigation_bar_height",
                "dimen", "android");
        //获取NavigationBar的高度
        int height = resources.getDimensionPixelSize(resourceId);
        return height;
    }

没错,你猜对了!这个方法在魅族pro6上有问题!明明是一个没有虚拟键的手机,结果人家非的给你返回了一个高度出来!!!就问你牛不牛!!


魅族

总结:以上方法还需更多的验证和完善,而且该方法均是没考虑刘海屏的情况下。别急,我已经看见产品大佬已经拿着需求向我走来了,祝我平安~


我还能写
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1、通过CocoaPods安装项目名称项目信息 AFNetworking网络请求组件 FMDB本地数据库组件 SD...
    阳明AGI阅读 16,019评论 3 119
  • 水果是我们日常生活经常接触,物美价廉而又健康诱人的食品,它多种艳丽的颜色和各种独特的形状,加上丰富的想象力和细致巧...
    齐梅齐阅读 2,437评论 25 53
  • 干涸身体内的太阳总不落山 开始憎恨黑不见底的明天 往事在房间里乱了位置 拉着松弛的眼睑不愿放手 蹒跚的梦被半夜惊醒...
    野马王阅读 396评论 9 13
  • 一直一直我都很迷茫自己是一个怎样的人 自己是以一个怎样的角色在这个社会中生存。。 自己想要什么,能做什么 以后自己...
    Lin小白阅读 582评论 0 0
  • 文、图/江南海北的雪 最近看到一本书《驴行北京》,这是户外徒步的亲身体验之作,是学者行者的跨界力作!也是第一本以驴...
    江南海北的雪阅读 273评论 1 2