android实现桌面移动悬浮窗口

 更新时间:2018年07月30日 09:03:00   作者:我守一座空城丶  
这篇文章主要为大家详细介绍了android实现桌面移动悬浮窗口,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

现在很多应用都有这样的功能,比如360等安全卫士,手机管家之内的应用。

效果图:

一、实现原理及移动思路

调用WindowManager,并设置WindowManager.LayoutParams的相关属性,通过WindowManager的addView方法创建View,这样产生出来的View根据WindowManager.LayoutParams属性不同,效果也就不同了。比如创建系统顶级窗口,实现悬浮窗口效果!然后通过覆写悬浮View中onTouchEvent方法来改变windowMananager.LayoutParams中x和y的值来实现自由移动悬浮窗口。

二、示例代码

先看看悬浮View的代码把

package com.example.suspend;
 
import android.content.Context;
import android.util.Log;
import android.view.MotionEvent;
import android.view.WindowManager;
import android.widget.ImageView;
 
public class MyFloatView extends ImageView {
 private float mTouchStartX;
 private float mTouchStartY;
 private float x;
 private float y;
 
 private WindowManager wm = (WindowManager) getContext()
  .getApplicationContext().getSystemService("window");
 // 此wmParams变量为获取的全局变量,用以保存悬浮窗口的属性
 private WindowManager.LayoutParams wmParams = ((MyApplication) getContext()
  .getApplicationContext()).getMywmParams();
 
 public MyFloatView(Context context) {
 super(context);
 }
 
 @Override
 public boolean onTouchEvent(MotionEvent event) {
 // TODO Auto-generated method stub
 // 获取相对屏幕的坐标,即以屏幕左上角为原点
 x = event.getRawX();
 y = event.getRawY() - 25; // 25是系统状态栏的高度
 Log.i("currP", "currX" + x + "====currY" + y);
 
 switch (event.getAction()) {
 case MotionEvent.ACTION_DOWN: // 捕获手指触摸按下动作
  // 获取相对View的坐标,即以此View左上角为原点
  mTouchStartX = event.getX();
  mTouchStartY = event.getY();
  Log.i("startP", "startX" + mTouchStartX + "====startY"
   + mTouchStartY);
  break;
 case MotionEvent.ACTION_MOVE://捕获手指触摸移动动作      
  updateViewPosition(); 
  break;
 case MotionEvent.ACTION_UP://捕获手指触摸离开动作 
  updateViewPosition(); 
  mTouchStartX=mTouchStartY=0; 
  break;
 default:
  break;
 }
 return true;
 }
 
 private void updateViewPosition() {
 //更新浮动窗口位置参数 
 wmParams.x = (int)(x - mTouchStartX);
 wmParams.y = (int)(y - mTouchStartY);
 wm.updateViewLayout(this, wmParams);//刷新显示  
 }
}

上面的wmParams变量(即WindowManager.LayoutParams)的存储采用了extends Application的方式来创建全局变量,
示例代码如下:

package com.example.suspend;
 
import android.app.Application;
import android.view.WindowManager;
 
public class MyApplication extends Application {
 private WindowManager.LayoutParams wmParams = new WindowManager.LayoutParams();
 
 @Override
 public void onCreate() {
 // TODO Auto-generated method stub
 super.onCreate();
 }
 
 public WindowManager.LayoutParams getMywmParams() {
 return wmParams;
 
 }
 
}

再来看一看Activity中的代码:

package com.example.suspend;
 
import android.app.Activity;
import android.graphics.PixelFormat;
import android.os.Bundle;
import android.view.Gravity;
import android.view.WindowManager;
import android.view.WindowManager.LayoutParams;
 
public class MainActivity extends Activity {
 private WindowManager wm;
 private WindowManager.LayoutParams wmParams;
 private MyFloatView myFV;
 
