美文网首页
TabLayout踩坑

TabLayout踩坑

作者: 卧槽欢爷 | 来源:发表于2016-12-07 15:16 被阅读1142次

TabLayout踩坑传奇

今天中午休息的时候,同事研究起了MD,对!就是传说中的Material design。他想要写一个MD主题的app,可是遇到个关于TabLayout的问题。本来想做出的效果是这个样子的。

想象中的效果想象中的效果 好了开工,写完一运行,额,完全不对啊!我的title呢??! imgimg title不见了!!title不见了!!

我好奇的过去瞅瞅,看了看。没有问题啊。

 private void initViews() {
        tabLayout.addTab(tabLayout.newTab().setText("ftm1"));
        tabLayout.addTab(tabLayout.newTab().setText("ftm2"));
        viewPager.setAdapter(new FragmentStatePagerAdapter(getSupportFragmentManager()) {
            @Override
            public Fragment getItem(int position) {
                return fragments.get(position);
            }

            @Override
            public int getCount() {
                return fragments == null ? 0 : fragments.size();
            }
        });
        tabLayout.setupWithViewPager(viewPager);
    }

设置了tab也加上了setupWithViewPager方法啊。查阅了一番资料,是调用setupWithViewPager方法将所有tab清除了。不说废话了上代码,看下是怎么移除tab的:

public void setupWithViewPager(@Nullable final ViewPager viewPager, boolean autoRefresh) {
        setupWithViewPager(viewPager, autoRefresh, false);
    }

    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 (mCurrentVpSelectedListener != null) {
            // If we already have a tab selected listener for the ViewPager, remove it
            removeOnTabSelectedListener(mCurrentVpSelectedListener);
            mCurrentVpSelectedListener = null;
        }

        if (viewPager != null) {
            mViewPager = viewPager;

            // Add our custom OnPageChangeListener to the ViewPager
            if (mPageChangeListener == null) {
                mPageChangeListener = new TabLayoutOnPageChangeListener(this);
            }
            mPageChangeListener.reset();
            viewPager.addOnPageChangeListener(mPageChangeListener);

            // Now we'll add a tab selected listener to set ViewPager's current item
            mCurrentVpSelectedListener = new ViewPagerOnTabSelectedListener(viewPager);
            addOnTabSelectedListener(mCurrentVpSelectedListener);

            final PagerAdapter adapter = viewPager.getAdapter();
            if (adapter != null) {
                // Now we'll populate ourselves from the pager adapter, adding an observer if
                // autoRefresh is enabled
                setPagerAdapter(adapter, autoRefresh);
            }

            // Add a listener so that we're notified of any adapter changes
            if (mAdapterChangeListener == null) {
                mAdapterChangeListener = new AdapterChangeListener();
            }
            mAdapterChangeListener.setAutoRefresh(autoRefresh);
            viewPager.addOnAdapterChangeListener(mAdapterChangeListener);

            // 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;
    }

在调用setupWithViewPager会调用到setPagerAdapter(null, false)看下这个方法里发生了什么吧。

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();
    }

setPagerAdapter里又调用了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
            if (mViewPager != null && adapterCount > 0) {
                final int curItem = mViewPager.getCurrentItem();
                if (curItem != getSelectedTabPosition() && curItem < getTabCount()) {
                    selectTab(getTabAt(curItem));
                }
            }
        }
    }

这个populateFromPagerAdapter()方法里将tabLayout里所有的tab移除了,又重新new了tab。所以原来add的tab没有显示,那怎么解决呢?再仔细看这个方法,对没错,addTab(newTab().setText(mPagerAdapter.getPageTitle(i)), false);在重新new tab的过程中有重新传title,这个值是通过mPagerAdapter.getPageTitle(i))获取的,那么我们在ViewPagerFragmentPagerAdapter重写getPageTitle方法,默认的PagerAdapter返回的是null。

public CharSequence getPageTitle(int position) {
   return null;
}

现在我们把代码修改成下面的样子:

private void initViews() {
        viewPager.setAdapter(new FragmentPagerAdapter(getSupportFragmentManager()) {
            @Override
            public Fragment getItem(int position) {
                return fragments.get(position);
            }

            @Override
            public int getCount() {
                return fragments == null ? 0 : fragments.size();
            }

            @Override
            public CharSequence getPageTitle(int position) {
                return titles.get(position);
            }
        });
        tabLayout.setupWithViewPager(viewPager);
    }

O(∩_∩)O哈哈~现在就是我们想要的样子啦。。。我写的Demo源码

参考资料

相关文章

  • TabLayout踩坑

    TabLayout踩坑传奇 今天中午休息的时候,同事研究起了MD,对!就是传说中的Material design。...

  • Tablayout踩坑与实践

    新手菜鸡一枚,跟着教程学Tablayout,结果第一步依赖包导入就遇坑。目标是可以使用android.suppor...

  • (转)TabLayout 踩坑之 onTabSelected 没

    最近项目中有个需求:一个页面顶部有3个tab,每一个tab分别展示一个不同的页面,点击tab 切换到对应页面。进入...

  • tablayout+viewpager踩坑记getParentF

    问题描述:一个页面最外层一个tablayout+viewpager+fragment,最内层Coordinarat...

  • JavaScrip-StepPitGuide《JavaScrip

    《JavaScript踩坑指南》JavaScrip-StepPitGuide? 《JavaScript踩坑指南》 ...

  • 算法踩坑6-二叉搜索树排序

    背景 接上面五篇文章算法踩坑-快速排序 算法踩坑2-插入排序 算法踩坑3-堆排序 算法踩坑4-冒泡排序 ...

  • 算法踩坑5-归并排序

    背景 接上面四篇文章算法踩坑-快速排序 算法踩坑2-插入排序 算法踩坑3-堆排序 算法踩坑4-冒泡排序 来...

  • D1094:踩坑的价值最大化

    是人就会踩坑,不踩坑理论上就不属于人类,踩坑是人之常情,能回头站在坑边反思,才是对踩过的的坑价值最大化的体现,要不...

  • 交互设计师所要避免的几个坑

    前言 工作中难免会踩到几个坑,即使现在不踩以后还会踩,只有踩过才会深刻记住,踩过说明爱过!但是踩过的坑必须把坑填满...

  • TabLayout 踩坑之 onTabSelected没有被回调

    一、 问题描述 最近项目中有个需求:一个页面顶部有3个tab,每一个tab分别展示一个不同的页面,点击tab 切换...

网友评论

      本文标题:TabLayout踩坑

      本文链接:https://www.haomeiwen.com/subject/jqndmttx.html