最近在做一个全局悬浮窗基于ChatGPT应用快Ai,需要悬浮于在其他应用上面,方便从悬浮窗中,和ChatGPT对话后,对ChatGPT返回的内容拖拽到其他应用内部。快Ai应用本身透明,通过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添加悬浮窗。在没有进行任何操作,正常情况下,可以调起科大讯飞进行录音转成文字发给ChatGPT。
问题点一:同事为了解决我还没来得及修复的windowManger.removeView改成exitProcess
问题,强行进行各种修改,最终还调用了activity
的finish
函数,把activity
干掉。最终导致无法调起科大讯飞的语音识别。总是报录音权限问题,找不到任何的问题点,网上资料都说没有给录音权限,其实是有的。最后通过代码回退,定位到是Activity
被干掉了,同事也承认他的愚蠢行为。
问题点二:在进行一些操作,例如授权跳转到设置之后,退出设置回到原先界面,科大讯飞调不起录音,还是报权限问题。在有了问题点一的经验后,在Activity的各个生命周期打印日志,发现但onResume
函数没有被回调到,也就是应用在后台运行时,该问题必现。
所以就一顿顿顿搜索后,找到官方文档: Android 9 对后台运行的应用增加了权限限制。
解决方法:
- 声明为系统应用,没问题。但我们想做通用软件。
- 增加前台服务。实测没效果。
- 在2的基础上,再添加一个属性:
android:foregroundServiceType="microphone"
。完美。
<service android:name=".ui.service.AiService"
android:foregroundServiceType="microphone"
/>
希望本文对君有用!