android悬浮功能实现,Android实现系统级悬浮按钮

本文实例为大家分享了Android系统级悬浮按钮的具体代码,供大家参考,具体内容如下

具体的需求

1、就是做一个系统级的悬浮按钮,就像iPhone 桌面的那个悬浮按钮效果一样,能随意拖动,并且手一放开,悬浮按钮就自动靠边。

2、可以点击并且可以随意拖动。

3、悬浮按钮自动靠边的时候,或者移动到边上的时候,自动隐藏半边。

4、横竖屏切换都兼容

1、就在WindowManager 里面添加View,这个View通过自定义控件来实现。

2、在onTouch里的MotionEvent.ACTION_MOVE事件里头,通过控制悬浮按钮的具体坐标来实现随意移动。

3、在onTouch里的MotionEvent.ACTION_UP事件里头,来控制悬浮按钮自动靠边,并且自动隐藏半边,不过在这里onTouch和onClick这两个事件是一起触发的,不过这也有解决办法,你可以在手放开的瞬间,通过移动的距离,来决定是否触发点击事件,,如果返回false,就会触发点击事件,如果返回true就会触发点击事件

4、通过自定义控件onLayout方法,来捕获横竖屏切换事件,

5、还有一个靠哪边停靠的问题,通过坐标来判读更靠近哪一边。就靠哪边停靠。

![以中间这个中心点为准,以更短的X轴画一个正方形]

c5715eab5672da60fdf9c67bfcb5d10b.png

下面是具体实现代码:

import android.content.Context;

import android.graphics.Canvas;

import android.graphics.Point;

import android.graphics.Rect;

import android.util.AttributeSet;

import android.view.MotionEvent;

import android.view.View;

import android.view.WindowManager;

import android.widget.ImageView;

import com.iapppay.openid.channel.LoginResultCallback;

import com.iapppay.openid.channel.OpenIDApplication;

import com.iapppay.openid.channel.util.DisplayUtil;

import com.iapppay.openid.channel.util.LogUtil;

import com.iapppay.openid.channel.util.Res;

/**

* Created by HuangTiebing 2017/2/14.

*/

