超简单,几行代码搞定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底部导航栏的主要内容,如果未能解决你的问题,请参考以下文章
带有 url 更新和超链接支持的 Flutter web 底部导航栏
Android 底部(简单,简易)导航栏(Fragment)