美文网首页Android开发之路Android技术知识Android开发
fragment+viewpager实现底部导航栏(二)

fragment+viewpager实现底部导航栏(二)

作者: 奔跑的佩恩 | 来源:发表于2017-06-24 23:09 被阅读212次
写在前面的话

首先需要说明的是,此篇文章是实现底部导航栏的第二部,而不是实现顶部导航栏的,注意注意啊。
之前已经写过一篇实现底部导航栏的文章,不知道请移步fragment+viewPager实现底部导航栏(一)

鉴于之前已经实现了底部导航栏的功能,为什么还要写这篇文章呢?此文目的有:

  • 对不需要显示小圆点的底部导航栏一个相对快捷的实现,不需要复杂的TextView实现底部布局
  • 在之前文章的基础上重新整理思路,让编码更顺畅,也算是对上篇文章的一个整理吧。

那么此篇文章除了没有加显示小圆点的布局外,在代码和布局上比之前相对简单的地方在哪里呢?一个是此篇文章用RadioButton实现底部布局,所以在布局上明显比之前简单是肯定的,再之之前是通过控制TextView的点击事件来实现fragment的实现的,所以代码上复杂繁琐些,而此篇文章由于有RadioGroup的原因,在点击事件上的处理会相对简洁些。

下面就来一次讲解实现吧。

提前准备

项目中用到基本的控件初始化,ButterKnife,所以添加依赖

 //butterKnife
    compile 'com.jakewharton:butterknife:8.5.1'
    //这条千万不能忘记!!
    annotationProcessor 'com.jakewharton:butterknife-compiler:8.5.1'

然后是基本的两个base类,BaseActivity和BaseFragment

public abstract class BaseActivity extends AppCompatActivity {

    protected View mLayoutView;//总布局
    protected Activity mContext;
    private Unbinder mUnbinder;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        mContext = this;
        AppActivityManager.getInstance().addActivity(this);

        if (getContentViewId() != 0) {
            mLayoutView = LayoutInflater.from(mContext).inflate(getContentViewId(), null);
            setContentView(mLayoutView);
            mUnbinder = ButterKnife.bind(this);
        }

        initData();
        setListener();
    }

    @Override
    protected void onResume() {
        //设置为横屏
        if (getRequestedOrientation() != ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE) {
            setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT);
        }
        super.onResume();
    }

    /**
     * 设置布局
     */
    public abstract int getContentViewId();


    /**
     * 界面数据初始化
     **/
    protected abstract void initData();

    /**
     * 界面UI事件监听
     **/
    protected abstract void setListener();

    @Override
    protected void onDestroy() {
        if (mUnbinder != null) {
            mUnbinder.unbind();
        }
        AppActivityManager.getInstance().finishActivity(this);
        super.onDestroy();
    }

    /**
     * 获取editText的值
     *
     * @param et
     * @return
     */
    protected String getTextOfEditText(EditText et) {
        if (et == null) {
            return null;
        }

        if (et.getText() == null) {
            return null;
        }

        if (StringUtil.isEmpty(et.getText().toString())) {
            return "";
        }
        return et.getText().toString().trim();
    }

    protected void showToast(String msg) {
        if (StringUtil.isNotEmpty(msg)) {
            ToastUtil.show(msg);
        }
    }

    protected void showShortToast(String msg){
        if (StringUtil.isNotEmpty(msg)) {
            ToastUtil.shortShow(msg);
        }
    }
}

下面是baseFragment,注意fragment相关的导入的都是 import android.support.v4.app.Fragment;

public abstract class BaseFragment extends Fragment{

    protected View mLayoutView;
    protected Context mContext;
    private Unbinder mUnbinder;

