Android 底部导航栏添加消息数目提示
写一篇短小精悍,好用的知识积累吧。开发中时常会出现信息提醒,新内容提示等等一堆问题。其实就是在各种控件或者是item上面加“小圆点”。网上一搜一大堆。。。但是感觉说的好多。我们只需要基本功能2333.
下面介绍三种方式吧,但是大体套路相同。
一、解需求思路
在 RadioGroup 的 RadioButton 上面直接加小圆点,对于我来说实现有点困难,因为我下面还有文字。搞不好,文字就挤没了。 所有我现在在原有的 RadioGroup 上面加一层覆盖物,类似于 ui 常常接触的图层。
二、布局
就是在原有的 RadioGroup 上覆盖一层透明的 button 并且保证位置相同相同
<?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" android:background="@color/colorGrayBg" tools:context="cn.ln80.happybirdcloud119.activity.MainActivity"> <android.support.v4.view.ViewPager android:id="@+id/vp_main_page" android:layout_width="match_parent" android:layout_height="match_parent" android:layout_marginBottom="61dp" /> <!--底栏--> <RadioGroup android:id="@+id/rg_main_bottom" android:layout_width="match_parent" android:layout_height="@dimen/iconsize_60" android:layout_alignParentBottom="true" android:background="@android:color/white" android:orientation="horizontal"> <RadioButton android:id="@+id/rb_main_task" style="@style/BottomRadio" android:drawableTop="@drawable/bg_bottom_task" android:text="可视化" android:textColor="@drawable/txt_bottom_notify" /> <RadioButton android:id="@+id/rb_main_notify" style="@style/BottomRadio" android:layout_height="match_parent" android:drawableTop="@drawable/bg_bottom_notify" android:text="通知" android:textColor="@drawable/txt_bottom_notify" /> <RadioButton android:id="@+id/rb_main_home" style="@style/BottomRadio" android:layout_marginTop="@dimen/widget_margin_5" android:checked="true" android:drawableTop="@drawable/bg_bottom_home" android:text="主页" android:textColor="@drawable/txt_bottom_notify" /> <RadioButton android:id="@+id/rb_main_inspect" style="@style/BottomRadio" android:drawableTop="@drawable/bg_bottom_inspect" android:text="巡检" android:textColor="@drawable/txt_bottom_notify" /> <RadioButton android:id="@+id/rb_main_mine" style="@style/BottomRadio" android:drawableTop="@drawable/bg_bottom_mine" android:text="我的" android:textColor="@drawable/txt_bottom_notify" /> </RadioGroup> <LinearLayout android:layout_width="match_parent" android:layout_height="@dimen/iconsize_60" android:layout_alignParentBottom="true" android:orientation="horizontal"> <Button android:id="@+id/btn_main_task" android:layout_width="0dp" android:layout_height="match_parent" android:layout_weight="1" android:background="@null" /> <Button android:id="@+id/btn_main_notification" android:layout_width="0dp" android:layout_height="match_parent" android:layout_weight="1" android:background="@null" /> <Button android:id="@+id/btn_main_home" android:layout_width="0dp" android:layout_height="match_parent" android:layout_weight="1" android:background="@null" /> <Button android:id="@+id/btn_main_inspect" android:layout_width="0dp" android:layout_height="match_parent" android:layout_weight="1" android:background="@null" /> <Button android:id="@+id/btn_main_my" android:layout_width="0dp" android:layout_height="match_parent" android:layout_weight="1" android:background="@null" /> </LinearLayout> </RelativeLayout>
三、第一种方式
import android.content.Context; import android.graphics.Color; import android.graphics.Typeface; import android.graphics.drawable.ShapeDrawable; import android.graphics.drawable.shapes.RoundRectShape; import android.util.AttributeSet; import android.util.Log; import android.util.TypedValue; import android.view.Gravity; import android.view.View; import android.view.ViewGroup; import android.widget.FrameLayout; import android.widget.LinearLayout; import android.widget.TabWidget; /** * Content: * Actor:韩小呆 ヾ(゚▽゚)ノ * Time: 2018/8/6 17:47 * Update: * Time: */ public class BadgeView extends android.support.v7.widget.AppCompatTextView { private boolean mHideOnNull = true; public BadgeView(Context context) { this(context, null); } public BadgeView(Context context, AttributeSet attrs) { this(context, attrs, android.R.attr.textViewStyle); } public BadgeView(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); init(); } private void init() { if (!(getLayoutParams() instanceof LinearLayout.LayoutParams)) { LinearLayout.LayoutParams layoutParams = new LinearLayout.LayoutParams( android.view.ViewGroup.LayoutParams.WRAP_CONTENT, android.view.ViewGroup.LayoutParams.WRAP_CONTENT, Gravity.END | Gravity.TOP); setLayoutParams(layoutParams); } // set default font setTextColor(Color.WHITE); setTypeface(Typeface.DEFAULT_BOLD); setTextSize(TypedValue.COMPLEX_UNIT_SP, 11); setPadding(dip2Px(5), dip2Px(1), dip2Px(5), dip2Px(1)); // set default background setBackground(9, Color.parseColor("#d3321b")); setGravity(Gravity.CENTER); // default values setHideOnNull(true); setBadgeCount(0); } public void setBackground(int dipRadius, int badgeColor) { int radius = dip2Px(dipRadius); float[] radiusArray = new float[]{radius, radius, radius, radius, radius, radius, radius, radius}; RoundRectShape roundRect = new RoundRectShape(radiusArray, null, null); ShapeDrawable bgDrawable = new ShapeDrawable(roundRect); bgDrawable.getPaint().setColor(badgeColor); setBackground(bgDrawable); } /** * @return Returns true if view is hidden on badge value 0 or null; */ public boolean isHideOnNull() { return mHideOnNull; } /** * @param hideOnNull the hideOnNull to set */ public void setHideOnNull(boolean hideOnNull) { mHideOnNull = hideOnNull; setText(getText()); } /* * (non-Javadoc) * * @see android.widget.TextView#setText(java.lang.CharSequence, android.widget.TextView.BufferType) */ @Override public void setText(CharSequence text, BufferType type) { if (isHideOnNull() && (text == null || text.toString().equalsIgnoreCase("0"))) { setVisibility(View.GONE); } else { setVisibility(View.VISIBLE); } super.setText(text, type); } public void setBadgeCount(int count) { setText(String.valueOf(count)); } public Integer getBadgeCount() { if (getText() == null) { return null; } String text = getText().toString(); try { return Integer.parseInt(text); } catch (NumberFormatException e) { return null; } } public void setBadgeGravity(int gravity) { FrameLayout.LayoutParams params = (FrameLayout.LayoutParams) getLayoutParams(); params.gravity = gravity; setLayoutParams(params); } public int getBadgeGravity() { FrameLayout.LayoutParams params = (FrameLayout.LayoutParams) getLayoutParams(); return params.gravity; } public void setBadgeMargin(int dipMargin) { setBadgeMargin(dipMargin, dipMargin, dipMargin, dipMargin); } public void setBadgeMargin(int leftDipMargin, int topDipMargin, int rightDipMargin, int bottomDipMargin) { FrameLayout.LayoutParams params = (FrameLayout.LayoutParams) getLayoutParams(); params.leftMargin = dip2Px(leftDipMargin); params.topMargin = dip2Px(topDipMargin); params.rightMargin = dip2Px(rightDipMargin); params.bottomMargin = dip2Px(bottomDipMargin); setLayoutParams(params); } public int[] getBadgeMargin() { FrameLayout.LayoutParams params = (FrameLayout.LayoutParams) getLayoutParams(); return new int[]{params.leftMargin, params.topMargin, params.rightMargin, params.bottomMargin}; } public void incrementBadgeCount(int increment) { Integer count = getBadgeCount(); if (count == null) { setBadgeCount(increment); } else { setBadgeCount(increment + count); } } public void decrementBadgeCount(int decrement) { incrementBadgeCount(-decrement); } /* * Attach the BadgeView to the TabWidget * * @param target the TabWidget to attach the BadgeView * * @param tabIndex index of the tab */ public void setTargetView(TabWidget target, int tabIndex) { View tabView = target.getChildTabViewAt(tabIndex); setTargetView(tabView); } /* * Attach the BadgeView to the target view * * @param target the view to attach the BadgeView */ public void setTargetView(View target) { if (getParent() != null) { ((ViewGroup) getParent()).removeView(this); } if (target == null) { return; } if (target.getParent() instanceof FrameLayout) { ((FrameLayout) target.getParent()).addView(this); } else if (target.getParent() instanceof ViewGroup) { ViewGroup parentContainer = (ViewGroup) target.getParent(); int groupIndex = parentContainer.indexOfChild(target); parentContainer.removeView(target); FrameLayout badgeContainer = new FrameLayout(getContext()); ViewGroup.LayoutParams parentLayoutParams = target.getLayoutParams(); badgeContainer.setLayoutParams(parentLayoutParams); target.setLayoutParams(new ViewGroup.LayoutParams( ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT)); parentContainer.addView(badgeContainer, groupIndex, parentLayoutParams); badgeContainer.addView(target); badgeContainer.addView(this); } else if (target.getParent() == null) { Log.e(getClass().getSimpleName(), "ParentView is needed"); } } /* * converts dip to px */ private int dip2Px(float dip) { return (int) (dip * getContext().getResources().getDisplayMetrics().density + 0.5f); } }
将这个代码直接 C、V 到自己的 类内,然后在使用的地方直接填写如下代码:
BadgeView badgeView = new BadgeView(this); badgeView.setTargetView(btnMainNotification); badgeView.setBadgeCount(12); badgeView.setBackground(20, getResources().getColor(R.color.colorRed));
这个自定义的代码是一位外国大佬搞出来的,原版代码在这,欢迎查阅:BadgeView。
这里面的一些属性:
//设置依附的控件 setTargetView(View) //设置显示的数字 setBadgeCount(int) //设置显示的位置 setBadgeGravity(Gravity) //设置背景色 setBackgroundColor() //设置背景图片 setBackgroundResource() //设置显示字体 比如说加粗、倾斜、啥的 setTypeface() //设置字体阴影 setShadowLayer()
四、第二种方式
引用这位大佬的依赖 :BadgeView ---> compile 'q.rorbin:badgeview:1.1.3'
代码代码中使用:
new QBadgeView(this).bindTarget(btnMainNotification).setBadgeText("99+").setBadgeBackgroundColor(getResources().getColor(R.color.colorRed));
可以返回一个 Badge 对象提高开发者操作。
一些常用属性
//设置Badge数字 setBadgeNumber(int i); //设置Badge文本 setBadgeText(String text); //设置文本字体大小 setBadgeTextSize(int size); // 设置文本颜色 setBadgeTextColor(); //设置是否显示精确模式数值 setExactMode(); //设置Badge相对于TargetView的位置 setBadgeGravity(); //设置外边距 setGravityOffset(); //设置内边距 setBadgePadding(); //设置背景色 setBadgeBackgroundColor(); //设置背景图片 setBadgeBackground(); //设置是否显示阴影 setShowShadow(); //打开拖拽消除模式并设置监听 setOnDragStateChangedListener(); //描边 stroke(); //隐藏Badge hide();
五、最后一种实现方式
六、总结
总体的实现思路就是,在底部导航栏覆盖一层透明的布局,然后在透明布局内的控件上面依赖(寄生)上小圆点。

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
Android开发之深度项目设计探索(二)
在《Android开发之深度项目设计探索(一)》 这篇文章中,主要描述了深度项目架构设计会涉及到的一些知识点,那么这篇文章主要介绍的是 RxLifecycle 使用及源码分析。 RxLifecycle : 在Android进行面试的时候,经常会被问到的就是Android的内存泄漏、泄漏场景、泄漏原因以及解决办法。如果你提到了Rxjava2,面试官可能会接连发问Rxjava2在使用过程中会有一些什么问题?这时就可以谈到Rxjava如果在一些特定场景没有及时解绑订阅可能会导致内存泄漏。 泄漏的原因是:当RxJava发布一个订阅后,此时页面执行了finish的生命周期,但订阅逻辑还未完成。如果没有及时取消订阅,就会导致Activity/Fragment无法被回收,从而引发内存泄漏。但是相应的,Rxjava系列也提供了解决办法,那就是使用RxLifecycle 。 RxLifecycle官方文档 通过官方文档了解发现,RxLifecycle库的使用和集成也是简单和快捷的 A:RxLifecycle对应的依赖(最新版本) // 这是RxLifecycle最主要的依赖 implementation...
- 下一篇
3.2 自定义控件基础 之 View的测量
本文对应项目的码云地址:https://gitee.com/wanchuanxy/AndroidHeroesTest/tree/master/3/SystemWidget 在现实生活中,如果我们去画一个图形,就必须知道它的大小和位置。Android系统在绘制View之前也必须对View进行测量,即告诉系统该画一个多大的View。这个过程在onMeasure()方法中进行。 Android系统给我们提供了一个设计短小精悍却功能强大的类----MeasureSpec类,我们可通过它来测量View。MeasureSpec是一个32位的值,其中高2位为测量的模式,低30位为测量的大小,在计算中使用位运算的原因是为了提高并优化速率。 测量模式为以下三种。 EXACTLY 即精确值模式,当我们将控件的layout_width或layout_height属性设定为具体数值时,比如android:layout_width="100dp",或者指定为match_parent属性时(占据父View的大小),系统使用的是EXACTLY模式。 AT_MOST 即最大值模式,当控件的layout_width或l...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- SpringBoot2更换Tomcat为Jetty,小型站点的福音
- CentOS8安装MyCat,轻松搞定数据库的读写分离、垂直分库、水平分库
- CentOS6,CentOS7官方镜像安装Oracle11G
- Jdk安装(Linux,MacOS,Windows),包含三大操作系统的最全安装
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- SpringBoot2全家桶,快速入门学习开发网站教程
- SpringBoot2编写第一个Controller,响应你的http请求并返回结果
- CentOS7安装Docker,走上虚拟化容器引擎之路
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- CentOS8编译安装MySQL8.0.19