热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android自定义APP全局悬浮按钮

这篇文章主要为大家详细介绍了Android自定义APP全局悬浮按钮,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

原本想通过framelayout实现一个悬浮在其他控件上的按钮,但是觉得很麻烦,需要各个界面都要动态填充.于是想到了悬浮窗,就自定一个ImageView用于显示全局按钮.

一、首先因为悬浮窗式的所以要添加权限,对于SDK>=23的需要动态获取权限,我这边用的是22的

 
  

二、通过application获取到全局性的WindowManager的params数据

 private WindowManager.LayoutParams wmParams=new WindowManager.LayoutParams();
  public WindowManager.LayoutParams getMywmParams(){
    return wmParams;
  }

三、自定义ImageView,并实现点击具有状态选择.其中写了一个回调接口用于对点击事件的处理

public class CustomeMovebutton extends ImageView {
  private final int statusHeight;
  int sW;
  int sH;
  private float mTouchStartX;
  private float mTouchStartY;
  private float x;
  private float y;
  private boolean isMove=false;
  private Context context;
  private WindowManager wm = (WindowManager) getContext().getApplicationContext().getSystemService(Context.WINDOW_SERVICE);
  private WindowManager.LayoutParams wmParams = ((MyApplication) getContext().getApplicationContext()).getMywmParams();
  private float mLastX;
  private float mLastY;
  private float mStartX;
  private float mStartY;
  private long mDownTime;
  private long mUpTime;
  private OnSpeakListener listener;

  public CustomeMovebutton(Context context) {
    this(context,null);
    this.cOntext= context;
  }
  public CustomeMovebutton(Context context, AttributeSet attrs) {
    this(context, attrs,-1);
  }
  public CustomeMovebutton(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs,defStyleAttr);
    sW = wm.getDefaultDisplay().getWidth();
    sH = wm.getDefaultDisplay().getHeight();
    statusHeight = getStatusHeight(context);
  }

  /**
   * 状态栏的高度
   *
   */
  public static int getStatusHeight(Context context) {
    int statusHeight = -1;
    try {
      Class clazz = Class.forName("com.android.internal.R$dimen");  //使用反射获取实例
      Object object = clazz.newInstance();
      int height = Integer.parseInt(clazz.getField("status_bar_height")
          .get(object).toString());
      statusHeight = context.getResources().getDimensionPixelSize(height);
    } catch (Exception e) {
      e.printStackTrace();
    }
    return statusHeight;
  }
  @Override
  public boolean onTouchEvent(MotionEvent event) {
    //获取相对屏幕的坐标,即以屏幕左上角为原点
    x = event.getRawX();
    y = event.getRawY() - statusHeight;  //statusHeight是系统状态栏的高度
    switch (event.getAction()) {
      case MotionEvent.ACTION_DOWN:  //按下
        setImageResource(R.drawable.btn_voice_pressed);
        mTouchStartX = event.getX();
        mTouchStartY = event.getY();
        mStartX = event.getRawX();
        mStartY = event.getRawY();
        mDownTime = System.currentTimeMillis();
        isMove = false;
        break;
      case MotionEvent.ACTION_MOVE:  //手指移动
        updateViewPosition();
        isMove = true;
        break;
      case MotionEvent.ACTION_UP:  //手抬起
        setImageResource(R.drawable.btn_voice_rest);
        mLastX = event.getRawX();
        mLastY = event.getRawY();
        mUpTime = System.currentTimeMillis();
        //按下到抬起的时间大于500毫秒,并且抬手到抬手绝对值大于20像素处理点击事件
        if(mUpTime - mDownTime <500){
          if(Math.abs(mStartX- mLastX )<20.0 && Math.abs(mStartY - mLastY) <20.0){
              if (listener!=null){
                listener.onSpeakListener();
              }
          }
        }

        break;
    }
    return true;
  }

  private void updateViewPosition() {
    wmParams.x = (int) (x - mTouchStartX);
    wmParams.y = (int) (y- mTouchStartY);
    wm.updateViewLayout(this, wmParams); //刷新显示
  }

  /**
   * 设置点击回调接口
   */
  public interface OnSpeakListener{
    void onSpeakListener();
  }
  public void setOnSpeakListener(OnSpeakListener listener){
    this.listener=listener;
  }
}

四、Activity中使用,其中有设置图片的参数和位置参数

public class MainActivity extends AppCompatActivity{

