天天看點

android推薦使用dialogFrament而不是alertDialog

dialogfragment在android 3.0時被引入。是一種特殊的fragment,用于在activity的内容之上展示一個模态的對話框。典型的用于:展示警告框,輸入框,确認框等等。

在dialogfragment産生之前,我們建立對話框:一般采用alertdialog和dialog。注:官方不推薦直接使用dialog建立對話框。

使用dialogfragment來管理對話框,當旋轉螢幕和按下後退鍵時可以更好的管理其聲明周期,它和fragment有着基本一緻的聲明周期。且dialogfragment也允許開發者把dialog作為内嵌的元件進行重用,類似fragment(可以在大螢幕和小螢幕顯示出不同的效果)。上面會通過例子展示這些好處~

使用dialogfragment至少需要實作oncreateview或者oncreatedialog方法。oncreateview即使用定義的xml布局檔案展示dialog。oncreatedialog即利用alertdialog或者dialog建立出dialog。

a)布局檔案,我們建立一個設定名稱的布局檔案:

<?xml version="1.0" encoding="utf-8"?>  

<relativelayout xmlns:android="http://schemas.android.com/apk/res/android"  

    android:layout_width="wrap_content"  

    android:layout_height="wrap_content" >  

    <textview  

        android:id="@+id/id_label_your_name"  

        android:layout_width="wrap_content"  

        android:layout_height="32dp"  

        android:gravity="center_vertical"  

        android:text="your name:" />  

    <edittext  

        android:id="@+id/id_txt_your_name"  

        android:layout_width="match_parent"  

        android:layout_height="wrap_content"  

        android:layout_torightof="@id/id_label_your_name"  

        android:imeoptions="actiondone"  

        android:inputtype="text" />  

    <button  

        android:id="@+id/id_sure_edit_name"  

        android:layout_alignparentright="true"  

        android:layout_below="@id/id_txt_your_name"  

        android:text="ok" />  

</relativelayout>  

b)繼承dialogfragment,重寫oncreateview方法

package com.example.zhy_dialogfragment;  

import android.app.dialogfragment;  

import android.os.bundle;  

import android.view.layoutinflater;  

import android.view.view;  

import android.view.viewgroup;  

public class editnamedialogfragment extends dialogfragment  

{  

    @override  

    public view oncreateview(layoutinflater inflater, viewgroup container,  

            bundle savedinstancestate)  

    {  

        view view = inflater.inflate(r.layout.fragment_edit_name, container);  

        return view;  

    }  

}  

c)測試運作:

main方法中調用:

public void showeditdialog(view view)  

        editnamedialogfragment editnamedialog = new editnamedialogfragment();  

        editnamedialog.show(getfragmentmanager(), "editnamedialog");  

效果圖:

android推薦使用dialogFrament而不是alertDialog

可以看到,對話框成功建立并顯示出來,不過預設對話框有個讨厭的标題,我們怎麼去掉呢:可以在oncreateview中調用getdialog().requestwindowfeature(window.feature_no_title);即可去掉。即:

        getdialog().requestwindowfeature(window.feature_no_title);  

android推薦使用dialogFrament而不是alertDialog

很完美的去掉了讨厭的标題。

在oncreatedialog中一般可以使用alertdialog或者dialog建立對話框,不過既然google不推薦直接使用dialog,我們就使用alertdialog來建立一個登入的對話框。

a)布局檔案

<linearlayout xmlns:android="http://schemas.android.com/apk/res/android"  

    android:layout_height="wrap_content"  

    android:orientation="vertical" >  

    <imageview  

        android:layout_height="64dp"  

        android:background="#ffffbb33"  

        android:contentdescription="@string/app_name"  

        android:scaletype="center"  

        android:src="@drawable/title" />  

        android:id="@+id/id_txt_username"  

        android:layout_marginbottom="4dp"  

        android:layout_marginleft="4dp"  

        android:layout_marginright="4dp"  

        android:layout_margintop="16dp"  

        android:hint="input username"  

        android:inputtype="textemailaddress" />  

        android:id="@+id/id_txt_password"  

        android:layout_marginbottom="16dp"  

        android:layout_margintop="4dp"  

        android:fontfamily="sans-serif"  

        android:hint="input password"  

        android:inputtype="textpassword" />  

</linearlayout>  

b)繼承dialogfragment重寫oncreatedialog方法

import android.app.alertdialog;  

import android.app.dialog;  

import android.content.dialoginterface;  

import android.widget.edittext;  

public class logindialogfragment extends dialogfragment  

    public dialog oncreatedialog(bundle savedinstancestate)  

        alertdialog.builder builder = new alertdialog.builder(getactivity());  

        // get the layout inflater  

        layoutinflater inflater = getactivity().getlayoutinflater();  

        view view = inflater.inflate(r.layout.fragment_login_dialog, null);  

        // inflate and set the layout for the dialog  

        // pass null as the parent view because its going in the dialog layout  

        builder.setview(view)  

                // add action buttons  

                .setpositivebutton("sign in",  

                        new dialoginterface.onclicklistener()  

                        {  

                            @override  

                            public void onclick(dialoginterface dialog, int id)  

                            {  

                            }  

                        }).setnegativebutton("cancel", null);  

        return builder.create();  

