前言

前面的一篇文章讲到了 Fragment 的基础知识,本章就让我们使用 Fragment+FrameLayout 布局实现底部导航栏。

实现代码步骤如下:

  • 定义布局文件
  • 写三个 Fragment 的布局文件
  • 写三个自定义的 Fragment 类
  • 编写 mainActivity

看完步骤有了一定的了解后我们就开始动手编码啦!!

思考第一步:怎么定义布局文件呢?

答:将我们定义的布局文件看成一个容器,可以使用各种布局,看各自的喜欢,这里我们使用 FrameLayout (帧布局,记住不要跟 Fragment 混淆了)作为放置 Fragment 的容器,底部栏可以看下图的构造

grafana导航栏面板_xml

可以看到一个底部栏的选项卡需要一个图片和文字构成,所以图片和文字可以用一个 RelativeLayout 包裹,然后放置在一个盛放三个选项卡的大布局中,可以选用线性布局,具体代码实现下面给出
activity_main.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

    <FrameLayout
        android:id="@+id/content"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1" >
    </FrameLayout>

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="70dp"
        android:background="#FFFFFF" >

        <RelativeLayout
            android:id="@+id/course_layout"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1" >

            <LinearLayout
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:layout_centerVertical="true"
                android:orientation="vertical" >

                <ImageView
                    android:id="@+id/course_image"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:layout_gravity="center_horizontal"
                    android:src="@drawable/ic_tabbar_course_normal" />

                <TextView
                    android:id="@+id/course_text"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:layout_gravity="center_horizontal"
                    android:text="课程"
                    android:textColor="#7597B3" />
            </LinearLayout>
        </RelativeLayout>

        <RelativeLayout
            android:id="@+id/found_layout"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1" >

            <LinearLayout
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:layout_centerVertical="true"
                android:orientation="vertical" >

                <ImageView
                    android:id="@+id/found_image"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:layout_gravity="center_horizontal"
                    android:src="@drawable/ic_tabbar_found_normal" />

                <TextView
                    android:id="@+id/found_text"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:layout_gravity="center_horizontal"
                    android:text="发现"
                    android:textColor="#7597B3" />
            </LinearLayout>
        </RelativeLayout>


        <RelativeLayout
            android:id="@+id/setting_layout"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1" >

            <LinearLayout
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:layout_centerVertical="true"
                android:orientation="vertical" >

                <ImageView
                    android:id="@+id/setting_image"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:layout_gravity="center_horizontal"
                    android:src="@drawable/ic_tabbar_settings_normal" />

                <TextView
                    android:id="@+id/setting_text"
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:layout_gravity="center_horizontal"
                    android:text="设置"
                    android:textColor="#7597B3" />
            </LinearLayout>
        </RelativeLayout>
    </LinearLayout>
</LinearLayout>

思考第二步:怎么写三个 Fragment 布局文件?

答:只需要在 layout 文件夹 bulid 三个 xml 文件,这里为了方便测试 Fragment 的布局内容我就只用了一个 TextView,然后复制三份改下颜色就可以了、

frag1.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:gravity="center"
    android:background="#FAECD8">
    <TextView
        android:text="日程Fragment"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />
</LinearLayout>

思考第三步:如何编写三个 Fragment 类?

答:只需要让自定义的类继承 Fragment 并且实现其中的 onCreateView()方法,使用inflater.inflate()将 Fragment 连接进 View ,也是复制三份,修改下参数即可
Fragment1.java

package com.example.administrator.tab;

import android.support.annotation.Nullable;
import android.support.v4.app.Fragment;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;

public class Fragment2 extends Fragment {
    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.frag2,container,false);
        return  view;
    }
}

思考最后一步:如何编写 MainActivity?

答:这点也是最难的一点,MainActivity 要实现的功能就是把整个界面呈现出来,由于我们这个测试界面比较简单,对象也比较少,实现的功能主要就是点击底部栏的同时切换 Fragment 内容,有这么一些步骤和逻辑
1.定义对象,以便通过事件操作对象,这里要定义的对象有 Fragment对象(展示内容),底部栏对象包括 ImageView(动态更换图片) 和 TestView(动态更换文字) ,layout(点击选项卡图片和文字都可切换)
2.重写点击事件,点击选项卡实现内容切换
3.书写重置选项的方法,当点击下一个选项时其他选项重置为可选状态

说了这么多,还是仔细看看具体代码吧!

package com.example.administrator.tab;

import android.graphics.Color;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentActivity;
import android.support.v4.app.FragmentManager;
import android.os.Bundle;
import android.support.v4.app.FragmentTransaction;
import android.view.View;
import android.widget.FrameLayout;
import android.widget.ImageView;
import android.widget.RelativeLayout;
import android.widget.TextView;

public class MainActivity extends FragmentActivity {

    //定义三个 Fragment 对象
    private Fragment1 frag1;
    private Fragment2 frag2;
    private Fragment3 frag3;
    //帧布局对象,就是用来存放 Fragment 的容器
    private FrameLayout fragLayout;

    //定义底部导航栏的三个布局
    private RelativeLayout course_layout;
    private RelativeLayout found_layout;
    private RelativeLayout setting_layout;