public class DragFloatActionButton extends ImageView implements View.OnTouchListener, View.OnClickListener {

public static String TAG = "DragFloatActionButton";

private Context context;

float lastX, lastY;

float originX, originY;

int screenWidth;

int screenHeight;

private int originWidth;

private WindowManager windowManager;

// // 此windowManagerParams变量为获取的全局变量,用以保存悬浮窗口的属性

private WindowManager.LayoutParams windowManagerParams;

private LoginResultCallback resultCallback; //悬浮按钮点击回调

public DragFloatActionButton(Context context, boolean isForceLogin, LoginResultCallback resultCallback) {

this(context, null);

OpenIDApplication.getInstance().setForceLogin(isForceLogin);

this.resultCallback = resultCallback;

}

public DragFloatActionButton(Context context, AttributeSet attrs) {

this(context, attrs, 0);

}

public DragFloatActionButton(Context context, AttributeSet attrs, int defStyleAttr) {

super(context, attrs, defStyleAttr);

this.context = context;

Point screenSize = DisplayUtil.getScreenSize(context);

screenWidth = screenSize.x;

screenHeight = screenSize.y;

setImageResource(Res.drawable(context, "ipay_float_btn_bg"));

setOnTouchListener(this);

setOnClickListener(this);

windowManager = (WindowManager) getContext().getApplicationContext().getSystemService(Context.WINDOW_SERVICE);

}

public int getOriginWidth() {

return originWidth;

}

public void setOriginWidth(int originWidth) {

this.originWidth = originWidth;

}

@Override

public boolean onTouch(View v, MotionEvent event) {

windowManagerParams = (WindowManager.LayoutParams) this.getLayoutParams();

//获取到状态栏的高度

Rect frame = new Rect();

getWindowVisibleDisplayFrame(frame);

int ea = event.getAction();

switch (ea) {

case MotionEvent.ACTION_DOWN:

lastX = event.getRawX();// 获取触摸事件触摸位置的原始X坐标

lastY = event.getRawY();

originX = lastX;

originY = lastY;

break;

case MotionEvent.ACTION_MOVE:

float dx = event.getRawX() - lastX;

float dy = event.getRawY() - lastY;

windowManagerParams.x += dx;

windowManagerParams.y += dy;

LogUtil.d(TAG, "移动距离:dx=" + dx + ",dy=" + dy);

showAllBtn();

lastX = (int) event.getRawX();

lastY = (int) event.getRawY();

break;

case MotionEvent.ACTION_UP:

float lastMoveDx = Math.abs(event.getRawX() - originX);

float lastMoveDy = Math.abs(event.getRawY() - originY);

LogUtil.d(TAG, "松开时,移动距离:lastMoveDx=" + lastMoveDx + ", lastMoveDy=" + lastMoveDy);

if (lastMoveDx < 10 && lastMoveDy < 10) { //移动距离太小,视为点击,

return false;

} else {

updateViewLayout(event);

isFirstClick = true;

return true;

}

}

return false;

}

/**

* 显示整个图标

*/

public void showAllBtn() {

windowManagerParams.width = originWidth;

windowManagerParams.height = originWidth;

setImageResource(Res.drawable(context, "ipay_float_btn_bg"));

windowManager.updateViewLayout(this, windowManagerParams); // 刷新显示

}

/**

* 悬浮按钮显示在左边

*/

private void showInLeft() {

windowManagerParams.x = 0;

windowManagerParams.width = originWidth / 2;

windowManagerParams.height = originWidth;

setImageResource(Res.drawable(context, "ipay_float_btn_left_hidden"));

windowManager.updateViewLayout(this, windowManagerParams); // 刷新显示

}

/**

* 悬浮按钮显示在右边

*/

private void showInRight() {

windowManagerParams.width = originWidth / 2;

windowManagerParams.height = originWidth;

windowManagerParams.x = screenWidth - windowManagerParams.width;

setImageResource(Res.drawable(context, "ipay_float_btn_right_hidden"));

windowManager.updateViewLayout(this, windowManagerParams); // 刷新显示

}

/**

* 悬浮按钮显示在上面

*/

private void showInTop() {

windowManagerParams.y = 0;

windowManagerParams.width = originWidth;

windowManagerParams.height = originWidth / 2;

setImageResource(Res.drawable(context, "ipay_float_btn_top_hidden"));

windowManager.updateViewLayout(this, windowManagerParams); // 刷新显示

}

/**

* 悬浮按钮显示在下面

*/

private void showInBottom() {

windowManagerParams.width = originWidth;

windowManagerParams.height = originWidth / 2;

windowManagerParams.y = screenHeight - windowManagerParams.width;

setImageResource(Res.drawable(context, "ipay_float_btn_bottom_hidden"));

windowManager.updateViewLayout(this, windowManagerParams); // 刷新显示

}

/**

* 更新悬浮图标

*

* @param event 手动移动事件

*/

public void updateViewLayout(MotionEvent event) {

Point center = new Point(screenWidth / 2, screenHeight / 2); //屏幕中心点

float xOffset, yOffset;//以屏幕中心点为原点,X轴和Y轴上的偏移量

if (event != null) {//手动移动的

xOffset = event.getRawX() - center.x;

yOffset = event.getRawY() - center.y;

} else {//自动隐藏

xOffset = lastX - center.x;

yOffset = lastY - center.y;

}

if (Math.abs(xOffset) >= Math.abs(yOffset)) {//向左或向右缩进隐藏

if (xOffset <= 0) { //向左缩进

showInLeft();

} else {

showInRight();

}

} else {//向上或向下缩进隐藏

if (yOffset <= 0) {//向上缩进

showInTop();

} else {

showInBottom();

}

}

}

@Override

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

super.onMeasure(widthMeasureSpec, heightMeasureSpec);

}

