TabLayout的使用:这一篇就够了

    TabLayout是design包中的控件继承HorizontalScrollView,用作页面切换的指示器,在分类页面中经常使用。结合ViewPager使用起来很方便。先看下具体效果。

TabLayout的基本使用

  • 添加依赖
  implementation 'com.android.support:design:26.1.0'
  • 布局
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    >


 <android.support.design.widget.TabLayout
     android:layout_width="match_parent"
     android:layout_height="wrap_content"
     />

<android.support.v4.view.ViewPager
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    />
  
  
</LinearLayout>
  • 常用属性
app:tabIndicatorColor :指示线的颜色
app:tabIndicatorHeight :指示线的高度
app:tabSelectedTextColor : tab选中时的字体颜色
app:tabMode="scrollable" : 默认是fixed,固定的;scrollable:可滚动的
  • 基本使用
public class MainActivity extends AppCompatActivity {
  String[] title = new String[] { "A", "B", "C", "D", "E", "F", "G", "H" };
  @Override protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    //默认的
    TabLayout normal = (TabLayout) findViewById(R.id.normal);
    //自定义布局
    TabLayout custom = (TabLayout) findViewById(R.id.custom);
    setTab(normal);
    for (int i = 0; i < title.length; i++) {
      custom.addTab(custom.newTab());
      String titleName = title[i];
      TabLayout.Tab tab = custom.getTabAt(i);
      View view = View.inflate(this,R.layout.tab_custom_layou,null);
      TextView text = (TextView) view.findViewById(R.id.text);
      text.setText(titleName);
      tab.setCustomView(view);
    }
  }

  private void setTab(TabLayout tabLayout) {
    for (int i = 0; i < title.length; i++) {
      tabLayout.addTab(tabLayout.newTab());
      tabLayout.getTabAt(i).setText(title[i]);
    }
  }
}
  • 结合ViewPager使用
public class MainActivity extends AppCompatActivity {
  String[] mTitle = new String[] { "A", "B", "C", "D", "E", "F", "G", "H" };
  List<Fragment> mFragments = new ArrayList<>();
  @Override protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    TabLayout tabLayout = (TabLayout) findViewById(R.id.tab_layout);
    ViewPager viewPager = (ViewPager) findViewById(R.id.view_pager);
    for (int i = 0; i < mTitle.length; i++) {
      TabFragment fragment = new TabFragment();
      Bundle bundle = new Bundle();
      bundle.putString(TabFragment.TEXT, mTitle[i]);
      fragment.setArguments(bundle);
      mFragments.add(fragment);
    }
  //绑定ViewPager
    tabLayout.setupWithViewPager(viewPager);
    TabViewPagerAdapter adapter = new TabViewPagerAdapter(getSupportFragmentManager(), mTitle,mFragments);
    viewPager.setAdapter(adapter);

  }

}



public class TabViewPagerAdapter extends FragmentStatePagerAdapter {
  private String[] titles;
  private List<Fragment> fragments;

  public TabViewPagerAdapter(FragmentManager fm,String[] titles,List<Fragment> fragments) {
    super(fm);
    this.titles = titles;
    this.fragments = fragments;
  }

  @Override public Fragment getItem(int position) {
    return fragments.get(position);
  }

  @Override public int getCount() {
    return fragments.size();
  }
  //设置指示器
  @Override public CharSequence getPageTitle(int position) {
    return titles[position];
  }
}
  • 效果

有时候需要修改指示器线条的样式和宽度,还要修改指示器的样式,这时候基本的使用无法满足需求,TabLayout自身也没有提供对应的方法。但是没关系我们可以通过反射来进行修改

  • 看源码中是如何实现指示器的,先找到实现指示器相关的类
 private final SlidingTabStrip mTabStrip;

通过源码查看SlidingTabStrip是继承LinearLayout的,通过draw方法去实现绘制指示器,mIndicatorLeft和mIndicatorRight变量是控制指示器的的宽度的

private class SlidingTabStrip extends LinearLayout {
        private int mSelectedIndicatorHeight;
        private final Paint mSelectedIndicatorPaint;

        int mSelectedPosition = -1;
        float mSelectionOffset;

        private int mLayoutDirection = -1;

        private int mIndicatorLeft = -1;
        private int mIndicatorRight = -1;

        private ValueAnimator mIndicatorAnimator;

        SlidingTabStrip(Context context) {
            super(context);
            setWillNotDraw(false);
            mSelectedIndicatorPaint = new Paint();
        }
                        ....省略代码

   @Override
        public void draw(Canvas canvas) {
            super.draw(canvas);
            ////这里是关键代码,画指示线,那么我么也可以利用底下的几个参数画整个背景,后面需要用到
            // Thick colored underline below the current selection
            if (mIndicatorLeft >= 0 && mIndicatorRight > mIndicatorLeft) {
                canvas.drawRect(mIndicatorLeft, getHeight() - mSelectedIndicatorHeight,
                        mIndicatorRight, getHeight(), mSelectedIndicatorPaint);
            }
        }
}
  • 对应的工具类
