Android TabLayout系列之简单使用

1 前言

在上一篇 Android TabLayout系列之属性 中我们介绍了TabLayout的属性,同时也给出了一些简单的效果图。但是没有具体到它的使用,今天就来看看TabLayout的简单使用。不知道大家留意到我们仿网易的效果布局中,我们明明是写的小写字母,字母就变成大写了,还有字体大小能改变否?我们一步一步来解决这些问题......

2 使用

介绍一种在实际开发中TabLayout较为常用的方式,那就是和ViewPager配合使用,实现联动。首先看看xml布局:

<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <android.support.design.widget.TabLayout
        android:id="@+id/tabLayout"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        app:tabBackground="@android:color/white"
        app:tabIndicatorColor="@android:color/holo_red_light"
        app:tabIndicatorHeight="2dp"
        app:tabMode="scrollable"
        app:tabSelectedTextColor="@android:color/holo_red_light"
        app:tabTextColor="@android:color/darker_gray"/>

    <android.support.v4.view.ViewPager
        android:id="@+id/viewPager"
        android:layout_width="match_parent"
        android:layout_height="match_parent"/>
</LinearLayout>

是不是很简单的布局,就只有一个TabLayout和ViewPager垂直排列,其实这个还不是官方的布局样式,官方的是这样的:

<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <android.support.v4.view.ViewPager
        android:id="@+id/viewPager"
        android:layout_width="match_parent"
        android:layout_height="match_parent">

        <android.support.design.widget.TabLayout
            android:id="@+id/tabLayout"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            app:tabBackground="@android:color/white"
            app:tabIndicatorColor="@android:color/holo_red_light"
            app:tabIndicatorHeight="2dp"
            app:tabMode="scrollable"
            app:tabSelectedTextColor="@android:color/holo_red_light"
            app:tabTextColor="@android:color/darker_gray"/>

    <android.support.v4.view.ViewPager/>
</LinearLayout>

第一种是我们常见到的,第二种是官方的,两种的区别是官方这种写法不用调用setupWithViewPager方法。接下来我们看看Activity的代码怎么实现的:

public class MainActivity extends BaseActivity {

    private TabLayout mTabLayout;
    private ViewPager mViewPager;
    private MainPagerAdapter mAdapter;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        AppStatusTracker.init(getApplication());
        super.onCreate(savedInstanceState);
    }

    @Override
    protected void initContentView() {
        setContentView(R.layout.activity_main);
    }

    @Override
    protected void initView() {
        mTabLayout = findView(tabLayout);
        mViewPager = findView(R.id.viewPager);
        for (int i = 0; i < 11; i++) {
            //为TabLayout添加10个tab并设置上文本
            mTabLayout.addTab(mTabLayout.newTab().setText("Tab " + i));
        }

        mAdapter = new MainPagerAdapter(getSupportFragmentManager());
        mViewPager.setAdapter(mAdapter);
        //官方推荐的绑定ViewPager方式
        mTabLayout.setupWithViewPager(mViewPager);
    }

    @Override
    protected void initData(@Nullable Bundle savedInstanceState) {

    }

    //ViewPager适配器  10个Fragment
    private class MainPagerAdapter extends FragmentPagerAdapter {
        public MainPagerAdapter(FragmentManager fm) {
            super(fm);
        }

        @Override
        public Fragment getItem(int position) {
            return BlankFragment.newInstance(position);
        }

        @Override
        public int getCount() {
            return 10;
        }
    }
}

接下来是BlankFragment的实现:

public class BlankFragment extends Fragment {
    private int index;

    public BlankFragment() {
        // Required empty public constructor
    }

    public static Fragment newInstance(int position) {
        BlankFragment fragment = new BlankFragment();
        fragment.index = position;
        return fragment;
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        return inflater.inflate(R.layout.fragment_blank, container, false);
    }

    @Override
    public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);
        ((TextView) view.findViewById(R.id.textView)).setText("this Tab " + index);
    }
}

这里需要先解释newTab,这个newTab是使用TabLayout中默认的Tab实现。从它的实现可以看到它支持设置图标,标题和内容描述(图标描述)。

    public static final class Tab {

        /**
         * An invalid position for a tab.
         *
         * @see #getPosition()
         */
        public static final int INVALID_POSITION = -1;

        private Object mTag;
        private Drawable mIcon;
        private CharSequence mText;
        private CharSequence mContentDesc;
        private int mPosition = INVALID_POSITION;
        private View mCustomView;

        TabLayout mParent;
        TabView mView;
  .....
    }

