美文网首页自定义view深入浅出AndroidAndroid知识
Android TabContainerView 实现底部导航栏

Android TabContainerView 实现底部导航栏

作者: 伪文艺大叔 | 来源:发表于2017-03-23 11:22 被阅读2129次

本篇文章已授权微信公众号 guolin_blog (郭霖)独家发布


xiaoguo.gif

上图效果大家应该都很熟悉了,基本市面上的App都会用到这种布局效果,实现起来也很简单,就是上面一个ViewPager,下面一个线性布局

TabContainerView就是把实现逻辑封装起来,让开发者可以通过更简单的代码实现这种布局效果,提高工作效率。

使用
TabContainerView tabContainerView = (TabContainerView) findViewById(R.id.container_tab);
//设置适配器配置数据
tabContainerView.setAdapter(new MainTabContainerAdapter(getSupportFragmentManager(),
new Fragment[] {new MainFragment(), new WorkFragment(), new AppFragment(), new MineFragment()}));

就是这么简单,调用两行代码就可以实现了,不过需要我们自己创建一个适配器,MainTabContainerAdapter就是自己创建的,它需要继承BaseAdapter来实现里面的抽象方法,BaseAdapter是此项目当中自定义的抽象类。

思路

TabContainerView是一个RelativeLayout布局,整个布局由两部分组成:底部布局,内容布局

底部布局为一个LinearLayout布局,里面的单个Tab也是一个LinarLayout布局;中间的内部区域是一个ViewPager。

实现

项目由6个类组成

//暴露给开发者的View,主要负责添加底部和ViewPager的布局
TabContainerView 
//底部布局的单个布局,包含单个布局的文本,图片属性信息
Tab
//底部布局的整体布局,负责Tab布局的添加和状态切换
TabHost
//Tab选中的监听
OnTabSelectedListener
//适配器提供了底部文本内容,图片内容,fragment内容
BaseAdapter
//内容ViewPager的适配器
TabViewPagerAdapter

我们先来分析调用的第一行代码
TabContainerView tabContainerView = (TabContainerView) findViewById(R.id.container_tab);

看看它的构造方法

 public TabContainerView(Context context, AttributeSet attrs) {
        super(context, attrs);
        init(context, attrs);
 }

构造方法调用了init方法

private void init(Context context, AttributeSet attrs) {
       initStyle(context, attrs);
       initTabHost(context);
       initDivideLine(context);
       initViewPager(context);

       tabHost.setContentViewPager(contentViewPager);
}

分别初始化了自定义属性,底部的TabHost,分割线,ViewPager等;
initStyle就是初始化一些自定义的属性,没啥好说的;
我们来看下initTabHost方法

 private void initTabHost(Context context) {
        tabHost = new TabHost(context);
        addView(tabHost.getRootView());
 }

 public TabHost(Context context) {
        this.context = context;
        initView();
 }

 private void initView() {
        rootView = new LinearLayout(context);
        rootView.setOrientation(LinearLayout.HORIZONTAL);
        rootView.setId(R.id.linearlayout_tab);

        RelativeLayout.LayoutParams rootViewLp = new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT);
        rootViewLp.addRule(RelativeLayout.ALIGN_PARENT_BOTTOM);
        rootView.setLayoutParams(rootViewLp);
 }

initTabHost方法里创建了一个TabHost,TabHost的构造方法里调用initView方法,方法创建RootView布局,设置长宽, 位置等属性,然后把TabHost的根布局添加到TabContainerView布局当中。

initDivideLine方法创建一个分割线View添加到布局当中
initViewPager方法创建一个ViewPager添加到布局当中

下面再来看看第二行代码里面做了些什么事情
tabContainerView.setAdapter(new MainTabContainerAdapter(getSupportFragmentManager(),
new Fragment[] {new MainFragment(), new WorkFragment(), new AppFragment(), new MineFragment()}));

 public void setAdapter(BaseAdapter baseAdapter) {
        setAdapter(baseAdapter, 0);
 }

 public void setAdapter(BaseAdapter baseAdapter, int index) {
        if (baseAdapter == null) return;
        tabHost.addTabs(baseAdapter, textSize, textColor, selectedTextColor);
        contentViewPager.setAdapter(new TabViewPagerAdapter(baseAdapter.getFragmentManager(), baseAdapter.getFragmentArray()));

        setCurrentItem(index);
 }

BaseAdapter是个抽象类

public abstract class BaseAdapter {

    /**
     *  tab数量
     */
    public abstract int getCount();

    /**
     * tab text 数组
     */
    public abstract String[] getTextArray();

    /**
     * tab icon 数组
     */
    public abstract int[] getIconImageArray();

    /**
     * tab icon 选中 数组
     */
    public abstract int[] getSelectedIconImageArray();

    /**
     * fragment 数组
     */
    public abstract Fragment[] getFragmentArray();

    public abstract FragmentManager getFragmentManager();

}

它提供了容器当中需要的文本,图片,还有内容区的fragment信息

方法里调用TabHost的addTabs方法给TabHost添加Tab

 tabHost.addTabs(baseAdapter, textSize, textColor, selectedTextColor);

 public void addTabs(BaseAdapter baseAdapter, int textSize, int textColor, int selectedTextColor) {
        int count = baseAdapter.getCount();
        String[] textArray = baseAdapter.getTextArray();
        int[] iconImageArray = baseAdapter.getIconImageArray();
        int[] selectedIconImageArray = baseAdapter.getSelectedIconImageArray();

        if (count == 0 || textArray == null || iconImageArray == null || selectedIconImageArray == null) return;
        if (textArray.length != count || iconImageArray.length != count || selectedIconImageArray.length != count) return;

        for (int i = 0; i < count; i++) {
            Tab tab = new Tab(context, textArray[i], textSize, textColor, selectedTextColor, iconImageArray[i], selectedIconImageArray[i], i);
            addTab(tab);
        }
    }

