今天设置AlertDialog不自动取消时采用以下代码,使用反射机制获取Dialog中的字段mShowing,
但一直报错:java.lang.NoSuchFieldException: mShowing
意思是未找到变量
public class DialogUtil {
public static void setClosable(DialogInterface dialog, boolean b) {
try {
Field field;
field = dialog.getClass().getSuperclass() .getDeclaredField("mShowing");//注意这一行
field.setAccessible(true);//使其获得访问private的权限,否则抛出第二个异常
field.set(dialog, b);
} catch (NoSuchFieldException e) {
e.printStackTrace();//这里抛出异常
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
}
}
}
后来将上面一行代码改为field = dialog.getClass().getSuperclass().getSuperclass().getDeclaredField("mShowing");注意,多了一个getSuperclass(),改了之后就行了
原因:getSuperclass()得到是超类,而在eclipse中AlertDialog的超类就是Dialog,但在Android studio中超类为AppCompatDialog,再超类才是Dialog,所以需要两次getSuperclass
AlertDialog之所以会自动消失,是因为系统在点击按钮后会自动调用dismiss()方法,但是我们可以通过修改该方法来实现目的. 系统调用dismiss时会检查对话框是否已经show了,如果我们调用setClosable方法,,就会修改了mShowing变量,就会骗过系统,以为对话框已经消失了,就不会继续执行dismiss了,这样就保留了对话框,
但是,这也导致了我们自己调用dismiss的时候也不能使对话框消失,所以在调用前需要再调用一次setClosable,并传入true参数. 一旦调用了setClosable,整个Dialog对象中的按钮都会受影响,所以如果两个按钮需要不同设置,就需要各调用一次
另外,设置点击框外对话框不消失的api为Builder. setCancelable()