    //定义底部导航栏的 ImageView 和 TextView
    //  1
    private ImageView course_image;
    private TextView course_text;

    //  2
    private ImageView found_image;
    private TextView found_text;

    //  3
    private ImageView setting_image;
    private TextView setting_text;

    //定义颜色值
    //定义要用的颜色值
    private int white = Color.parseColor("#FFFFFF");
    private int gray = Color.parseColor("#FFA0A1A7");
    private int blue = Color.parseColor("#FF1F39C9");
    //定义 FragmentManager 对象,需要继承 FragmentActivity否则报错
    FragmentManager fManager;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        fManager = getSupportFragmentManager();
        initView();
    }
    //组件初始化
    private void initView() {
        //课程
        course_layout = (RelativeLayout) findViewById(R.id.course_layout);
        course_image = (ImageView) findViewById(R.id.course_image);
        course_text = (TextView) findViewById(R.id.course_text);

        //发现
        found_layout = (RelativeLayout) findViewById(R.id.found_layout);
        found_image = (ImageView) findViewById(R.id.found_image);
        found_text = (TextView) findViewById(R.id.found_text);

        //设置
        setting_layout = (RelativeLayout) findViewById(R.id.setting_layout);
        setting_image = (ImageView) findViewById(R.id.setting_image);
        setting_text = (TextView) findViewById(R.id.setting_text);

        //点击事件
        course_layout.setOnClickListener(new LayoutOnClickListener());
        found_layout.setOnClickListener(new LayoutOnClickListener());
        setting_layout.setOnClickListener(new LayoutOnClickListener());
    }
    //重写点击事件
    class  LayoutOnClickListener implements View.OnClickListener{
        @Override
        public void onClick(View v) {
                switch (v.getId()){
                    case R.id.course_layout:
                        ChoiceTab(0);
                        break;
                    case R.id.found_layout:
                        ChoiceTab(1);
                        break;
                    case R.id.setting_layout:
                        ChoiceTab(2);
                        break;
                    default:break;
                }
        }
    }
    //定义一个选择 tab 后的处理
    public void ChoiceTab(int index){
        //构建 FragmentTransction 执行 Tab事务
        FragmentTransaction transaction = fManager.beginTransaction();
        //重置选项:没有此步骤会使 tab 之间跳转混乱
        clearchoice();
        //隐藏 Fragment 内容,没有此步骤会使得容器内容混乱
       // hideFragment(transaction);
        //对传入参数选项进行处理并显示容器内容
        switch (index){
            case 0:
                course_image.setImageResource(R.drawable.ic_tabbar_course_pressed);
                course_text.setTextColor(blue);
                course_layout.setBackgroundResource(R.drawable.ic_tabbar_bg_click);
                if(frag1 == null){//如果frag1为空(第一次加载)则添加到界面上
                    frag1 = new Fragment1();
                    transaction.add(R.id.content,frag1);
                }else {//如果不为空则直接显示出来
                    transaction.replace(R.id.content, frag1);
                }
                break;
            case 1:
                found_image.setImageResource(R.drawable.ic_tabbar_found_pressed);
                found_text.setTextColor(blue);
                found_layout.setBackgroundResource(R.drawable.ic_tabbar_bg_click);
                if(frag2 == null){
                    frag2 = new Fragment2();
                    transaction.add(R.id.content,frag2);
                }else {
                    transaction.replace(R.id.content, frag2);
                }
                break;
            case 2:
                setting_image.setImageResource(R.drawable.ic_tabbar_settings_pressed);
                setting_text.setTextColor(blue);
                setting_layout.setBackgroundResource(R.drawable.ic_tabbar_bg_click);
                if(frag3 == null){
                    frag3 = new Fragment3();
                    transaction.add(R.id.content,frag3);
                }else {
                   // transaction.show(frag3);
                    transaction.replace(R.id.content, frag3);

                }
                break;
            default:break;
        }
        transaction.commit();
    }

    //定义重置选项的方法
    private void clearchoice() {
        //课程选项重置
        course_image.setImageResource(R.drawable.ic_tabbar_course_normal);
        course_layout.setBackgroundColor(white);
        course_text.setTextColor(gray);

        //发现选项重置
        found_image.setImageResource(R.drawable.ic_tabbar_found_normal);
        found_layout.setBackgroundColor(white);
        found_text.setTextColor(gray);

        //设置选项重置
        setting_image.setImageResource(R.drawable.ic_tabbar_settings_normal);
        setting_layout.setBackgroundColor(white);
        setting_text.setTextColor(gray);
    }
    //定义一个隐藏 Fragment 的方法,将已经显示的内容隐藏
    private void hideFragment(FragmentTransaction transaction) {
        if(frag1 != null){
            transaction.hide(frag1);
        }
        if (frag2 != null){
            transaction.hide(frag2);
        }
        if(frag3 != null){
            transaction.hide(frag3);
        }
    }
}

有人可能会疑问为什么写的隐藏 Fragment 的方法没有用,那是因为我用 transaction.replace()方法界面被取代了,如果是用 transaction.show的同学,则要使用此隐藏方法否则界面会混乱。

关于 Fragment 实现底部导航栏就简单介绍到这里