美文网首页自定义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