Android实现悬浮窗全系统版本

 更新时间:2019年11月18日 16:33:25   作者:林一天  
这篇文章主要为大家详细介绍了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悬浮窗 源码下载

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

相关文章

  • android使用Ultra-PullToRefresh实现下拉刷新自定义代码

    android使用Ultra-PullToRefresh实现下拉刷新自定义代码

    本篇文章主要介绍了android使用Ultra-PullToRefresh实现下拉刷新新自定义,具有一定的参考价值,感兴趣的小伙伴们可以参考一下。
    2017-02-02
  • JetPack Compose底部导航栏的实现方法详解

    JetPack Compose底部导航栏的实现方法详解

    开发一个新项目,底部导航栏一般是首页的标配,在以前的xml布局中,我们可以很轻松的是用谷歌提供的BottomNavigationView或者自定义来实现底部导航的功能,在Compose中也有也提供了一个类似的控件androidx.compose.material.BottomNavigation
    2022-09-09
  • Kotlin语言使用WebView示例介绍

    Kotlin语言使用WebView示例介绍

    随着后台技术的不断发展,App前端的应用都布置了Web页面的界面,这个界面就是由WebView组件渲染出来的。WebView由如下优点:可以直接显示和渲染Web页面或者网页;可以直接调用网络上或者本地的html文件,也可以和JavaScript交互使用
    2022-09-09
  • Android面试题问答整理

    Android面试题问答整理

    今天小编就为大家分享一篇关于Android面试题问答整理,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2018-12-12
  • android重力感应开发之微信摇一摇功能

    android重力感应开发之微信摇一摇功能

    这篇文章主要为大家详细介绍了android重力感应开发之微信摇一摇功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-05-05
  • 简单实现Android滚动公告栏

    简单实现Android滚动公告栏

    这篇文章主要为大家详细介绍了如何简单实现Android滚动公告栏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-01-01
  • 详解Android 硬布局item的高级写法

    详解Android 硬布局item的高级写法

    这篇文章主要介绍了详解Android 硬布局item的高级写法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-12-12
  • Android 中ViewPager中使用WebView的注意事项

    Android 中ViewPager中使用WebView的注意事项

    这篇文章主要介绍了Android 中ViewPager中使用WebView的注意事项的相关资料,希望通过本文大家在使用过程中遇到这样的问题解决,需要的朋友可以参考下
    2017-09-09
  • Android后端服务器的搭建方法

    Android后端服务器的搭建方法

    本篇文章主要介绍了Android后端服务器的搭建方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-07-07
  • Android自定义弹出窗口PopupWindow使用技巧

    Android自定义弹出窗口PopupWindow使用技巧

    这篇文章主要介绍了Android自定义弹出窗口PopupWindow使用技巧,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-11-11

最新评论