Android开发悬浮窗踩坑解决
正文
最近在做一个全局悬浮窗的基于ChatGPT应用快Ai,需要悬浮于其他应用上面,方便从悬浮窗ChatGPT返回的内容可以拖拽到其他应用内部。自身应用透明,通过WindowManger添加悬浮窗。类似现在很多应用跳转到其他应用,会悬浮一个小按钮,方便用户点击调回自身一样。只不过快Ai窗口比较大,但不全屏。
碰到以下几个问题:
1、悬浮窗中EditText无法获得弹出键盘
主要是没有明白下面两个属性的作用,在网上搜索之后直接设置了。
WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
设置FLAG_NOT_FOCUSABLE
,悬浮窗外的点击才有效,会把事件分发给悬浮窗底层的其他应用Activity
。如果设置了FLAG_NOT_FOCUSABLE
,那么屏幕上弹窗之外的地方能够点击、但是弹窗上的EditText
键盘也不会弹出来。
此时悬浮窗外的事件是不会触发悬浮窗内View
的onToucheEvent
函数,可以通过添加WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH
标志位,但无法拦截事件。
WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
屏幕上弹窗之外的地方能够点击、弹窗上的EditText也可以输入、键盘能够弹出来。
所以根据业务需要,我只需要添加WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
标志位即可。
2、悬浮窗无法录音
通过Activity调起Service,然后在Service通过WindowManager进行添加悬浮窗。在进行没有任何操作,可以正常调起科大讯飞进行录音。
问题点一:同事为了解决我还没来得及修复的windowManger.removeView改成exitProcess
问题,强行各种修改,最终还调用了activity
的finish
函数,把activity
干掉。最终导致无法调起科大讯飞的语音识别。总是报录音权限问题,找不到任何的问题点,最后代码回退,定位到Activity
被干掉,同事也承认他的愚蠢行为。
问题点二:在进行一些操作,例如授权跳转到设置之后,退出设置回到原先界面,调不录音,还是权限。定位答应透明Activity
的生命周期,发现onResume
函数没有被调用到,所以应用现在在后台运行。
所以就一顿顿顿搜索,官方文档: Android 9 对后台运行的应用增加了权限限制。
解决方法:
- 声明为系统应用,没问题。但我们想做通用软件。
- 增加前台服务。实测没效果。
- 在2的基础上,再添加一个属性:
android:foregroundServiceType="microphone"
。完美。
<service android:name=".ui.service.AiService"
android:foregroundServiceType="microphone"
/>
希望本文对君有用!