Android实现悬浮窗全系统版本

悬浮窗是在系统上显示的内容,好像微信视频聊天时的小窗口一样,在退出软件后依然存在的一个窗口,本博客以窗口中放一个button组件为例,简单展示悬浮窗,其中包括了对Android 6.0以下、Android 6.0到Android 8.0、Android 8.0以上版本的处理,下面开始介绍实现方法:

1、MainActivity中的代码

public Button mFloatingButton;

 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);

  //初始化view
  initView();
 }
 //初始化view
 private void initView() {
  mFloatingButton=(Button) findViewById(R.id.floating_btn);
  mFloatingButton.setOnClickListener(this);
 }

 public void startFloatingButtonService(View view) {
  Log.e("测试流程", "测试流程");
  if (FloatingService_Button.isStarted) {
   Log.e("测试流程2", "测试流程2");
   return;
  }
  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {//判断系统版本
   if (!Settings.canDrawOverlays(this)) {
    Toast.makeText(this, "当前无权限,请授权", Toast.LENGTH_SHORT);
    Log.e("测试流程3", "测试流程3");
    startActivityForResult(new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION, Uri.parse("package:" + getPackageName())), 0);
   } else {
    Log.e("测试流程4", "测试流程4");
    startService(new Intent(MainActivity.this, FloatingService_Button.class));
   }
  } else {
   startService(new Intent(MainActivity.this, FloatingService_Button.class));
  }

 }

 @Override
 protected void onActivityResult(int requestCode, int resultCode, Intent data) {
  if (requestCode == 0) {
   if (!Settings.canDrawOverlays(this)) {
    Log.e("测试流程5", "测试流程5");
    Toast.makeText(this, "授权失败", Toast.LENGTH_SHORT).show();
   } else {
    Log.e("测试流程6", "测试流程6");
    Toast.makeText(this, "授权成功", Toast.LENGTH_SHORT).show();
    startService(new Intent(MainActivity.this, FloatingService_Button.class));
   }
  }
 }

 @Override
 public void onClick(View v) {
  switch (v.getId()){
   case R.id.floating_btn :
    startFloatingButtonService(v);
    break;
  }
 }

思路简单解释:点击弹出悬浮窗按钮时,获取版本并判断“Build.VERSION.SDK_INT >= Build.VERSION_CODES.M”如果系统版本在6.0以下这不需要请求权限,如果系统版本在6.0以上需要进行权限检测以及请求,获取权限后,弹出悬浮框

2、activity_main.xml代码

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 xmlns:tools="http://schemas.android.com/tools"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
 android:orientation="vertical"
 tools:context="com.example.jack_lin.suspendwindow.MainActivity">

 <Button
  android:id="@+id/floating_btn"
  style="@style/floatingBtn"
  android:text="@string/floating_btn" />

</LinearLayout>

简单解释:xml中没什么特别东西,线性布局中放一个按钮

3、FloatingService_Button的代码

