> Dialog是一個常見的顯示在當前activity之上的小窗口。下面的activity會失去焦點,而dialog回接受用戶輸入。dialog常用在與程序直接相關聯(lián)的通知和短小的activity中。 Android API支持以下幾種dialog: 它可以包含0、1、2、3個按鈕,或者一個列表或者多選單選按鈕等,它是一個功能最強大的dialog接口,詳細信息可參考下面的章節(jié)。 它會顯示一個進度條或者進度環(huán),因為他是AlertDialog的子類,所有野支持按鈕。 用來選擇日期的對話框。 用來選擇時間的。 如果你想要定制自己的dialog,你可以繼承Dialog對象,或者它的任何一個子類,并且定義一個新的布局。 Dialog 總是被當做activity的一部分來創(chuàng)建和顯示。你可以在activity的onCreateDialog(int)方法中創(chuàng)建一個dialog。當你使用這個方法,android系統(tǒng)會自動的管理每個dialog的狀態(tài)并且關聯(lián)到所在的activity中,讓這個activity成為dialog的管理者。每個dialog都會繼承activity的某些特性。例如,當dialog打開時,按下menu彈出的是所在activity的菜單,調節(jié)的是所在activity的音量。 注意:如果你決定在onCreateDialog()方法之外建立dialog,他將不會連接到activity中,此時,你可以使用setOwnerActivity(Activity)方法來綁定activity。 當你顯示dialog時,調用showDialog(int)來傳遞一個dialog的id句柄。 當一個dialog首次顯示時,android會在實例化dialog的activity中調用onCreateDialog(int)方法。回調方法會傳遞相同的id給showDialog(int)。當 創(chuàng)建完一個dialog后,會再方法的最后返回這個對象。 在dialog顯示前,android回調用可選的方法 :onPrepareDialog(int,Dialog)。如果你想在每次調用dialog時改變一些配置的話,你可以定義這個方法。OnPrepareDialog(int,Dialog)方法會在每次調用dialog時調用,而onCreateDialog(int)方法只會調用一次。如果你不定義onPrepareDialog()方法,那么打開的dialog會保持上一次的狀態(tài)。這個方法也會傳遞dialog的id句柄。 定義這兩個onXXX()方法最好使用一個switch結構來檢測Id參數,每一個case項都應該創(chuàng)建自己的dialog。例如。想象一個 游戲使用兩個不同的dialog,一個暫停一個結束游戲:
- static final int DIALOG_PAUSED_ID = 0;
static final int DIALOG_GAMEOVER_ID = 1;
復制代碼然后,再onCreateDialog(int)里根據id創(chuàng)建dialog: - protected Dialog onCreateDialog(int id) {
Dialog dialog;
switch(id) {
case DIALOG_PAUSED_ID:
// do the work to define the pause Dialog
break;
case DIALOG_GAMEOVER_ID:
// do the work to define the game over Dialog
break;
default:
dialog = null;
}
return dialog;
}
復制代碼注意:在例子中沒有詳寫,因為定義dialog屬于另外的章節(jié)。現(xiàn)在可以調用showDilaog(int)來顯示一個dialog了: - showDialog(DIALOG_PAUSED_ID);
復制代碼 調用dialog的dismiss()方法可以隱藏正在顯示的dialog,如果必要的話,可以調用activity的dismissDialog(int)方法,他倆效果是一樣的。如果使用的onCreateDialog(int)方法來管理dialog的狀態(tài),那么每次當你的dialog消失時,對話框的狀態(tài)都會被activity保存著。如果不太需要這個對話框或者不希望activity保留dialog的狀態(tài),可以調用removeDialog(int)方法。它會刪除任何關于dialog的引用,如果dialog正在顯示,此方法會讓dialog隱藏。隱藏dialog監(jiān)聽器的使用如果你想讓activity在dialog隱藏時執(zhí)行某些動作,那么你可以建立一個監(jiān)聽器。首先定義DialogInterface.OnDismissListerner 接口,這個接口只有一個方法,onDismiss(DialogInterface),當dialog隱藏時被調用,然后傳遞OnDismissListener 對象給setOnDismissLister()方法。然而,注意dialog也可以是取消,用戶讓這個dialog取消也是一種特殊的情況。當用戶按下back鍵時,或者調用cancel()方法時會發(fā)生這種情況。當一個dialog被取消時,OnDismissLister監(jiān)聽器仍然會收到通知,但如果你喜歡的到明確的取消消息,可以注冊DialogInterface.OnCancelLister監(jiān)聽器。
AlertDialog時Dialog的子類,Dilaog絕大多數是這個強大類型,你可以在以下情況下使用:@ 一個標題@ 一個文本信息@ 一個兩個或者三個按鈕@ 一個單選或者多選列表建立AlertDialog,使用AlertDialog.Builder子類。使用AlertDialog.Builder(Context)方法來獲得一個Builder,并且使用它的公共方法來定義AlertDialog所有的屬性。最后,調用create()方法來顯示。下面顯示了如何定義AlertDialog.Builder類的一些屬性,如果在onCreateDialog()方法中使用了例子中的代碼,你可以返回結果對話框來顯示這個dialog。
創(chuàng)建一個上圖所示包含按鈕的AlertDialog,可以使用setXXXButton()方法: - AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setMessage("Are you sure you want to exit?")
.setCancelable(false)
.setPositiveButton("Yes", new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int id) {
MyActivity.this.finish();
}
})
.setNegativeButton("No", new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int id) {
dialog.cancel();
}
});
AlertDialog alert = builder.create();
復制代碼首先,通過setMessage(CharSequence)為dialog添加一個message,然后通過setCancelable(boolean)方法讓此dialog無法通過按back鍵來取消。每個按鈕都需要調用setXXXButton()方法,例如setPositiveButton()方法,DialogInterface.OnClickListener()類會定義按下按鈕所要做的處理。注意:每種類型的按鈕只能加一個,這就是說,你不能添加多于一個的positive按鈕。最多能添加三個按鈕,positive, neutral, 和 negative.他們名字所顯示的功能并未實現(xiàn),但能幫你記住要實現(xiàn)的功能。
如上圖所示,使用setItems()方法添加可選列表: - final CharSequence[] items = {"Red", "Green", "Blue"};
AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setTitle("Pick a color");
builder.setItems(items, new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int item) {
Toast.makeText(getApplicationContext(), items[item], Toast.LENGTH_SHORT).show();
}
});
AlertDialog alert = builder.create();
復制代碼首先,使用setTitle(CharSequence)方法設置標題,然后使用setItem()方法添加可選列表,這個列表會接收一個item數組來顯示,DialogInterFace.OnClickListener類會定義他們的點擊事件。 通過setMultiChoiceItems()方法或 setSingleChoiceItems()方法來分別建立一個多選按鈕列表或者單選列表,如果再onCreateDialog()方法中建立了其中一種列表,android會為你管理這個list。當activity處于 活動狀態(tài)時,dialog會記住當才選中項,如果退出了程序,選擇結果便會丟失。注意:當用戶離開或者暫停activity時,如果你想保存選擇狀態(tài),你必須在整個activity的生命周期中保存這個設置。永久的保存所選項,甚至當前進程完全被關閉,你需要使用數據存儲方式來保存。建立一個如上圖所示的列表dialog,代碼和上面的例子相同,只需要把setItems()方法改為setSingleChoiceItems()方法即可。 - final CharSequence[] items = {"Red", "Green", "Blue"};
AlertDialog.Builder builder = new AlertDialog.Builder(this);
builder.setTitle("Pick a color");
builder.setSingleChoiceItems(items, -1, new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int item) {
Toast.makeText(getApplicationContext(), items[item], Toast.LENGTH_SHORT).show();
}
});
AlertDialog alert = builder.create();
復制代碼setSingleChoiseItems()方法的第二個參數是checkedItem的id值,從0開始對應著位置,如果返回”-1“表明沒有選中任何項。
ProgressDialog時AlertDialog的子類,它會顯示一個表示進度的圓形動畫,來表示一個進度或者任務正在運行,也可以時一個進度條,能清晰的表示出進度。他也能添加按鈕,比如取消一個下載進程。調用ProgressDialog.show()方法可以顯示進程對話框,例如,上圖的對話框可通過如下代碼生成: - ProgressDialog dialog = ProgressDialog.show(MyActivity.this, "",
"Loading. Please wait...", true);
復制代碼第一個參數是程序的Context引用,四二個為標題,第三個為顯示的信息,最后一個為類型,(當創(chuàng)建進度條時才會用到,下節(jié)討論)。默認的進度條為圓形的樣式,如果你想生成一個通過具體數值來顯示任務的加載情況的進度條,下一節(jié)會討論。進度條的顯示顯示一個進度條要經過以下幾個步驟:1-使用ProgressDialog(Context)方法初始化2-使用setProgressStyle(int)方法設置類型。3-調用show()方法顯示,或者在onCreateDialog(int)方法里返回一個ProgressDialog。4-你可以調用setProgress(int)方法,根據整體的任務完成度來設置一個具體進度值,或者使用incrementPressBy(int)來設置一個增長值。例如: - ProgressDialog progressDialog;
progressDialog = new ProgressDialog(mContext);
progressDialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
progressDialog.setMessage("Loading...");
progressDialog.setCancelable(false);
復制代碼設置代碼非常簡單,大部分代碼是在dialog參與進程并且更新的功能里。你會發(fā)現(xiàn),另起一個線程來做這個工作是很有必要的,要把消息傳遞給activity的UI線程里需要用到 Handler 消息機制。如果你并不熟悉使用額外的線程,那么看這個例子:這個例子使用了第二個線程來跟蹤任務的進度(實際上只是在數值上加到100),線程通過 Handler 發(fā)了一個Message 給主activity,然后主activity更新ProgressDialog。 - package com.example.progressdialog;
import android.app.Activity;
import android.app.Dialog;
import android.app.ProgressDialog;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
public class NotificationTest extends Activity {
static final int PROGRESS_DIALOG = 0;
Button button;
ProgressThread progressThread;
ProgressDialog progressDialog;
/** Called when the activity is first created. */
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
// Setup the button that starts the progress dialog
button = (Button) findViewById(R.id.progressDialog);
button.setOnClickListener(new OnClickListener(){
public void onClick(View v) {
showDialog(PROGRESS_DIALOG);
}
});
}
protected Dialog onCreateDialog(int id) {
switch(id) {
case PROGRESS_DIALOG:
progressDialog = new ProgressDialog(NotificationTest.this);
progressDialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
progressDialog.setMessage("Loading...");
progressThread = new ProgressThread(handler);
progressThread.start();
return progressDialog;
default:
return null;
}
}
// Define the Handler that receives messages from the thread and update the progress
final Handler handler = new Handler() {
public void handleMessage(Message msg) {
int total = msg.getData().getInt("total");
progressDialog.setProgress(total);
if (total >= 100){
dismissDialog(PROGRESS_DIALOG);
progressThread.setState(ProgressThread.STATE_DONE);
}
}
};
/** Nested class that performs progress calculations (counting) */
private class ProgressThread extends Thread {
Handler mHandler;
final static int STATE_DONE = 0;
final static int STATE_RUNNING = 1;
int mState;
int total;
ProgressThread(Handler h) {
mHandler = h;
}
public void run() {
mState = STATE_RUNNING;
total = 0;
while (mState == STATE_RUNNING) {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
Log.e("ERROR", "Thread Interrupted");
}
Message msg = mHandler.obtainMessage();
Bundle b = new Bundle();
b.putInt("total", total);
msg.setData(b);
mHandler.sendMessage(msg);
total++;
}
}
/* sets the current state for the thread,
* used to stop the thread */
public void setState(int state) {
mState = state;
}
}
}
復制代碼 如果你想自定義dialog的布局,你可以自己創(chuàng)建一個dialog布局。定義好之后,傳遞根View對象或者資源ID到setContextView(View)方法。例如,如上圖的dialog:1-建立一個xml布局文件custom_dialog.xml;
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/layout_root"
android:orientation="horizontal"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:padding="10dp"
>
<ImageView android:id="@+id/image"
android:layout_width="wrap_content"
android:layout_height="fill_parent"
android:layout_marginRight="10dp"
/>
<TextView android:id="@+id/text"
android:layout_width="wrap_content"
android:layout_height="fill_parent"
android:textColor="#FFF"
/>
</LinearLayout>
復制代碼 這個xml在LinearLayout里定義了一個ImageView和TextView。 2-設置上面的布局為dialog的context view ,并且定義ImageView和TextView兩個元素。
- Context mContext = getApplicationContext();
Dialog dialog = new Dialog(mContext);
dialog.setContentView(R.layout.custom_dialog);
dialog.setTitle("Custom Dialog");
TextView text = (TextView) dialog.findViewById(R.id.text);
text.setText("Hello, this is a custom dialog!");
ImageView image = (ImageView) dialog.findViewById(R.id.image);
image.setImageResource(R.drawable.android);
復制代碼實例化dialog后,使用setContextView(int)方法設置自定義的布局?,F(xiàn)在dialog便有了一個自定義的布局,你可以使用findViewById(int)方法來獲得或者修改布局。3-完成了,現(xiàn)在你可以顯示自定義的dialog了。一個dialog必須有一個title,如果你沒有調用setTitile()方法,那么會標題處會顯示空,但dialog仍然可見,如果你不想顯示標題,只有寫一個自己的dialog類了。然而,因為一個AlertDialog使用AlertDialog.builder類創(chuàng)建起來非常簡單,你不必使用setContextView(int)方法。但必須使用setView(view)方法代替。這個方法會接受一個view參數,你需要從xml中得到根view元素。得到xml布局,通過LayoutInflater類的getLayoutflater()方法(或者getSystemService()方法),然后調用inflate(int,ViewGroup)方法,第一個參數是xml文件id,第二個參數是根view的id,在這點上,你可以使用inflated 布局來獲得xml中的view對象并且定義ImageView和TextView對象,然后實例化AlertDialog.Builder類并且使用setView(View)方法來設置布局。這有一個自定義dialog布局文件的例子: - AlertDialog.Builder builder;
AlertDialog alertDialog;
Context mContext = getApplicationContext();
LayoutInflater inflater = (LayoutInflater) mContext.getSystemService(LAYOUT_INFLATER_SERVICE);
View layout = inflater.inflate(R.layout.custom_dialog,
(ViewGroup) findViewById(R.id.layout_root));
TextView text = (TextView) layout.findViewById(R.id.text);
text.setText("Hello, this is a custom dialog!");
ImageView image = (ImageView) layout.findViewById(R.id.image);
image.setImageResource(R.drawable.android);
builder = new AlertDialog.Builder(mContext);
builder.setView(layout);
alertDialog = builder.create();
復制代碼使用自定義布局這種方式來生成dialog,可以讓你使用更高級的特性,比如管理按鈕、列表、標題、圖標等。
|