美文网首页自定义控件
BottomNavigationView的属性设置

BottomNavigationView的属性设置

作者: 弥勒佛_1959 | 来源:发表于2019-04-01 10:52 被阅读38次

    备注:BottomNavigationView中5个菜单已经是极限了,再多它就崩溃了。

    个性化

    更换标题(title)

    更换menu item中title即可。

    更换图标(Icon)

    更换图标在介绍menu的时候也说过了,AS可以帮你生成Verctor的图标,理论上各个场景的图标,都有提供。但是,注意这里的但是!AS自动帮我们生成的是VectorDrawable,本身没有什么问题,相对png来说体积小,可是5.0以下的设备不支持,不支持...

    目前来说,5.0以下的市场还不能丢,所以我们就老老实实的换成png吧。

    可是,实现起来,却是这样的...

    Bottom Navigation 图标错误

    一脸懵逼.png

    虽然不知道为什么,但是可以肯定的是一定是自己的姿势不对,于是赶紧翻源码,不翻不知道,一翻更懵逼,就这个BottomNavigationView原来实现这么复杂,但是还是硬着头皮看下去。下面来理一下给icon着色的步骤。

    首先我们没有给icon设置tintList的时候,BottomNavigationView会生成一个默认的ColorStateList:

    privateColorStateListcreateDefaultColorStateList(intbaseColorThemeAttr){finalTypedValue value =newTypedValue();if(!getContext().getTheme().resolveAttribute(baseColorThemeAttr, value,true)) {returnnull;        }        ColorStateList baseColor = AppCompatResources.getColorStateList(                getContext(), value.resourceId);if(!getContext().getTheme().resolveAttribute(                android.support.v7.appcompat.R.attr.colorPrimary, value,true)) {returnnull;        }intcolorPrimary = value.data;intdefaultColor = baseColor.getDefaultColor();returnnewColorStateList(newint[][]{                DISABLED_STATE_SET,                CHECKED_STATE_SET,                EMPTY_STATE_SET        },newint[]{                baseColor.getColorForState(DISABLED_STATE_SET, defaultColor),                colorPrimary,                defaultColor        });    }

    可以看到,选中状态是使用的colorPrimary而未选中状态则使用的默认颜色,这个默认颜色可以通过属性android.R.attr.textColorSecondary去源码中查看。发现跟实际的效果表现一致.

    那么问题来了,图片呢,怎么会变成一个小色块呢,接着往下看,图标颜色设置的代码,进入BottomNavigationMenuView.setIconTintList,又调用了BottomNavigationItemView.setIconTintList,再继续又调用了DrawableCompat.setTintList,最终是调用了Drawable.setTintList,而具体的实现则在BitmapDrawble.setColorFilter。

    这里有一篇关于setColorFilter的介绍,可以参考下。这样就不难解释为什么是一个小色块,吐槽下我们UED的切图,居然图标周边不是透明的,而是白色,白色...

    所以,划重点了!!!图标周边要是透明的!图标周边要是透明的!!图标周边要是透明的!!!

    周边透明.png

    非透明的区域,都会被着色,着色分选中[android.R.attr.state_checked]和未选中[-android.R.attr.state_checked]对,没看错,就是-,减号,负号

    现在换成周边透明的图标(这次更彻底,除了线条都是透明的),效果图如下:

    有效图标

    作者:seph_von

    链接:https://www.jianshu.com/p/7b2d842267ab

    还有一种方式就是直接由用户设置图标,不利用系统的着色方案

    <?xml version="1.0" encoding="utf-8"?>

    <Selector xmlns:android="http://schemas.android.com/apk/res/android">

    <item android:state_checked="false" android:drawable="@mipmap/icon_nav_02_de"/>

    <item android:state_checked="true" android:drawable="@mipmap/icon_nav_02_hi"/>

    <.selector>

    然后在BottomNavigationView中设置

    app:itemIconTint="@null"

    更换文字颜色

    在drawable文件加载创建文件 drawable_colcor.xml

    <?xml version="1.0" encoding="utf-8"?>

    <Selector xmlns:android="http://schemas.android.com/apk/res/android">

    <item android:state_checked="false" android:color="@color/icon_nav_02_de"/>

    <item android:state_checked="true" android:color="@color/icon_nav_02_hi"/>

    </Selector >

    然后在BottomNavigationView中设置

    app:itemTextColor="@drawable/drawable_color"

    修改图标颜色

    现在基本知道了换图标的注意点,以及着色的流程,所以如果要给图标换个颜色的话,就简单了。BottomNavigationView提供了自定义属性R.styleable.BottomNavigationView_itemIconTint,因此在布局文件里添加itemIconTint的属性就可以了

    color_state_menu_navi.xml

    效果图如下(颜色有些浮夸,请忽略):

    设置Bottom Navigation颜色

    如果是在代码中实现的话,是这样的

    privatefuninitNavigationColor(){        val states = Array(2) { IntArray(1) }        states[0][0] = -android.R.attr.state_checked        states[1][0] = android.R.attr.state_checked        val colors = IntArray(2)        colors[0] = ContextCompat.getColor(this@BottomNaviActivity, android.R.color.red)        colors[1] = ContextCompat.getColor(this@BottomNaviActivity, R.color.green)        val csl = ColorStateList(states, colors)        navigation.itemTextColor = csl        navigation.itemIconTintList = csl    }

    是的,你没有看错,未选中的状态是选中状态前面加一个负号

    作者:seph_von

    链接:https://www.jianshu.com/p/7b2d842267ab

    来源:简书

    点击效果

    仔细看,会发现点击时,字体和图标会有一点位移:字体会变大,而图标也会相应的向上挪。大部分情况下,我们是不需要这样的效果的,那么怎么修改呢?

    翻了下源码,居然没有提供接口,那是不是就没有办法了呢?也不是!我们来看下item的布局文件:

    居然有两个TextView,再看BottomNavigationItemView里点击响应部分的实现

    if(checked) {                LayoutParams iconParams = (LayoutParams) mIcon.getLayoutParams();                iconParams.gravity = Gravity.CENTER_HORIZONTAL| Gravity.TOP;                iconParams.topMargin = mDefaultMargin + mShiftAmount;                mIcon.setLayoutParams(iconParams);                mLargeLabel.setVisibility(VISIBLE);                mSmallLabel.setVisibility(INVISIBLE);                mLargeLabel.setScaleX(1f);                mLargeLabel.setScaleY(1f);                mSmallLabel.setScaleX(mScaleUpFactor);                mSmallLabel.setScaleY(mScaleUpFactor);            }else{                LayoutParams iconParams = (LayoutParams) mIcon.getLayoutParams();                iconParams.gravity = Gravity.CENTER_HORIZONTAL |Gravity.TOP;                iconParams.topMargin = mDefaultMargin;                mIcon.setLayoutParams(iconParams);                mLargeLabel.setVisibility(INVISIBLE);                mSmallLabel.setVisibility(VISIBLE);                mLargeLabel.setScaleX(mScaleDownFactor);                mLargeLabel.setScaleY(mScaleDownFactor);                mSmallLabel.setScaleX(1f);                mSmallLabel.setScaleY(1f);            }

    也就是说,在选中和非选中的状态,是显示不一样的TextView,而且图标距离上边距的距离也不一样,那么我们先来看icon的上边距变化的区别就在mShiftAmount,这个是在构造函数中赋值的:

    intinactiveLabelSize =                res.getDimensionPixelSize(R.dimen.design_bottom_navigation_text_size);intactiveLabelSize = res.getDimensionPixelSize(                R.dimen.design_bottom_navigation_active_text_size);        mDefaultMargin = res.getDimensionPixelSize(R.dimen.design_bottom_navigation_margin);        mShiftAmount = inactiveLabelSize - activeLabelSize;

    看到这里就柳暗花明了,原来mShiftAmount的值就是两个TextView的字体大小的差,接下来就简单了,如果设置两个TextView的字体大小一样的话,就解决了所有的问题。

    尽管并没有提供设置字体大小的接口,但是我们可以通过重新定义R.dimen.design_bottom_navigation_text_size和R.dimen.design_bottom_navigation_margin的值来设置这两个TextView的大小。

    因此,只要在values.xml中新增两个属性即可:

    14dp14dp

    如果要设置icon距离上边距的距离,也可以通过重新定义R.dimen.design_bottom_navigation_margin来实现。

    作者:seph_von

    链接:https://www.jianshu.com/p/7b2d842267ab

    来源:简书

    简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。

    相关文章

      网友评论

        本文标题:BottomNavigationView的属性设置

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