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

别只盯着Canvas了!用Surface+SurfaceFlinger手搓一个“悬浮球”应用(Android 13+)

别只盯着Canvas了用SurfaceSurfaceFlinger手搓一个“悬浮球”应用Android 13在Android开发中Canvas可能是大多数开发者最熟悉的绘图工具但它只是UI渲染的冰山一角。如果你想让应用拥有类似系统悬浮球那样独立于Activity窗口、可穿透点击的炫酷效果就必须深入理解Surface和SurfaceFlinger这套底层图形系统。本文将带你从零实现一个真正的悬浮球应用过程中你会掌握如何绕过Activity窗口直接与SurfaceFlinger交互理解SurfaceView与普通View的本质区别处理触摸事件穿透等高级交互问题优化性能避免出现鬼影等常见问题1. 为什么需要直接操作Surface传统View体系的所有UI最终都会通过HWUI渲染到Activity的Surface上。但当我们想要实现以下特性时就必须直接操作Surface悬浮球的三大核心需求独立于Activity即使主界面退出仍能显示穿透点击下方内容可继续响应触摸高性能渲染避免动画卡顿提示从Android 13开始系统对TYPE_APPLICATION_OVERLAY类型窗口的管理更加严格这也是我们选择Surface方案的技术背景对比三种实现方案的优劣方案独立性穿透点击性能兼容性普通View❌❌★★★★★★★★WindowManager✔️需特殊权限★★★★★SurfaceSurfaceFlinger✔️✔️★★★★★★★★★2. 搭建Surface绘图环境2.1 创建SurfaceView的替代方案传统SurfaceView有诸多限制如不能变形、动画生硬我们需要更灵活的解决方案class FloatingSurface(context: Context) : View(context) { private val surfaceHolder: SurfaceHolder private val surfaceCallback object : SurfaceHolder.Callback { override fun surfaceCreated(holder: SurfaceHolder) { // Surface准备好后启动绘制线程 renderThread RenderThread(holder) renderThread.start() } //...其他回调方法 } init { // 关键配置设置透明背景和正确的Z-order setZOrderOnTop(true) holder.setFormat(PixelFormat.TRANSLUCENT) holder.addCallback(surfaceCallback) } }必须注意的三个细节setZOrderOnTop(true)确保Surface显示在最上层透明格式允许下方内容透出单独的渲染线程避免阻塞主线程2.2 配置WindowManager参数val params WindowManager.LayoutParams( width, height, WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY, WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE or WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS or WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL, PixelFormat.TRANSLUCENT ).apply { gravity Gravity.TOP or Gravity.START x initialX y initialY } windowManager.addView(floatingSurface, params)参数解析表参数作用必要性TYPE_APPLICATION_OVERLAY允许悬浮在其他应用上方★★★★★FLAG_NOT_FOCUSABLE不获取焦点实现点击穿透★★★★FLAG_NOT_TOUCH_MODAL将触摸事件传递给下层窗口★★★★FLAG_LAYOUT_NO_LIMITS允许超出屏幕边界★★3. 实现高性能渲染循环3.1 定制渲染线程inner class RenderThread(private val holder: SurfaceHolder) : Thread() { private val frameInterval 16 // 60fps private var running true override fun run() { val canvas holder.lockCanvas() try { // 初始绘制 drawBaseCircle(canvas) } finally { holder.unlockCanvasAndPost(canvas) } while (running) { val startTime SystemClock.elapsedRealtime() val canvas holder.lockCanvas() try { // 动画帧绘制 drawAnimationFrame(canvas) } finally { holder.unlockCanvasAndPost(canvas) } // 精确控制帧率 val renderTime SystemClock.elapsedRealtime() - startTime if (renderTime frameInterval) { sleep(frameInterval - renderTime) } } } fun stopRendering() { running false join() } }性能优化要点使用SystemClock.elapsedRealtime()保证时间精度动态调整sleep时间维持稳定帧率确保每次lock/unlock成对调用3.2 避免Surface闪烁的技巧常见问题快速拖动时出现画面撕裂或闪烁解决方案fun drawAnimationFrame(canvas: Canvas) { // 1. 先清空画布保留alpha通道 canvas.drawColor(Color.TRANSPARENT, PorterDuff.Mode.CLEAR) // 2. 使用双缓冲技术 val bufferCanvas holder.lockHardwareCanvas() try { // 所有绘制操作在bufferCanvas上完成 drawToBuffer(bufferCanvas) canvas.drawBitmap(bitmap, 0f, 0f, null) } finally { bufferCanvas.release() } }4. 高级交互事件处理4.1 实现可拖拽悬浮球override fun onTouchEvent(event: MotionEvent): Boolean { when (event.action) { MotionEvent.ACTION_DOWN - { lastX event.rawX lastY event.rawY } MotionEvent.ACTION_MOVE - { val dx event.rawX - lastX val dy event.rawY - lastY // 更新窗口位置 params.x dx.toInt() params.y dy.toInt() windowManager.updateViewLayout(this, params) lastX event.rawX lastY event.rawY } } return true }注意必须使用rawX/rawY获取屏幕绝对坐标常规的x/y是相对于视图的坐标4.2 点击穿透的精细控制当悬浮球需要部分区域可点击时override fun onTouchEvent(event: MotionEvent): Boolean { val hitRect Rect() getHitRect(hitRect) return if (hitRect.contains(event.x.toInt(), event.y.toInt())) { // 处理有效区域触摸 true } else { // 允许事件穿透 false } }区域检测的三种方案对比矩形检测性能最好但精度低圆形检测适合圆形悬浮球Path检测最灵活但计算开销大5. 生产环境中的实战经验在实际项目中我们遇到过几个典型问题问题1Surface在锁屏后消失原因系统默认会销毁非Activity窗口解决方案params.privateFlags params.privateFlags or WindowManager.LayoutParams.PRIVATE_FLAG_SHOW_FOR_ALL_USERS问题2横竖屏切换时位置错乱修复方案override fun onConfigurationChanged(newConfig: Configuration) { // 重新计算位置 resetPosition() }问题3低端设备上动画卡顿优化策略fun setRenderQuality(highQuality: Boolean) { renderThread.frameInterval if (highQuality) 16 else 33 if (highQuality) { holder.setFixedSize(width, height) } else { holder.setSizeFromLayout() } }在华为MatePad Pro上实测数据质量模式平均帧率CPU占用内存消耗高性能58fps12%45MB高质量60fps18%52MB普通View实现42fps25%68MB

相关文章:

别只盯着Canvas了!用Surface+SurfaceFlinger手搓一个“悬浮球”应用(Android 13+)

别只盯着Canvas了!用SurfaceSurfaceFlinger手搓一个“悬浮球”应用(Android 13) 在Android开发中,Canvas可能是大多数开发者最熟悉的绘图工具,但它只是UI渲染的冰山一角。如果你想让应用拥有类似系统悬浮球那样独立于A…...

Rust的#[derive(Hash, PartialEq, Eq)]派生宏

Rust语言中的派生宏是简化代码的利器,其中#[derive(Hash, PartialEq, Eq)]的组合尤为实用。它允许开发者通过一行代码自动为结构体或枚举实现多个关键trait,大幅提升开发效率。对于需要哈希计算或相等比较的场景,这个宏能避免大量重复劳动。本…...

OpenClaw从入门到应用——Agent:记忆(Memory)

通过OpenClaw实现副业收入:《OpenClaw赚钱实录:从“养龙虾“到可持续变现的实践指南》 OpenClaw 的记忆是 agent 工作区中的纯 Markdown 文件。这些文件是事实来源;模型只“记住”写入磁盘的内容。 记忆搜索工具由活动的记忆插件提供&#…...

浦语灵笔2.5-7B多场景:跨境电商、智慧医疗、智能制造、数字政务四大方向

浦语灵笔2.5-7B多场景实战:解锁跨境电商、智慧医疗、智能制造、数字政务四大方向 你是不是经常遇到这样的场景?电商运营需要快速理解海量商品图片,医生需要辅助分析复杂的医学影像,工厂质检员要处理成千上万的零件照片&#xff0…...

模块化多智能体建模架构深度解析:Mesa如何重塑复杂系统仿真范式

模块化多智能体建模架构深度解析:Mesa如何重塑复杂系统仿真范式 【免费下载链接】mesa Mesa is an open-source Python library for agent-based modeling, ideal for simulating complex systems and exploring emergent behaviors. 项目地址: https://gitcode.c…...

UCIe多模块链路训练实战:当你的4个Module训练结果不一致时,MMPL是怎么“和稀泥”的?

UCIe多模块链路训练实战:当你的4个Module训练结果不一致时,MMPL是怎么“和稀泥”的? 在芯片物理层设计中,UCIe(Universal Chiplet Interconnect Express)的多模块(Multi-Module)配置…...

如何为Unity游戏去除马赛克:5个高效插件的完整配置指南

如何为Unity游戏去除马赛克:5个高效插件的完整配置指南 【免费下载链接】UniversalUnityDemosaics A collection of universal demosaic BepInEx plugins for games made in Unity3D engine 项目地址: https://gitcode.com/gh_mirrors/un/UniversalUnityDemosaics…...

KUKA C4与C2控制柜软限位修改:从HMI界面到系统配置文件的实战指南

1. KUKA控制柜软限位修改的核心价值 刚接触工业机器人的朋友可能对"软限位"这个概念比较陌生。简单来说,软限位就像是给机器人划定的电子围栏,告诉它"你最多只能走到这里"。和硬限位不同,软限位是通过软件设置的&#xf…...

不止于APK:用bsdiff玩转Android文件差分,从游戏资源到配置文件的增量更新思路

超越APK:用bsdiff构建Android全文件增量更新体系 当游戏资源包从200MB增长到300MB时,用户每次更新都要重新下载整个文件?配置文件微调几个参数却要推送完整文件?这些场景正是二进制差分技术大显身手的舞台。bsdiff作为高效的二进制…...

数字记忆管家:三步构建你的个人AI数据资产库

数字记忆管家:三步构建你的个人AI数据资产库 【免费下载链接】WeChatMsg 提取微信聊天记录,将其导出成HTML、Word、CSV文档永久保存,对聊天记录进行分析生成年度聊天报告 项目地址: https://gitcode.com/GitHub_Trending/we/WeChatMsg …...

Python asyncio 调度机制性能优化

Python asyncio调度机制性能优化 在现代高并发的网络应用中,Python的asyncio库凭借其高效的异步IO能力成为开发者的首选。随着业务复杂度的提升,默认的调度机制可能无法充分发挥性能潜力。本文将深入探讨asyncio调度机制的性能优化策略,帮助…...

23岁亿万富豪创立的Mercor,陷员工舞弊、安全漏洞与文化困境

数据标注初创公司Mercor年化营收破10亿美元,却面临系列挑战数据标注初创公司Mercor由三名二十多岁的年轻人于2023年创立,今年早些时候,公司年化营收突破10亿美元。然而,如今这家公司正面临一系列挑战,包括员工挪用公款…...

Honey Select 2画质飞跃攻略:DHH、Graphics插件深度对比与材质编辑器进阶调校

Honey Select 2画质飞跃攻略:DHH、Graphics插件深度对比与材质编辑器进阶调校 当默认画质无法满足你对虚拟世界的视觉期待时,Honey Select 2的模组生态提供了从基础优化到专业级渲染的全套解决方案。本文将带你深入两款核心画质插件的技术内核&#xff0…...

如何在英雄联盟国服免费体验所有皮肤?R3nzSkin工具完全指南

如何在英雄联盟国服免费体验所有皮肤?R3nzSkin工具完全指南 【免费下载链接】R3nzSkin-For-China-Server Skin changer for League of Legends (LOL) 项目地址: https://gitcode.com/gh_mirrors/r3/R3nzSkin-For-China-Server 你是否曾经羡慕别人拥有那些稀有…...

暗黑3智能按键助手:5分钟快速上手,彻底告别手指疲劳的终极指南

暗黑3智能按键助手:5分钟快速上手,彻底告别手指疲劳的终极指南 【免费下载链接】D3keyHelper D3KeyHelper是一个有图形界面,可自定义配置的暗黑3鼠标宏工具。 项目地址: https://gitcode.com/gh_mirrors/d3/D3keyHelper 还在为暗黑3高…...

别再死记硬背了!从勾股定理到斜率乘积-1,一个初中生也能懂的几何证明

从勾股定理到斜率关系:一场初中生也能玩转的几何推理游戏 数学课本上那些冷冰冰的公式,常常让学生们望而生畏。特别是当老师要求"记住这个结论"时,很多同学的第一反应是机械背诵,而不是理解背后的逻辑。今天&#xff0c…...

新手必看!Xinference-v1.17.1 Docker部署常见错误排查

新手必看!Xinference-v1.17.1 Docker部署常见错误排查 1. 部署前的准备工作 1.1 系统环境检查 在开始部署Xinference之前,确保你的系统满足以下基本要求: Docker版本:Docker 20.10.0或更高版本NVIDIA驱动(GPU版本需…...

UVC Gadget实战:从V4L2到USB端点的视频流转发引擎

1. UVC Gadget技术全景解析 想象一下你手里有个树莓派开发板,想把它变成一台能即插即用的USB摄像头——这就是UVC Gadget技术的魔力所在。作为连接V4L2视频框架和USB端点的桥梁,它本质上是个视频流转发引擎,把本地的视频源(比如/d…...

GD32定时器时钟源配置避坑指南:为什么你的定时时间总是不对?

GD32定时器时钟源配置避坑指南:为什么你的定时时间总是不对? 在嵌入式开发中,精确的定时控制是许多功能实现的基础。然而,不少开发者在初次接触GD32系列MCU的定时器时,常常会遇到一个令人困惑的问题:明明按…...

B站CC字幕终极下载工具:三步轻松获取多语言字幕文件

B站CC字幕终极下载工具:三步轻松获取多语言字幕文件 【免费下载链接】BiliBiliCCSubtitle 一个用于下载B站(哔哩哔哩)CC字幕及转换的工具; 项目地址: https://gitcode.com/gh_mirrors/bi/BiliBiliCCSubtitle BiliBiliCCSubtitle是一款专业的B站CC字幕下载与转…...

QueryExcel技术解密:多Excel文件并行检索工具深度解析与实战指南

QueryExcel技术解密:多Excel文件并行检索工具深度解析与实战指南 【免费下载链接】QueryExcel 多Excel文件内容查询工具。 项目地址: https://gitcode.com/gh_mirrors/qu/QueryExcel 在现代企业数据管理中,Excel文件作为最常见的数据存储格式之一…...

Browser Harness:轻便自修复工具,让大语言模型自由完成浏览器任务!

导航菜单与平台功能 导航可进行切换,提供登录、外观设置等选项。平台包含多个方面的功能: 在 AI 代码创作方面,有 GitHub Copilot 借助 AI 编写更优质代码、GitHub Spark 构建并部署智能应用、GitHub Models 管理并比较提示词、MCP Registry&…...

Pixel Aurora Engine部署教程:Docker Compose多服务协同(API+Web+DB)

Pixel Aurora Engine部署教程:Docker Compose多服务协同(APIWebDB) 1. 项目概览 Pixel Aurora Engine是一款融合AI生成技术与复古像素美学的创意工具。通过Docker Compose,我们可以轻松部署包含以下核心服务的完整系统&#xff…...

告别官方启动器:Starward如何彻底改变你的米哈游游戏体验

告别官方启动器:Starward如何彻底改变你的米哈游游戏体验 【免费下载链接】Starward Game Launcher for miHoYo - 米家游戏启动器 项目地址: https://gitcode.com/gh_mirrors/st/Starward 你是否厌倦了官方启动器的缓慢加载、功能单一和繁琐操作?…...

别再死记硬背了!用这个‘非交叠时钟’时序图,彻底搞懂SAR ADC下级板采样

非交叠时钟时序图解析:SAR ADC下级板采样的可视化学习路径 在模数转换器(ADC)设计中,逐次逼近型(SAR)架构因其高能效比和中等精度优势,成为工业测量、医疗设备和消费电子领域的首选方案。然而&…...

保姆级教程:用Huggingface Hub命令行工具上传你的第一个PyTorch模型(含大文件LFS配置)

从零到一:Huggingface Hub命令行高效部署PyTorch模型全指南 当你完成了一个PyTorch模型的训练,下一步自然是想把它分享给社区或者团队成员。Huggingface Hub作为模型托管平台,提供了完整的命令行工具链,让开发者能够高效地上传和管…...

Translumo:打破语言壁垒的终极屏幕实时翻译神器

Translumo:打破语言壁垒的终极屏幕实时翻译神器 【免费下载链接】Translumo Advanced real-time screen translator for games, hardcoded subtitles in videos, static text and etc. 项目地址: https://gitcode.com/gh_mirrors/tr/Translumo 你是否曾在玩外…...

别再只盯着快充了!聊聊USB PD电源那些‘看不见’的硬核要求,比如过流保护和电容充放电

USB PD电源设计的隐形战场:工程师必须掌握的五大安全设计准则 当消费者为手机充电速度提升10分钟而欢呼时,鲜少有人关注到背后电源工程师们为那1%的安全边际所付出的努力。USB PD协议看似简单的电压电流转换背后,隐藏着一个由精密时序控制、多…...

从公式到代码:手把手推导STM32舵机控制PWM的占空比计算(附两种角度表示法)

从公式到代码:手把手推导STM32舵机控制PWM的占空比计算(附两种角度表示法) 第一次接触舵机控制时,看到代码里那些神秘的"(角度135)/9"或"角度*20/1805"公式,你是否也感到一头雾水?这些…...

java面向对象的学习主线

java基础的语法,是为了执行一段程序。 这篇文章的主体是 有一个逻辑思路,解决javase是什么知识体系, 为什么要学这套知识体系。 我们想一个概念。 假如,我们要执行一段程序。 我们总不能 光执行一个类里面的psvm方法 于是我们就创…...