Tab默认图标在标题的上方,还有xml里面设置TabItem的话,最终也是设置到Tab上。关于Tab就说明这些,具体的大家可以自行测试。
可以看到关于TabLayout的实现也很简单,就是给布局里面的TextView设置了文本显示当前是第几个Tab。布局就更简单了,FrameLayout中嵌套了一个TextView,就不单独给出了。接下来我们就可以来瞅一瞅实际的使用效果了。


3 “填坑”与源码解析

  • 源码解析
    看到上面的效果图是不是觉得很怪异?没错,我们所设置的Tab上的文字没了,只有下面的指示条。相信大家一通搜索后,相信大家都知道了,最常见的就是分析源码后得出结论,被remove了!!!接下来我们也来装下文化人,看看怎么回事?
    首先是TabLayout的setupWithViewPager方法,一顿单击后会到达如下源码位置:
private void setupWithViewPager(@Nullable final ViewPager viewPager, boolean autoRefresh,
            boolean implicitSetup) {
        if (mViewPager != null) {
            // If we've already been setup with a ViewPager, remove us from it
            if (mPageChangeListener != null) {
                mViewPager.removeOnPageChangeListener(mPageChangeListener);
            }
            if (mAdapterChangeListener != null) {
                mViewPager.removeOnAdapterChangeListener(mAdapterChangeListener);
            }
        }
......
......
            if (adapter != null) {
                // Now we'll populate ourselves from the pager adapter, adding an observer if
                // autoRefresh is enabled
                setPagerAdapter(adapter, autoRefresh);
            }
......
......

            // Now update the scroll position to match the ViewPager's current item
            setScrollPosition(viewPager.getCurrentItem(), 0f, true);
        } else {
            // We've been given a null ViewPager so we need to clear out the internal state,
            // listeners and observers
            mViewPager = null;
            setPagerAdapter(null, false);
        }

        mSetupViewPagerImplicitly = implicitSetup;
    }

这里没什么说的就是设置判断移除、设置监听等操作,我们最主要去找到网上最多叙述问题所在的源码位置,接下来我们看setPagerAdapter:

    void setPagerAdapter(@Nullable final PagerAdapter adapter, final boolean addObserver) {
        if (mPagerAdapter != null && mPagerAdapterObserver != null) {
            // If we already have a PagerAdapter, unregister our observer
            mPagerAdapter.unregisterDataSetObserver(mPagerAdapterObserver);
        }

        mPagerAdapter = adapter;

        if (addObserver && adapter != null) {
            // Register our observer on the new adapter
            if (mPagerAdapterObserver == null) {
                mPagerAdapterObserver = new PagerAdapterObserver();
            }
            adapter.registerDataSetObserver(mPagerAdapterObserver);
        }

        // Finally make sure we reflect the new adapter
        populateFromPagerAdapter();
    }

这段代码主要是判断之前的mPagerAdapter是否为空,不为空就移除DataSetObserver监听,将新的adapter设置进来并注册上DataSetObserver监听。这里我们捕获方法populateFromPagerAdapter一枚,我们再看看它的实现:

    void populateFromPagerAdapter() {
        removeAllTabs();

        if (mPagerAdapter != null) {
            final int adapterCount = mPagerAdapter.getCount();
            for (int i = 0; i < adapterCount; i++) {
                addTab(newTab().setText(mPagerAdapter.getPageTitle(i)), false);
            }

            // Make sure we reflect the currently set ViewPager item
            ......
        }
    }

哇塞!removeAllTabs,大家说得最多的一个方法,观看方法名就很吓人了,remove all tabs....看看它到底干了什么:

    public void removeAllTabs() {
        // Remove all the views
        for (int i = mTabStrip.getChildCount() - 1; i >= 0; i--) {
            removeTabViewAt(i);
        }

        for (final Iterator<Tab> i = mTabs.iterator(); i.hasNext();) {
            final Tab tab = i.next();
            i.remove();
            tab.reset();
            sTabPool.release(tab);
        }

        mSelectedTab = null;
    }