@Override

protected void onLayout(boolean changed, int left, int top, int right, int bottom) {

super.onLayout(changed, left, top, right, bottom);

Point screenSize = DisplayUtil.getScreenSize(context);

if (screenWidth != screenSize.x) {//屏幕旋转切换

screenWidth = screenSize.x;

screenHeight = screenSize.y;

lastY = windowManagerParams.x;

lastX = windowManagerParams.y;

windowManagerParams.x = (int) lastX;

windowManagerParams.y = (int) lastY;

updateViewLayout(null);

}

}

private boolean isFirstClick = true;

@Override

protected void onDraw(Canvas canvas) {

super.onDraw(canvas);

}

@Override

public void onClick(View v) {

LogUtil.d(TAG, "执行点击事件");

if (!isFirstClick) {

OpenIDApplication.getInstance().floatBtnClick(context, OpenIDApplication.getInstance().isForceLogin(), resultCallback);

} else {//半隐藏状态,点击显示全部

isFirstClick = false;

showAllBtn();

}

}

}

调用实现代码,这里注意有个问题,弹出系统级的悬浮窗,需要配置权限:

并且Android 6.0以上的手机,还要弹出对话框问用户是否运行,如果这个用户拒绝了,就不能弹出系统级的悬浮窗了,还有个别手机厂商修改了android源码,还需要进系统设置里去允许这个应用弹出悬浮窗。这样的话就体验感非常不好,不过这里有个小技巧,按下面方式设置为toast类型就完全解决,既不用配置权限,也不弹出窗来向用户获取权限,完全解决问题。

WindowManager.LayoutParams windowManagerParams = new WindowManager.LayoutParams(WindowManager.LayoutParams.TYPE_TOAST,

WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,

PixelFormat.TRANSLUCENT);

具体实现代码如下:

DragFloatActionButton floatBtn = new DragFloatActionButton(context, isForceLogin, mResultCallback);

WindowManager windowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);

// 设置LayoutParams(全局变量)相关参数

WindowManager.LayoutParams windowManagerParams = new WindowManager.LayoutParams(WindowManager.LayoutParams.TYPE_TOAST,

WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,

PixelFormat.TRANSLUCENT);

/**

* 注意,flag的值可以为:

* 下面的flags属性的效果形同“锁定”。

* 悬浮窗不可触摸,不接受任何事件,同时不影响后面的事件响应。

* LayoutParams.FLAG_NOT_TOUCH_MODAL 不影响后面的事件

* LayoutParams.FLAG_NOT_FOCUSABLE 不可聚焦

* LayoutParams.FLAG_NOT_TOUCHABLE 不可触摸

*/

// 调整悬浮窗口至左上角,便于调整坐标

windowManagerParams.gravity = Gravity.LEFT | Gravity.TOP;

// 以屏幕左上角为原点,设置x、y初始值

windowManagerParams.x = 0;

windowManagerParams.y = 0;

// 设置悬浮窗口长宽数据

floatBtn.measure(0, 0);

floatBtn.setOriginWidth(floatBtn.getMeasuredWidth() - 50);

windowManagerParams.width = floatBtn.getOriginWidth();

windowManagerParams.height = windowManagerParams.width;

// 显示myFloatView图像

windowManager.addView(floatBtn, windowManagerParams);

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/539374.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

oracle decode_错过血亏!一文搞懂Oracle锁相关视图及相关操作

本文主要研究锁的相关视图&#xff0c;以及锁的相关操作&#xff0c;通过视图查锁的问题。 一、v$transaction视图第一个视图是v$transaction&#xff0c;就是Oracle数据库所有活动的事务数&#xff0c;所有活动的事务每一个活动的事务在这里有一行。v$transactionXIDUSN表示当…

Linux文件系统与命令行

