在Activity的onCreate方法中显示PopupWindow导致异常的原因分析及解决方案

一、前言
        在某些情况下,我们需要一进入Activity就显示PopupWindow,比如常见的选择界面。但由于PopupWindow是依附于Activity的,如果Activity没有创建完成,Activity还没完全显示出来就显示PopupWindow的话,会出现异常现象。

二、问题复现
        我在Activity的onCreate()方法中调用如下方法:
[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. public void show( ){  
  2.     ifnull != mPopupWindow ){  
  3.         mPopupWindow.showAtLocation(mView, Gravity.CENTER, 00);  
  4.     }  
  5. }  

        运行程序的时候出现如下异常:



三、解决方案
        在StackOverflow上搜索这个问题你会发现,都没有原因分析,但我在《Android开发精要》一书中找到了答案(P158):

        PopupWindow不像对话框那样从屏幕的固定位置弹出,而是依赖于锚点控件对象的位置,所谓锚点控件对象,就是界面组件中的某个控件,PopupWindow的展示和功能都是以它为核心,作为锚点控件的扩展交互界面,以增强该控件对象的功能。

        弹出窗口与描点控件有着紧密的联系,在构造并展示弹出窗口前,需要保证锚点控件所在的控件树已经与窗口管理服务建立连接,因为在弹出窗口的展示过程中,需要通过该窗口对象来获取相关信息。在界面组件的构造过程中,窗口连接的建立是个异步过程,也就是说,当Activity.onCreate()等函数被调用时,界面与窗口管理服务的双向通信连接尚未建立,如果在此时构造弹出窗口则会抛出异常。因此,如果期望在界面组件展现之处便构造弹出窗口,可以将弹出窗口对象构造也转换成一个异步过程。

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. // 在界面组件onCreate函数中调用  
  2. final View anchor = findViewById(R.id.anchor);  
  3.     anchor.post(new Runnable(){  
  4.     @Override  
  5.     public void run(){  
  6.         // 构造和展现弹出窗口  
  7.         PopupWindow window = createWindow();  
  8.         window.showAsDropDown(anchor);  
  9.     }  
  10. });  

        在与窗口管理服务未建立连接之前,界面组件将通过View.post函数发送过来的消息放入一个静态队列当中,在通信连接建立完成后,再从该队列中读取消息并一一执行。因此,通过这样的实现模型可以保证弹出窗口展现时窗口通信连接已经构建成功。


         所以对于上面的问题,最简单的处理方法是,异步显示PopupWindow就好了:
[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. public void show( ){  
  2.     mView.post( new Runnable( ) {  
  3.         @Override  
  4.         public void run() {  
  5.             ifnull != mPopupWindow ){  
  6.                 mPopupWindow.showAtLocation(mView, Gravity.CENTER, 00);  
  7.             }  
  8.         }  
  9.     });  
  10. }  

四、题外话

        今天想做一个在PopupWindow里面播放视频的功能,结果发现SurfaceView在PopupWindow中是无法正常显示的,如果需要显示SurfaceView,建议用View、Fragment或者Dialog Activity代替PopupWindow。我在StackOverflow上查了一下,说是SurfaceView必须要依附于window,但PopupWindow是依附于View的,所以Surfaceview在PopupWindow中无法正常创建,可参见:SurfaceView not working inside PopupWindow

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值