    @Override
    public void onAttach(Context context) {
        super.onAttach(context);
        mContext = context;
    }

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        mLayoutView = inflater.inflate(getLayoutId(), container, false);
        mUnbinder= ButterKnife.bind(this,mLayoutView);//绑定framgent
        onCreateFragmentView(inflater, container, savedInstanceState);
        return mLayoutView;
    }

    protected void onCreateFragmentView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        initData();
        setListener();
    }

    @Override
    public void onDestroy() {
        if(mUnbinder!=null){
            mUnbinder.unbind();
        }
        super.onDestroy();
    }

    protected abstract int getLayoutId();
    protected abstract void initData();
    protected abstract void setListener();

    /**
     * 获取editText的值
     *
     * @param et
     * @return
     */
    protected String getTextOfEditText(EditText et){
        if (et == null) {
            return null;
        }
        if (et.getText() == null) {
            return null;
        }
        if (StringUtil.isEmpty(et.getText().toString())) {
            return "";
        }
        return et.getText().toString().trim();
    }

    protected void showToast(String msg) {
        if (StringUtil.isNotEmpty(msg)) {
            ToastUtil.show(msg);
        }
    }

    protected void showShortToast(String msg){
        if (StringUtil.isNotEmpty(msg)) {
            ToastUtil.shortShow(msg);
        }
    }
}

第一步,准备一个基本的TextView的style供xml文件中使用,在res/values/styles.xml中新建style

  <style name="tv_text">
        <item name="android:lineSpacingExtra">@dimen/dp_1</item>
        <item name="android:lineSpacingMultiplier">1.0</item>
    </style>

第二步,写一个RadioButton样式的通用style---tab_menu_item

<style name="tab_menu_item">
        <item name="android:layout_width">0dp</item>
        <item name="android:layout_weight">1</item>
        <item name="android:layout_height">match_parent</item>
        <item name="android:background">@drawable/tab_menu_bg</item>
        <item name="android:button">@null</item>
        <item name="android:gravity">center</item>
        <item name="android:paddingTop">3dp</item>
        <item name="android:textColor">@drawable/tab_menu_text</item>
        <item name="android:textSize">18sp</item>
    </style>

background中tab_menu_bg.xml文件如下:

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:drawable="@drawable/tab_menu_pressed"  android:state_checked="true" />
    <item android:drawable="@drawable/tab_menu_normal"  android:state_checked="false"/>
</selector>

tab_menu_pressed.xml代码如下

<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android" >
    <solid android:color="#d4dbe2"></solid>
</shape>

tab_menu_normal.xml代码如下

<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android" >
    <solid android:color="@color/white"></solid>
</shape>

textColo中tab_menu_text.xml对应如下

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:color="@color/color_f79e03" android:state_checked="true" />
    <item android:color="@color/color_596b76" android:state_checked="false"/>
</selector>

第二步,写一个Frgment的样例吧,ManagerFragment代码如下:

public class ManagerFragment extends BaseFragment{


    @Override
    protected int getLayoutId() {
        return R.layout.fragment_manager;
    }

    @Override
    protected void initData() {

    }

    @Override
    protected void setListener() {

    }
}

对应布局fragment_manager.xml如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="guanli"/>

</LinearLayout>

其它的ReportFragment和InfoFragment仿照上面的ManagerFragment写即可。

第三步,涉及到ViewPager的话,肯定少不了对应的适配器MainPagerAdapter了

/***
 * 标签主页适配
 *
 * @author pei
 * @version 1.0
 * @create 2016-6-21
 */
public class MainPagerAdapter extends FragmentPagerAdapter {

    private List<Fragment> mFragmentList;

    public MainPagerAdapter(FragmentManager fm, List<Fragment> mFragmentList) {
        super(fm);
        this.mFragmentList = mFragmentList;
    }

    @Override
    public Fragment getItem(int position) {
        // TODO Auto-generated method stub
        return mFragmentList == null ? null : mFragmentList.get(position);
    }

    @Override
    public int getCount() {
        // TODO Auto-generated method stub
        return mFragmentList == null ? 0 : mFragmentList.size();
    }

    @Override
    public void destroyItem(View container, int position, Object object) {
        //        super.destroyItem(container, position, object);
    }
}

第四步,ViewPager对应的滑动监听,我自己写了一个TabOnPageChangeListener来实现ViewPager.OnPageChangeListener,为的是去掉viewpager滑动到边界出现的阴影,另外为了减少mainActivity中的代码干扰,只暴露一个ViewPager滑动完毕的方法。

package com.orderform.Listener;

import android.support.v4.view.ViewPager;
import android.support.v4.widget.EdgeEffectCompat;

import java.lang.reflect.Field;

/***
 * TabMenuFragmentActivity  viewpager滑动监听(禁用滑到边界显示黑边)
 *
 * @author pei
 * @version 1.0
 * @create 2016-6-21
 */
public class TabOnPageChangeListener implements ViewPager.OnPageChangeListener {