什么是命令行? 接收键盘命令并将其传给操作系统执行的程序(用于输入和管理命令的程序),统称命令行,也叫: Shell&#xff0c;几乎所有Linux发行版都提供了一个 Shell 程序,叫做: Bash (Bourne-Again Shell, 因为最初的 Shell 是由 Steve Bourne 编写的原始 Unix 程序, Again 表…

android 自定义菜单栏,GitHub - earthWo/AndroidBottomNavigation: android 底部菜单栏,自定义样式,自定义菜单数量,添加滚动动画和水波纹动画...

AndroidBottomNavigation截图使用方法gradle:compile com.whitelife.library:library:1.0.1maven:com.whitelife.librarylibrary1.0pomandroid:id"id/bottom_navigation"android:layout_width"match_parent"android:layout_height"56dp"android:…

windows怎么打开python_windows怎么打开Python

Windows中运行Python的两种运行方式认识编程环境 1 直接命令行启用Python。当然&#xff0c;如果直接在cmd中输入python&#xff0c;需要在windows中的path环境变量中做好设置。 此时&#xff0c;cmd中运行python就可以出现 “>>>” 符号。意味着python进入了交互运行…

AOE网与关键路径简介

前面我们说过的拓扑排序主要是为解决一个工程能否顺序进行的问题&#xff0c;但有时我们还需要解决工程完成需要的最短时间问题。如果我们要对一个流程图获得最短时间&#xff0c;就必须要分析它们的拓扑关系&#xff0c;并且找到当中最关键的流程&#xff0c;这个流程的时间就…

Java 集合体系详解——List体系有序集合

引言 面向对象语言对事物的体现必然是以对象的形式&#xff0c;Java工程师为了方便多多个对象的操作&#xff0c;就对对象进行存储&#xff0c;集合就是存储对象的一种方式&#xff0c;他们的底层都是基于不同的数据结构。当然集合和数组一样都是容器&#xff0c;数组也是可以存…

alert闪一下就没了_尾部贯穿式镀铬银饰条除了丑,还能闪瞎眼

尾部贯穿式镀铬银饰条&#xff0c;在2010年代成为诸多汽车品牌车型争相采用的新世纪新标配&#xff0c;配以双边排气&#xff0c;让整个车尾看起来层次感强烈&#xff0c;视觉收窄&#xff0c;几十万的奥迪A8L有&#xff0c;十几万的斯柯达速派有&#xff0c;A级车有&#xff0…

docker 指定网卡_Docker | Docker技术基础梳理(五) Docker网络管理

为什么需要容器的网络管理&#xff1f;容器的网络默认与宿主机、与其他容器相互隔离&#xff0c;且容器中可以运行一些网络应用&#xff0c;比如nginx、web应用、数据库等&#xff0c;如果需要让外部也可以访问这些容器中运行的网络应用&#xff0c;那么就需要配置网络来实现。…

Android安装两次才成功,Android应用从市场安装完成打开与桌面打开,被启动两次的问题...

问题描述&#xff1a;1、从Android应用市场下载并安装应用&#xff0c;安装完成后&#xff0c;当前界面下方会出现“打开”按钮&#xff0c;这时候我们点击“打开”&#xff0c;会启动应用&#xff0c;进入到应用的启动页面&#xff0c;然后进入应用的主界面&#xff0c;这个时…

事务保存点

在SQL Server中使用rollback会回滚所有的未提交事务状态&#xff0c;但是有些时候我们只需要回滚部分语句&#xff0c;把不需要回滚的语句提到事务外面来&#xff0c;虽然是个方法&#xff0c;但是却破坏了事务的ACID。 SQL中使用事务保存点 即可解决这个问题. 一.SQL 事务中存…

鼎信诺审计前端取数工具_给2019前端的5个建议

2019 农历新年即将到来&#xff0c;是时候总结一下团队过去一年的技术沉淀。过去一年我们支撑的数据相关业务突飞猛进&#xff0c;其中两个核心平台级产品代码量分别达到30万行和80万行&#xff0c;TS 模块数均超过1000个&#xff0c;协同开发人员增加到20人。由于历史原因&…