  private WindowManager wm;
  private WindowManager.LayoutParams wmParams;
  private CustomeMovebutton CustomeMovebutton;
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    wm = (WindowManager) getApplicationContext().getSystemService(Context.WINDOW_SERVICE);
    DisplayMetrics dm = getResources().getDisplayMetrics();
    int widthPixels = dm.widthPixels;
    int heightPixels = dm.heightPixels;
    wmParams = ((MyApplication) getApplication()).getMywmParams();
    wmParams.type = WindowManager.LayoutParams.TYPE_SYSTEM_ALERT;
    wmParams.format= PixelFormat.RGBA_8888;//设置背景图片
    wmParams.flags= WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE ;//
    wmParams.gravity = Gravity.LEFT|Gravity.TOP;//
    wmParams.x = widthPixels-150;  //设置位置像素
    wmParams.y = heightPixels-110;
    wmParams.width=200; //设置图片大小
    wmParams.height=200;
    CustomeMovebutton = new CustomeMovebutton(getApplicationContext());
    CustomeMovebutton.setImageResource(R.drawable.btn_voice_rest);
    wm.addView(CustomeMovebutton, wmParams);
    CustomeMovebutton.setOnSpeakListener(new CustomeMovebutton.OnSpeakListener() {
      @Override
      public void onSpeakListener() {
        Toast.makeText(MainActivity.this, "点击事件", Toast.LENGTH_SHORT).show();
      }
    });
  }

  @Override
  protected void onDestroy() {
    super.onDestroy();
    if(CustomeMovebutton != null){
      wm.removeView(CustomeMovebutton);
    }
  }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 深入解析Android自定义View面试题
    本文探讨了Android Launcher开发中自定义View的重要性,并通过一道经典的面试题,帮助开发者更好地理解自定义View的实现细节。文章不仅涵盖了基础知识,还提供了实际操作建议。 ... [详细]
  • 国内BI工具迎战国际巨头Tableau,稳步崛起
    尽管商业智能(BI)工具在中国的普及程度尚不及国际市场,但近年来,随着本土企业的持续创新和市场推广,国内主流BI工具正逐渐崭露头角。面对国际品牌如Tableau的强大竞争,国内BI工具通过不断优化产品和技术,赢得了越来越多用户的认可。 ... [详细]
  • 本文介绍了如何使用jQuery根据元素的类型(如复选框)和标签名(如段落)来获取DOM对象。这有助于更高效地操作网页中的特定元素。 ... [详细]
  • 本文将详细介绍如何使用剪映应用中的镜像功能,帮助用户轻松实现视频的镜像效果。通过简单的步骤,您可以快速掌握这一实用技巧。 ... [详细]
  • 深入理解Cookie与Session会话管理
    本文详细介绍了如何通过HTTP响应和请求处理浏览器的Cookie信息,以及如何创建、设置和管理Cookie。同时探讨了会话跟踪技术中的Session机制,解释其原理及应用场景。 ... [详细]
  • 本文介绍如何在 Xcode 中使用快捷键和菜单命令对多行代码进行缩进,包括右缩进和左缩进的具体操作方法。 ... [详细]
  • 本文介绍了一款用于自动化部署 Linux 服务的 Bash 脚本。该脚本不仅涵盖了基本的文件复制和目录创建,还处理了系统服务的配置和启动,确保在多种 Linux 发行版上都能顺利运行。 ... [详细]
  • 在Linux系统中配置并启动ActiveMQ
    本文详细介绍了如何在Linux环境中安装和配置ActiveMQ,包括端口开放及防火墙设置。通过本文,您可以掌握完整的ActiveMQ部署流程,确保其在网络环境中正常运行。 ... [详细]
  • Android 渐变圆环加载控件实现
    本文介绍了如何在 Android 中创建一个自定义的渐变圆环加载控件,该控件已在多个知名应用中使用。我们将详细探讨其工作原理和实现方法。 ... [详细]
  • 如何在WPS Office for Mac中调整Word文档的文字排列方向
    本文将详细介绍如何使用最新版WPS Office for Mac调整Word文档中的文字排列方向。通过这些步骤,用户可以轻松更改文本的水平或垂直排列方式,以满足不同的排版需求。 ... [详细]
  • 本文总结了在使用Ionic 5进行Android平台APK打包时遇到的问题,特别是针对QRScanner插件的改造。通过详细分析和提供具体的解决方法,帮助开发者顺利打包并优化应用性能。 ... [详细]
  • 理解存储器的层次结构有助于程序员优化程序性能,通过合理安排数据在不同层级的存储位置,提升CPU的数据访问速度。本文详细探讨了静态随机访问存储器(SRAM)和动态随机访问存储器(DRAM)的工作原理及其应用场景,并介绍了存储器模块中的数据存取过程及局部性原理。 ... [详细]
  • 360SRC安全应急响应:从漏洞提交到修复的全过程
    本文详细介绍了360SRC平台处理一起关键安全事件的过程,涵盖从漏洞提交、验证、排查到最终修复的各个环节。通过这一案例,展示了360在安全应急响应方面的专业能力和严谨态度。 ... [详细]
  • 几何画板展示电场线与等势面的交互关系
    几何画板是一款功能强大的物理教学软件,具备丰富的绘图和度量工具。它不仅能够模拟物理实验过程,还能通过定量分析揭示物理现象背后的规律,尤其适用于难以在实际实验中展示的内容。本文将介绍如何使用几何画板演示电场线与等势面之间的关系。 ... [详细]
  • 本文介绍如何通过Windows批处理脚本定期检查并重启Java应用程序,确保其持续稳定运行。脚本每30分钟检查一次,并在需要时重启Java程序。同时,它会将任务结果发送到Redis。 ... [详细]
author-avatar
津pig
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有