public static boolean isStarted = false;

 private WindowManager windowManager;
 private WindowManager.LayoutParams layoutParams;

 private Button button;

 @Override
 public void onCreate() {
  super.onCreate();
  Log.e("进入服务1", "进入服务1");
  isStarted = true;
  windowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
  layoutParams = new WindowManager.LayoutParams();
  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
   layoutParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
  } else {
   layoutParams.type = WindowManager.LayoutParams.TYPE_PHONE;
  }
  layoutParams.format = PixelFormat.RGBA_8888;
  layoutParams.gravity = Gravity.LEFT | Gravity.TOP;
  layoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
  layoutParams.width = 500;
  layoutParams.height = 100;
  layoutParams.x = 300;
  layoutParams.y = 300;
 }

 @Nullable
 @Override
 public IBinder onBind(Intent intent) {
  Log.e("进入服务2", "进入服务2");

  return null;
 }

 @Override
 public int onStartCommand(Intent intent, int flags, int startId) {
  Log.e("进入服务3", "进入服务3");
  showFloatingWindow();
  return super.onStartCommand(intent, flags, startId);
 }

 private void showFloatingWindow() {
  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {//判断系统版本
   if (Settings.canDrawOverlays(this)) {
    button = new Button(getApplicationContext());
    button.setText("我是个button窗口");
    button.setBackgroundColor(Color.BLUE);
    windowManager.addView(button, layoutParams);

    button.setOnTouchListener(new FloatingOnTouchListener());
   }
  } else {
   button = new Button(getApplicationContext());
   button.setText("我是个button窗口");
   button.setBackgroundColor(Color.BLUE);
   windowManager.addView(button, layoutParams);

   button.setOnTouchListener(new FloatingOnTouchListener());
  }
 }

 private class FloatingOnTouchListener implements View.OnTouchListener {
  private int x;
  private int y;

  @Override
  public boolean onTouch(View view, MotionEvent event) {
   switch (event.getAction()) {
    case MotionEvent.ACTION_DOWN:
     x = (int) event.getRawX();
     y = (int) event.getRawY();
     break;
    case MotionEvent.ACTION_MOVE:
     int nowX = (int) event.getRawX();
     int nowY = (int) event.getRawY();
     int movedX = nowX - x;
     int movedY = nowY - y;
     x = nowX;
     y = nowY;
     layoutParams.x = layoutParams.x + movedX;
     layoutParams.y = layoutParams.y + movedY;
     windowManager.updateViewLayout(view, layoutParams);
     break;
    default:
     break;
   }
   return false;
  }
 }

思路简单解释:首先获取WindowManager服务,然后定义并设置在window上显示的layoutParams(此处需注意Android 8.0以上版本中LayoutParam里的type变量变为TYPE_APPLICATION_OVERLAY与Android 8.0以下版本LayoutParam里的type变量TYPE_PHONE不一样,需要通过判断系统版本进行区分),然后定义并设置在layoutParams上面显示的Button按钮以及监听事件(此处的监听事件主要是悬浮窗口拖动的监听)最后将设置好的button与layoutParams添加入window中

4、AndroidManifest.xml中权限添加

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

5、效果图

源码下载地址:Android悬浮窗 源码下载

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

(0)