通过方法得到文本,图片等数组,然后通过循环创建Tab对象,添加到TabHost布局当中,我们来看看Tab的构造方法

 public Tab(Context context, String text, int textSize, int textColor, int selectedTextColor, int iconImage, int selectedIconImage, int index) {
        this.context = context;
        this.text = text;
        this.textSize = textSize;
        this.textColor = textColor;
        this.selectedTextColor = selectedTextColor;

        this.iconImage = iconImage;
        this.selectedIconImage = selectedIconImage;
        this.index = index;

        init();
    }

    private void init() {
        initView();

        rootView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                tabSelected();
            }
        });
    }

    private void initView() {
        rootView = new LinearLayout(context);
        rootView.setOrientation(LinearLayout.VERTICAL);
        rootView.setGravity(Gravity.CENTER_HORIZONTAL);
        rootView.setPadding(0, 25, 0, 0);
        LinearLayout.LayoutParams rootViewLp = new LinearLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT);
        rootViewLp.weight = 1;

        rootView.setLayoutParams(rootViewLp);

        /**
         *  icon view
         */
        iconImageView = new ImageView(context);
        iconImageView.setImageResource(iconImage);
        iconImageView.setLayoutParams(new LinearLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT));
        rootView.addView(iconImageView);

        /**
         *  text view
         */
        textTextView = new TextView(context);
        textTextView.setText(text);
        textTextView.setTextColor(textColor);
        textTextView.setTextSize(textSize);
        textTextView.setLayoutParams(new LinearLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT));
        rootView.addView(textTextView);
    }

Tab的构造方法就是给文本和图片的属性设值,添加监听,创建Tab需要的文本,图片布局
接下来给ViewPager设置适配器,添加fragment

 contentViewPager.setAdapter(new TabViewPagerAdapter(baseAdapter.getFragmentManager(), baseAdapter.getFragmentArray()));

这样底部的Tab和内容区域的ViewPager数据都填充完成了

接下来需要设置点击状态的切换,从Tab类的init方法里可以看出给RootView添加了点击事件,onClick方法会调用TabHost设置给Tab的监听回调类,下面的代码就是TabHost给Tab添加的监听

//TabHost里给Tab添加Tab选中的监听
private void addTabChangeListener(Tab tab) {
      tab.setOnTabSelectedListener(new OnTabSelectedListener() {
           @Override
            public void onTabSelected(Tab tab) {
                contentViewPager.setCurrentItem(tab.getIndex());
            }
        });
}

onTabSelected方法里设置contentViewPager当前选中的Item,然后会回调到ViewPager监听类OnPageChangeListener的onPageSelected方法

 public void onPageSelected(int position) {
       tabHost.onChangeTabHostStatus(position);
       Tab selectedTab = tabHost.getTabForIndex(position);
       if (onTabSelectedListener != null && selectedTab != null) 
           onTabSelectedListener.onTabSelected(selectedTab);
 }

首先调用onChangeTabHostStatus方法

 public void onChangeTabHostStatus(int index) {
        for (int i = 0, size = tabList.size(); i < size; i++) {
            Tab tab = tabList.get(i);
            tab.setTabIsSelected(index == i ? true : false);
        }
  }
.```
循环TabList,根据index判断Tab状态的选中与否
然后调用回调监听onTabSelectedListener的onTabSelected方法

到此为止整个项目的实现过程就分析完了。

####结束语
整个项目的实现并没有任何难度,把它封装成一个View是为了以后在项目中更好更快的实现这种效果,提升开发效率

想看完整代码的可以移步至:https://github.com/chenpengfei88/TabContainerView
欢迎大家Star,Follow,谢谢。

TabContainerView V2.0版本(http://www.jianshu.com/p/9aaff43bbf9f  )

我还有一篇封装布局状态切换的文章,大家有兴趣也可以看看
http://www.jianshu.com/p/9d53893b3eda

相关文章

网友评论

  • d5f7a50d45f2:请问一下,导航栏的位置在哪里设置?我想把导航栏设置到顶部,但是木有找到设置的方法
    伪文艺大叔: @寂夜听风语 暂时没有这个功能
  • 73ece15c815f:大哥,你这个少于四个tab 就不显示了,蛋疼
    伪文艺大叔: @小呢个李 没事
    73ece15c815f:@伪文艺大叔 谢谢了,做了一下午东西 脑壳晕。。我忘记把icon 提出来了。。。。谢谢大神了咯。你这个好用啊,我扩展了一下,tab 可上可下。。以后一直用。
    伪文艺大叔:getCount放回的int值要和getTextArray,getIconImageArray,getSelectedIconImageArray大小要一致
  • 年才下:请问一下,底部布局是如何整合到整个布局里去的?感觉光靠findViewById和setAdapter两行代码就只传入了fragment集合,根本就没有实现传入图片和item项的操作啊
    年才下:@伪文艺大叔 好的,明白了,谢谢你的回答。不过我感觉要是能直接给出设置图片和标题的方法,这个封装就更完美了。
    伪文艺大叔:你看下我代码里MainTabContainerAdapter这个类就知道了
  • cbbe7f5c7ae9:撸主写的非常好,学习了
    伪文艺大叔: @cbbe7f5c7ae9 谢谢

本文标题:Android TabContainerView 实现底部导航栏

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