它果真和它的命名一样,遍历remove了所有的tab,并且将当前选中也置为null。怎么办?难道Google故意留下这么个坑?我们继续看上面的populateFromPagerAdapter,会发现removeAllTabs后,它会判断adapter是否为空,不为空就调用了addTab(newTab().setText(mPagerAdapter.getPageTitle(i)), false)添加上了新的tab。

  • "填坑"思路
  1. 从上面的分析,我们发现如果要有tab还得去重写adapter的getPageTitle方法。再看一段官方文档中的说明:

If you're using a ViewPager together with this layout, you can call setupWithViewPager(ViewPager)
to link the two together. This layout will be automatically populated from the PagerAdapter
's page titles.

就是说官方推荐我们使用setupWithViewPager(ViewPager)来关联Tablayout和Viewpager,且TabLayout会自动填充PagerAdapter的Title。也就是说它会自动创建tab,并绑定为adapter的page
title。这下我们来改造下Activity:

public class MainActivity extends BaseActivity {

    private TabLayout mTabLayout;
    private ViewPager mViewPager;
    private MainPagerAdapter mAdapter;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        AppStatusTracker.init(getApplication());
        super.onCreate(savedInstanceState);
    }

    @Override
    protected void initContentView() {
        setContentView(R.layout.activity_main);
    }

    @Override
    protected void initView() {
        mTabLayout = findView(tabLayout);
        mViewPager = findView(R.id.viewPager);
        mAdapter = new MainPagerAdapter(getSupportFragmentManager());
        mViewPager.setAdapter(mAdapter);
        //官方推荐的绑定ViewPager方式
        mTabLayout.setupWithViewPager(mViewPager);
    }

    @Override
    protected void initData(@Nullable Bundle savedInstanceState) {

    }

    //ViewPager适配器  10个Fragment
    private class MainPagerAdapter extends FragmentPagerAdapter {
        public MainPagerAdapter(FragmentManager fm) {
            super(fm);
        }

        @Override
        public Fragment getItem(int position) {
            return BlankFragment.newInstance(position);
        }

        //TabLayout会根据当前page的title自动绑定tab
        @Override
        public CharSequence getPageTitle(int position) {
            return "Tab " + position;
        }

        @Override
        public int getCount() {
            return 10;
        }
    }
}

这就是新的Activity实现,改动很小。我们首先删除了initView中关于tab的初始化操作,然后重写了FragmentPagerAdapter的getPageTitle方法。


  1. 我们既然知道他会自动为我们绑定tab,那么我们可以利用它自动帮我绑定tab,而不去重写getPageTitle方法,在绑定后去设置关于tab的显示。
public class MainActivity extends BaseActivity {

    private TabLayout mTabLayout;
    private ViewPager mViewPager;
    private MainPagerAdapter mAdapter;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        AppStatusTracker.init(getApplication());
        super.onCreate(savedInstanceState);
    }

    @Override
    protected void initContentView() {
        setContentView(R.layout.activity_main);
    }

    @Override
    protected void initView() {
        mTabLayout = findView(tabLayout);
        mViewPager = findView(R.id.viewPager);
        mAdapter = new MainPagerAdapter(getSupportFragmentManager());
        mViewPager.setAdapter(mAdapter);
        //官方推荐的绑定ViewPager方式
        mTabLayout.setupWithViewPager(mViewPager);
        int tabCount = mTabLayout.getTabCount();
        for (int i = 0; i < tabCount; i++) {
            //这里tab可能为null 根据实际情况处理吧
            mTabLayout.getTabAt(i).setText("Tab" + i);
        }
    }

    @Override
    protected void initData(@Nullable Bundle savedInstanceState) {

    }

    //ViewPager适配器  10个Fragment
    private class MainPagerAdapter extends FragmentPagerAdapter {
        public MainPagerAdapter(FragmentManager fm) {
            super(fm);
        }

        @Override
        public Fragment getItem(int position) {
            return BlankFragment.newInstance(position);
        }

        @Override
        public int getCount() {
            return 10;
        }
    }
}

