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

在Activity的onCreate方法中显示PopupWindow导致异常的原因分析及解决方案
一、前言
        在某些情况下,我们需要一进入Activity就显示PopupWindow,比如常见的选择界面。但由于PopupWindow是依附于Activity的,如果Activity没有创建完成,Activity还没完全显示出来就显示PopupWindow的话,会出现异常现象。

二、问题复现
        我在Activity的onCreate()方法中调用如下方法:
    public void show( ){
        if( null != mPopupWindow ){
            mPopupWindow.showAtLocation(mView, Gravity.CENTER, 0, 0);
        }
    }

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

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

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

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

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

// 在界面组件onCreate函数中调用
final View anchor = findViewById(R.id.anchor);
    anchor.post(new Runnable(){
    @Override
    public void run(){
        // 构造和展现弹出窗口
        PopupWindow window = createWindow();
        window.showAsDropDown(anchor);
    }
});

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


        所以对于上面的问题,最简单的处理方法是,异步显示PopupWindow就好了:
    public void show( ){
        mView.post( new Runnable( ) {
            @Override
            public void run() {
                if( null != mPopupWindow ){
                    mPopupWindow.showAtLocation(mView, Gravity.CENTER, 0, 0);
                }
            }
        });
    }

四、题外话

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