Android系统原理与源码分析方案
出处:awey 发布于:2011-09-03 20:44:44
本文以下就是如何随心所欲地控制AlertDialog的介绍。
现在我们来看看个需求:如果某个应用需要弹出一个对话框。当单击“确定“按钮时完成某些工作,如果这些工作失败,对话框不能关闭。而当成功完成工作后,则关闭对话框。当然,无论何程度情况,单击“取消”按钮都会关闭对话框。
这个需求并不复杂,也并不过分。但使用过AlertDialog的读者都知道,无论单击的哪个按钮,无论按钮单击事件的执行情况如何,对话框是肯定要关闭的。也就是说,用户无法控制对话框的关闭动作。实际上,关闭对话框的动作已经在Android SDK写死了,并且未给使用者留有任何接口。但我的座右铭是“宇宙中没有什么是不能控制的”。
既然要控制对放框的关闭行为,首先就得分析是哪些类、哪些代码使这个对话框关闭的。进入AlertDialog类的源代码。在AlertDialog中只定义了一个变量:mAlert。这个变量是AlertCONtroller类型。AlertController类是Android的内部类,在com.android.internal.app包中,无法通过普通的方式访问。也无法在Eclipse中通过按Ctrl键跟踪进源代码。但可以直接在Android源代码中找到AlertController.java。我们再回到AlertDialog类中。AlertDialog类实际上只是一个架子。象设置按钮、设置标题等工作都是由AlertController类完成的。因此,AlertController类才是关键。
找到AlertController.java文件。打开后不要感到头晕哦,这个文件中的代码是很多地。不过这么多代码对本文的主题也没什么用处。下面就找一下控制按钮的代码。
在AlertController类的开头就会看到如下的代码:
View.OnClickListener mButtonHandler = new View.OnClickListener() {
public void onClick{
Message m = null;
if {
m = Message.obtain;
} else if {
m = Message.obtain;
} else if {
m = Message.obtain;
}
if {
m.sendToTarget();
}
// Post a message so we dismiss after the above handlers are executed
mHandler.obtainMessage
.sendToTarget();
}
};
从这段代码中可以猜出来,前几行代码用来触发对话框中的三个按钮的单击事件,而的代码则用来关闭对话框。
mHandler.obtainMessage
.sendToTarget();
上面的代码并不是直接来关闭对话框的,而是通过一个Handler来处理,代码如下:
private stATIc final class ButtonHandler extends Handler {
// Button clicks have Message.what as the BUTTON{1,2,3} constant
private staTIc final int MSG_DISMISS_DIALOG = 1;
private WeakReference《DialogInterface》 mDialog;
public ButtonHandler {
mDialog = new WeakReference《DialogInterface》;
}
@Override
public void handleMessage {
switch {
case DialogInterface.BUTTON_POSITIVE:
case DialogInterface.BUTTON_NEGATIVE:
case DialogInterface.BUTTON_NEUTRAL:
((DialogInterface.OnClickListener) msg.obj)。onClick(mDialog.get(), msg.what);
break;
case MSG_DISMISS_DIALOG:
((DialogInterface) msg.obj)。dismiss();
}
}
}
从上面代码的可以找到
((DialogInterface) msg.obj)。dismiss();。在AlertController处理对话框按钮时会为每一个按钮添加一个onclick事件。而这个事件类的对象实例就是上面的mButtonHandler。在这个单击事件中首先会通过发送消息的方式调用为按钮设置的单击事件(也就是通过setPositiveButton等方法的第二个参数设置的单击事件),在触发完按钮的单击事件后,会通过发送消息的方式调用dismiss方法来关闭对话框。而在AlertController类中定义了一个全局的mHandler变量。在AlertController类中通过ButtonHandler类来对象来为mHandler赋值。因此,我们只要使用我们自己Handler对象替换ButtonHandler就可以阻止调用dismiss方法来关闭对话框。下面先在自己的程序中建立一个新的ButtonHandler类。
class ButtonHandler extends Handler
{
private WeakReference《DialogInterface》 mDialog;
public ButtonHandler
{
mDialog = new WeakReference《DialogInterface》;
}
@Override
public
void handleMessage
{
switch
{
case DialogInterface.BUTTON_POSITIVE:
case DialogInterface.BUTTON_NEGATIVE:
case DialogInterface.BUTTON_NEUTRAL:
((DialogInterface.OnClickListener) msg.obj)。onClick(mDialog
.get(), msg.what);
break;
}
}
}
我们可以看到,上面的类和AlertController中的ButtonHandler类很像,只是支掉了switch语句的一个case子句和相关的代码。
由于mHandler是private变量,因此,在这里需要使用Java的反射技术来为mHandler赋值。由于在AlertDialog类中的mAlert变量同样也是private,因此,也需要使用同样的反射技术来获得mAlert变量。代码如下:
先建立一个AlertDialog对象
AlertDialog alertDialog = new AlertDialog.Builder
.setTitle
.setMessage
.setIcon
.setPositiveButton
new OnClickListener()
{
@Override
public void onClick
{
}
})。setNegativeButton()
{
@Override
public void onClick
{
dialog.dismiss();
}
})。create();
上面的对话框很普通,单击哪个按钮都会关闭对话框。下面在调用show方法之前来修改一个mHandler变量的值,OK,下面我们就来见证奇迹的时刻。
try
{
Field field = alertDialog1.getClass()。getDeclaredField;
field.setAccessible;
//
获得mAlert变量的值
Object obj = field.get;
field = obj.getClass()。getDeclaredField;
field.setAccessible;
//
修改mHandler变量的值,使用新的ButtonHandler类
field.set; }
catch {
}
//
显示对话框
alertDialog.show();
我们发现,如果加上try
catch语句,单击对话框中的确定按钮不会关闭对话框,单击取消按钮则会关闭对话框。如果去了try…catch代码段,对话框又会恢复正常了。
以上代码解决了问题,但需要的代码还比较多,所以我们采用了另一种方法来阻止关闭对话框。
这种方法需要用点技巧。由于系统通过调用dismiss来关闭对话框,那么我们可以在dismiss方法上做点文章。在系统调用dismiss方法时会首先判断对话框是否已经关闭,如果对话框已经关闭了,就会退出dismiss方法而不再继续关闭对话框了。因此,我们可以欺骗一下系统,当调用dismiss方法时我们可以让系统以为对话框已经关闭(虽然对话框还没有关闭),这样dismiss方法就失效了,这样即使系统调用了dismiss方法也无法关闭对话框了。
下面让我们回到AlertDialog的源代码中,再继续跟踪到AlertDialog的父类Dialog的源代码中。找到dismissDialog方法。实际上,dismiss方法是通过dismissDialog方法来关闭对话框的,dismissDialog方法的代码如下:
private
void dismissDialog() {
if {
if Log.v;
return;
}
if {
if Log.v;
return;
}
mWindowManager.removeView;
mDecor = null;
mWindow.closeAllPanels();
onStop();
mShowing = false;
sendDismissMessage();
}
该方法后面的代码不用管它,先看if{…}这段代码。这个mShowing变量就是判断对话框是否已关闭的。因此,我们在代码中通过设置这个变量就可以使系统认为对话框已经关闭,就不再继续关闭对话框了。由于mShowing也是private变量,因此,也需要反射技术来设置这个变量。我们可以在对话框按钮的单击事件中设置mShowing,代码如下:
try
{
Field field = dialog.getClass()
.getSuperclass()。getDeclaredField;
field.setAccessible(true);
//
将mShowing变量设为false,表示对话框已关闭
field.set;
dialog.dismiss();
}
catch
{
}
将上面的代码加到哪个按钮的单击事件代码中,哪个按钮就再也无法关闭对话框了。如果要关闭对话框,只需再将mShowing设为true即可。要注意的是,在一个按钮里设置了mShowing变量,也会影响另一个按钮的关闭对话框功能,因此,需要在每一个按钮的单击事件里都设置mShowing变量的值。
版权与免责声明
凡本网注明“出处:维库电子市场网”的所有作品,版权均属于维库电子市场网,转载请必须注明维库电子市场网,https://www.dzsc.com,违反者本网将追究相关法律责任。
本网转载并注明自其它出处的作品,目的在于传递更多信息,并不代表本网赞同其观点或证实其内容的真实性,不承担此类作品侵权行为的直接责任及连带责任。其他媒体、网站或个人从本网转载时,必须保留本网注明的作品出处,并自负版权等法律责任。
如涉及作品内容、版权等问题,请在作品发表之日起一周内与本网联系,否则视为放弃相关权利。
- 嵌入式系统的应用领域2024/11/12 17:16:41
- 使用 PLECSPIL 开发嵌入式控件2024/9/9 17:08:41
- 什么是嵌入式系统?嵌入式系统的优缺点及应用2024/9/5 17:49:34
- KEIL编程小技巧2024/8/13 17:34:35
- 什么是MCU?一文快速了解MCU基础知识2024/7/15 17:39:55