可以看到只是将initView中初始化Tab的位置调整到setupWithViewPager的后面,设置上我们需要的标题。这里效果和上面一样就不给出来占篇幅了。

  1. 上面我们依旧利用了setupWithViewPager自动为我们绑定tab的实现,在分析源码的时候发现它会调用ViewPager的addOnPageChangeListener和TabLayout的addOnTabSelectedListener,实现TabLayout与ViewPager的关联。那么我们就自己来实现关联,不去使用官方推荐的setupWithViewPager方法绑定。
public class MainActivity extends BaseActivity {

    private TabLayout mTabLayout;
    private ViewPager mViewPager;
    private MainPagerAdapter mAdapter;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        AppStatusTracker.init(getApplication());
        super.onCreate(savedInstanceState);
    }

    @Override
    protected void initContentView() {
        setContentView(R.layout.activity_main);
    }

    @Override
    protected void initView() {
        mTabLayout = findView(tabLayout);
        mViewPager = findView(viewPager);
        mAdapter = new MainPagerAdapter(getSupportFragmentManager());
        mViewPager.setAdapter(mAdapter);
        //初始化tab
        for (int i = 0; i < 10; i++) {
            mTabLayout.addTab(mTabLayout.newTab().setText("item" + i));
        }
        //自己实现关联
        mViewPager.addOnPageChangeListener(new TabLayout.TabLayoutOnPageChangeListener(mTabLayout));
        mTabLayout.addOnTabSelectedListener(new TabLayout.ViewPagerOnTabSelectedListener(mViewPager));
    }

    @Override
    protected void initData(@Nullable Bundle savedInstanceState) {

    }

    //ViewPager适配器  10个Fragment
    private class MainPagerAdapter extends FragmentPagerAdapter {
        public MainPagerAdapter(FragmentManager fm) {
            super(fm);
        }

        @Override
        public Fragment getItem(int position) {
            return BlankFragment.newInstance(position);
        }

        @Override
        public int getCount() {
            return 10;
        }
    }
}

这段代码,在initView中移除了官方推荐的设置相关代码,自己初始化了tab并实现了关联。注意这里我们改变了Tab的标题作为区分。效果也和上面一样,只是改变了Tab的标题。

4 其他问题

这样TabLayout的简单使用就介绍完了,这里还有两个问题一个是TabLayout的标题怎么就是大写了,还有它的文字大小怎么改呢?还记得我怎么在上一篇Android TabLayout系列之属性中介绍的tabTextAppearance属性么?我们就利用它来改变标题字母大小写和文字大小问题。先开看看TabLayout源码中的默认使用:

      mTabTextAppearance = a.getResourceId(R.styleable.TabLayout_tabTextAppearance,
      R.style.TextAppearance_Design_Tab);


      <style name="TextAppearance.Design.Tab" parent="TextAppearance.AppCompat.Button">
          <item name="android:textSize">@dimen/design_tab_text_size</item>
          <item name="android:textColor">?android:textColorSecondary</item>
          <item name="textAllCaps">true</item>
      </style>

我们为Tablayout添加上tabTextAppearance属性:

      app:tabTextAppearance="@style/TabLayoutStyle"

这里需要把具体的属性定义到stytle中,这里有两种方式:

    <style name="TabLayoutStyle" parent="TextAppearance.Design.Tab">
        <item name="android:textSize">16sp</item>
        <item name="textAllCaps">false</item>
    </style>
      <style name="TabLayoutStyle">
          <item name="android:textSize">16sp</item>
         //<item name="textAllCaps">false</item>
          <item name="android:textAllCaps">false</item>
      </style>

需要注意 parent="TextAppearance.Design.Tab"时,textAllCaps没有android,当不继承的时候有没有都可以。其实这里的字母大小写,不能算个坑。我们可以看material设计中所有tab的字母都是大写,估计歪果仁标题都习惯大写吧,或者是为了符合material设计,所以默认的Tab被设置成了大写。最后运行效果如下:


这里附上我的build.gradle,有可能不同版本会有不一样的效果,所以大家实际使用的时候需要注意一下:

android {
    compileSdkVersion 25
    buildToolsVersion "25.0.1"

    defaultConfig {
        applicationId "com.joker.demo"
        minSdkVersion 19
        targetSdkVersion 25
        versionCode 1
        versionName "1.0"

        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"

    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
}

TabLayou的简单使用就介绍这么多,其实真的是简单使用,主要是中间插入了一段源码分析,所以看起来多了点。
附:

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

推荐阅读更多精彩内容