当前位置: 首页 > news >正文

安卓基础(悬浮窗)

悬浮窗

import android.app.Service;
import android.content.Context;
import android.graphics.PixelFormat;
import android.os.IBinder;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.View;
import android.view.WindowManager;public class FloatingWindowService extends Service {private WindowManager windowManager;private View floatingView;@Overridepublic void onCreate() {super.onCreate();// 获取 WindowManager 对象windowManager = (WindowManager) getSystemService(WINDOW_SERVICE);// 加载悬浮窗布局floatingView = LayoutInflater.from(this).inflate(R.layout.floating_window, null);// 设置悬浮窗的布局参数WindowManager.LayoutParams params = new WindowManager.LayoutParams(WindowManager.LayoutParams.WRAP_CONTENT,WindowManager.LayoutParams.WRAP_CONTENT,WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY,WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,android.graphics.PixelFormat.TRANSLUCENT);// 设置悬浮窗的位置params.gravity = Gravity.TOP | Gravity.START;params.x = 0;params.y = 100;// 将悬浮窗添加到 WindowManager 中windowManager.addView(floatingView, params);}@Overridepublic void onDestroy() {super.onDestroy();if (floatingView != null) {// 移除悬浮窗windowManager.removeView(floatingView);}}@Overridepublic IBinder onBind(android.content.Intent intent) {return null;}
}

参数说明

// 设置窗口的宽度,使其根据内部内容自动调整大小,能刚好包裹住内容
WindowManager.LayoutParams.WRAP_CONTENT, 
// 设置窗口的高度,同样使其根据内部内容自动调整大小,能刚好包裹住内容
WindowManager.LayoutParams.WRAP_CONTENT, 
// 指定窗口类型为应用覆盖层,可让窗口显示在其他应用之上(Android 8.0 及以上需特定权限)
WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY, 
// 设置窗口的标志位,让窗口不获取焦点,触摸事件可穿透传递给下方窗口
WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE, 
// 设置窗口的像素格式为半透明,窗口会呈现半透明效果,可看到下方内容
android.graphics.PixelFormat.TRANSLUCENT 

窗口管理.布局参数 参数 = 窗口管理.布局参数

(实例)窗口管理.添加视图(加载xml布局,参数)

悬浮窗拖拽功能

import android.app.Service;
import android.content.Context;
import android.graphics.PixelFormat;
import android.os.IBinder;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.WindowManager;public class FloatingWindowService extends Service {private WindowManager windowManager;private View floatingView;private WindowManager.LayoutParams params;private int initialX;private int initialY;private float initialTouchX;private float initialTouchY;@Overridepublic void onCreate() {super.onCreate();// 获取 WindowManager 对象windowManager = (WindowManager) getSystemService(WINDOW_SERVICE);// 加载悬浮窗布局floatingView = LayoutInflater.from(this).inflate(R.layout.floating_window, null);// 设置悬浮窗的布局参数params = new WindowManager.LayoutParams(WindowManager.LayoutParams.WRAP_CONTENT,WindowManager.LayoutParams.WRAP_CONTENT,WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY,WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,android.graphics.PixelFormat.TRANSLUCENT);// 设置悬浮窗的位置params.gravity = Gravity.TOP | Gravity.START;params.x = 0;params.y = 100;// 将悬浮窗添加到 WindowManager 中windowManager.addView(floatingView, params);// 为悬浮窗设置触摸监听器以实现拖拽功能floatingView.setOnTouchListener(new View.OnTouchListener() {@Overridepublic boolean onTouch(View v, MotionEvent event) {switch (event.getAction()) {case MotionEvent.ACTION_DOWN:// 记录初始位置和触摸点initialX = params.x;initialY = params.y;initialTouchX = event.getRawX();initialTouchY = event.getRawY();return true;case MotionEvent.ACTION_UP:return true;case MotionEvent.ACTION_MOVE:// 计算位置变化并更新悬浮窗位置params.x = initialX + (int) (event.getRawX() - initialTouchX);params.y = initialY + (int) (event.getRawY() - initialTouchY);windowManager.updateViewLayout(floatingView, params);return true;}return false;}});}@Overridepublic void onDestroy() {super.onDestroy();if (floatingView != null) {// 移除悬浮窗windowManager.removeView(floatingView);}}@Overridepublic IBinder onBind(android.content.Intent intent) {return null;}
}    

加载xml布局.设置触碰监听

MotionEvent event

event.getAction() //当前事件行动

MotionEvent.ACTION_DOWN

MotionEvent.ACTION_UP

MotionEvent.ACTION_MOVE

windowManager.updateViewLayout(floatingView, params); //悬浮窗更新


悬浮窗有两个按钮,一个是全截图,另一个是无障碍点击

floating_window.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="vertical"android:padding="8dp"><Buttonandroid:id="@+id/full_screenshot_button"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="全截图" /><Buttonandroid:id="@+id/perform_accessibility_click_button"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="执行无障碍点击" />
</LinearLayout>    

FloatingWindowService.java

import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.graphics.PixelFormat;
import android.os.IBinder;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.WindowManager;
import android.widget.Button;public class FloatingWindowService extends Service {private WindowManager windowManager;private View floatingView;private WindowManager.LayoutParams params;private int initialX;private int initialY;private float initialTouchX;private float initialTouchY;@Overridepublic void onCreate() {super.onCreate();// 获取 WindowManager 对象windowManager = (WindowManager) getSystemService(WINDOW_SERVICE);// 加载悬浮窗布局floatingView = LayoutInflater.from(this).inflate(R.layout.floating_window, null);// 设置悬浮窗的布局参数params = new WindowManager.LayoutParams(WindowManager.LayoutParams.WRAP_CONTENT,WindowManager.LayoutParams.WRAP_CONTENT,WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY,WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,android.graphics.PixelFormat.TRANSLUCENT);// 设置悬浮窗的位置params.gravity = Gravity.TOP | Gravity.START;params.x = 0;params.y = 100;// 将悬浮窗添加到 WindowManager 中windowManager.addView(floatingView, params);// 为悬浮窗设置触摸监听器以实现拖拽功能floatingView.setOnTouchListener(new View.OnTouchListener() {@Overridepublic boolean onTouch(View v, MotionEvent event) {switch (event.getAction()) {case MotionEvent.ACTION_DOWN:// 记录初始位置和触摸点initialX = params.x;initialY = params.y;initialTouchX = event.getRawX();initialTouchY = event.getRawY();return true;case MotionEvent.ACTION_UP:return true;case MotionEvent.ACTION_MOVE:// 计算位置变化并更新悬浮窗位置params.x = initialX + (int) (event.getRawX() - initialTouchX);params.y = initialY + (int) (event.getRawY() - initialTouchY);windowManager.updateViewLayout(floatingView, params);return true;}return false;}});// 找到两个按钮并设置点击监听器Button fullScreenshotButton = floatingView.findViewById(R.id.full_screenshot_button);Button performAccessibilityClickButton = floatingView.findViewById(R.id.perform_accessibility_click_button);fullScreenshotButton.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {// 处理全截图逻辑,这里只是示例,需要根据实际情况实现// 比如调用截图方法performFullScreenshot();}});performAccessibilityClickButton.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {// 处理执行无障碍点击逻辑,这里只是示例,需要根据实际情况实现// 比如调用无障碍点击方法performAccessibilityClick();}});}private void performFullScreenshot() {// 这里实现全截图的具体逻辑// 例如使用 MediaProjection 进行截图// 由于代码较为复杂,这里只是占位// 你可以参考 Android 官方文档或相关教程来实现}private void performAccessibilityClick() {// 这里实现执行无障碍点击的具体逻辑// 例如通过 AccessibilityService 来模拟点击操作// 由于代码较为复杂,这里只是占位// 你可以参考 Android 官方文档或相关教程来实现}@Overridepublic void onDestroy() {super.onDestroy();if (floatingView != null) {// 移除悬浮窗windowManager.removeView(floatingView);}}@Overridepublic IBinder onBind(Intent intent) {return null;}
}    

从加载xml布局找到两个按钮

// 找到两个按钮并设置点击监听器
Button fullScreenshotButton = floatingView.findViewById(R.id.full_screenshot_button);
Button performAccessibilityClickButton = floatingView.findViewById(R.id.perform_accessibility_click_button);

两个按钮分别进行一个点击监听

fullScreenshotButton.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {// 处理全截图逻辑,这里只是示例,需要根据实际情况实现// 比如调用截图方法performFullScreenshot();}});performAccessibilityClickButton.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {// 处理执行无障碍点击逻辑,这里只是示例,需要根据实际情况实现// 比如调用无障碍点击方法performAccessibilityClick();}});}

相关文章:

  • 在Windows系统中使用Docker发布镜像到镜像仓库
  • 数据结构4.0
  • IP伪装、代理池与分布式爬虫
  • AAA GitOps详解
  • 怎样提升社交机器人闲聊能力
  • 对比表格:数字签名方案、密钥交换协议、密码学协议、后量子密码学——密码学基础
  • GPU集群监控系统开发实录:基于Prometheus+Grafana的算力利用率可视化方案
  • 滑动窗口leetcode 904
  • 施磊老师rpc(三)
  • C++总结01-类型相关
  • 探索内容智能化的关键解决方案
  • 【Linux】进程优先级与进程切换理解
  • tp5 php获取农历年月日干支甲午
  • 55.[前端开发-前端工程化]Day02-包管理工具npm等
  • C++继承基础总结
  • 【AI面试准备】电商购物车AI测试设计与实施
  • Javase 基础入门 —— 07 接口
  • P1434 [SHOI2002] 滑雪
  • Redis持久化:
  • 如何实现一个虚拟dom
  • 印巴冲突升级,巴防长称已击落5架印度战机
  • 央行行长详解降准:将释放长期流动性1万亿,整体存款准备金率平均水平降至6.2%
  • 奥迪4S店内揭车衣时遭“连环车损”,双方因赔偿分歧陷僵局
  • 外交部:中欧关系50年发展最宝贵经验是相互尊重,求同存异
  • 山东滕州一车辆撞向公交站台致多人倒地,肇事者被控制,案件已移交刑警
  • 繁荣活跃!“五一”假期全国重点零售和餐饮企业销售额同比增长6.3%