相关推荐

  • 不依赖于Activity的Android全局悬浮窗的实现

    前言 当我们在手机上使用360安全卫士时,手机屏幕上时刻都会出现一个小浮动窗口,点击该浮动窗口可跳转到安全卫士的操作界面,而且该浮动窗口不受其他activity的覆盖影响仍然可见(多米音乐也有相关的和主界面交互的悬浮小窗口).那么这种不受Activity界面影响的悬浮窗口是怎么实现的呢? Android悬浮窗实现 实现基础 Android悬浮窗实现使用WindowManager WindowManager介绍 通过Context.getSystemService(Context.WINDOW_S

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

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

  • Android实现类似qq微信消息悬浮窗通知功能

    实现方法:(需要开启悬浮窗通知权限.允许应用在其他应用上显示) 一.利用headsup 悬挂式Notification,他是5.0中新增的,也就是API中的Headsup的Notification,可以在不打断用户操作的时候,给用户通知 二.使用Window创建悬浮窗 当window属性设置为FLAGE_NOT_FOCUSABLE表示不需要获取焦点,也不需要接受各种输入事件,此标记会同时启用FLAGE_NOT_TOUCH_MODEL,最终事件会直接传递给下层具有焦点的Widow FLAGE_NO

  • Android实现桌面悬浮窗、蒙板效果实例代码

    现在很多安全类的软件,比如360手机助手,百度手机助手等等,都有一个悬浮窗,可以飘浮在桌面上,方便用户使用一些常用的操作. 今天这篇文章,就是介绍如何实现桌面悬浮窗效果的. 首先,看一下效果图. 悬浮窗一共分为两个部分,一个是平常显示的小窗口,另外一个是点击小窗口显示出来的二级悬浮窗口. 首先,先看一下这个项目的目录结构. 最关键的就是红框内的四个类. 首先,FloatWindowService是一个后台的服务类,主要负责在后台不断的刷新桌面上的小悬浮窗口,否则会导致更换界面之后,悬浮窗口也会随

  • Android 8.0如何完美适配全局dialog悬浮窗弹出

    前言 最近项目targetSdkVersion升级到了26,出现很多问题趟了很多坑,其中就包括本篇的需要解决的问题:全局dialog 不显示. 出现场景 有时候我们需要在App中弹dialog,但是却不知道依附的是哪个Activity,这个时候通常会启动一个service来依附,显示一个全局的dialog. Android 6.0出现的悬浮窗权限 为什么从6.0说起? 在Android6.0之后,使用悬浮窗功能需要申请开启悬浮窗权限,在API23以下版本编译,悬浮窗权限关闭的,但是没有权限限制,

  • Android应用内悬浮窗的实现方案示例

    1.悬浮窗的基本介绍 悬浮窗,大家应该也不陌生,凌驾于应用之上的一个小弹窗,实现上很简单,就是添加一个系统级别的窗口,Android中通过WindowManagerService( WMS)来管理所有的窗口,对于WMS来说,管你是Activity.Toast.Dialog,都不过是通过WindowManagerGlobal.addView()添加的一个个View. Android中的窗口分为三个级别: 1.1 应用窗口,比如Activity的窗口; 1.2 子窗口,依赖于父窗口,比如PopupW

  • Android实现类似360,QQ管家那样的悬浮窗

    一.前言: 我手机从来不装这些东西,不过,有次看到同事的android手机上,有个QQ管家在桌面上浮着,同事拖动管家时,管家就变成一只鸟,桌面下方还有个弹弓,桌面顶部有只乌鸦,把管家也就是鸟拖动到弹弓那,然后,松手,鸟就飞出去.这个过程是动画过程,做的事,实际上是清楚内存. 二:原理: 其实,没什么原理,用到的就是WindowManager以及WindowManager.LayoutParams,对这个LayoutParams做文章,当设置为属性后,然后,创建一个View,将这个View添加到W

  • Android 悬浮窗权限各机型各系统适配大全(总结)

    这篇博客主要介绍的是 Android 主流各种机型和各种版本的悬浮窗权限适配,但是由于碎片化的问题,所以在适配方面也无法做到完全的主流机型适配,这个需要大家的一起努力,这个博客的名字永远都是一个将来时. 悬浮窗适配 悬浮窗适配有两种方法:第一种是按照正规的流程,如果系统没有赋予 APP 弹出悬浮窗的权限,就先跳转到权限授权界面,等用户打开该权限之后,再去弹出悬浮窗,比如 QQ 等一些主流应用就是这么做得:第二种就是利用系统的漏洞,绕过权限的申请,简单粗暴,这种方法我不是特别建议,但是现在貌似有些

  • Android悬浮窗屏蔽悬浮窗外部所有的点击事件的实例代码

    Android可以在所有应用上方添加View,就是给WindowManager添加一个View,在创建的View的时候可以给这个View设置LayoutParams(android.view.WindowManager.LayoutParams.LayoutParams())属性,默认是屏蔽View区域以外的所有点击事件,即在显示View的时候,屏幕的其它区域都不可点击,包括菜单和返回键等. 当给LayoutParams设置合适的flag后,其它区域就可以响应点击了. 示例代码: /** * 创

  • android 添加随意拖动的桌面悬浮窗口

    用过新版本android 360手机助手都人都对 360中只在桌面显示一个小小悬浮窗口羡慕不已吧? 其实实现这种功能,主要有两步: 1.判断当前显示的是为桌面.这个内容我在前面的帖子里面已经有过介绍,如果还没看过的赶快稳步看一下哦. 2.使用windowManager往最顶层添加一个View .这个知识点就是为本文主要讲解的内容哦.在本文的讲解中,我们还会讲到下面的知识点: a.如果获取到状态栏的高度 b.悬浮窗口的拖动 c.悬浮窗口的点击事件 有开始之前,我们先来看一下效果图:  接下来我们来

随机推荐