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();

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

(0)

相关推荐

  • Android自定义View实现拖动选择按钮

    本文为大家分享了Android实现拖动选择按钮的具体代码,供大家参考,具体内容如下 效果图 View代码 第一步:自定义属性 <declare-styleable name="DragView"> <attr name="icon_drag" format="reference"/> <attr name="color_circle" format="color"/> &

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

    本文介绍了android 应用内部悬浮可拖动按钮简单实现代码,分享给大家,具体如下: 可以悬浮在activity上面,在加载fragment时悬浮按钮不会消失 实现方式很简单,因为是在应用内部拖动的,只需要通过Activity获取WindowManager,然后将要拖动的view设置上去就行 设置代码: WindowManager wm = (WindowManager) activity.getSystemService(Context.WINDOW_SERVICE); DisplayMetr

  • Android应用内悬浮窗Activity的简单实现

    目录 前言 缩放方法 悬浮样式 点击穿透空白 移动悬浮窗 例子的完整代码 styles.xml layout activity 运行效果 小结 前言 悬浮窗是一种比较常见的需求.例如把视频通话界面缩小成一个悬浮窗,然后用户可以在其他界面上处理事情. 本文给出一个简单的应用内悬浮窗实现.可缩小activity和还原大小.可悬浮在同一个app的其他activity上.使用TouchListener监听触摸事件,拖动悬浮窗. 缩放方法 缩放activity需要使用WindowManager.Layou

  • Android顶部工具栏和底部工具栏的简单实现代码

    废话少说,直接上图,有图有真相. 这两个工具栏全是用布局来实现的.底部工具栏布局代码: 代码 复制代码 代码如下: < xmlns:android="http://schemas.android.com/apk/res/android"    android:background="@drawable/bottom"    android:layout_width="fill_parent"    android:layout_height

  • Qt实现可拖动按钮

    本文实例为大家分享了Qt实现可拖动按钮的具体代码,供大家参考,具体内容如下 直接上代码 self-contained.h #ifndef SELFCONTAINED_H #define SELFCONTAINED_H #include <QWidget> #include <QPainter> #include <QTimer> #include <QImage> #include <QMouseEvent> #include <QVect

  • android控件实现单击拖动效果

    本文实例为大家分享了android控件实现单击拖动效果的具体代码,供大家参考,具体内容如下 分析 setOnClickListener setOnClickListener 可以接收控件的单击动作,无返回值 要想拖动控件,就要利用 setOnTouchListener 自己定义了 setOnTouchListener 通过按下时的坐标和移动时的坐标之间的差值,来计算移动的距离,然后更改控件的位置以达到拖动的目的 因 Touch 事件执行完成后如果返回false则会继续执行单击事件,不是我们想要的

  • 圣诞节,写个程序练练手————Android 全界面悬浮按钮实现

    开始我以为悬浮窗口,可以用Android中得PopupWindow 来实现,虽然也实现了,但局限性非常大.比如PopupWindow必须要有载体View,也就是说,必须要指定在那个View的上面来实现.以该View为相对位置,来显示PopupWindow.这就局限了其智能在用户交互的窗口上,相对的显示.而无法自由的拖动位置和在桌面显示. 于是查阅了一些资料,有两种实现方法.一种是自定义Toast,Toast是运行于所有界面之上的,也就是说没有界面可以覆盖它.另一种是Android中得Compat

  • 微信小程序实现可拖动悬浮图标(包括按钮角标的实现)

    在制作商城类微信小程序的过程中,我们经常会碰到需要增加可拖动悬浮图标的情况,本文简单的介绍一下可拖动悬浮按钮的实现. 运行截图: 主要代码: js: var startPoint Page({ data: { //按钮位置参数 buttonTop: 0, buttonLeft: 0, windowHeight: '', windowWidth: '', //角标显示数字 corner_data:0, }, onLoad:function(){ //定义角标数字 this.setData({ co

  • Android 中FloatingActionButton(悬浮按钮)实例详解

    Android 中FloatingActionButton(悬浮按钮)实例详解 一.介绍 这个类是继承自ImageView的,所以对于这个控件我们可以使用ImageView的所有属性 二.使用准备, 在as 的 build.grade文件中写上 compile 'com.android.support:design:22.2.0' 三.使用说明 <android.support.design.widget.FloatingActionButton android:id="@+id/floa

  • android编程实现悬浮窗体的方法

    本文实例讲述了android编程实现悬浮窗体的方法.分享给大家供大家参考,具体如下: 突然对悬浮窗体感兴趣,查资料做了个小Demo,效果是点击按钮后,关闭当前Activity,显示悬浮窗口,窗口可以拖动,双击后消失.效果图如下: 它的使用原理很简单,就是借用了WindowManager这个管理类来实现的. 1.首先在AndroidManifest.xml中添加使用权限: 复制代码 代码如下: <uses-permission android:name="android.permission

  • Android编程实现悬浮窗获取并显示当前内存使用量的方法

    本文实例讲述了Android编程实现悬浮窗获取并显示当前内存使用量的方法.分享给大家供大家参考,具体如下: 运行效果: 其中: 这一块就是悬浮窗,可以随意拖动,动态显示当前内存使用量. 下面看一下代码是如何实现的: 悬浮窗的实现是用了一个service,为什么要用service呢?了解service特点的大体就会明白.下面看一下: public class FloatService extends Service { WindowManager wm = null; WindowManager.

随机推荐