在Activity的onCreate方法中显示PopupWindow导致异常的原因分析及解决方案
2015-07-27 22:08
465 查看
一、前言
在某些情况下,我们需要一进入Activity就显示PopupWindow,比如常见的选择界面。但由于PopupWindow是依附于Activity的,如果Activity没有创建完成,Activity还没完全显示出来就显示PopupWindow的话,会出现异常现象。
二、问题复现
我在Activity的onCreate()方法中调用如下方法:
?
运行程序的时候出现如下异常:
三、解决方案
在StackOverflow上搜索这个问题你会发现,都没有原因分析,但我在《Android开发精要》一书中找到了答案(P158):
PopupWindow不像对话框那样从屏幕的固定位置弹出,而是依赖于锚点控件对象的位置,所谓锚点控件对象,就是界面组件中的某个控件,PopupWindow的展示和功能都是以它为核心,作为锚点控件的扩展交互界面,以增强该控件对象的功能。
弹出窗口与描点控件有着紧密的联系,在构造并展示弹出窗口前,需要保证锚点控件所在的控件树已经与窗口管理服务建立连接,因为在弹出窗口的展示过程中,需要通过该窗口对象来获取相关信息。在界面组件的构造过程中,窗口连接的建立是个异步过程,也就是说,当Activity.onCreate()等函数被调用时,界面与窗口管理服务的双向通信连接尚未建立,如果在此时构造弹出窗口则会抛出异常。因此,如果期望在界面组件展现之处便构造弹出窗口,可以将弹出窗口对象构造也转换成一个异步过程。
?
在与窗口管理服务未建立连接之前,界面组件将通过View.post函数发送过来的消息放入一个静态队列当中,在通信连接建立完成后,再从该队列中读取消息并一一执行。因此,通过这样的实现模型可以保证弹出窗口展现时窗口通信连接已经构建成功。
所以对于上面的问题,最简单的处理方法是,异步显示PopupWindow就好了:
?
另外一篇的解决方案:经过查询android api,我找到了onWindowFocusChanged这个方法,Activity生命周期中,onStart, onResume, onCreate都不是真正visible的时间点,真正的visible时间点是onWindowFocusChanged()函数被执行时。所以如果要实现这个功能,可以在onWindowFocusChanged方法中进行show pop up window
的操作。
@Override
public void onWindowFocusChanged(boolean hasFocus) {
if (hasFocus) {
showLockScreenPopUpWindow();
}
}
参数hasFoucs用来判断当前的activity是否获取到了用户的焦点。
四、题外话
今天想做一个在PopupWindow里面播放视频的功能,结果发现SurfaceView在PopupWindow中是无法正常显示的,如果需要显示SurfaceView,建议用View、Fragment或者Dialog Activity代替PopupWindow。我在StackOverflow上查了一下,说是SurfaceView必须要依附于window,但PopupWindow是依附于View的,所以Surfaceview在PopupWindow中无法正常创建。
在某些情况下,我们需要一进入Activity就显示PopupWindow,比如常见的选择界面。但由于PopupWindow是依附于Activity的,如果Activity没有创建完成,Activity还没完全显示出来就显示PopupWindow的话,会出现异常现象。
二、问题复现
我在Activity的onCreate()方法中调用如下方法:
?
三、解决方案
在StackOverflow上搜索这个问题你会发现,都没有原因分析,但我在《Android开发精要》一书中找到了答案(P158):
PopupWindow不像对话框那样从屏幕的固定位置弹出,而是依赖于锚点控件对象的位置,所谓锚点控件对象,就是界面组件中的某个控件,PopupWindow的展示和功能都是以它为核心,作为锚点控件的扩展交互界面,以增强该控件对象的功能。
弹出窗口与描点控件有着紧密的联系,在构造并展示弹出窗口前,需要保证锚点控件所在的控件树已经与窗口管理服务建立连接,因为在弹出窗口的展示过程中,需要通过该窗口对象来获取相关信息。在界面组件的构造过程中,窗口连接的建立是个异步过程,也就是说,当Activity.onCreate()等函数被调用时,界面与窗口管理服务的双向通信连接尚未建立,如果在此时构造弹出窗口则会抛出异常。因此,如果期望在界面组件展现之处便构造弹出窗口,可以将弹出窗口对象构造也转换成一个异步过程。
?
所以对于上面的问题,最简单的处理方法是,异步显示PopupWindow就好了:
?
的操作。
@Override
public void onWindowFocusChanged(boolean hasFocus) {
if (hasFocus) {
showLockScreenPopUpWindow();
}
}
参数hasFoucs用来判断当前的activity是否获取到了用户的焦点。
四、题外话
今天想做一个在PopupWindow里面播放视频的功能,结果发现SurfaceView在PopupWindow中是无法正常显示的,如果需要显示SurfaceView,建议用View、Fragment或者Dialog Activity代替PopupWindow。我在StackOverflow上查了一下,说是SurfaceView必须要依附于window,但PopupWindow是依附于View的,所以Surfaceview在PopupWindow中无法正常创建。
相关文章推荐
- error: variably modified 'table' at file scope
- shell脚本:shell的基本元素-6 重定向与管道
- 要成为linux网站运维工程师必须要掌握的技能
- 要成为linux网站运维工程师必须要掌握的技能
- Odoo(OpenERP)应用实践:代发货管理
- Linux Mint---fcitx中文,日语输入法
- Unix/Linux运维首选工具Xmanager Enterprise 3.0的使用教程
- Linux下的tar压缩解压缩命令详解
- iPad popView封装
- zoj 3888 Twelves Monkeys 二分+线段树维护次小值
- Tomcat中Servlet与浏览器之间传值乱码解决办法
- Linux Mint---安装docky
- 软件架构师应该知道的97件事
- linux-命令
- Linux Mint---开启桌面三维特效
- Centos7安装图形界面
- POJ2115 C Looooops(扩展欧几里得)
- linux c++循环缓冲区模板类
- linux中的nm命令简介
- ubuntu下opencv的配置