由于系统通过调用dismiss来关闭对话框,那么我们可以在dismiss方法上做点文章。在系统调用dismiss方法时会首先判断对话框是否已经关闭,如果对话框已经关闭了,就会退出dismiss方法而不再继续关闭对话框了。因此,我们可以欺骗一下系统,当调用dismiss方法时我们可以让系统以为对话框已经关闭(虽然对话框还没有关闭),这样dismiss方法就失效了,这样即使系统调用了dismiss方法也无法关闭对话框了。
下面让我们回到AlertDialog的源代码中,再继续跟踪到AlertDialog的父类Dialog的源代码中。找到dismissDialog方法。实际上,dismiss方法是通过dismissDialog方法来关闭对话框的:
下面让我们回到AlertDialog的源代码中,再继续跟踪到AlertDialog的父类Dialog的源代码中。找到dismissDialog方法。实际上,dismiss方法是通过dismissDialog方法来关闭对话框的:
- private void dismissDialog() {
- if (mDecor == null ) {
- if (Config.LOGV) Log.v(LOG_TAG,
- " [Dialog] dismiss: already dismissed, ignore " );
- return ;
- }
- if ( ! mShowing) {
- if (Config.LOGV) Log.v(LOG_TAG,
- " [Dialog] dismiss: not showing, ignore " );
- return ;
- }
- mWindowManager.removeView(mDecor);
- mDecor = null ;
- mWindow.closeAllPanels();
- onStop();
- mShowing = false ;
- sendDismissMessage();
- }
该方法后面的代码不用管它,先看 if(!mShowing){ … }这段代码。这个 mShowing变量就是判断对话框是否已关闭的。因此,我们在代码中通过设置这个变量就可以使系统认为对话框已经关闭,就不再继续关闭对话框了。由于 mShowing也是 private变量,因此,也需要反射技术来设置这个变量。我们可以在对话框按钮的单击事件中设置 mShowing,代码如下:
不关闭对话框:
- try {
- Field field = dialog.getClass().getSuperclass().getDeclaredField("mShowing");
- field.setAccessible(true);
- field.set(dialog, false);
- } catch (Exception e) {
- e.printStackTrace();
- }
关闭对话框:
- try {
- Field field = dialog.getClass().getSuperclass().getDeclaredField("mShowing");
- field.setAccessible(true);
- field.set(dialog, true);
- } catch (Exception e) {
- e.printStackTrace();
- }
- builder.setPositiveButton("确认", new DialogInterface.OnClickListener(){
- public void onClick(DialogInterface dialog, int which) {
- //设置不关闭对对话框
- try {
- Field field = dialog.getClass().getSuperclass().getDeclaredField("mShowing");
- field.setAccessible(true);
- field.set(dialog, false);
- } catch (Exception e) {
- e.printStackTrace();
- }
- //A条件,正常响应之后,关闭对话框
- //B条件,未符合要求,保留对话框
- if(A条件){
- //……
- //关闭对话框
- try {
- Field field = dialog.getClass().getSuperclass().getDeclaredField("mShowing");
- field.setAccessible(true);
- field.set(dialog, true);
- } catch (Exception e) {
- e.printStackTrace();
- }
- }else{
- //……
- }
- }
- });