android 应用内部悬浮可拖动按钮简单实现代码

 更新时间:2017年10月30日 17:17:49   作者:君莫醉  
本篇文章主要介绍了android 应用内部悬浮可拖动按钮简单实现代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

本文介绍了android 应用内部悬浮可拖动按钮简单实现代码,分享给大家,具体如下:

可以悬浮在activity上面,在加载fragment时悬浮按钮不会消失

实现方式很简单,因为是在应用内部拖动的,只需要通过Activity获取WindowManager,然后将要拖动的view设置上去就行
设置代码:

WindowManager wm = (WindowManager) activity.getSystemService(Context.WINDOW_SERVICE);
    DisplayMetrics dm = new DisplayMetrics();
    activity.getWindowManager().getDefaultDisplay().getMetrics(dm);
    //通过像素密度来设置按钮的大小
    dpi = dpi(dm.densityDpi);
    //屏宽
    screenWidth = wm.getDefaultDisplay().getWidth();
    //屏高
    screenHeight = wm.getDefaultDisplay().getHeight();
    //布局设置
    wmParams = new WindowManager.LayoutParams();
    // 设置window type
    wmParams.type = WindowManager.LayoutParams.TYPE_APPLICATION;
    wmParams.format = PixelFormat.RGBA_8888; // 设置图片格式,效果为背景透明
    wmParams.gravity = Gravity.LEFT | Gravity.TOP;
    // 设置Window flag
    wmParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
    wmParams.width = dpi;
    wmParams.height = dpi;
    wmParams.y = (screenHeight - dpi) >> 1;
    wm.addView(this, wmParams);

控件的大小根据像素密度来进行设置的

  /**
   * 根据密度选择控件大小
   *
   */
  private int dpi(int densityDpi) {
    if (densityDpi <= 120) {
      return 36;
    } else if (densityDpi <= 160) {
      return 48;
    } else if (densityDpi <= 240) {
      return 72;
    } else if (densityDpi <= 320) {
      return 96;
    }
    return 108;
  }

主要的处理问题就是控件的拖动问题,通过重写onTouchEvent方法进行处理

源码:

import android.app.Activity;
import android.content.Context;
import android.graphics.PixelFormat;
import android.util.DisplayMetrics;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.WindowManager;
import android.widget.RelativeLayout;
import android.widget.TextView;


/**
 * Created by xiang on 2016/12/28.
 *
 * im悬浮窗视图
 */

public class ChatView extends RelativeLayout{

  // 悬浮栏位置
  private final static int LEFT = 0;
  private final static int RIGHT = 1;
  private final static int TOP = 3;
  private final static int BUTTOM = 4;

  private int dpi;
  private int screenHeight;
  private int screenWidth;
  private WindowManager.LayoutParams wmParams;
  private WindowManager wm;
  private float x, y;
  private float mTouchStartX;
  private float mTouchStartY;
  private boolean isScroll;

  public ChatView(Activity activity) {
    super(activity);
    LayoutInflater.from(activity).inflate(R.layout.view_chat, this);
    setBackgroundResource(R.drawable.chat_btn);
    wm = (WindowManager) activity.getSystemService(Context.WINDOW_SERVICE);
    DisplayMetrics dm = new DisplayMetrics();
    activity.getWindowManager().getDefaultDisplay().getMetrics(dm);
    //通过像素密度来设置按钮的大小
    dpi = dpi(dm.densityDpi);
    //屏宽
    screenWidth = wm.getDefaultDisplay().getWidth();
    //屏高
    screenHeight = wm.getDefaultDisplay().getHeight();
    //布局设置
    wmParams = new WindowManager.LayoutParams();
    // 设置window type
    wmParams.type = WindowManager.LayoutParams.TYPE_APPLICATION;
    wmParams.format = PixelFormat.RGBA_8888; // 设置图片格式,效果为背景透明
    wmParams.gravity = Gravity.LEFT | Gravity.TOP;
    // 设置Window flag
    wmParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
    wmParams.width = dpi;
    wmParams.height = dpi;
    wmParams.y = (screenHeight - dpi) >> 1;
    wm.addView(this, wmParams);
    hide();
  }


  /**
   * 根据密度选择控件大小
   *
   */
  private int dpi(int densityDpi) {
    if (densityDpi <= 120) {
      return 36;
    } else if (densityDpi <= 160) {
      return 48;
    } else if (densityDpi <= 240) {
      return 72;
    } else if (densityDpi <= 320) {
      return 96;
    }
    return 108;
  }

  public void show() {
    if (isShown()) {
      return;
    }
    setVisibility(View.VISIBLE);
  }


  public void hide() {
    setVisibility(View.GONE);
  }

  public void destory() {
    hide();
    wm.removeViewImmediate(this);
  }