c)調用

public void showlogindialog(view view)  

        logindialogfragment dialog = new logindialogfragment();  

        dialog.show(getfragmentmanager(), "logindialog");  

android推薦使用dialogFrament而不是alertDialog

可以看到通過重寫oncreatedialog同樣可以實作建立對話框,效果還是很nice的。

從dialog傳遞資料給activity,可以使用“fragment interface pattern”的方式,下面通過一個改造上面的登入框來展示這種模式。

改動比較小,直接貼代碼了:

    private edittext musername;  

    private edittext mpassword;  

    public interface logininputlistener  

        void onlogininputcomplete(string username, string password);  

        musername = (edittext) view.findviewbyid(r.id.id_txt_username);  

        mpassword = (edittext) view.findviewbyid(r.id.id_txt_password);  

                                logininputlistener listener = (logininputlistener) getactivity();  

                                listener.onlogininputcomplete(musername  

                                        .gettext().tostring(), mpassword  

                                        .gettext().tostring());  

拿到username和password的引用,在點選登入的時候,把activity強轉為我們自定義的接口:logininputlistener,然後将使用者輸入的資料傳回。

mainactivity中需要實作我們的接口logininputlistener,實作我們的方法,就可以實作當使用者點選登陸時,獲得我們的帳号密碼了:

c)  mainactivity  

import com.example.zhy_dialogfragment.logindialogfragment.logininputlistener;  

import android.app.activity;  

import android.widget.toast;  

public class mainactivity extends activity implements logininputlistener  

    protected void oncreate(bundle savedinstancestate)  

        super.oncreate(savedinstancestate);  

        setcontentview(r.layout.activity_main);  

    public void showlogindialog(view view)  

    public void onlogininputcomplete(string username, string password)  

        toast.maketext(this, "帳号:" + username + ",  密碼 :" + password,  

                toast.length_short).show();  

效果:

android推薦使用dialogFrament而不是alertDialog

我們希望,一個對話框在大螢幕上以對話框的形式展示,而小螢幕上則直接嵌入目前的actvity中。這種效果的對話框,隻能通過重寫oncreateview實作。下面我們利用上面的editnamedialogfragment來顯示。

editnamedialogfragment我們已經編寫好了,直接在mainactivity中寫調用

public void showdialogindifferentscreen(view view)  

        fragmentmanager fragmentmanager = getfragmentmanager();  

        editnamedialogfragment newfragment = new editnamedialogfragment();  

        boolean mislargelayout = getresources().getboolean(r.bool.large_layout) ;  

        log.e("tag", mislargelayout+"");  

        if (mislargelayout )  

        {  

            // the device is using a large layout, so show the fragment as a  

            // dialog  

            newfragment.show(fragmentmanager, "dialog");  

        } else  

            // the device is smaller, so show the fragment fullscreen  

            fragmenttransaction transaction = fragmentmanager  

                    .begintransaction();  

            // for a little polish, specify a transition animation  

            transaction  

                    .settransition(fragmenttransaction.transit_fragment_open);  

            // to make it fullscreen, use the 'content' root view as the  

            // container  

            // for the fragment, which is always the root view for the activity  

            transaction.replace(r.id.id_ly, newfragment)  

                    .commit();  

        }  

可以看到,我們通過讀取r.bool.large_layout,然後根據得到的布爾值,如果是大螢幕則直接以對話框顯示,如果是小螢幕則嵌入我們的activity布局中

這個r.bool.large_layout是我們定義的資源檔案:

在預設的values下建立一個bools.xml

<resources>  

    <bool name="large_layout">false</bool>  

</resources>  

然後在res下建立一個values-large,在values-large下再建立一個bools.xml

    <bool name="large_layout">true</bool>  

最後測試:

android推薦使用dialogFrament而不是alertDialog
android推薦使用dialogFrament而不是alertDialog

左邊為模拟器,右邊為我的手機~~~~~

當使用者輸入帳号密碼時,忽然旋轉了一下螢幕,帳号密碼不見了~~~是不是會抓狂

傳統的new alertdialog在螢幕旋轉時,第一不會儲存使用者輸入的值,第二還會報異常,因為activity銷毀前不允許對話框未關閉。而通過dialogfragment實作的對話框則可以完全不必考慮旋轉的問題。

我們直接把上面登入使用alertdialog建立的登入框,拷貝到mainactivity中直接調用:

public void showlogindialogwithoutfragment(view view)  

        alertdialog.builder builder = new alertdialog.builder(this);  

        layoutinflater inflater = this.getlayoutinflater();  

        builder.setview(inflater.inflate(r.layout.fragment_login_dialog, null))  

                                // sign in the user ...  

                        }).setnegativebutton("cancel", null).show();  

下面我分别點選兩種方式建立的登入框,看效果圖:

android推薦使用dialogFrament而不是alertDialog

可以看到,傳統的dialog旋轉螢幕時就消失了,且背景log會報異常~~~使用dialogfragment則不受影響。

好了,關于dialogfragment的介紹結束~~~~

有任何疑問請留言

<a target="_blank" href="http://download.csdn.net/detail/lmj623565791/7638357">源碼點選下載下傳</a>

繼續閱讀