Android TabContainerView 实现底部导航栏效果 V1.0

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

推荐阅读更多精彩内容