Fragment的应用之底部导航栏的实现(一)

效果图

废话少说,直接上效果图

底部导航栏(一).gif

最近因为实验室的一个项目要大改,一直没时间来学习学习新的东西写篇博客,不过断断续续花了两周的时间总算是把项目弄好了。这篇博客主要是一周前写的Fragment的后续,都知道现在主流APP基本都会有底部导航栏,比如微信,微博,虎扑什么的无一例外使用了底部导航的设计。
本博客所记录的底部导航的实现可能在目前的实际应用中的应用价值不大,这里主要是为了巩固Fragment的使用。

底部导航栏布局

开始我准备使用TextView来实现底部导航的tab,但是我发现在textview中,使用drawable_top属性来指定文字上方附带的图标时,对于图标的大小有很大的限制,不能对图标进行缩放。所以我最后还是选择了使用TextView+ImageView的Linearlayout的方式来组成tab。

资源文件

因为底部导航栏的每个tab需要有点击和未点击两种UI状态,所以需要为其准备资源文件。
例如图中的第一个tab主页,为Imageview建立一个xml资源文件home_back.xml;

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_selected="true" android:drawable="@drawable/ic_home_select"/>
    <item android:drawable="@drawable/ic_home"/>
</selector>

然后为TextView建立一个xml资源文件text_color_back.xml;

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_selected="true" android:color="@color/colorTextSelect"/>
    <item android:color="@android:color/white"/>
</selector>

其他的三个tab的资源文件跟上面形式一样,照葫芦画瓢就行,这里安利两个图标网站,Android Material 材料风格图标LOGO生成器阿里巴巴图标库,只能说见到这两个网站有一种想见恨晚的感觉。

主界面布局

准备好了资源文件之后就可以编写主界面的布局,主界面的布局主要就包括底部的四个竖直方向LinearLayout,而每一个LinearLayout都是由ImageView和TextView组成;
布局另外的部分就主要是提供给显示Fragment,这里用FrameLayout进行定位,主界面布局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="com.android.tu.fragmenttest.MainActivity">


    <LinearLayout
        android:id="@+id/tab_linear"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentBottom="true"
        android:orientation="horizontal"
        android:background="@color/colorPrimary">
        <LinearLayout
            android:id="@+id/linear_home"
            style="@style/TabLinear">

            <ImageView
                style="@style/TabImageStyle"
                android:src="@drawable/home_back"/>
            <TextView
                android:text="主页"
                android:textColor="@drawable/text_color_back"
                style="@style/TabTextStyle"/>

        </LinearLayout>
        <LinearLayout
            android:id="@+id/linear_list"
            style="@style/TabLinear">

            <ImageView
                style="@style/TabImageStyle"
                android:src="@drawable/list_back"/>
            <TextView
                android:text="消息"
                android:textColor="@drawable/text_color_back"
                style="@style/TabTextStyle"/>

        </LinearLayout>
        <LinearLayout
            android:id="@+id/linear_polymer"
            style="@style/TabLinear">

            <ImageView
                style="@style/TabImageStyle"
                android:src="@drawable/polymer_back"/>
            <TextView
                android:text="扩展"
                android:textColor="@drawable/text_color_back"
                style="@style/TabTextStyle"/>

        </LinearLayout>
        <LinearLayout
            android:id="@+id/linear_user"
            style="@style/TabLinear">

            <ImageView
                style="@style/TabImageStyle"
                android:src="@drawable/user_back" />
            <TextView
                android:text="设置"
                android:textColor="@drawable/text_color_back"
                style="@style/TabTextStyle"/>

        </LinearLayout>

    </LinearLayout>

    <FrameLayout
        android:id="@+id/fragment_frame"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_above="@+id/tab_linear">

    </FrameLayout>

</RelativeLayout>

其他的就不多说了,有一点要说的就是在底部的布局中都用到了style,因为这四个tab中的控件都有多个属性相同,所以就提取出来为一个style,在styles.xml文件中定义,例如

<style name="TabImageStyle">
        <item name="android:layout_width">35dp</item>
        <item name="android:layout_height">35dp</item>
        <item name="android:layout_gravity">center_horizontal</item>
</style>

这是底部ImageView提取出来的共同属性,其他的以此类推。

Fragment布局

在实际应用中一般一个Fragmen会对应一个自己的xml布局文件,但在此demo为了演示简单就只用了一个布局文件。

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

    <TextView
        android:id="@+id/fragment_text"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"
        android:gravity="center"
        android:textSize="30sp"/>


</RelativeLayout>

自定义Fragment

布局文件都完成了,下面就是处理各种逻辑交互了,首先自定义一个Fragment,引入相应布局

public class FragmentTest extends Fragment{

    private String fragmentText;

    private TextView fragmentTextView;

    public FragmentTest(String fragmentText) {
        this.fragmentText=fragmentText;
    }

    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        View view=inflater.inflate(R.layout.test_fragment,container,false);
        fragmentTextView= (TextView) view.findViewById(R.id.fragment_text);
        fragmentTextView.setText(fragmentText);
        return view;
    }
}