    private ViewPager mViewPager;
    private OnSelectedFragmentListener mOnSelectedFragmentListener;

    private EdgeEffectCompat leftEdge;//去除viewpager滑到边界的黑边
    private EdgeEffectCompat rightEdge;//去除viewpager滑到边界的黑边

    public TabOnPageChangeListener(ViewPager viewPager) {
        this.mViewPager = viewPager;
        init();
    }

    public void setOnSelectedFragmentListener(OnSelectedFragmentListener onSelectedFragmentListener) {
        this.mOnSelectedFragmentListener = onSelectedFragmentListener;
    }

    private void init() {
        try {
            Field leftEdgeField = mViewPager.getClass().getDeclaredField("mLeftEdge");
            Field rightEdgeField = mViewPager.getClass().getDeclaredField("mRightEdge");
            if (leftEdgeField != null && rightEdgeField != null) {
                leftEdgeField.setAccessible(true);
                rightEdgeField.setAccessible(true);
                leftEdge = (EdgeEffectCompat) leftEdgeField.get(mViewPager);
                rightEdge = (EdgeEffectCompat) rightEdgeField.get(mViewPager);
            }
        } catch (NoSuchFieldException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

    @Override
    public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
        // 禁用滑到边界显示黑边
        if (leftEdge != null && rightEdge != null) {
            leftEdge.finish();
            rightEdge.finish();
            leftEdge.setSize(0, 0);
            rightEdge.setSize(0, 0);
        }
    }


    @Override
    public void onPageScrollStateChanged(int state) {
        //state的状态有三个,0表示什么都没做,1正在滑动,2滑动完毕
    }

    @Override
    public void onPageSelected(int position) {
        if (mOnSelectedFragmentListener != null) {
            mOnSelectedFragmentListener.selectedFragment(position);
        }
    }

    public interface OnSelectedFragmentListener {

        void selectedFragment(int index);
    }
}

第五步,终于要开始mainActivity的相关了,先上mainActivity的布局activity_main.xml代码吧

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".ui.main.MainActivity">


    <include
        layout="@layout/layout_title_back_tv"
        android:layout_width="match_parent"
        android:layout_height="@dimen/dp_50"
        android:background="@color/white"></include>

    <RadioGroup
        android:id="@+id/rg_tab_bar"
        android:layout_width="match_parent"
        android:layout_height="@dimen/dp_56"
        android:layout_alignParentBottom="true"
        android:background="@color/white"
        android:orientation="horizontal">

        <RadioButton
            android:id="@+id/rb_manager"
            style="@style/tab_menu_item"
            android:drawableTop="@drawable/tab_manager_bg"
            android:text="@string/tab_manager"/>

        <RadioButton
            android:id="@+id/rb_report"
            style="@style/tab_menu_item"
            android:drawableTop="@drawable/tab_report_bg"
            android:text="@string/tab_report" />

        <RadioButton
            android:id="@+id/rb_info"
            style="@style/tab_menu_item"
            android:drawableTop="@drawable/tab_info_bg"
            android:text="@string/tab_info" />

    </RadioGroup>

    <android.support.v4.view.ViewPager
        android:id="@+id/vpager"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_above="@+id/rg_tab_bar"
        android:layout_below="@+id/title_rl" />

</RelativeLayout>

这里涉及到一个标题栏的问题,给出inculd的布局layout_title_back_tv.xml代码吧:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/title_rl"
    android:layout_width="match_parent"
    android:layout_height="?attr/actionBarSize"
    android:background="@color/color_00000000"
    android:focusable="true"
    android:focusableInTouchMode="true"
    android:gravity="center_vertical">

    <ImageView
        android:id="@+id/title_back"
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
        android:layout_alignParentLeft="true"
        android:paddingLeft="@dimen/dp_15"
        android:paddingRight="@dimen/dp_15"
        android:src="@mipmap/ic_back_black"
        android:visibility="visible" />

    <TextView
        android:id="@+id/title_name"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        style="@style/tv_text"
        android:layout_centerHorizontal="true"
        android:layout_centerVertical="true"
        android:layout_gravity="center_horizontal"
        android:singleLine="true"
        android:textColor="@color/color_ffffff"
        android:textSize="@dimen/sp_14" />

