Показать диалог из фрагмента? - PullRequest
114 голосов
/ 22 марта 2011

У меня есть несколько фрагментов, которые должны показывать обычный диалог. В этих диалоговых окнах пользователь может выбрать ответ «да / нет», и тогда фрагмент должен вести себя соответственно.

Теперь класс Fragment не имеет метода onCreateDialog() для переопределения, поэтому я предполагаю, что мне нужно реализовать диалоги снаружи, в содержащем Activity. Это нормально, но тогда Activity нужно как-то сообщить выбранный ответ фрагменту. Конечно, я мог бы использовать здесь шаблон обратного вызова, так что фрагмент регистрируется в Activity с классом слушателя, и Activity сообщит об ответе через это или что-то в этом роде.

Но это, кажется, довольно большой беспорядок для такой простой задачи, как отображение «простого» диалога «да-нет» во фрагменте. Кроме того, мой Fragment будет менее автономным.

Есть ли какой-нибудь более чистый способ сделать это?

Edit:

Ответ на этот вопрос на самом деле не объясняет подробно, как следует использовать DialogFragments для отображения диалоговых окон из фрагментов. Итак, AFAIK, путь должен быть:

  1. Показать фрагмент.
  2. При необходимости создайте экземпляр DialogFragment.
  3. Установите исходный фрагмент в качестве цели этого DialogFragment с помощью .setTargetFragment().
  4. Показать фрагмент DialogFragment с .show () из исходного фрагмента.
  5. Когда пользователь выбирает какую-либо опцию в этом DialogFragment, уведомляет исходный фрагмент об этом выборе (например, пользователь нажал «да»), вы можете получить ссылку на исходный фрагмент с помощью .getTarget ().
  6. Отклонить фрагмент диалога.

Ответы [ 7 ]

37 голосов
/ 23 марта 2011

Вместо этого следует использовать DialogFragment .

29 голосов
/ 28 марта 2011

Я должен с осторожностью усомниться в ранее принятом ответе, что использование DialogFragment является наилучшим вариантом.Предполагаемая (основная) цель DialogFragment, по-видимому, состоит в том, чтобы отображать фрагменты, которые сами являются диалогами, а не отображать фрагменты, которые имеют диалогов для отображения.

Я полагаючто использование активности фрагмента в качестве посредника между диалогом и фрагментом является предпочтительным вариантом.

24 голосов
/ 23 апреля 2013

Вот полный пример фрагмента «Да / Нет»:

Класс:

public class SomeDialog extends DialogFragment {

    @Override
    public Dialog onCreateDialog(Bundle savedInstanceState) {
        return new AlertDialog.Builder(getActivity())
            .setTitle("Title")
            .setMessage("Sure you wanna do this!")
            .setNegativeButton(android.R.string.no, new OnClickListener() {
                @Override
                public void onClick(DialogInterface dialog, int which) {
                    // do nothing (will close dialog)
                }
            })
            .setPositiveButton(android.R.string.yes,  new OnClickListener() {
                @Override
                public void onClick(DialogInterface dialog, int which) {
                    // do something
                }
            })
            .create();
    }
}

Для запуска диалога:

        FragmentTransaction ft = getSupportFragmentManager().beginTransaction();
        // Create and show the dialog.
        SomeDialog newFragment = new SomeDialog ();
        newFragment.show(ft, "dialog");

Вы также можете позволить классу реализовать onClickListener и использовать его вместо встроенных слушателей.

Обратный вызов к действию

Если вы хотите реализовать обратный вызов, вот как это делается В вашей деятельности:

YourActivity extends Activity implements OnFragmentClickListener

и

@Override
public void onFragmentClick(int action, Object object) {
    switch(action) {
        case SOME_ACTION:
        //Do your action here
        break;
    }
}

Класс обратного вызова:

public interface OnFragmentClickListener {
    public void onFragmentClick(int action, Object object);
}