public class ProxyDrawable extends Drawable {
  private View view;
  private Paint paint;

  private int indicatorPaddingLeft;
  private int indicatorPaddingRight;
  private int indicatorHeight;

  private RectF drawRectF;

  public ProxyDrawable(@NonNull View view,int indicatorHeight) {
    this.view = view;
    this.indicatorHeight = indicatorHeight;
    paint = new Paint();
    drawRectF = new RectF();
  }

  public void setIndicatorPaddingLeft(int indicatorPaddingLeft) {
    this.indicatorPaddingLeft = indicatorPaddingLeft;
  }

  public void setIndicatorPaddingRight(int indicatorPaddingRight) {
    this.indicatorPaddingRight = indicatorPaddingRight;
  }

  public void setIndicatorColor(int color) {
    paint.setColor(color);
  }

  @Override public void draw(@NonNull Canvas canvas) {
    //这里通过反射获取SlidingTabStrip的两个变量,源代码画的是下划线,我们现在画的是带圆角的矩形
    int mIndicatorLeft = getIntValue("mIndicatorLeft") + indicatorPaddingLeft;
    int mIndicatorRight = getIntValue("mIndicatorRight") - indicatorPaddingRight;
    int mSelectedIndicatorHeight = getIntValue("mSelectedIndicatorHeight");
    int height = view.getHeight();
    /*mSelectedIndicatorHeight / 2*/
    int radius = 0;
    if (mIndicatorLeft >= 0 && mIndicatorRight > mIndicatorLeft) {
      drawRectF.set(mIndicatorLeft, height - mSelectedIndicatorHeight - DensityUtil.dip2px(indicatorHeight), mIndicatorRight,
          height - DensityUtil.dip2px(indicatorHeight));
      canvas.drawRoundRect(drawRectF, radius, radius, paint);
    }
  }

  private int getIntValue(String name) {
    try {
      Field f = view.getClass().getDeclaredField(name);
      f.setAccessible(true);
      Object obj = f.get(view);
      return (Integer) obj;
    } catch (Exception e) {
      e.printStackTrace();
    }
    return 0;
  }

  @Override public void setAlpha(@IntRange(from = 0, to = 255) int alpha) {

  }

  @Override public void setColorFilter(@Nullable ColorFilter colorFilter) {

  }

  @Override public int getOpacity() {
    return PixelFormat.UNKNOWN;
  }

  • 进行修改
     View tabStrip = mArticleTypeTab.getChildAt(0);
      if (tabStrip != null) {
        ProxyDrawable proxyDrawable = new ProxyDrawable(tabStrip, 7);
        proxyDrawable.setIndicatorColor(Res.color(R.color.tax_describe_text_bottom));
        proxyDrawable.setIndicatorPaddingLeft(DensityUtil.dip2px(getActivity(), 25));
        proxyDrawable.setIndicatorPaddingRight(DensityUtil.dip2px(getActivity(), 25));
        tabStrip.setBackground(proxyDrawable);
      }

  • 注意:存在的问题是画的下划线仍然存在,需要在布局将其隐藏
    <android.support.design.widget.TabLayout
        android:id="@+id/article_type_tab"
        android:layout_width="match_parent"
        android:layout_height="48dp"
        android:layout_marginTop="5dp"
        app:tabBackground="@null"
        app:tabIndicatorColor="@color/transparent"
        app:tabIndicatorHeight="4dp"
        app:tabMinWidth="70dp"
        />
  </android.support.design.widget.AppBarLayout>

TabLayout切换的监听 实现文字大小变换

    mArticleTypeTab.addOnTabSelectedListener(new TabLayout.OnTabSelectedListener() {
      @Override public void onTabSelected(TabLayout.Tab tab) {
        int position = tab.getPosition();
        trackSensors(position);
        View view = tab.getCustomView();
        if (view != null) {
          TextView textView = (TextView) view.findViewById(R.id.title);
          textView.setTextSize(16);
        }
      }

      @Override public void onTabUnselected(TabLayout.Tab tab) {
        View view = tab.getCustomView();
        if (view != null) {
          TextView textView = (TextView) view.findViewById(R.id.title);
          textView.setTextSize(14);
        }
      }

      @Override public void onTabReselected(TabLayout.Tab tab) {

      }
    });

  • 效果

有时候TabLayout的Tab指示器之间还需要有分隔线,TabLayout很贴心的帮我们提供了对应的方法

      LinearLayout tabStrip = (LinearLayout) loginType.getChildAt(0);
      tabStrip.setShowDividers(LinearLayout.SHOW_DIVIDER_MIDDLE);
      tabStrip.setDividerDrawable(
          ContextCompat.getDrawable(getActivity(), R.drawable.shape_tab_divider));
      tabStrip.setDividerPadding(DensityUtil.dip2px(18));

                  -----shape----shape_tab_divider------

<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android">
<solid android:color="@color/divider_color"/>
  <size android:width="1dp"/>
</shape>

  • 效果

TabLayout的常用方法就讲解到这里啦,欢迎补充!

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