 @Override
 protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_main);
 // 创建悬浮窗口
 createFloatView();
 }
 
 private void createFloatView() {
 myFV = new MyFloatView(getApplicationContext());
 myFV.setImageResource(R.drawable.ic_launcher);
 // 获取WindowManager
 wm = (WindowManager) getApplicationContext().getSystemService("window");
 // 设置LayoutParams(全局变量)相关参数
 wmParams = ((MyApplication) getApplication()).getMywmParams();
 
 wmParams.type = LayoutParams.TYPE_PHONE;// 设置window type
 wmParams.format = PixelFormat.RGBA_8888;// 设置图片格式,效果为背景透明
 // 设置Window flag
 wmParams.flags = LayoutParams.FLAG_NOT_TOUCH_MODAL
  | LayoutParams.FLAG_NOT_FOCUSABLE;
 /*
  * 
  * 下面的flags属性的效果形同“锁定”。
  * 
  * 悬浮窗不可触摸,不接受任何事件,同时不影响后面的事件响应。
  * 
  * wmParams.flags=LayoutParams.FLAG_NOT_TOUCH_MODAL
  * 
  * | LayoutParams.FLAG_NOT_FOCUSABLE
  * 
  * | LayoutParams.FLAG_NOT_TOUCHABLE;
  */
 wmParams.gravity = Gravity.LEFT | Gravity.TOP;// 调整悬浮窗口至左上角,便于调整坐标
 // 以屏幕左上角为原点,设置x、y初始值
 wmParams.x = 0;
 wmParams.y = 0;
 // 设置悬浮窗口长宽数据
 wmParams.width = 40;
 wmParams.height = 40;
 // 显示myFloatView图像
 wm.addView(myFV, wmParams);
 }
 
 @Override
 protected void onDestroy() {
 // TODO Auto-generated method stub
 super.onDestroy();
 // 在程序退出(Activity销毁)时销毁悬浮窗口
 wm.removeView(myFV);
 }
}

最后,别忘了在AndroidManifest.xml中添加权限:

<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
  <uses-permission android:name="android.permission.SYSTEM_OVERLAY_WINDOW" />

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

相关文章

  • Android设计模式系列之单例模式

    Android设计模式系列之单例模式

    单例模式,可以说是GOF的23种设计模式中最简单的一个。接下来通过本文给大家实例讲解android设计模式系列之单例模式的相关知识,感兴趣的朋友一起看看吧
    2016-09-09
  • Room Kotlin API的使用入门教程

    Room Kotlin API的使用入门教程

    这篇文章主要介绍了Room Kotlin API使用入门教程,帮助大家更好的理解和学习使用并且测试 Room Kotlin API,感兴趣的朋友可以了解下
    2021-04-04
  • Android 实现APP可切换多语言步骤详解

    Android 实现APP可切换多语言步骤详解

    如果是单独给app加上国际化,其实很容易,创建对应的国家资源文件夹即可,如values-en,values-pt,这篇文章主要介绍了Android 实现APP可切换多语言,需要的朋友可以参考下
    2023-11-11
  • Android实现状态栏白底黑字效果示例代码

    Android实现状态栏白底黑字效果示例代码

    这篇文章主要介绍了Android实现状态栏白底黑字效果的相关资料,实现后的效果非常适合日常开发中使用,文中给出了详细的示例代码供大家参考学习,需要的朋友们下面随着小编来一起学习学习吧。
    2017-10-10
  • Android基于Intent实现Activity之间数据传递的方法

    Android基于Intent实现Activity之间数据传递的方法

    这篇文章主要介绍了Android基于Intent实现Activity之间数据传递的方法,结合实例形式分析了Activity之间数据传递操作的相关技巧,代码备有较为详尽的注释,需要的朋友可以参考下
    2016-11-11
  • 直接应用项目中的Android图片缓存技术

    直接应用项目中的Android图片缓存技术

    这篇文章主要为大家详细介绍了直接应用项目中的Android图片缓存技术,简单、方便、高效,感兴趣的小伙伴们可以参考一下
    2016-04-04
  • Android中AlarmManager+Notification实现定时通知提醒功能

    Android中AlarmManager+Notification实现定时通知提醒功能

    本篇文章主要介绍了Android中AlarmManager+Notification实现定时通知提醒功能,非常具有实用价值,需要的朋友可以参考下
    2017-10-10
  • Android编程心得分享——JSON学习过程

    Android编程心得分享——JSON学习过程

    在我们初步学习JSON时我们都知道JSON作为现在比较流行的数据交换格式,有着它的许多优点,这里将我学习JSON的过程记录如下
    2013-06-06
  • 深入浅析Android消息机制

    深入浅析Android消息机制

    在Android中,线程内部或者线程之间进行信息交互时经常会使用消息,这些基础的东西如果我们熟悉其内部的原理,将会使我们容易、更好地架构系统,避免一些低级的错误,通过本文给大家介绍android消息机制,感兴趣的朋友一起学习吧
    2016-04-04
  • Android 实现单指滑动双指缩放照片demo及过程解析

    Android 实现单指滑动双指缩放照片demo及过程解析

    这篇文章主要为大家介绍了Android 实现单指滑动双指缩放照片demo及过程解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-04-04

最新评论