美文网首页
全面屏虚拟键适配方案

全面屏虚拟键适配方案

作者: maplesnow | 来源:发表于2018-11-14 09:47 被阅读226次

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

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

虚拟键显示
虚拟键隐藏

所以,我们要做的很简单,就是监听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上有问题!明明是一个没有虚拟键的手机,结果人家非的给你返回了一个高度出来!!!就问你牛不牛!!


魅族

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


我还能写

相关文章

  • 全面屏虚拟键适配方案

    写之前先吐槽下自己,工作了这么多年,终于能静下心来写博客了 最近公司有个需求,要实现类似于抖音的小视频全屏播放的样...

  • iOS 关于全面屏适配的方案及UI在不同尺寸下适配方案

    iOS 关于全面屏适配的方案及UI在不同尺寸下适配方案 iOS 关于全面屏适配的方案及UI在不同尺寸下适配方案

  • 2018-11-29

    iOS 关于全面屏适配的方案及UI在不同尺寸下适配方案 - 掘金

  • Android全面屏适配

    Android全面屏适配 方案1: AndroidManifest.xml 文件添加属性: 应用适配建议采用me...

  • 安卓适配

    ------刘海屏------屏幕适配 | Android 刘海屏适配总结Android刘海屏适配方案Androi...

  • Android--待总结:全面屏适配方案

    [全面屏适配方案 ]https://mp.weixin.qq.com/s/HhLaWrub-QIvY-9cElo7XQ

  • Android虚拟按键处理

    随着全面屏的出现,Android厂商为了更高的屏占屏,纷纷抛弃实体按键的设计,转而采用虚拟按键的方案。由于Andr...

  • Android适配之版本适配

    这篇文章用来记录学习和开发时遇到的版本适配问题,持续更新 全面屏、刘海屏的适配:Android 9 支持最新的全面...

  • 全面屏适配

    简单记录一下,预备知识来源:安卓屏幕完美适配方案——独家秘笈 先来点预备知识 屏幕尺寸(in) 手机对角线的物理尺...

  • 全面屏适配

    官方方法 只需要在application中添加

网友评论

      本文标题:全面屏虚拟键适配方案

      本文链接:https://www.haomeiwen.com/subject/haddfqtx.html