超简单,几行代码搞定Android底部导航栏

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了超简单,几行代码搞定Android底部导航栏相关的知识,希望对你有一定的参考价值。

参考技术A 咳咳,答应过年增加新功能的,没想到拖到现在,延迟了一个来月,尴尬,尴尬
那个,我们先忽略这尴尬的事情吧,进入正题才是最重要滴

老规矩,先上效果图:

跟原来的图有个很明显的区别,你们也一定都发现了,对不对。那么显眼的小红点,一定都看到了吧。

当然除了这个,还增加了一项功能,虽然不是很明显,但相信也有小伙伴发现了吧,截图的这俩手机屏幕明显大小不同,但是底部导航栏的大小还是相差不大滴。

是的,你们没有看多,这次不仅增加了小红点功能,还增加了底部导航栏的适配,你没有听错,以后底部导航栏也不用那些dp、sp了,都按照UI妹子们标注的px来就可以了,再也不用为了底部导航栏去跟UI妹子解释啥叫dp了。

好了,效果图展示完了,现在该进入枯燥的使用介绍了。

由于这次改动有点大,所以,先介绍下上个稳定版本的用法,到底是用最新的,还是用原来的,就看各位小伙伴的意愿了

上个稳定版本是1.1.3的,引用方式如下
compile 'com.hjm:BottomTabBar:1.1.3'

具体用法如下(备注都加好了,我也就不多废话了):

最新版本是1.2.2的,引用方式如下
compile 'com.hjm:BottomTabBar:1.2.2'

其实1.2.0与1.1.3区别并不大,只有4点改动:

现在默认的,分割线高度都是设置的1个像素。这里以后也固定都用这个默认的高度了,不再对外提供修改的方法。

这就是新增加的适配了,多的也不说了,你们都懂的

标准尺寸,就是UI妹子给你提供的效果图的屏幕尺寸,只要在init()方法里添加上标准尺寸,你就可以放肆的使用px了

这个方法就是控制小红点显示的方法了,index就是需要显示或者隐藏小红点的TabItem,isShow是一个boolean类型的参数,他是控制小红点是否显示的,如果为true,就会显示小红点;如果为false,就会隐藏小红点

1.2.2版本新增了两个方法

介绍到这里,超简单的底部导航栏,第二阶段就可以告一段落了。以后还会持续优化,完善的。
第三阶段我打算封装一下有中间凸起的底部导航栏,这个功能我本地已经做了,但是封装进去的时候,封装的不理想,这次就没有上线,留作下次了。

最后,再上个 GitHub 地址

Android 底部(简单,简易)导航栏(Fragment)

 

首先编写activity的布局

<?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"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <RelativeLayout
        android:id="@+id/tab_rg"
        android:layout_width="match_parent"
        android:layout_height="48dp"
        android:background="@color/white"
        >
        <TextView
            android:id="@+id/txt_topbar"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:layout_centerInParent="true"
            android:gravity="center"
            android:text="message"
            android:textColor="@color/text_gray"
            android:textSize="18dp"/>
        <View
            android:layout_width="match_parent"
            android:layout_height="2px"
            android:layout_alignParentBottom="true"
            android:background="@color/text_gray"/>
    </RelativeLayout>
    <LinearLayout
        android:id="@+id/tab_group"
        android:layout_width="match_parent"
        android:layout_height="56dp"
        android:background="@color/white"
        android:layout_alignParentBottom="true"
        android:orientation="horizontal"
        >
        <TextView
            android:id="@+id/rd_menu_index"
            style="@style/tab_menu_item"
            android:text="index"
            android:textColor="@drawable/tab_text"

            />
        <TextView
            android:id="@+id/rd_menu_Rand"
            style="@style/tab_menu_item"
            android:text="Rand"
            android:textColor="@drawable/tab_text"

            />
    </LinearLayout>

    <View
        android:id="@+id/div_tab_bar"
        android:layout_width="match_parent"
        android:layout_height="2px"
        android:background="@color/text_gray"
        android:layout_above="@id/tab_group"/>
    <FrameLayout
        android:id="@+id/fragment_container"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_above="@id/tab_group"
        android:layout_below="@id/tab_rg"
        android:background="@color/white"/>


</RelativeLayout>

 

效果:

乍一看代码那么多,实际上在这个方法里底部导航并不需要那么多,上面的relativelayout布局实际上对底部导航并没有多大用,底部导航只需要下面的layout布局,然后将其设为水平布局,主要的还是 android:layout_alignParentBottom="true" 这是将layout布局放在父组件的底部,之后在其内部添加textview就可以大概实现上方的效果了,当然样式要好看的话,我们还需要将textview设置权重。

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

可能你已经发现了这个layout_weight并没有在textview控件内写,而是写在style下的item里,这个属性可以抽取相同的属性。这个一般写在values包下的themes.xml里,在对应的布局里使用 style = " " 就可以获取到属性了。