Затем для выполнения обратного вызова из фрагмента необходимо убедиться, что слушатель подключен следующим образом:

@Override
public void onAttach(Activity activity) {
    super.onAttach(activity);
    try {
        mListener = (OnFragmentClickListener) activity;
    } catch (ClassCastException e) {
        throw new ClassCastException(activity.toString() + " must implement listeners!");
    }
}

И обратный вызов выполняется так:

mListener.onFragmentClick(SOME_ACTION, null); // null or some important object as second parameter.
12 голосов
/ 31 июля 2014

Для меня это было следующее -

MyFragment:

public class MyFragment extends Fragment implements MyDialog.Callback
{
    ShowDialog activity_showDialog;

    @Override
    public void onAttach(Activity activity)
    {
        super.onAttach(activity);
        try
        {
            activity_showDialog = (ShowDialog)activity;
        }
        catch(ClassCastException e)
        {
            Log.e(this.getClass().getSimpleName(), "ShowDialog interface needs to be     implemented by Activity.", e);
            throw e;
        }
    }

    @Override
    public void onClick(View view) 
    {
        ...
        MyDialog dialog = new MyDialog();
        dialog.setTargetFragment(this, 1); //request code
        activity_showDialog.showDialog(dialog);
        ...
    }

    @Override
    public void accept()
    {
        //accept
    }

    @Override
    public void decline()
    {
        //decline
    }

    @Override
    public void cancel()
    {
        //cancel
    }

}

MyDialog:

public class MyDialog extends DialogFragment implements View.OnClickListener
{
    private EditText mEditText;
    private Button acceptButton;
    private Button rejectButton;
    private Button cancelButton;

    public static interface Callback
    {
        public void accept();
        public void decline();
        public void cancel();
    }

    public MyDialog()
    {
        // Empty constructor required for DialogFragment
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)
    {
        View view = inflater.inflate(R.layout.dialogfragment, container);
        acceptButton = (Button) view.findViewById(R.id.dialogfragment_acceptbtn);
        rejectButton = (Button) view.findViewById(R.id.dialogfragment_rejectbtn);
        cancelButton = (Button) view.findViewById(R.id.dialogfragment_cancelbtn);
        acceptButton.setOnClickListener(this);
        rejectButton.setOnClickListener(this);
        cancelButton.setOnClickListener(this);
        getDialog().setTitle(R.string.dialog_title);
        return view;
    }

    @Override
    public void onClick(View v)
    {
        Callback callback = null;
        try
        {
            callback = (Callback) getTargetFragment();
        }
        catch (ClassCastException e)
        {
            Log.e(this.getClass().getSimpleName(), "Callback of this class must be implemented by target fragment!", e);
            throw e;
        }

        if (callback != null)
        {
            if (v == acceptButton)
            {   
                callback.accept();
                this.dismiss();
            }
            else if (v == rejectButton)
            {
                callback.decline();
                this.dismiss();
            }
            else if (v == cancelButton)
            {
                callback.cancel();
                this.dismiss();
            }
        }
    }
}

Активность:

public class MyActivity extends ActionBarActivity implements ShowDialog
{
    ..

    @Override
    public void showDialog(DialogFragment dialogFragment)
    {
        FragmentManager fragmentManager = getSupportFragmentManager();
        dialogFragment.show(fragmentManager, "dialog");
    }
}

Макет DialogFragment:

<?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"
    android:layout_gravity="center"
    android:orientation="vertical" >

    <TextView
        android:id="@+id/dialogfragment_textview"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="20dp"
        android:layout_centerHorizontal="true"
        android:layout_marginBottom="10dp"
        android:text="@string/example"/>

    <Button
        android:id="@+id/dialogfragment_acceptbtn"
        android:layout_width="200dp"
        android:layout_height="wrap_content"
        android:layout_marginTop="20dp"
        android:layout_centerHorizontal="true"
        android:layout_below="@+id/dialogfragment_textview"
        android:text="@string/accept"
        />

    <Button
        android:id="@+id/dialogfragment_rejectbtn"
        android:layout_width="200dp"
        android:layout_height="wrap_content"
        android:layout_marginTop="10dp"
        android:layout_alignLeft="@+id/dialogfragment_acceptbtn"
        android:layout_below="@+id/dialogfragment_acceptbtn"
        android:text="@string/decline" />

     <Button
        android:id="@+id/dialogfragment_cancelbtn"
        android:layout_width="200dp"
        android:layout_height="wrap_content"
        android:layout_marginTop="10dp"
        android:layout_marginBottom="20dp"
        android:layout_alignLeft="@+id/dialogfragment_rejectbtn"
        android:layout_below="@+id/dialogfragment_rejectbtn"
        android:text="@string/cancel" />

     <Button
        android:id="@+id/dialogfragment_heightfixhiddenbtn"
        android:layout_width="200dp"
        android:layout_height="20dp"
        android:layout_marginTop="10dp"
        android:layout_marginBottom="20dp"
        android:layout_alignLeft="@+id/dialogfragment_cancelbtn"
        android:layout_below="@+id/dialogfragment_cancelbtn"
        android:background="@android:color/transparent"
        android:enabled="false"
        android:text=" " />
</RelativeLayout>

Как видно из названия dialogfragment_heightfixhiddenbtn, я просто не мог найти способ исправить то, что высота нижней кнопки была уменьшена вдвое, несмотря на то, что я сказал wrap_content, поэтому я добавилскрытая кнопка, чтобы вместо этого «разрезать» пополам.Извините за взлом.

3 голосов
/ 09 мая 2015
 public void showAlert(){


     AlertDialog.Builder alertDialog = new AlertDialog.Builder(getActivity());
     LayoutInflater inflater = getActivity().getLayoutInflater();
     View alertDialogView = inflater.inflate(R.layout.test_dialog, null);
     alertDialog.setView(alertDialogView);

     TextView textDialog = (TextView) alertDialogView.findViewById(R.id.text_testDialogMsg);
     textDialog.setText(questionMissing);

     alertDialog.setPositiveButton("Ok", new DialogInterface.OnClickListener() {
         public void onClick(DialogInterface dialog, int which) {
             dialog.cancel();
         }
     });
     alertDialog.show();

}

где .test_dialog имеет пользовательский xml

1 голос
/ 22 октября 2017

Я сам новичок и, честно говоря, не смог найти удовлетворительного ответа, который мог бы понять или реализовать.

Так вот внешняя ссылка, которая мне действительно помогла мне достичь того, чего я хотел.Это очень просто и легко следовать.

http://www.helloandroid.com/tutorials/how-display-custom-dialog-your-android-application

ЭТО ЧТО Я ПОПЫТАЛ ДОСТИГНУТЬ С КОДОМ:

У меня есть MainActivity, в котором находится фрагмент,Я хотел, чтобы в верхней части макета появилось диалоговое окно для запроса ввода данных пользователем, а затем соответствующей обработки. Смотрите скриншот

Вот как выглядит onCreateView моего фрагмента

@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {

    View rootView = inflater.inflate(R.layout.fragment_home_activity, container, false);

    Button addTransactionBtn = rootView.findViewById(R.id.addTransactionBtn);

    addTransactionBtn.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View view) {
            Dialog dialog = new Dialog(getActivity());
            dialog.setContentView(R.layout.dialog_trans);
            dialog.setTitle("Add an Expense");
            dialog.setCancelable(true);

            dialog.show();

        }
    });

Надеюсь, он вам поможет

Дайте мне знать, если естьспутанность сознания.:)

0 голосов
/ 06 сентября 2017
    public static void OpenDialog (Activity activity, DialogFragment fragment){

    final FragmentManager fm = ((FragmentActivity)activity).getSupportFragmentManager();

    fragment.show(fm, "tag");
}
...