  @Override
  public boolean onTouchEvent(MotionEvent event) {
    // 获取相对屏幕的坐标, 以屏幕左上角为原点
    x = event.getRawX();
    y = event.getRawY();
    switch (event.getAction()) {
      case MotionEvent.ACTION_DOWN:
        // setBackgroundDrawable(openDrawable);
        // invalidate();
        // 获取相对View的坐标,即以此View左上角为原点
        mTouchStartX = event.getX();
        mTouchStartY = event.getY();
        break;
      case MotionEvent.ACTION_MOVE:
        if (isScroll) {
          updateViewPosition();
        } else {
          // 当前不处于连续滑动状态 则滑动小于图标1/3则不滑动
          if (Math.abs(mTouchStartX - event.getX()) > dpi / 3
              || Math.abs(mTouchStartY - event.getY()) > dpi / 3) {
            updateViewPosition();
          } else {
            break;
          }
        }
        isScroll = true;
        break;
      case MotionEvent.ACTION_UP:
        // 拖动
        if (isScroll) {
          autoView();
          // setBackgroundDrawable(closeDrawable);
          // invalidate();
        } else {
          // 当前显示功能区,则隐藏
          // setBackgroundDrawable(openDrawable);
          // invalidate();

        }
        isScroll = false;
        mTouchStartX = mTouchStartY = 0;
        break;
    }
    return true;
  }

  /**
   * 自动移动位置
   */
  private void autoView() {
    // 得到view在屏幕中的位置
    int[] location = new int[2];
    getLocationOnScreen(location);
    //左侧
    if (location[0] < screenWidth / 2 - getWidth() / 2) {
      updateViewPosition(LEFT);
    } else {
      updateViewPosition(RIGHT);
    }
  }

  /**
   * 手指释放更新悬浮窗位置
   *
   */
  private void updateViewPosition(int l) {
    switch (l) {
      case LEFT:
        wmParams.x = 0;
        break;
      case RIGHT:
        int x = screenWidth - dpi;
        wmParams.x = x;
        break;
      case TOP:
        wmParams.y = 0;
        break;
      case BUTTOM:
        wmParams.y = screenHeight - dpi;
        break;
    }
    wm.updateViewLayout(this, wmParams);
  }

  // 更新浮动窗口位置参数
  private void updateViewPosition() {
    wmParams.x = (int) (x - mTouchStartX);
    //是否存在状态栏(提升滑动效果)
    // 不设置为全屏(状态栏存在) 标题栏是屏幕的1/25
    wmParams.y = (int) (y - mTouchStartY - screenHeight / 25);
    wm.updateViewLayout(this, wmParams);
  }
}

使用方法:

//传入上下文Activity
ChatView chatView = new ChatView(this);
chatView.show();

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

相关文章

  • 手把手教你用Android自定义饼状图

    手把手教你用Android自定义饼状图

    最近在项目中需要用到简单的饼形图统计,我们就没必要去找什么开源的,可以尝试自己画一下。现在将实现的过程分享给大家,有需要的朋友们可以参考借鉴,下面来一起看看吧。
    2016-10-10
  • Android动画之雷达扫描效果

    Android动画之雷达扫描效果

    雷达扫描效果在我们日常会经常看到,比如在新浪微博上有一个雷达功能,感觉类似于微信附近的人。只是多了一个类似于雷达扫描效果的动画,某些知名安全软件也有这样的雷达效果,因此今天在这里小编带着大家学习一下。
    2016-08-08
  • Android应用开发中Fragment间通信的实现教程

    Android应用开发中Fragment间通信的实现教程

    这篇文章主要介绍了Android应用开发中Fragment间通信的实现教程,包括接口的定义实现与Fragment通信的动静态加载等,需要的朋友可以参考下
    2016-02-02
  • Android应用中使用XmlSerializer序列化XML数据的教程

    Android应用中使用XmlSerializer序列化XML数据的教程

    这篇文章主要介绍了Android应用中使用XmlSerializer序列化XML数据的教程,XmlSerializer序列化XML同时也是将数据写为XML格式的基本方法,需要的朋友可以参考下
    2016-04-04
  • Android Lock锁实现原理详细分析

    Android Lock锁实现原理详细分析

    这篇文章主要介绍了Android Lock锁实现原理,Lock接口的实现类提供了比使用synchronized关键字更加灵活和广泛的锁定对象操作,而且是以面向对象的方式进行对象加锁
    2023-02-02
  • 详解Android 8.0以上系统应用如何保活

    详解Android 8.0以上系统应用如何保活

    这篇文章主要介绍了详解Android 8.0以上系统应用如何保活,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-08-08
  • android studio3.3.1代码提示忽略大小写的设置

    android studio3.3.1代码提示忽略大小写的设置

    这篇文章主要介绍了android studio3.3.1代码提示忽略大小写的设置,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-03-03
  • Android OnFocuChangeListener焦点事件详解

    Android OnFocuChangeListener焦点事件详解

    这篇文章主要为大家详细介绍了Android OnFocuChangeListener焦点事件,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-09-09
  • Android避免内存溢出(Out of Memory)方法汇总

    Android避免内存溢出(Out of Memory)方法汇总

    这篇文章主要为大家详细介绍了Android避免内存溢出Out of Memory方法汇总,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-01-01
  • Android自定义View app更新动画详解

    Android自定义View app更新动画详解

    这篇文章给大家分享了Android自定义View app更新动画的相关代码以及知识点内容,有兴趣的朋友参考学习下。
    2018-07-07

最新评论