前言

在开发Android应用时,加新功能是必不可少的,我们加入了新的功能,有的一看界面就可以看出来,但是有的新功能就比较隐蔽,也就是用户很难知道你添加了这个新功能,这个时候就需要用户在打开我们的应用时给出一些提示,说明我们在哪里添加了新功能,点击哪里可以看到这个新功能。这时我们第一时间想到的可能是Toast,因为它用法简单,又不影响用户操作,但是它有个缺点,就是不能明确的指示是哪里添加了新功能,除非你用文字描述出来。为此,我基于Toast编写了一个小组件FloatTextToast(下面遇到的这个名字代替我写的这个组件),他和Toast的用法一样简单,并且弥补了Toast的缺点,也更显得更好看。

效果图

你可以学到

  1. Toast的基本用法
  2. Android的消息机制,如何创建自己的消息队列
  3. 怎样在Activity启动时获取一个View的width、height、top、left等属性

基本思路

  1. 首先你要有一个处理好的9 PNG的图片,用于自适应文字显示,关于9 PNG处理可以参考Android Doc
  2. 要显示在哪个View的下面,就要知道这个目标View的位置
  3. 把要显示的文本放在一个TextView里,使用Toast的setView方法设置Toast要显示的View。
  4. 根据得到的位置,最后就是使用Toast的setGravity方法把要显示的内容放到正确的位置显示出来即可。
总的来说首先就是要知道目标View,根据targetView计算出要显示提示的位置,然后根据位置使用Toast把提示的文本显示出来。但是这里有几个难点,下面就一一解决

Activity加载完成时获取targetVIew的宽高和位置属性

我们加入了新的功能提示,自然会在用户打开这个界面的时候就提示,但是在UI没有渲染完成绑定倒Window上的时候,是不能获取倒targetView的width、height和position的,那么我们怎么才能知道targetView的这些属性呢?Activity的onAttachedToWindow回调方法是不能用的,况且它是在API 5加上的,以前的API中并没有。不过我们还有一种方法,那就是在显示提示的时候获取targetView的属性,如果获取不到(为0)就一直获取,直到获取到为止,这其实是一个轮询。为了达到这一目的,我们在开发者调用FloatTextToast.show()的时候使用Android的Message机制轮询获取一个targetView的属性,如果获取到,就会显示提示文字了。在此之前先看下FloatTextToast构造函数,可以对它有个大概的了解,防止后面的代码中出现的成员变量不认识。

[java] view plaincopyprint?
  1. private FloatTextToast(Context context,View targetView) {
  2. this.mTargetView = targetView;
  3. this.mContext= context;
  4. mToast=new Toast(mContext);
  5. mContentView=new TextView(mContext);
  6. mContentView.setBackgroundResource(R.drawable.float_text_toast_bg);
  7. mContentView.setTextColor(Color.BLACK);
  8. mContentView.setTextSize(TypedValue.COMPLEX_UNIT_DIP,16);
  9. mToast.setView(mContentView);
  10. //初始化一个Handler线程
  11. mHandlerThread=new HandlerThread("FloatTextToast");
  12. mHandlerThread.start();
  13. mHandler=new FloatTextToastHandler(mHandlerThread.getLooper());
  14. }

