Android利用WindowManager生成悬浮按钮及悬浮菜单

发布时间 - 2026-01-10 22:42:44    点击率:

简介

本文模仿实现的是360手机卫士基础效果,同时后续会补充一些WindowManager的原理知识。

整体思路

360手机卫士的内存球其实就是一个没有画面的应用程序,整个应用程序的主体是一个Service。我们的程序开始以后,启动一个service,同时关闭activity即可:

public class MainActivity extends Activity {
  private static final String TAG = MainActivity.class.getSimpleName();
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    startService(new Intent(this, FloatWindowService.class));
    finish();
  }
}
import android.os.IBinder;
import android.util.Log;

import java.util.Timer;
import java.util.TimerTask;

public class FloatWindowService extends Service {
  private static final String TAG = FloatWindowService.class.getSimpleName();

  public FloatWindowService() {
  }
  @Override
  public int onStartCommand(Intent intent, int flags, int startId) {
    Log.d(TAG, "on start command");
    FloatWindowManager.instance(getApplicationContext()).createFloatWindow();
    return super.onStartCommand(intent, flags, startId);
  }

  @Override
  public IBinder onBind(Intent intent) {
    // TODO: Return the communication channel to the service.
    throw new UnsupportedOperationException("Not yet implemented");
  }

}

我们要注意的是,传统的Service默认是运行在UI线程中的,这点与封装了一个Thread和Handler的intentService不同,所以我们可以直接在Service中更改UI相关的内容。

再来看一下FloatWindowManager中的方法:

  public void createFloatWindow() {
    if (isWindowShowing()) return;
    WindowManager windowManager = getWindowManger(context);
    int screenWidth = windowManager.getDefaultDisplay().getWidth();
    int screenHeight = windowManager.getDefaultDisplay().getHeight();
    if (floatLayout == null) {
      floatLayout = new FloatLayout(context);
      if (smallLayoutParams == null) {
        smallLayoutParams = new WindowManager.LayoutParams();
        smallLayoutParams.type = WindowManager.LayoutParams.TYPE_PHONE;
        smallLayoutParams.format = PixelFormat.RGBA_8888;
        smallLayoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
            | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
        smallLayoutParams.gravity = Gravity.LEFT | Gravity.TOP;
        smallLayoutParams.width = FloatLayout.viewWidth;
        smallLayoutParams.height = FloatLayout.viewHeight;
        smallLayoutParams.x = screenWidth;
        smallLayoutParams.y = screenHeight / 2;
      }
    }
    windowManager.addView(floatLayout,smallLayoutParams);
  }

以及自定义的View:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:id="@+id/small_layout"
  android:background="@drawable/bg_small"
  android:orientation="vertical" android:layout_width="60dip"
  android:layout_height="25dip">
<TextView
  android:layout_width="match_parent"
  android:gravity="center"
  android:text="悬浮窗"
  android:layout_height="match_parent" />
</LinearLayout>
public class FloatLayout extends LinearLayout {
  public static int viewWidth;
  public static int viewHeight;
  private WindowManager windowManager;
  public FloatLayout(final Context context) {
    super(context);
    windowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
    LayoutInflater.from(context).inflate(R.layout.small_layout, this);
    View view = findViewById(R.id.small_layout);
    viewWidth = view.getLayoutParams().width;
    viewHeight = view.getLayoutParams().height;
    setOnTouchListener(new OnTouchListener() {
      @Override
      public boolean onTouch(View v, MotionEvent event) {
        FloatWindowManager.instance(context).createFloatMenu();
        return true;
      }
    });
  }

}

自定义的View除了加载了一个布局,就是设置了一个Touch监听器,用于点击悬浮窗弹出菜单。注意这里要使用 view.getLayoutParams() 来获取视图的宽和高,因为在构造方法中,这个View并没有被measure完成,所以采用view.getHeight得到的宽高是0。

创建菜单的方法类似,同样通过WindowManager:

  public void createFloatMenu() {
    if (menuLayout != null) return;
    Log.d(TAG, "create float menu");
    WindowManager windowManager = getWindowManger(context);
    if (menuLayout == null){
      menuLayout = new MenuLayout(context);
      menuLayoutParams = new WindowManager.LayoutParams();
      menuLayoutParams.type = WindowManager.LayoutParams.TYPE_PHONE;
      menuLayoutParams.format = PixelFormat.RGBA_8888;

    }
    windowManager.addView(menuLayout,menuLayoutParams);

  }

自定义的菜单将背景设置成半透明,同时分成上下两部分,上部分点击删除菜单,下部分是一些展示的内容:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:orientation="vertical" android:layout_width="match_parent"
  android:background="#96000000"
  android:layout_height="match_parent">
<LinearLayout
  android:layout_width="match_parent"
  android:id="@+id/trans_part"
  android:orientation="horizontal"
  android:layout_weight="1"
  android:layout_height="0dp"></LinearLayout>
  <LinearLayout
    android:layout_width="match_parent"
    android:layout_weight="1"
    android:background="@color/colorPrimary"
    android:layout_height="0dp">
    <TextView
      android:layout_width="match_parent"
      android:text="存放content"
      android:layout_height="match_parent" />

  </LinearLayout>