这段代码基本就是前一篇文中说到的Fragment的基本使用的知识了,在onCreatView中加载布局文件,另外还给Fragment定义了一个带String类型的构造函数,用于让TextView来显示不同文字以区分不同的Fragment

底部导航的实现

前面的工作基本都是在做准备,在MainActivity中实现底部导航的功能才是重点。
首先可以说一下实现的大体思路,要导航就是点击相应的tab就切换显示对应的Fragment,所以其实很简单。

  1. 首先为每一个tab的LinearLayout设置点击事件的监听,当点击tab时,将所有的Fragment都隐藏起来;
  2. 然后先把所有的tab设置为未选中的状态,再把点击的tab设为选中的状态;
  3. 最后就是切换到对应Fragment,先判断当前tab对应的Fragment是否有初始化,若已经初始化就直接显示出来,若没有初始化,就先利用fragment的带一个String参数的构造函数初始化一个,然后利用FragmentTransaction的实例将该Fragment添加到对应的布局;

代码如下:

public class MainActivity extends AppCompatActivity implements View.OnClickListener{

    LinearLayout homeLinear;

    LinearLayout listLinear;

    LinearLayout polyLinear;

    LinearLayout userLinear;

    FragmentTest fragmentHome,fragmentList,fragmentPoly,fragmentUser;

    private FragmentManager mfragmentManger;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        homeLinear= (LinearLayout) findViewById(R.id.linear_home);
        listLinear= (LinearLayout) findViewById(R.id.linear_list);
        polyLinear= (LinearLayout) findViewById(R.id.linear_polymer);
        userLinear= (LinearLayout) findViewById(R.id.linear_user);
        homeLinear.setOnClickListener(this);
        listLinear.setOnClickListener(this);
        polyLinear.setOnClickListener(this);
        userLinear.setOnClickListener(this);
        mfragmentManger = getSupportFragmentManager();
        homeLinear.performClick();
    }

    @Override
    public void onClick(View view) {
        FragmentTransaction fragmentTransaction = mfragmentManger.beginTransaction();//只能是局部变量,不能为全局变量,否则不能重复commit
                                                                                     //FragmentTransaction只能使用一次
        hideAllFragment(fragmentTransaction);
        switch (view.getId()){
            case R.id.linear_home:
                setAllFalse();
                homeLinear.setSelected(true);
                if (fragmentHome==null){
                    fragmentHome=new FragmentTest("Home");
                    fragmentTransaction.add(R.id.fragment_frame,fragmentHome);
                }else{
                    fragmentTransaction.show(fragmentHome);
                }
                break;
            case R.id.linear_list:
                setAllFalse();
                listLinear.setSelected(true);
                if(fragmentList==null){
                    fragmentList=new FragmentTest("List");
                    fragmentTransaction.add(R.id.fragment_frame,fragmentList);
                }else {
                    fragmentTransaction.show(fragmentList);
                }
                break;
            case R.id.linear_polymer:
                setAllFalse();
                polyLinear.setSelected(true);
                if(fragmentPoly==null){
                    fragmentPoly=new FragmentTest("Polymer");
                    fragmentTransaction.add(R.id.fragment_frame,fragmentPoly);
                }else {
                    fragmentTransaction.show(fragmentPoly);
                }
                break;
            case R.id.linear_user:
                setAllFalse();
                userLinear.setSelected(true);
                if(fragmentUser==null){
                    fragmentUser=new FragmentTest("User");
                    fragmentTransaction.add(R.id.fragment_frame,fragmentUser);
                }else {
                    fragmentTransaction.show(fragmentUser);
                }
                break;
        }
        fragmentTransaction.commit();//记得必须要commit,否则没有效果

    }

    private void hideAllFragment(FragmentTransaction fragmentTransaction) {
        if(fragmentHome!=null){
            fragmentTransaction.hide(fragmentHome);
        }
        if(fragmentList!=null){
            fragmentTransaction.hide(fragmentList);
        }
        if(fragmentPoly!=null){
            fragmentTransaction.hide(fragmentPoly);
        }
        if(fragmentUser!=null){
            fragmentTransaction.hide(fragmentUser);
        }
    }

    private void setAllFalse() {
        homeLinear.setSelected(false);
        listLinear.setSelected(false);
        polyLinear.setSelected(false);
        userLinear.setSelected(false);
    }
}

这样就已经实现了底部导航的效果,其实很简单,主要的知识都是在Fragment的基本使用中提到过的,就有一个performClick方法我是第一次用,这个是为了模拟一次点击事件,让主界面打开的时候初始化某一个Fragment。

这篇文章因为主要是为了巩固Fragment的使用,没有什么新的知识点,所以以代码为主,后续还会写一些底部导航栏的其他实现方式。

欢迎批评指正,小弟不胜感激。

2017.4.10 15:37
806实验室

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

推荐阅读更多精彩内容