欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

TabLayout设置下划线(Indicator)宽

程序员文章站 2022-06-01 07:50:59
...

作者:w余生请多指教

地址:http://blog.csdn.net/u013134391/article/details/70833903


因为TabLayout中系统是强制设置所有TabView的宽度为最宽那个TabView的宽度,而下划线宽度即为TabView宽度,所以需要自定义下划线宽度时,解决方法如下:

//了解源码得知:下划线的宽度是根据TabView的宽度来设置的  
tabLayout.post(() -> {  
  
    try {  
        //拿到tabLayout的mTabStrip属性  
        Field mTabStripField = tabLayout.getClass().getDeclaredField("mTabStrip");  
        mTabStripField.setAccessible(true);  
  
        LinearLayout mTabStrip = (LinearLayout) mTabStripField.get(tabLayout);  
  
        int dp10 = SM.dip2px(getContext(), 10);  
  
        for (int i = 0; i < mTabStrip.getChildCount(); i++) {  
            View tabView = mTabStrip.getChildAt(i);  
  
            //拿到tabView的mTextView属性  
            Field mTextViewField = tabView.getClass().getDeclaredField("mTextView");  
            mTextViewField.setAccessible(true);  
  
            TextView mTextView = (TextView) mTextViewField.get(tabView);  
  
            tabView.setPadding(0, 0, 0, 0);  
  
            //因为想要的效果是 下划线与字体宽度同步,所以测量mTextView的宽度  
            int width = 0;  
            width = mTextView.getWidth();  
            if (width == 0) {  
                mTextView.measure(0, 0);  
                width = mTextView.getMeasuredWidth();  
            }  
  
            //设置tab左右间距为10dp  注意这里不能使用Padding 因为源码中下划线的宽度是根据 tabView的宽度来设置的  
            LinearLayout.LayoutParams params = (LinearLayout.LayoutParams) tabView.getLayoutParams();  
            params.width = width ;  
            params.leftMargin = dp10;  
            params.rightMargin = dp10;  
            tabView.setLayoutParams(params);  
  
            tabView.invalidate();  
        }  
  
    } catch (NoSuchFieldException e) {  
        e.printStackTrace();  
    } catch (IllegalAccessException e) {  
        e.printStackTrace();  
    }  
  
});  


问题解决思路

第一反应是找系统的方法和属性,发现只有设置tabIndicatorHeight的属性 并没有宽度的属性;接着百度,一百度就看到几个博客,宣称可以解决这个问题,我们先看看他们的解决方案:传送门

这种解决方案仅限于所有的tabView的text字数都是相同字数,比如所有的图中所有的tab字数都是2个。其实思路是错的,没有研究源码详细实现。

他的思路是设置tabView的padding为0,并且设置了margin。这种方案错误的原因是,tablayout会强制设置tabView的宽度为  几个tabView中最宽的宽度,比如4个字的tabview和2个字的tabview的组合,两个tabview的宽度强制为4个字的tabview的宽度。

下面会证实这一点:

TabLayout设置下划线(Indicator)宽

那只有查源码了呗,tab的创建是 tablayout.addTab();方法构造的 具体代码如下

tabLayout.addTab(tabLayout.newTab().setText("生鲜食品"));  

直接查这个方法,通过几个重载方法(addTab(Tab tab)->addTab(Tab tab,boolean setSelected)->addTab( Tab tab, int position, boolean setSelecte);  跳转如下代码

TabLayout设置下划线(Indicator)宽

从注释看就是添加一个tab到这个layout上  具体实现是在addTabView(Tab tab)里面,继续看这个方法

TabLayout设置下划线(Indicator)宽

可以看到最后添加到mTabStrip中,我们再来看看TabView里面有什么东西

TabLayout设置下划线(Indicator)宽

从属性可以看出TabView可以自定义的,而且并没有发现Indicator线的痕迹,猜测他可能放在layout(mTabStrip)里面,那就来看mTabStrip

TabLayout设置下划线(Indicator)宽

查看类中,发现mSelectedIndicatorHeight,眼睛一亮,下划线高度!!,就是画线的地方。追踪mIndicatorLeft和mIndicatorRight的来路,几经追踪,发现如下代码

TabLayout设置下划线(Indicator)宽

如图,selectedTitle就是TabView,直接获取了左边坐标和右边坐标,也就说是线的宽度就是tabview的宽度,那疑问又来了,为什么我们两个字的tabView和4个字的tabView是一样宽度,先去看看SlidingTabStrip的onMeasure方法,如下图

TabLayout设置下划线(Indicator)宽

第一个for循环干的事就是记录下来所有tabView中的最大宽度,第二个循环就是把所有的tabView的宽度设置为第一个循环得到的最大宽!!!

罪魁祸首是找到了,这时候能动态代理一个重写onMeasure方法的SlidingTabStrip对象塞进去,也可以解决这个问题,你会发现SlidingTabStrip是private的!!!!!!!

思路一转,系统是强制设置所有tabview的宽度为 最宽那个tabview的宽度,那重新设置一遍tabView的宽度即可,解决问题(其实中间还尝试过调用setIndicatorPosition方法,但是系统源码,在多个时期调用这个方法,所以毙掉了)


那最上面的解决方案就来了:

1、通过反射拿到SlidingTabStrip,通过遍历拿到tabview,继续通过反射拿到textview,然后设置Tabview的宽度为textview的宽度

2、为了美观我们可以设置一下tabview的margin,不设置会连在一起


PS:需要注意的事

1.因为用到了反射,所以混淆的时候要注意

2.如果app:tabMode="fixed",每个TabView的weight都为1,设置width是没用的,目前的解决办法是可以先拿到TabView的宽度减去TextView宽度除以2,得到TabView的左右margin,设置上去就行了(看看谁有更好的办法,希望提出来如何解决)