</LinearLayout>
public class MenuLayout extends LinearLayout {
  public MenuLayout(final Context context) {
    super(context);
    LayoutInflater.from(context).inflate(R.layout.transparent_layout,this);
    View view = findViewById(R.id.trans_part);
    view.setOnClickListener(new OnClickListener() {
      @Override
      public void onClick(View v) {
        FloatWindowManager.instance(context).removeMenuLayout();
      }
    });
  }
}

可以看见,实现悬浮窗,其实就是通过windowManager.addView时,在LayoutParam 的type设置为TYPE_PHONE,这样你的视图就是系统级视图,可以覆盖在全部程序的最上面。其余的,更多的是自定义View的知识。

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


# Android  # WindowManager  # 悬浮按钮  # 悬浮菜单  # Android开发悬浮按钮 Floating ActionButton的实现方法  # Android仿知乎悬浮功能按钮FloatingActionButton效果  # Android自定义可拖拽的悬浮按钮DragFloatingActionButton  # Android开发之FloatingActionButton悬浮按钮基本使用、字体、颜色用法示例  # android 应用内部悬浮可拖动按钮简单实现代码  # Android实现系统级悬浮按钮  # Android悬浮按钮的使用方法  # Android悬浮窗按钮实现点击并显示/隐藏多功能列表  # Android自定义APP全局悬浮按钮  # Android自定义悬浮按钮效果  # 的是  # 自定义  # 应用程序  # 是一个  # 要注意  # 再来  # 弹出  # 可以直接  # 看一下  # 设置为  # 装了  # 两部分  # 要使  # 可以看见  # 大家多多  # 设置成  # 加载  # onBind  # TODO  # return 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 高防服务器租用首荐平台,企业级优惠套餐快速部署  浅谈redis在项目中的应用  Laravel如何实现用户角色和权限系统_Laravel角色权限管理机制  Laravel怎么配置不同环境的数据库_Laravel本地测试与生产环境动态切换【方法】  Laravel如何实现本地化和多语言支持?(i18n教程)  Laravel模型关联查询教程_Laravel Eloquent一对多关联写法  标准网站视频模板制作软件,现在有哪个网站的视频编辑素材最齐全的,背景音乐、音效等?  Internet Explorer官网直接进入 IE浏览器在线体验版网址  Python面向对象测试方法_mock解析【教程】  浅谈javascript alert和confirm的美化  php静态变量怎么调试_php静态变量作用域调试技巧【解答】  北京网站制作公司哪家好一点,北京租房网站有哪些?  Laravel集合Collection怎么用_Laravel集合常用函数详解  如何为不同团队 ID 动态生成多个“认领值班”按钮  Laravel项目如何进行性能优化_Laravel应用性能分析与优化技巧大全  Laravel怎么使用Markdown渲染文档_Laravel将Markdown内容转HTML页面展示【实战】  Laravel如何实现图片防盗链功能_Laravel中间件验证Referer来源请求【方案】  中山网站制作网页,中山新生登记系统登记流程?  如何用花生壳三步快速搭建专属网站?  宙斯浏览器文件分类查看教程 快速筛选视频文档与图片方法  如何快速生成专业多端适配建站电话?  Python高阶函数应用_函数作为参数说明【指导】  如何快速建站并高效导出源代码?  Laravel队列任务超时怎么办_Laravel Queue Timeout设置详解  Win11怎么关闭资讯和兴趣_Windows11任务栏设置隐藏小组件  Laravel Facade的原理是什么_深入理解Laravel门面及其工作机制  如何在Tomcat中配置并部署网站项目?  Laravel如何实现数据导出到CSV文件_Laravel原生流式输出大数据量CSV【方案】  原生JS获取元素集合的子元素宽度实例  iOS发送验证码倒计时应用  Laravel如何实现RSS订阅源功能_Laravel动态生成网站XML格式订阅内容【教程】  Laravel如何构建RESTful API_Laravel标准化API接口开发指南  Laravel如何实现登录错误次数限制_Laravel自带LoginThrottles限流配置【方法】  Android利用动画实现背景逐渐变暗  Python企业级消息系统教程_KafkaRabbitMQ高并发应用  如何快速搭建高效香港服务器网站?  简历没回改:利用AI润色让你的文字更专业  矢量图网站制作软件,用千图网的一张矢量图做公司app首页,该网站并未说明版权等问题,这样做算不算侵权?应该如何解决?  Laravel如何升级到最新的版本_Laravel版本升级流程与兼容性处理  googleplay官方入口在哪里_Google Play官方商店快速入口指南  如何用搬瓦工VPS快速搭建个人网站?  Laravel如何处理CORS跨域问题_Laravel项目CORS配置与解决方案  Java解压缩zip - 解压缩多个文件或文件夹实例  node.js报错:Cannot find module &#39;ejs&#39;的解决办法  Laravel如何获取当前用户信息_Laravel Auth门面获取用户ID  Laravel的路由模型绑定怎么用_Laravel Route Model Binding简化控制器逻辑  微信小程序 wx.uploadFile无法上传解决办法  CSS3怎么给轮播图加过渡动画_transition加transform实现【技巧】  laravel服务容器和依赖注入怎么理解_laravel服务容器与依赖注入解析  Python文件异常处理策略_健壮性说明【指导】