private FloatTextToast(Context context,View targetView) { this.mTargetView = targetView; this.mContext= context; mToast=new Toast(mContext); mContentView=new TextView(mContext); mContentView.setBackgroundResource(R.drawable.float_text_toast_bg); mContentView.setTextColor(Color.BLACK); mContentView.setTextSize(TypedValue.COMPLEX_UNIT_DIP,16); mToast.setView(mContentView); //初始化一个Handler线程 mHandlerThread=new HandlerThread("FloatTextToast"); mHandlerThread.start(); mHandler=new FloatTextToastHandler(mHandlerThread.getLooper()); }

自定义自己的消息循环机制

要想在一个自定义的组件中使用Message机制,一定要有自己的Looper机制,我们不能使用Activity的Looper,因为主Looper可能会有其他的Message需要处理,这就会导致我们的show方法会延迟调用,这样效果就不好了,所以要有一个专门的Looper来处理此Message。要声明自己的Looper,就需要HandlerThread这个类的配合了,这可是个好东西,使用它你会很容易的创建一个自己的线程用于处理你Message。使用方法很简单,如下代码:
[java] view plaincopyprint?
  1. //初始化一个Handler线程
  2. mHandlerThread=new HandlerThread("FloatTextToast");
  3. mHandlerThread.start();
  4. mHandler=new FloatTextToastHandler(mHandlerThread.getLooper());

//初始化一个Handler线程 mHandlerThread=new HandlerThread("FloatTextToast"); mHandlerThread.start(); mHandler=new FloatTextToastHandler(mHandlerThread.getLooper());
这样就声明了一个HandlerThread并且让它运行,运行之后我们就可以获取一个属于该Thread的Looper,然后把Message发送给这个Looper,那么这个线程就可以处理你发送的消息了。。看看我们的自定义Handler

[java] view plaincopyprint?
  1. private class FloatTextToastHandler extends Handler{
  2. public FloatTextToastHandler(Looper looper) {
  3. super(looper);
  4. }
  5. @Override
  6. public void handleMessage(Message msg) {
  7. switch(msg.what){
  8. case WHAT_SHOW:
  9. showInHandler();
  10. }
  11. }
  12. }

private class FloatTextToastHandler extends Handler{ public FloatTextToastHandler(Looper looper) { super(looper); } @Override public void handleMessage(Message msg) { switch(msg.what){ case WHAT_SHOW: showInHandler(); } } }
它需要传递一个Looper作为构造参数声明,意思就是使用这个Looper处理我发send的Message的意思。上面的代码

[java] view plaincopyprint?
  1. mHandler=new FloatTextToastHandler(mHandlerThread.getLooper());

mHandler=new FloatTextToastHandler(mHandlerThread.getLooper());
正是我们使用自己开启的线程处理我们的Message的意思。下面看下我们的showInHandler()方法是怎么处理的。

[java] view plaincopyprint?
  1. /**在Handler调用的show方法,主要为了等待{@link #mTargetView}的位置*/
  2. private void showInHandler(){
  3. int[] targetPos=getTargetViewPos();
  4. if(targetPos[0]==0&&targetPos[1]==0){
  5. mHandler.sendEmptyMessageDelayed(WHAT_SHOW, 100);
  6. }else{
  7. final Rect contentPos=getContentViewPos(targetPos);
  8. mToast.setGravity(Gravity.LEFT|Gravity.TOP, contentPos.left, contentPos.top);
  9. mToast.show();
  10. }
  11. }

/**在Handler调用的show方法,主要为了等待{@link #mTargetView}的位置*/ private void showInHandler(){ int[] targetPos=getTargetViewPos(); if(targetPos[0]==0&&targetPos[1]==0){ mHandler.sendEmptyMessageDelayed(WHAT_SHOW, 100); }else{ final Rect contentPos=getContentViewPos(targetPos); mToast.setGravity(Gravity.LEFT|Gravity.TOP, contentPos.left, contentPos.top); mToast.show(); } }
该方法其实就是在获取targetVIew的位置,如果获取不到,则向自定义的Looper里发送一个Message重新调用该函数,如果得到了位置,那么就调用Toast的setGravity方法设置好要显示文本的位置,然后显示即可。

获取要显示文本的位置

要获取显示的位置,就要知道targetVIew的位置以及它的宽、高,这样就能计算要显示文本的位置了。View组件都有一个函数,可以把自己在Window里的坐标转换为一个数组。

[java] view plaincopyprint?
  1. private int[] getTargetViewPos(){
  2. final int[] targetPos=new int[2];
  3. mTargetView.getLocationInWindow(targetPos);
  4. return targetPos;
  5. }

private int[] getTargetViewPos(){ final int[] targetPos=new int[2]; mTargetView.getLocationInWindow(targetPos); return targetPos; }这样,就返回了targetView的xy坐标了。光有targetView的坐标还不够,还要有contentView最终要显示的位置。

[java] view plaincopyprint?
  1. /**
  2. * 计算获取浮动文本显示的位置,把浮动文本放在targetView的中心处
  3. * @return 一个包含top和left的Rect
  4. */
  5. private  Rect getContentViewPos(int[] targetPos){
  6. final Rect windowVisibleRect=new Rect();
  7. final View targetView=mTargetView;
  8. final TextView contentView=mContentView;
  9. //状态栏高度
  10. targetView.getWindowVisibleDisplayFrame(windowVisibleRect);
  11. int statusBarHeight=windowVisibleRect.top;
  12. //背景图那个三角箭头的位置
  13. final TextPaint textPaint=contentView.getPaint();
  14. int contentW=(int)textPaint.measureText((String)contentView.getText());
  15. int arrowPos=(int)(contentW*(30.0/160));
  16. final Rect rect = new Rect();
  17. rect.left = targetPos[0]+targetView.getWidth()/2-arrowPos;
  18. rect.top = targetPos[1]-statusBarHeight + targetView.getHeight();
  19. return rect;
  20. }

/** * 计算获取浮动文本显示的位置,把浮动文本放在targetView的中心处 * @return 一个包含top和left的Rect */ private Rect getContentViewPos(int[] targetPos){ final Rect windowVisibleRect=new Rect(); final View targetView=mTargetView; final TextView contentView=mContentView; //状态栏高度 targetView.getWindowVisibleDisplayFrame(windowVisibleRect); int statusBarHeight=windowVisibleRect.top; //背景图那个三角箭头的位置 final TextPaint textPaint=contentView.getPaint(); int contentW=(int)textPaint.measureText((String)contentView.getText()); int arrowPos=(int)(contentW*(30.0/160)); final Rect rect = new Rect(); rect.left = targetPos[0]+targetView.getWidth()/2-arrowPos; rect.top = targetPos[1]-statusBarHeight + targetView.getHeight(); return rect; }
这个函数的功能就是让文本显示在targetView的下方的横向中间的位置,也就是文本的背景尖角三角要指向targetView横向中间的位置,这样才好看些。为了这个才需要使用Paint测量文本的宽度,所以这也是该组件的一个缺陷,不能显示String格式之外的字符,比如SpannableString。

完整的组件代码

上面是对组件代码的拆分讲解,是为了说明我们当时实现这个组件的想法以及步骤,下面就整体把代码列出来,明了的看一下。

[java] view plaincopyprint?
  1. /**
  2. * 浮动的文本显示。根据一个提供的View,可以把文本显示到该View的下面.
  3. * 可以设置显示的时间,多了该时间后自动消失。目前只支持纯文本{@link String}类型的显示
  4. * 因为要计算显示文本的宽度。
  5. * @author michael_li(飞雪无情)
  6. * @since 2011-12-10 下午04:57:36
  7. */
  8. public class FloatTextToast {
  9. public static final int LENGTH_LONG=Toast.LENGTH_LONG;
  10. public static final int LENGTH_SHORT=Toast.LENGTH_SHORT;
  11. private static final int WHAT_SHOW=1;
  12. private Context mContext;
  13. private View mTargetView;
  14. private Toast mToast;
  15. private  TextView mContentView;
  16. private HandlerThread mHandlerThread;
  17. private FloatTextToastHandler mHandler;
  18. private FloatTextToast(Context context,View targetView) {
  19. this.mTargetView = targetView;
  20. this.mContext= context;
  21. mToast=new Toast(mContext);
  22. mContentView=new TextView(mContext);
  23. mContentView.setBackgroundResource(R.drawable.float_text_toast_bg);
  24. mContentView.setTextColor(Color.BLACK);
  25. mContentView.setTextSize(TypedValue.COMPLEX_UNIT_DIP,16);
  26. mToast.setView(mContentView);
  27. //初始化一个Handler线程
  28. mHandlerThread=new HandlerThread("FloatTextToast");
  29. mHandlerThread.start();
  30. mHandler=new FloatTextToastHandler(mHandlerThread.getLooper());
  31. }
  32. /**
  33. * 生成一个FloatTextToast
  34. * @param context Activity 上下文
  35. * @param targetView  目标View,浮动文本要显示在哪个View下面
  36. * @param text 要显示的文本
  37. * @param duration 浮动文本显示的时间 {@link #LENGTH_LONG} {@link #LENGTH_SHORT}
  38. * @return 一个FloatTextToast,可以调用{@link #show()}显示
  39. */
  40. public static FloatTextToast makeText(Context context,View targetView, String text, int duration) {
  41. final FloatTextToast floatToast=new FloatTextToast(context,targetView);
  42. final TextView contentView=floatToast.mContentView;
  43. contentView.setText(text);
  44. floatToast.mToast.setDuration(duration);
  45. return floatToast;
  46. }
  47. /**
  48. * 显示浮动文本
  49. */
  50. public void show(){
  51. mHandler.sendEmptyMessage(WHAT_SHOW);
  52. }
  53. /**在Handler调用的show方法,主要为了等待{@link #mTargetView}的位置*/
  54. private void showInHandler(){
  55. int[] targetPos=getTargetViewPos();
  56. if(targetPos[0]==0&&targetPos[1]==0){
  57. mHandler.sendEmptyMessageDelayed(WHAT_SHOW, 100);
  58. }else{
  59. final Rect contentPos=getContentViewPos(targetPos);
  60. mToast.setGravity(Gravity.LEFT|Gravity.TOP, contentPos.left, contentPos.top);
  61. mToast.show();
  62. }
  63. }
  64. private int[] getTargetViewPos(){
  65. final int[] targetPos=new int[2];
  66. mTargetView.getLocationInWindow(targetPos);
  67. return targetPos;
  68. }
  69. /**
  70. * 计算获取浮动文本显示的位置,把浮动文本放在targetView的中心处
  71. * @return 一个包含top和left的Rect
  72. */
  73. private  Rect getContentViewPos(int[] targetPos){
  74. final Rect windowVisibleRect=new Rect();
  75. final View targetView=mTargetView;
  76. final TextView contentView=mContentView;
  77. //状态栏高度
  78. targetView.getWindowVisibleDisplayFrame(windowVisibleRect);
  79. int statusBarHeight=windowVisibleRect.top;
  80. //背景图那个三角箭头的位置
  81. final TextPaint textPaint=contentView.getPaint();
  82. int contentW=(int)textPaint.measureText((String)contentView.getText());
  83. int arrowPos=(int)(contentW*(30.0/160));
  84. final Rect rect = new Rect();
  85. rect.left = targetPos[0]+targetView.getWidth()/2-arrowPos;
  86. rect.top = targetPos[1]-statusBarHeight + targetView.getHeight();
  87. return rect;
  88. }
  89. private class FloatTextToastHandler extends Handler{
  90. public FloatTextToastHandler(Looper looper) {
  91. super(looper);
  92. }
  93. @Override
  94. public void handleMessage(Message msg) {
  95. switch(msg.what){
  96. case WHAT_SHOW:
  97. showInHandler();
  98. }
  99. }
  100. }
  101. }

/** * 浮动的文本显示。根据一个提供的View,可以把文本显示到该View的下面. * 可以设置显示的时间,多了该时间后自动消失。目前只支持纯文本{@link String}类型的显示 * 因为要计算显示文本的宽度。 * @author michael_li(飞雪无情) * @since 2011-12-10 下午04:57:36 */ public class FloatTextToast { public static final int LENGTH_LONG=Toast.LENGTH_LONG; public static final int LENGTH_SHORT=Toast.LENGTH_SHORT; private static final int WHAT_SHOW=1; private Context mContext; private View mTargetView; private Toast mToast; private TextView mContentView; private HandlerThread mHandlerThread; private FloatTextToastHandler mHandler; private FloatTextToast(Context context,View targetView) { this.mTargetView = targetView; this.mContext= context; mToast=new Toast(mContext); mContentView=new TextView(mContext); mContentView.setBackgroundResource(R.drawable.float_text_toast_bg); mContentView.setTextColor(Color.BLACK); mContentView.setTextSize(TypedValue.COMPLEX_UNIT_DIP,16); mToast.setView(mContentView); //初始化一个Handler线程 mHandlerThread=new HandlerThread("FloatTextToast"); mHandlerThread.start(); mHandler=new FloatTextToastHandler(mHandlerThread.getLooper()); } /** * 生成一个FloatTextToast * @param context Activity 上下文 * @param targetView 目标View,浮动文本要显示在哪个View下面 * @param text 要显示的文本 * @param duration 浮动文本显示的时间 {@link #LENGTH_LONG} {@link #LENGTH_SHORT} * @return 一个FloatTextToast,可以调用{@link #show()}显示 */ public static FloatTextToast makeText(Context context,View targetView, String text, int duration) { final FloatTextToast floatToast=new FloatTextToast(context,targetView); final TextView contentView=floatToast.mContentView; contentView.setText(text); floatToast.mToast.setDuration(duration); return floatToast; } /** * 显示浮动文本 */ public void show(){ mHandler.sendEmptyMessage(WHAT_SHOW); } /**在Handler调用的show方法,主要为了等待{@link #mTargetView}的位置*/ private void showInHandler(){ int[] targetPos=getTargetViewPos(); if(targetPos[0]==0&&targetPos[1]==0){ mHandler.sendEmptyMessageDelayed(WHAT_SHOW, 100); }else{ final Rect contentPos=getContentViewPos(targetPos); mToast.setGravity(Gravity.LEFT|Gravity.TOP, contentPos.left, contentPos.top); mToast.show(); } } private int[] getTargetViewPos(){ final int[] targetPos=new int[2]; mTargetView.getLocationInWindow(targetPos); return targetPos; } /** * 计算获取浮动文本显示的位置,把浮动文本放在targetView的中心处 * @return 一个包含top和left的Rect */ private Rect getContentViewPos(int[] targetPos){ final Rect windowVisibleRect=new Rect(); final View targetView=mTargetView; final TextView contentView=mContentView; //状态栏高度 targetView.getWindowVisibleDisplayFrame(windowVisibleRect); int statusBarHeight=windowVisibleRect.top; //背景图那个三角箭头的位置 final TextPaint textPaint=contentView.getPaint(); int contentW=(int)textPaint.measureText((String)contentView.getText()); int arrowPos=(int)(contentW*(30.0/160)); final Rect rect = new Rect(); rect.left = targetPos[0]+targetView.getWidth()/2-arrowPos; rect.top = targetPos[1]-statusBarHeight + targetView.getHeight(); return rect; } private class FloatTextToastHandler extends Handler{ public FloatTextToastHandler(Looper looper) { super(looper); } @Override public void handleMessage(Message msg) { switch(msg.what){ case WHAT_SHOW: showInHandler(); } } } }

此组件和Toast的实现方法一样,所以上手不难,只需使用makeText静态方法生成一个即可

[java] view plaincopyprint?
  1. FloatTextToast.makeText(Context context, View targetView, String text, int duration).show()

FloatTextToast.makeText(Context context, View targetView, String text, int duration).show()就这么简单,传进去几个参数,show出即可,和Toast一样好用。

小结

这里主要是通过类之间的组合编写一个一个FloatTextToast组件,便于在应用中提示一些信息,不光局限于新功能的提示,还有其他的点击查看个人信息等等,就如上面的效果图一样。这里主要的难点就在于Activity启动获取targetView的状态,这里采用了不受影响的自定义的消息机制,能及时的获取targetView的状态。这里也采用的Toast的队列机制,这样就能够更好的一个个的提示,让用户看完一个再显示另外一个,不至于一下子全显示出来,而用户没有时间看。这里还采用了Paint用于测量文本的真实宽度,所以也有了一些缺陷,如果哪位有更好的方法,也可以留言告知我,不胜感激。
附上组件源代码和效果图的Demo下载 http://download.csdn.net/detail/michael__li/3904636

转载于:https://www.cnblogs.com/zhwl/archive/2012/03/17/2402863.html

基于Android的浮动组件,可以用于应用中的新功能展示等等。相关推荐

  1. 卡路里计算JAVA_pedometer 这是一个基于android平台的软件,用于计算你走路的步数,算出消耗的卡路里,以达到健 238万源代码下载...

    详细说明:这是一个基于android平台的软件,用于计算你走路的步数,算出消耗的卡路里,以达到健康的管理.-This is an android platform based software, us ...

  2. android开发收藏功能实现,Android使用Realm数据库如何实现App中的收藏功能

    Android使用Realm数据库如何实现App中的收藏功能 发布时间:2021-05-07 11:20:34 来源:亿速云 阅读:63 作者:小新 这篇文章主要介绍了Android使用Realm数据 ...

  3. js php通讯录,基于aotu.js实现微信自动添加通讯录中的联系人功能

    什么是Auto.JS? Auto.JS是Android平台上的JavaScript自动化工具. 它的本质是可执行自己编写的简易Javascript脚本的,尤其可以在开启"无障碍模式" ...

  4. chrome android 功能大全,Android版Chrome 55 大更新,多个新功能袭来

    原标题:Android版Chrome 55 大更新,多个新功能袭来 前几天,发布了桌面版Chrome 55 稳定版,而今日Android版Chrome 55 也正式到来,相比之间只是性能优化和BUG修 ...

  5. android8.1新功能,EMUI 8.0 +Android 8.1 你可能从未见过的新功能!

    原标题:EMUI 8.0 +Android 8.1 你可能从未见过的新功能! 被誉为国产机皇,唯一能叫板苹果的华为,在10月16日发布了旗下第一代搭载人工智能AI处理器芯片的手机,这款手机可以说是集齐 ...

  6. 基于android的网络音乐播放器-网络音乐的搜索和展示(五)

    作为android初学者,最近把疯狂android讲义和疯狂Java讲义看了一遍,看到书中介绍的知识点非常多,很难全部记住,为了更好的掌握基础知识点,我将开发一个网络音乐播放器-EasyMusic来巩 ...

  7. Android 8.0 正式发布,Google 带来了哪些新功能?

    文/屠敏 2017 年 8 月 21 日,随着日全食的到来,此前一直猜测是 OREO(奥利奥)还是 Orellete(加泰罗尼亚的点心)的 Android 8.0 最终拉开帷幕,Google 正式采取 ...

  8. birt预览能有内容发布后没内容_谷歌突然推出Android 11开发者预览版 新版带来部分新功能和改进...

    谷歌刚刚在安卓开发者网站放出 Android 11 版的开发者预览版,该版本主要面向开发者提供用来测试新版功能. 在新版本里目前谷歌已经发布部分新功能和改进,随着开发者版本不断更迭后续更多新功能和改进 ...

  9. 谷歌不支持调用摄像头麦克风_谷歌突然推出Android 11开发者预览版 新版带来部分新功能和改进...

    谷歌刚刚在安卓开发者网站放出 Android 11 版的开发者预览版,该版本主要面向开发者提供用来测试新版功能. 在新版本里目前谷歌已经发布部分新功能和改进,随着开发者版本不断更迭后续更多新功能和改进 ...

最新文章

  1. 苹果/三星/华为纷纷布局人工智能 将AI作为公司发展新动力
  2. 东芝收购协议达成 富士通正式退出硬盘市场
  3. 难点电路详解之负反馈放大器电路(1)
  4. linux sed错误sed: -e expression #1, unknown option to `s'解决办法
  5. cf451E. Devu and Flowers(产生不同多重集数量)
  6. 在eclipse中指定启动时java的位置
  7. javascript中的||运算符
  8. 美国太空部队加入美国情报系统,以确保太空的安全
  9. linux8如何开启多个桌面,CentOS8安装GNOME3桌面并设置开机启动图形界面
  10. 日志过滤工具 LogViewer Pro
  11. html按钮 字 颜色代码,html中按钮的字体颜色怎么设置?
  12. ‘字符型‘变量和‘字符串型‘变量
  13. 560套Axure低保真原型打包下载!各行各业产品经理、交互设计师必备资源库!!!
  14. PC游戏史上十大最经典RPG
  15. Metrics-server
  16. android activity pause,关于android:onPause()和onStop()在Activity中
  17. 2018 年 Google 设计亮点回顾
  18. 海尔微型计算机一键还原怎么操作,电脑的一键还原在哪_电脑一键还原详细教程_电脑一键还原怎么操作...
  19. udb和udbx的区别_DB2 UDB,WebSphere和iBATIS
  20. vue(4) - 收藏集 - 掘金

热门文章

  1. js获取当前Frame在父页面中的id
  2. 【转】Linux root修改密码失败
  3. 基于oracle的sql优化
  4. Linux文字分段裁剪命令cut(转)
  5. Win7 + VS2015 + Python3.6编译
  6. 用栈实现队列和用队列实现栈
  7. 6.Python补充_Python之道
  8. Java反射机制大神必学系列之 ,高级与低级的差别在哪里?
  9. Head First设计模式之原型模式
  10. js之字面量、对象字面量的访问、关键字in的用法