    <TextView
        android:id="@+id/title_right"
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
        style="@style/tv_text"
        android:layout_alignParentRight="true"
        android:layout_gravity="right"
        android:paddingLeft="@dimen/dp_15"
        android:paddingRight="@dimen/dp_15"
        android:textSize="@dimen/sp_12"
        android:gravity="center"
        android:textColor="@color/color_ffffff"
        android:visibility="gone" />

第六步,MainActivity代码

public class MainActivity extends BaseActivity implements View.OnClickListener, RadioGroup.OnCheckedChangeListener, TabOnPageChangeListener.OnSelectedFragmentListener {

    public static Intent newIndexIntent(Context context) {
        Intent newIntent = new Intent(context, MainActivity.class);
        return newIntent;
    }

    @BindView(R.id.title_rl)
    RelativeLayout mTitleLayout;
    @BindView(R.id.title_back)
    ImageView mTilteBack;
    @BindView(R.id.title_name)
    TextView mTilteName;
    @BindView(R.id.title_right)
    TextView mTilteRight;

    @BindView(R.id.rg_tab_bar)
    RadioGroup mRgTabBar;
    @BindView(R.id.rb_manager)
    RadioButton mRbManager;
    @BindView(R.id.rb_report)
    RadioButton mRbReport;
    @BindView(R.id.rb_info)
    RadioButton mRbInfo;
    @BindView(R.id.vpager)
    ViewPager mViewPager;

    private List<Fragment> mFragmentList;
    private MainPagerAdapter mainPagerAdapter;
    private TabOnPageChangeListener mTabOnPageChangeListener;

    @Override
    public int getContentViewId() {
        return R.layout.activity_main;
    }

    @Override
    protected void initData() {
        mFragmentList = new ArrayList<>();
        mFragmentList.add(0, new ManagerFragment());
        mFragmentList.add(1, new ReportFragment());
        mFragmentList.add(2, new InfoFragment());

        mTabOnPageChangeListener = new TabOnPageChangeListener(mViewPager);
        mainPagerAdapter = new MainPagerAdapter(getSupportFragmentManager(), mFragmentList);
        mViewPager.setOffscreenPageLimit(3);// 设置预加载Fragment个数
        mViewPager.setAdapter(mainPagerAdapter);
        mViewPager.setCurrentItem(0);// 设置当前显示标签页为第一页
        //获取第一个单选按钮,并设置其为选中状态
        mRbManager.setChecked(true);
    }

    @Override
    protected void setListener() {
        mRgTabBar.setOnCheckedChangeListener(this);
        mViewPager.addOnPageChangeListener(mTabOnPageChangeListener);
        mTabOnPageChangeListener.setOnSelectedFragmentListener(this);
        mTilteBack.setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        switch (v.getId()) {
            case R.id.title_back:
                AppActivityManager.getInstance().appExit(mContext);
                break;
            case 1:
                break;
            default:
                break;
        }
    }

    @Override
    public void onCheckedChanged(RadioGroup group, @IdRes int checkedId) {
        setSelected();
        switch (checkedId) {
            case R.id.rb_manager:
                mRbManager.setChecked(true);
                mViewPager.setCurrentItem(0);
                break;
            case R.id.rb_report:
                mRbReport.setChecked(true);
                mViewPager.setCurrentItem(1);
                break;
            case R.id.rb_info:
                mRbInfo.setChecked(true);
                mViewPager.setCurrentItem(2);
                break;
            default:
                break;
        }
    }

    @Override
    public void selectedFragment(int index) {
        setSelected();
        switch (index) {
            case 0:
                mRbManager.setChecked(true);
                mViewPager.setCurrentItem(0);
                break;
            case 1:
                mRbReport.setChecked(true);
                mViewPager.setCurrentItem(1);
                break;
            case 2:
                mRbInfo.setChecked(true);
                mViewPager.setCurrentItem(2);
                break;
        }
    }

    //重置所有文本的选中状态
    private void setSelected() {
        mRbManager.setChecked(false);
        mRbReport.setChecked(false);
        mRbInfo.setChecked(false);
    }
}

好了,到此你就可以欢乐的撸码了,这次的逻辑应该比上次清楚多了,下面给出我的效果图吧


2.gif

ok,今天就讲到这里吧,谢谢诶。

相关文章

网友评论

    本文标题:fragment+viewpager实现底部导航栏(二)

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