最近在优化一个dialog,里面使用AndroidTreeView展示树状数据。由于数据量颇多,每次打开dialog都会卡顿一两秒,用户体验不好。于是想第一次加载完dialog后,缓存dialog,再打开时就不用重新生成了。
开始以为很简单,结果也花了点时间。到底实现功能了,在这里记录一下。
创建Dialog使用的是DialogFragment,官方不推荐直接使用Dialog创建对话框,具体使用方法就不说了。
使用AlertDialog.Builder的setXXXButton方法,就可以设置Positive、Negative、Neutral三种按钮的onClick动作,代码如下:
builder.setPositiveButton(R.string.common_ok, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
}
}
});
目标是隐藏Dialog,理论上在onClick方法里调用Dialog的hide()方法就可以了。实际没有这么简单,onClick方法调用完毕后,会自动调用Dialog的dismiss方法,导致Dialog被销毁,无法实现隐藏。
从setPositiveButton()开始跟踪源码,方法会为AlertDialog.Builder的AlertController.AlertParams设置参数,记录OnClickListener方法:
public Builder setPositiveButton(int textId, final OnClickListener listener) {
P.mPositiveButtonText = P.mContext.getText(textId);
P.mPositiveButtonListener = listener;
return this;
}
Dialog创建的时候,会设置按钮的listener,最终由mButtonHandler处理事件。从最后一句代码可以知道,Dialog会自动调用dismiss方法关闭:
private final View.OnClickListener mButtonHandler = new View.OnClickListener() {
@Override
public void onClick(View v) {
final Message m;
if (v == mButtonPositive && mButtonPositiveMessage != null) {
m = Message.obtain(mButtonPositiveMessage);
} else if (v == mButtonNegative && mButtonNegativeMessage != null) {
m = Message.obtain(mButtonNegativeMessage);
} else if (v == mButtonNeutral && mButtonNeutralMessage != null) {
m = Message.obtain(mButtonNeutralMessage);
} else {
m = null;
}
if (m != null) {
m.sendToTarget();
}
// Post a message so we dismiss after the above handlers are executed
mHandler.obtainMessage(ButtonHandler.MSG_DISMISS_DIALOG, mDialog)
.sendToTarget();
}
};
@Override
public void dismiss() {
if (Looper.myLooper() == mHandler.getLooper()) {
dismissDialog();
} else {
mHandler.post(mDismissAction);
}
}
void dismissDialog() {
if (mDecor == null || !mShowing) {
return;
}
if (mWindow.isDestroyed()) {
Log.e(TAG, "Tried to dismissDialog() but the Dialog's window was already destroyed!");
return;
}
try {
mWindowManager.removeViewImmediate(mDecor);
} finally {
if (mActionMode != null) {
mActionMode.finish();
}
mDecor = null;
mWindow.closeAllPanels();
onStop();
mShowing = false;
sendDismissMessage();
}
}
参考别人的解决方法,想要阻止dialog自动关闭,只要将mShowing设置为false即可。需要使用反射,代码如下:
try {
Field field = dialog.getClass().getSuperclass().getSuperclass().getDeclaredField("mShowing");
field.setAccessible(true);
field.set(dialog, false);
} catch (Exception e) {
e.printStackTrace();
}
最后实现Dialog点击按钮隐藏可以如下代码所写:
builder.setNegativeButton(R.string.common_cancel, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
try {
Field field = dialog.getClass().getSuperclass().getSuperclass().getDeclaredField("mShowing");
field.setAccessible(true);
field.set(dialog, false);
} catch (Exception e) {
e.printStackTrace();
}
mAlertDialog.hide();
}
});
Dialog的hide()很简单,只是将View设置为GONE:
public void hide() {
if (mDecor != null) {
mDecor.setVisibility(View.GONE);
}
}
Dialog的重新显示比较简单,只要调用show()方法即可,但要注意,mShowing在Dialog隐藏的时候已经被设置为false,查看show()的代码如下:
public void show() {
if (mShowing) {
if (mDecor != null) {
if (mWindow.hasFeature(Window.FEATURE_ACTION_BAR)) {
mWindow.invalidatePanelMenu(Window.FEATURE_ACTION_BAR);
}
mDecor.setVisibility(View.VISIBLE);
}
return;
}
//省略其他代码
}
想要被缓存Dialog正常地显示出来,需要将mShowing重新设置为true,才能调用到Decor.setVisibility(View.VISIBLE);
try {
Field field = alertDialog.getClass().getSuperclass().getSuperclass().getDeclaredField("mShowing");
field.setAccessible(true);
field.set(alertDialog, true);
} catch (Exception e) {
e.printStackTrace();
}
dialogFragment.getDialog().show();
当页面跳转不需要Dialog时,需要记得主动销毁。我是在Activity中缓存DialogFragment对象,一开始在onDestory时调用dismiss,但应用收起后台再返回前台会报错,在onStop时调用也不行,只有在onPause时才行。我估计在onStop时会做一些缓存操作,而手动缓存的Fragment影响了系统的操作。
知道了一些,但又知道我不知道很多。对于Activity的生命周期理解还不够,接下来要重点研究。
第一次发表博客,感觉不错。接下来,在工作中遇到的问题和新学习的知识,都要尽量记录,每个星期发一篇!如果亲爱的你有任何建议,将告诉我,谢谢。