Hadoop HA

HA概念&#xff1a; high avalability 高可用性。 hadoop 1.x非ha设计 Secondnode对元数据的可靠性有了保障&#xff0c;但服务的可用性不高。 即&#xff1a;当Namenode节点宕机了&#xff0c;整个hadoop就不能使用了&#xff0c;影响了client的使用。 hadoop 2.x的ha设计 新…

紫光展锐处理器有那些手机用_酷派将发千元5G手机,国产紫光展锐加持,主打性价比...

↑↑↑点击上方蓝字订阅每日最新热点手机资讯数年之前&#xff0c;“中华酷联”是国产智能手机的四大代表。不过随着越来越多的强力竞争者入局&#xff0c;中兴、酷派、联想渐渐衰败&#xff0c;仅剩华为屹立手机行业顶端。但是酷派似乎从未想过放弃&#xff0c;最近便要发布一…

jelly bean android,Jelly Bean占Android系统份额突破10%

Android系统份额图(腾讯科技配图)腾讯科技讯(清雨)北京时间1月4日消息&#xff0c;据国外媒体报道&#xff0c;微博)周四发布最新数据显示&#xff0c;Android 4.1版本和Android 4.2版本的Jelly Bean在Android系统中的份额超过了10%&#xff0c;另外Android 4.0版本的ICS的份额…

妲己机器人需要什么条件才能使用_estar零封YTG:平头哥快乐电竞,只有妲己没亚瑟,差评...

2020KPL秋季赛常规赛第8周最后1个比赛日的第2场比赛&#xff0c;结果已经尘埃落定了。而最终的比赛结果是&#xff1a;estarpro轻松以3比0的大比分零封战胜YTG。有一说一&#xff0c;这一场比赛真的是毫无悬念啊&#xff0c;甚至双方交手的第1小局比赛&#xff0c;estarpro只用…

python离线录音转文字_Python将文字转成语音并读出来的实例详解

前言 本篇文章主要介绍&#xff0c;如何利用Python来实现将文字转成语音。将文字转成语音主要有两种不同的实现方法&#xff1a;先将文字转成语音&#xff0c;然后再通过读取语音实现发音、直接调用系统内置的语音引擎实现发音&#xff0c;后一种方法的实现主要利用第三方库。 …

魅族15系统是android,魅族15系列评测:性能够用王者荣耀优化

硬件性能&#xff1a;中配够用&#xff0c;高配优秀硬件方面&#xff0c;文章前面的参数表已经写得很清楚&#xff0c;魅族15搭载的是高通骁龙660处理器&#xff0c;并配备4GB的运行内存&#xff1b;魅族15 Plus则搭载三星Exynos 8895&#xff0c;配备6GB运行内存。在目前的移动…

.net 怎么循环得到数组里的值_HashMap 底层实现、加载因子、容量值及死循环

写在前面&#xff1a;2020年面试必备的Java后端进阶面试题总结了一份复习指南在Github上&#xff0c;内容详细&#xff0c;图文并茂&#xff0c;有需要学习的朋友可以Star一下&#xff01;GitHub地址&#xff1a;abel-max/Java-Study-NoteHashMap 简介HashMap 是一个基于哈希表…

hdfs命令

bin/hdfs dfs命令 appendToFile Usage: hdfs dfs -appendToFile <localsrc> ... <dst> 追加一个或者多个文件&#xff08;linux文件&#xff09; <localsrc> ...到hdfs制定文件<dst>中.也可以从命令行读取输入. hdfs dfs -appendToFile localfile /use…

eclipse jdk配置_eclipse的安装和jdk的配置(JAVA)

首先需要到eclipse官网下载(eclipse.org)点击download进入新界面点击download 64bit进入新界面 点击划线的&#xff0c;点击download也许但是比较慢&#xff0c;点击划线的会出现扩展选项&#xff0c;选择距离你比较近的节点(速度比较快)作者选的是C…