《電子技術應用》
您所在的位置:首頁 > 其他 > 設計應用 > Android系統原理與源碼分析
Android系統原理與源碼分析
OFweek電子工程網
摘要: 眾所周知,AlertDialog類對于顯示對話。關于AlertDialog的基本用法在這里就不詳細介紹了,網上有很多,讀者可以自己搜索。那么本文要介紹的是如何隨心所欲地控制AlertDialog。
Abstract:
Key words :

  眾所周知,AlertDialog類對于顯示對話。關于AlertDialog的基本用法在這里就不詳細介紹了,網上有很多,讀者可以自己搜索。那么本文要介紹的是如何隨心所欲地控制AlertDialog。

  現在我們來看看第一個需求:如果某個應用需要彈出一個對話框。當單擊“確定“按鈕時完成某些工作,如果這些工作失敗,對話框不能關閉。而當成功完成工作后,則關閉對話框。當然,無論何程度情況,單擊“取消”按鈕都會關閉對話框。

  這個需求并不復雜,也并不過分(雖然我們可以自己弄個Activity來完成這個工作,也可在View上自己放按鈕,但這顯示有些大炮打蚊子了,如果對話框上只有一行文本,費這么多勁太不值了)。但使用過AlertDialog的讀者都知道,無論單擊的哪個按鈕,無論按鈕單擊事件的執行情況如何,對話框是肯定要關閉的。也就是說,用戶無法控制對話框的關閉動作。實際上,關閉對話框的動作已經在Android" title="Android">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(View v) {

  Message m = null;

  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);

  }

  if (m != null) {

  m.sendToTarget();

  }

  // Post a message so we dismiss after the above handlers are executed

  mHandler.obtainMessage(ButtonHandler.MSG_DISMISS_DIALOG, mDialogInterface)

  .sendToTarget();

  }

  };

  從這段代碼中可以猜出來,前幾行代碼用來觸發對話框中的三個按鈕(Positive、Negative和Neutral)的單擊事件,而最后的代碼則用來關閉對話框(因為我們發現了MSG_DISMISS_DIALOG、猜出來的)。

  mHandler.obtainMessage(ButtonHandler.MSG_DISMISS_DIALOG, mDialogInterface)

  .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(DialogInterface dialog) {

  mDialog = new WeakReference《DialogInterface》(dialog);

  }

  @Override

  public void handleMessage(Message msg) {

  switch (msg.what) {

  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(DialogInterface dialog)

  {

  mDialog = new WeakReference《DialogInterface》(dialog);

  }

  @Override

  public

  void handleMessage(Message msg)

  {

  switch (msg.what)

  {

  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子句(用于調用dismiss方法)和相關的代碼。

  眾所周知,AlertDialog類對于顯示對話。關于AlertDialog的基本用法在這里就不詳細介紹了,網上有很多,讀者可以自己搜索。那么本文要介紹的是如何隨心所欲地控制AlertDialog。

  現在我們來看看第一個需求:如果某個應用需要彈出一個對話框。當單擊“確定“按鈕時完成某些工作,如果這些工作失敗,對話框不能關閉。而當成功完成工作后,則關閉對話框。當然,無論何程度情況,單擊“取消”按鈕都會關閉對話框。

  這個需求并不復雜,也并不過分(雖然我們可以自己弄個Activity來完成這個工作,也可在View上自己放按鈕,但這顯示有些大炮打蚊子了,如果對話框上只有一行文本,費這么多勁太不值了)。但使用過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(View v) {

  Message m = null;

  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);

  }

  if (m != null) {

  m.sendToTarget();

  }

  // Post a message so we dismiss after the above handlers are executed

  mHandler.obtainMessage(ButtonHandler.MSG_DISMISS_DIALOG, mDialogInterface)

  .sendToTarget();

  }

  };

  從這段代碼中可以猜出來,前幾行代碼用來觸發對話框中的三個按鈕(Positive、Negative和Neutral)的單擊事件,而最后的代碼則用來關閉對話框(因為我們發現了MSG_DISMISS_DIALOG、猜出來的)。

  mHandler.obtainMessage(ButtonHandler.MSG_DISMISS_DIALOG, mDialogInterface)

  .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(DialogInterface dialog) {

  mDialog = new WeakReference《DialogInterface》(dialog);

  }

  @Override

  public void handleMessage(Message msg) {

  switch (msg.what) {

  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(DialogInterface dialog)

  {

  mDialog = new WeakReference《DialogInterface》(dialog);

  }

  @Override

  public

  void handleMessage(Message msg)

  {

  switch (msg.what)

  {

  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子句(用于調用dismiss方法)和相關的代碼。

  下面我們就要為AlertController中的mHandler重新賦值。由于mHandler是private變量,因此,在這里需要使用Java的反射技術來為mHandler賦值。由于在AlertDialog類中的mAlert變量同樣也是private,因此,也需要使用同樣的反射技術來獲得mAlert變量。代碼如下:

  先建立一個AlertDialog對象

  AlertDialog alertDialog = new AlertDialog.Builder(this)

  .setTitle(“abc”)

  .setMessage(“content”)

  .setIcon(R.drawable.icon)

  .setPositiveButton( “確定”,

  new OnClickListener()

  {

  @Override

  public void onClick(DialogInterface dialog,

  int which)

  {

  }

  }).setNegativeButton(“取消”, new OnClickListener()

  {

  @Override

  public void onClick(DialogInterface dialog, int which)

  {

  dialog.dismiss();

  }

  }).create();

  上面的對話框很普通,單擊哪個按鈕都會關閉對話框。下面在調用show方法之前來修改一個mHandler變量的值,OK,下面我們就來見證奇跡的時刻。

  try

  {

  Field field = alertDialog1.getClass().getDeclaredField(“mAlert”);

  field.setAccessible(true);

  //

  獲得mAlert變量的值

  Object obj = field.get(alertDialog1);

  field = obj.getClass().getDeclaredField(“mHandler”);

  field.setAccessible(true);

  //

  修改mHandler變量的值,使用新的ButtonHandler類

  field.set(obj, new ButtonHandler(alertDialog1));

  }

  catch (Exception e)

  {

  }

  //

  顯示對話框

  alertDialog.show();

  我們發現,如果加上try

  catch語句,單擊對話框中的確定按鈕不會關閉對話框(除非在代碼中調用dismiss方法),單擊取消按鈕則會關閉對話框(因為調用了dismiss方法)。如果去了try…catch代碼段,對話框又會恢復正常了。

  雖然上面的代碼已經解決了問題,但需要編寫的代碼仍然比較多,為此,我們也可采用另外一種方法來阻止關閉對話框。這種方法不需要定義任何的類。

  這種方法需要用點技巧。由于系統通過調用dismiss來關閉對話框,那么我們可以在dismiss方法上做點文章。在系統調用dismiss方法時會首先判斷對話框是否已經關閉,如果對話框已經關閉了,就會退出dismiss方法而不再繼續關閉對話框了。因此,我們可以欺騙一下系統,當調用dismiss方法時我們可以讓系統以為對話框已經關閉(雖然對話框還沒有關閉),這樣dismiss方法就失效了,這樣即使系統調用了dismiss方法也無法關閉對話框了。

  下面讓我們回到AlertDialog的源代碼中,再繼續跟蹤到AlertDialog的父類Dialog的源代碼中。找到dismissDialog方法。實際上,dismiss方法是通過dismissDialog方法來關閉對話框的,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);

  //

  將mShowing變量設為false,表示對話框已關閉

  field.set(dialog, false);

  dialog.dismiss();

  }

  catch (Exception e)

  {

  }

  將上面的代碼加到哪個按鈕的單擊事件代碼中,哪個按鈕就再也無法關閉對話框了。如果要關閉對話框,只需再將mShowing設為true即可。要注意的是,在一個按鈕里設置了mShowing變量,也會影響另一個按鈕的關閉對話框功能,因此,需要在每一個按鈕的單擊事件里都設置mShowing變量的值。

此內容為AET網站原創,未經授權禁止轉載。
亚洲一区二区欧美_亚洲丝袜一区_99re亚洲国产精品_日韩亚洲一区二区
国产主播一区二区| 精品成人a区在线观看| 久久不射电影网| 在线一区二区三区四区五区| 亚洲欧洲在线一区| 久久精品一区| 久久av一区二区三区| 亚洲欧美日韩国产另类专区| 亚洲午夜性刺激影院| 亚洲精品乱码久久久久久黑人| 亚洲成色999久久网站| 狠狠色噜噜狠狠色综合久| 国产午夜精品麻豆| 国产精品自在欧美一区| 国产精品视频久久| 国产精品网站视频| 国产精品一区二区a| 国产精品丝袜久久久久久app| 欧美视频在线一区| 国产精品狠色婷| 国产精品久久久999| 欧美三级网址| 国产精品啊v在线| 国产精品美女久久福利网站| 国产美女精品免费电影| 国产婷婷色一区二区三区四区| 国产无一区二区| 黄色在线成人| 亚洲二区在线观看| 亚洲精品乱码久久久久| 亚洲久久成人| 亚洲午夜在线| 亚洲欧美精品一区| 久久精品国产亚洲aⅴ| 亚洲国产精品久久久久婷婷老年 | 久久精品视频免费播放| 久久爱另类一区二区小说| 亚洲国产成人tv| 亚洲另类春色国产| 亚洲永久免费观看| 欧美一区二区观看视频| 久久久亚洲国产美女国产盗摄| 另类图片国产| 欧美日韩免费一区| 国产美女精品在线| 亚洲国产成人一区| 一本一本久久a久久精品综合妖精| 亚洲一区图片| 亚洲国产精品一区| 一区二区三区视频在线看| 欧美一区二区三区四区在线| 毛片一区二区三区| 欧美日韩中字| 韩国视频理论视频久久| 亚洲精品少妇| 亚洲欧美影音先锋| 亚洲日本成人女熟在线观看| 亚洲综合社区| 免费在线观看成人av| 欧美日韩亚洲视频| 国产香蕉久久精品综合网| 亚洲黑丝在线| 午夜精品一区二区三区四区| 亚洲国产欧美一区二区三区久久| 亚洲视频中文字幕| 久久人体大胆视频| 欧美日韩一区二区在线视频| 国产性做久久久久久| 亚洲狼人综合| 久久国产精品99精品国产| 在线视频你懂得一区二区三区| 欧美一区二区三区另类| 欧美高清视频在线| 国产精品尤物| 亚洲巨乳在线| 亚洲国产一区二区在线| 午夜精品福利视频| 欧美剧在线观看| 亚洲欧美国产不卡| 欧美日韩精品免费| 国产一区二区三区高清| 亚洲精品综合久久中文字幕| 欧美中文字幕| 亚洲欧美国产精品桃花| 欧美成人激情视频免费观看| 国产日产欧产精品推荐色| 99国产精品私拍| 91久久久久久| 欧美一区二区精品久久911| 欧美日韩国产小视频| 伊人婷婷久久| 欧美一区二区三区成人| 午夜视频一区| 欧美三级在线| 亚洲片国产一区一级在线观看| 久久成人国产| 欧美制服第一页| 国产精品美女久久久久久免费 | 亚洲欧美日韩一区| 欧美日本网站| 亚洲国产成人久久| 久久精品人人| 性久久久久久久久| 欧美午夜免费影院| 亚洲人体大胆视频| 亚洲国产一区在线观看| 久久婷婷人人澡人人喊人人爽| 国产精品视频免费在线观看| 一区二区日韩精品| 一区二区三区三区在线| 欧美日韩国产色视频| 亚洲精选一区二区| 日韩午夜剧场| 欧美激情精品久久久久久| 尤物九九久久国产精品的分类| 欧美一区二区三区免费视频| 先锋影音久久| 国产欧美日韩激情| 亚洲综合色婷婷| 亚洲男人第一网站| 国产精品嫩草久久久久| 中文在线资源观看视频网站免费不卡| 一区二区三区久久网| 欧美亚男人的天堂| 在线亚洲一区| 西西裸体人体做爰大胆久久久| 国产精品私拍pans大尺度在线| 亚洲天堂免费观看| 午夜在线视频一区二区区别| 国产欧美日韩一级| 欧美一区二区三区视频在线观看| 久久疯狂做爰流白浆xx| 国产一区二区高清| 久久精品国产久精国产一老狼| 久久只精品国产| 亚洲电影免费观看高清完整版在线观看| 亚洲国产高清一区二区三区| 欧美成人综合| 亚洲精品免费在线| 国产女人aaa级久久久级| 亚洲一区区二区| 制服丝袜激情欧洲亚洲| 欧美视频一区二区三区…| 亚洲性av在线| 久久精品色图| 在线免费观看视频一区| 亚洲美女黄网| 欧美视频在线观看免费| 午夜精品亚洲| 欧美成人69av| 夜夜爽99久久国产综合精品女不卡| 亚洲女人av| 一区二区视频欧美| 99国产一区| 国产日韩精品在线播放| 亚洲激情在线观看视频免费| 欧美日韩精品一区视频| 亚洲欧美日韩第一区| 免费亚洲网站| 亚洲视频图片小说| 久久久久久亚洲精品杨幂换脸| 亚洲第一区中文99精品| 在线亚洲免费| 国产有码在线一区二区视频| 亚洲人成精品久久久久| 欧美视频日韩视频| 久久av一区二区| 欧美日本精品一区二区三区| 一区二区三区高清| 午夜在线a亚洲v天堂网2018| 狠狠色狠色综合曰曰| 亚洲午夜av电影| 国产在线拍揄自揄视频不卡99 | 久久精品30| 亚洲精品国产精品国自产在线 | 亚洲一区二区三区四区五区黄| 久久频这里精品99香蕉| 亚洲精品乱码久久久久久久久| 午夜在线a亚洲v天堂网2018| 亚洲国产另类精品专区| 亚洲欧美日韩综合| 亚洲高清av| 久久av红桃一区二区小说| 亚洲人成精品久久久久| 久久精品久久99精品久久| 亚洲精品乱码久久久久久| 久久久久久久性| 一区二区日韩伦理片| 蜜臀av性久久久久蜜臀aⅴ| 亚洲一区免费观看| 欧美激情精品久久久久| 亚洲视频自拍偷拍| 久久美女性网| 黄色av日韩| 亚洲日本aⅴ片在线观看香蕉| 亚洲视频免费在线观看| 老司机精品视频一区二区三区| 一区二区欧美视频| 日韩天堂在线视频| 久久综合色88|