BottomNavigationView的属性设置

备注: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

来源:简书

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

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,684评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,143评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,214评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,788评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,796评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,665评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,027评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,679评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,346评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,664评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,766评论 1 331
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,412评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,015评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,974评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,073评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,501评论 2 343

推荐阅读更多精彩内容