Android–使用反射修改TabLayout下划线宽度

  • Post author:
  • Post category:其他


更新 2017.08.17

通过反射修改动画不够流畅,另外交互设计师还在增加分割线等要求…,基于TabLayout修改了代码写了自定义控件PPDTabLayout,支持调整下划线长度,增加了分割线

项目地址:

https://github.com/fengruisd/Hotpot

实现方法与原理

先上代码,这里是基于22的support包进行的反射,其他版本的代码没有去看,这里应该没有差别

public class MyActivity extends BaseActivity {


    private TabLayout tabLayout;

    // 通过反射改变下划线的长度
    private Field mIndicatorLeft;
    private Field mIndicatorRight;
    private View mTabStripObject;
    // 动画的最后位置会跳一下,原因没有分析出来,通过设置最小宽度限制一下
    private int minWidth;
    private final int positionOffset = 100;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        // 其他初始化代码
        ...
        changeTabLayoutLength();
    }

    private void changeTabLayoutLength() {
        minWidth = getResources().getDisplayMetrics().widthPixels / 4;
        try {
            Field mTabStrip = TabLayout.class.getDeclaredField("mTabStrip");
            mTabStrip.setAccessible(true);
            mIndicatorLeft = mTabStrip.getType().getDeclaredField("mIndicatorLeft");
            mIndicatorLeft.setAccessible(true);
            mIndicatorRight = mTabStrip.getType().getDeclaredField("mIndicatorRight");
            mIndicatorRight.setAccessible(true);
            mTabStripObject = (View) mTabStrip.get(tabLayout);
            mTabStripObject.getViewTreeObserver().addOnPreDrawListener(this);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @Override
    public boolean onPreDraw() {
        try {
            int left = (int) mIndicatorLeft.get(mTabStripObject);
            int right = (int) mIndicatorRight.get(mTabStripObject);
            if(right - left - positionOffset * 2 > minWidth) {
                mIndicatorLeft.set(mTabStripObject, left + positionOffset);
                mIndicatorRight.set(mTabStripObject, right - positionOffset);
            }
        } catch (Exception e) {
        }
        return true;
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        try {
            mTabStripObject.getViewTreeObserver().removeOnPreDrawListener(this);
        } catch (Exception e) {
        }
    }
}
原理

TabLayout下划线是由SlidingTabScrip实现的,在animateIndicatorToPosition添加 了动画更新下划线位置,在onDraw方法中进行绘制,所以只要修改通过反射修改下划线的绘制位置就可以了,即mIndicatorLeft和mIndicatorRight。

这里还有一个小问题没有解决,在最后动画结束单独时候,这两个值会有一个跳动,估计是反射修改值之后影响了原来的计算,但是具体影响了哪里没有分析出来,所以在上面的实现代码中加了一个minWidth进行限制了取值范围,规避了这个问题。

TabLayout.java

    @Override
    protected void onDraw(Canvas canvas) {
        // Thick colored underline belowthe current selection
        if (mIndicatorLeft >= 0 && mIndicatorRight > mIndicatorLeft) {
            canvas.drawRect(mIndicatorLeft, getHeight() - mSelectedIndicatorHeight,
                    mIndicatorRight, getHeight(), mSelectedIndicatorPaint);
        }
    }



版权声明:本文为fengrui_sd原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。