Android错误 - NullPointerException查看调度键事件

时间:2011-08-01 09:16:21

标签: android runtime nullpointerexception dispatch

当我在我的应用程序中浏览时,我有时会遇到此错误。我不确定这个错误是随机的,我不知道为什么。日志没有显示我的应用程序的任何特定内容?

我希望有人可以给我一个提示或暗示为什么会出现此错误。 (最后一次,当我尝试打开一个弹出窗口,一个Quickaction对话框,在使用相同的Quickaction Dialog第二次出现之后,它再次在我第三次尝试之后再次工作。现在它一直在工作,又一次吗?)

08-01 11:09:15.980: ERROR/AndroidRuntime(9579): FATAL EXCEPTION: main
08-01 11:09:15.980: ERROR/AndroidRuntime(9579): java.lang.NullPointerException
08-01 11:09:15.980: ERROR/AndroidRuntime(9579):     at android.widget.PopupWindow$PopupViewContainer.dispatchKeyEvent(PopupWindow.java:1445)
08-01 11:09:15.980: ERROR/AndroidRuntime(9579):     at android.view.ViewRoot.deliverKeyEventToViewHierarchy(ViewRoot.java:2664)
08-01 11:09:15.980: ERROR/AndroidRuntime(9579):     at android.view.ViewRoot.deliverKeyEvent(ViewRoot.java:2629)
08-01 11:09:15.980: ERROR/AndroidRuntime(9579):     at android.view.ViewRoot.handleMessage(ViewRoot.java:1935)
08-01 11:09:15.980: ERROR/AndroidRuntime(9579):     at android.os.Handler.dispatchMessage(Handler.java:99)
08-01 11:09:15.980: ERROR/AndroidRuntime(9579):     at android.os.Looper.loop(Looper.java:143)
08-01 11:09:15.980: ERROR/AndroidRuntime(9579):     at android.app.ActivityThread.main(ActivityThread.java:4196)
08-01 11:09:15.980: ERROR/AndroidRuntime(9579):     at java.lang.reflect.Method.invokeNative(Native Method)
08-01 11:09:15.980: ERROR/AndroidRuntime(9579):     at java.lang.reflect.Method.invoke(Method.java:507)
08-01 11:09:15.980: ERROR/AndroidRuntime(9579):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
08-01 11:09:15.980: ERROR/AndroidRuntime(9579):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
08-01 11:09:15.980: ERROR/AndroidRuntime(9579):     at dalvik.system.NativeStart.main(Native Method)

这里有更多细节:

QuickAction qa = QuickactionBuilder.showSaveForgetQuickaction();

                qa.setOnActionItemClickListener(new QuickAction.OnActionItemClickListener() {
                    @Override
                    public void onItemClick(int pos) {
                        if (pos == 0) { // SAVE
                            Toast.makeText(ctxHolder.getCtx(), "TODO SAVE", Toast.LENGTH_SHORT).show();
                        } else if (pos == 1) { // REJECT
                            Toast.makeText(ctxHolder.getCtx(), "REJECTED", Toast.LENGTH_SHORT).show();
                        } else { // CANCEL
                            Toast.makeText(ctxHolder.getCtx(), "CANCEL", Toast.LENGTH_SHORT).show();
                        }
                    }
                });

                qa.show(v);

QuickactionBuilder只为我创建QuickAction对话框。 这很简单: (仅显示该方法的基础知识)

QuickAction qa = new QuickAction
ActionItem ai = new ActionItem() 
// some setters ...

qa.add(ai)

return qa;

3 个答案:

答案 0 :(得分:1)

我遇到了与QuickAction相同的问题 我收到了用户的崩溃,虽然我无法在我的设备上重现它们 所以,如果有人仍然需要解决方案,我很乐意分享它

这个想法是避免在PopupWindow中使用PopupViewContainer,因为崩溃错误在其dispatchKeyEvent(KeyEvent事件)方法中。我能做到的唯一方法是保持backgroundDrawable == null。然后你必须自己处理外部触摸和后退按钮。

PopupWindows类:

public PopupWindows(Context context) {
    mContext    = context;

    mWindow     = new PopupWindow(context);
    mWindow.setBackgroundDrawable(null);
    ......
}

protected void onShow() {   

    if (mRootView == null) {
        return;
    }

    if (mRootView instanceof CustomRelativeLayout) {
        ((CustomRelativeLayout)mRootView).setDispatchKeyEventListener(new CustomRelativeLayout.OnDispatchKeyEventListener() {

            @Override
            public void onDispatchKeyEvent(KeyEvent event) {
                if (event.getKeyCode() == KeyEvent.KEYCODE_BACK && event.getRepeatCount() == 0 && mWindow.isShowing()) {
                    dismiss();
                }
            }
        });
    }       

    mRootView.setOnTouchListener(new View.OnTouchListener() {

        @Override
        public boolean onTouch(View v, MotionEvent event) {
            if (event.getAction() == MotionEvent.ACTION_DOWN && mWindow.isShowing()) {
                dismiss();
                return true;
            }
            return false;
        }
    });

}

请勿在此处设置任何可绘制的内容:

protected void preShow() {
    if (mRootView == null) 
        throw new IllegalStateException("setContentView was not called with a view to display.");

    onShow();

    mWindow.setWidth(WindowManager.LayoutParams.WRAP_CONTENT);
    mWindow.setHeight(WindowManager.LayoutParams.WRAP_CONTENT);
    mWindow.setTouchable(true);
    mWindow.setFocusable(true);
    mWindow.setOutsideTouchable(true);

    mWindow.setContentView(mRootView);
}

接下来的一切都是经典的

CustomRelativeLayout:

public class CustomRelativeLayout extends RelativeLayout {

public CustomRelativeLayout(Context context, AttributeSet attrs, int defStyle) {
    super(context, attrs, defStyle);
}

public CustomRelativeLayout(Context context, AttributeSet attrs) {
    super(context, attrs);
}

public CustomRelativeLayout(Context context) {
    super(context);
}

private OnDispatchKeyEventListener mOnDispatchKeyEventListener;



public void setDispatchKeyEventListener(OnDispatchKeyEventListener listener) {
    mOnDispatchKeyEventListener = listener;
}

@Override
public boolean dispatchKeyEvent(KeyEvent event) {
    if (mOnDispatchKeyEventListener != null) {
        mOnDispatchKeyEventListener.onDispatchKeyEvent(event);
    }
    return super.dispatchKeyEvent(event);
}

@Override
public boolean onTouchEvent(MotionEvent event) {
    return super.onTouchEvent(event);
}   

public static interface OnDispatchKeyEventListener {
    public void onDispatchKeyEvent(KeyEvent event);
}
}

popup_vertical.xml和popup_horizo​​ntal.xml:

只需使用com.your.package.CustomRelativeLayout而不是顶级RelativeLayout

希望这会有所帮助

答案 1 :(得分:0)

您必须在UI线程上执行一些昂贵的操作。我建议你将这些操作移动到一个单独的非UI线程。发布更多代码来识别疼痛区域。

答案 2 :(得分:0)

这个答案现在很老了。我找到了解决方案,这个问题是由Quickaction Library istelf引起的。我的解决方案是使用不同的库。

它不同但它的效果也很好。该库还提供了一些其他功能。

检查出来:http://greendroid.cyrilmottier.com/