android 主页底部菜单tab切换标签

虽然现在谷歌官方也提供了不少控件可解决这个效果,第三方也不少,也几乎所有人都会实现这种效果,但本人仍想记录一下。


这里的实现是使用RadioGroup+FrameLayout实现,自己定义个FragmentTabAdapter,不使用viewpage。

这里从简单的开始吧!

首先tab栏的实现,定义选中和非选中的drawable大家都会的了所以这里就不多说了,对于RadioButton的风格样式肯定是不满足需要的,所以进行下处理吧!先看代码,如下

------------------------------------------------
<style name="tabBarButton" parent="android:Widget.CompoundButton.RadioButton">
        <item name="android:drawablePadding">0dp</item>
        <item name="android:button">@null</item>
        <item name="android:textSize">11dp</item>
        <!--部分机型,如三星N7100,不加padding会出现偏移,导致第一个左边多出很多空白,最后一个靠边-->
        <item name="android:padding">0dp</item>
        <item name="android:maxLines">1</item>
        <item name="android:gravity">center</item>
        <item name="android:layout_gravity">center</item>
        <item name="android:layout_weight">1.0</item>
        <item name="android:layout_width">0dp</item>
        <item name="android:layout_height">wrap_content</item>
        <item name="android:textColor">@color/selector_black_a1</item>
    </style>
---------------------------------------------

在布局中只需设置drawableTop和text即可,如下代码(或者这两个属性直接在代码里写两个数组,进行for循环addView进去也是很方便的)

------------------------------------------------
<RadioButton
            style="@style/tabBarButton"
            android:drawableTop="@android:drawable/btn_star"
            android:text="第三方tool" />
-----------------------------------------------

布局这些都是比较简单的,剩下就是FragmentTabAdapter类了。

这里适配器需接受一下参数(或者也使用抽象类,一些数据外部使用的时候再实现)

1、List<Fragment>   该主页需切换的几个Fragment

2、RadioGroup 我们布局中的RadioGroup,为其添加OnCheckedChangeListener

3、fragmentContentId,布局中FrameLayout的id

4、FragmentActivity,这里只是为了获取FragmentManager,如果不是继承FragmentActivity的,也可改为FragmentManager。这里直接接收activity是为了以防有莫名的需求需要activity对象

所以构造方法如下:

------------------------------------------------
public FragmentTabAdapter(FragmentActivity fragmentActivity, List<Fragment> fragments, int fragmentContentId, RadioGroup rgs) {
        this.fragments = fragments;
        this.rgs = rgs;
        this.fragmentActivity = fragmentActivity;
        this.fragmentContentId = fragmentContentId;
        this.cuurentCheckId = rgs.getCheckedRadioButtonId();

        // 默认显示第一页
        FragmentTransaction ft = fragmentActivity.getSupportFragmentManager().beginTransaction();
        ft.add(fragmentContentId, fragments.get(0));
        ft.commit();
        rgs.setOnCheckedChangeListener(this);
    }
------------------------------------------------

因为我们需要为RadioGroup 添加事件,所以直接就实现 RadioGroup.OnCheckedChangeListener接口了,主要是监听RadioGroup改变选中view时,实时的改变显示的Fragment,实现如下:

----------------------------------------------- 
@Override
    public void onCheckedChanged(RadioGroup radioGroup, int checkedId) {

        for (int i = 0; i < rgs.getChildCount(); i++) {
            if (rgs.getChildAt(i).getId() == checkedId) {
                if (null == onRgsCheckedListener ||
                        !onRgsCheckedListener.onResCheckedChangedBefor(radioGroup, checkedId, i, cuurentCheckId, currentTab)) {
                    Fragment fragment = fragments.get(i);
                    FragmentTransaction ft = obtainFragmentTransaction(i);

                    getCurrentFragment().onPause(); // 暂停当前tab

                    if (fragment.isAdded()) {
                        fragment.onResume(); // 启动目标tab的onResume()
                    } else {
                        ft.add(fragmentContentId, fragment);
                    }
                    showTab(i); // 显示目标tab
                    ft.commit();
                    cuurentCheckId = checkedId;

                    // 如果设置了切换tab额外功能功能接口
                    if (null != onRgsCheckedListener) {
                        onRgsCheckedListener.onRgsCheckedChanged(radioGroup, checkedId, i);
                    }
                }
            }

        }
    }
------------------------------------------------

代码中的showTab()方法则是控制哪个Fragment显示的方法:

-----------------------------------------------
 /**
     * 切换tab
     *
     * @param idx
     */
    private void showTab(int idx) {
        for (int i = 0; i < fragments.size(); i++) {
            Fragment fragment = fragments.get(i);
            FragmentTransaction ft = obtainFragmentTransaction(idx);//我们加个切换动画

            if (idx == i) {
                ft.show(fragment);
            } else {
                ft.hide(fragment);
            }
            ft.commit();
        }
        currentTab = idx; // 更新目标tab为当前tab
    }

    /**
     * 获取一个带动画的FragmentTransaction
     *
     * @param index
     * @return
     */
    private FragmentTransaction obtainFragmentTransaction(int index) {
        FragmentTransaction ft = fragmentActivity.getSupportFragmentManager().beginTransaction();
        // 设置切换动画
        if (index > currentTab) {
            ft.setCustomAnimations(R.anim.slide_left_in, R.anim.slide_left_out);
        } else {
            ft.setCustomAnimations(R.anim.slide_right_in, R.anim.slide_right_out);
        }
        return ft;
    }

------------------------------------------------


至此我们的适配器已经完成了,有人会注意到在onCheckedChanged方法中好像有预留了个接口,是的,这边预留了个接口,以满足特殊需求的需要,如小红点,登录才可切换,双击滚到顶部等,这里不想强制实现所有的方法,所以直接写成了空实现的类,可据需继承实现,代码如下:

/**
     * 切换tab额外功能功能接口
     */
    public static class OnRgsCheckedListener {


        /**改变tab之后
         * @param radioGroup
         * @param checkedId 选择的RadioButton的id
         * @param index 选择的下标
         */
        void onRgsCheckedChanged(RadioGroup radioGroup, int checkedId, int index) {
		//RadioGroup 改变选中View之后呃回调,用于
        }

        /**在改变tab之前
         * @param radioGroup
         * @param checkedId 选择的RadioButton的id
         * @param index  选择的下标
         * @param preCheckedId 当前的RadioButton的id
         * @param preIndex 当前的下标
         * @return
         */
        boolean onResCheckedChangedBefor(RadioGroup radioGroup, int checkedId,
                                         int index, int preCheckedId, int preIndex) {
	    //如某些子页面需要登录后才可访问,即可在此拦截,返回true则不进行页面切换
		
            return false;
        }
    }

另外顺便在此记一下,这中实现,当我们app存在bug,导致app闪退自动重启时,我们的MainActivity是会被回收重新生成调用Oncreate方法的,而系统却默认把fragment缓存了,没有回收,所以但app闪退自动重启后你会发现会有两个以上的Fragment会重叠在一起,我的解决方法是在oncreate方法中先将存在的Fragment全部移除掉,代码如下:

//避免activity蹦溃回收,fragment没回收,activity重启导致fragment重叠现象,

        List<Fragment> temF = getSupportFragmentManager().getFragments();
        if (temF != null) {
            for (Fragment f : temF) {
                removeFragment(f);
            }
        }



原文地址:http://blog.csdn.net/lanqi_x/article/details/78039324




发布了13 篇原创文章 · 获赞 8 · 访问量 3万+

猜你喜欢

转载自blog.csdn.net/lanqi_x/article/details/78039324