创建fragment布局和类

这个布局是用作底部导航导向的页面

fragment布局:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@color/white"
    android:orientation="vertical"
    tools:context=".FirstFragment">

    <TextView
        android:id="@+id/txt_content"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_gravity="center"
        android:text="hello"
        android:textColor="@color/text_gray"
        android:textSize="20sp"/>

</LinearLayout>

fragment类:

package com.example.try2;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.appcompat.app.AppCompatActivity;
import androidx.fragment.app.Fragment;

import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;

public class FirstFragment extends Fragment 
    private String context;
    private TextView mTextView;

    public FirstFragment(String context) 
        this.context = context;
    

    @Nullable
    @Override
    public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup     
        container, @Nullable Bundle savedInstanceState) 
        View view = inflater.inflate(R.layout.activity_first_fragment, container, false);
        mTextView = (TextView)view.findViewById(R.id.txt_content);
        mTextView.setText(context);
        return view;
    

 

先来说明一下onCreateView的每个参数

LayoutInflater:是用来找res/layout/下的xml布局文件,并且实例化。使用的inflate方法和findViewById很像不过后者是找xml布局文件下的具体widget控件(如 Button、TextView等)。

ViewGroup container:表示容器。

Bundle savedInstanceState:保存当前的状态,在活动的生命周期中,只要离开了可见阶段,活动可能就会被进程终止,这种机制能保存当时的状态。

 

编写activity的类

package com.example.try2;

import static android.media.CamcorderProfile.get;

import androidx.appcompat.app.AppCompatActivity;
import androidx.fragment.app.FragmentManager;
import androidx.fragment.app.FragmentTransaction;
//import android.app.FragmentTransaction;


import android.graphics.drawable.Drawable;
import android.os.Bundle;
import android.view.View;
import android.view.Window;
import android.widget.FrameLayout;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity implements View.OnClickListener

    private TextView topBar;
    private TextView tab_index;
    private TextView tab_Rand;

    private FrameLayout ly_content;

    private FirstFragment f1, f2, f3, f4;
//    private second f2;
    private FragmentManager fragmentManager;


    @Override
    protected void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        bindView();

    

    private void bindView() 
        topBar = (TextView) this.findViewById(R.id.txt_topbar);
        tab_index = (TextView) this.findViewById(R.id.rd_menu_index);
        tab_Rand = (TextView) this.findViewById(R.id.rd_menu_Rand);

        tab_index.setOnClickListener(this);
        tab_Rand.setOnClickListener(this);
    

    //重置所有文本的选中状态
    public void selected() 
        tab_index.setSelected(false);
        tab_Rand.setSelected(false);
    

    //隐藏Fragment
    public void hideAllFragment(FragmentTransaction transaction) 
        if(f1 != null) 
            transaction.hide(f1);
        
        if(f2 != null) 
            transaction.hide(f2);
        
        if(f3 != null) 
            transaction.hide(f3);
        
        if(f4 != null) 
            transaction.hide(f4);
        
    

    @Override
    public void onClick(View view) 
        FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
        hideAllFragment(transaction);
        switch (view.getId()) 
            case R.id.rd_menu_index:
                selected();
                tab_index.setSelected(true);
                if(f1 == null) 
                    f1 = new FirstFragment("第一个Fragment");
                    transaction.add(R.id.fragment_container, f1);
                 else 
                    transaction.show(f1);
                
                break;
            case R.id.rd_menu_Rand:
                selected();
                tab_Rand.setSelected(true);
                if(f2 == null) 
                    f2 = new FirstFragment("第二个Fragment");
                    transaction.add(R.id.fragment_container, f2);
                 else 
                    transaction.show(f2);
                
                break;
        
        transaction.commit();
    

效果:

 

 

实际上这是同一个类,不过在fragment里添加了构造方法,switch里使用时添加不同的文本,但本质上还是两个页面,毕竟都开辟了空间,如果想要有不同排版的页面,可以像fragment布局和类在写一个相似的,之后在switch里使用就可以了

最后到这里就差不多就结束了,如果还想在导航处添加一些图片可以使用drawableLeft,drawableRight,drawableTop,drawableBottom等属性。

最后感谢各位阅读,如果文章中有错误或者您有什么需要补充,也欢迎您直接留言评论。

 

 

以上是关于超简单,几行代码搞定Android底部导航栏的主要内容,如果未能解决你的问题,请参考以下文章

Android如何隐藏底部虚拟按键

android 底部导航栏

带有 url 更新和超链接支持的 Flutter web 底部导航栏

Android 底部(简单,简易)导航栏(Fragment)

超好用的移动框架--Ionic--构建APP侧边栏 底部选项卡 轮播图 加载动画

Android 底部导航栏在 webview 中登录后显示