Android: поддержка DialogFragment с пользовательским макетом не показана

В моем приложении был подкласс DialogFragment (android.support.v4.app.DialogFragment), который долгое время работал нормально. Сам диалог был построен в обратном вызове onCreateDialog().

Однако теперь я хочу перейти к новому макету диалогов и решил создать полностью настраиваемый диалог. Насколько мне известно, для этого требуется удалить переопределение onCreateDialog() и раздуть иерархию View в обратном вызове onCreateView(). Я так и сделал.

В результате получается очень странное поведение - когда нужно отобразить диалоговое окно, экран затемняется, но макет диалогового окна не отображается (поведение кнопки «назад» также соответствует функциональности диалогового окна):

введите описание изображения здесь

Я попытался вернуться к старой реализации (которая использует обратный вызов onCreateDialog()) - она ​​все еще работает.

У меня вопрос: что я делаю не так?

Код диалога:

/**
 * A dialog that can show title and message and has two buttons. User's actions performed
 * in this dialog will be posted to {@link EventBus} as {@link PromptDialogDismissedEvent}.
 */
public class PromptDialog extends BaseDialog {


    /* package */ static final String ARG_TITLE = "ARG_TITLE";
    /* package */ static final String ARG_MESSAGE = "ARG_MESSAGE";
    /* package */ static final String ARG_POSITIVE_BUTTON_CAPTION = "ARG_POSITIVE_BUTTON_CAPTION";
    /* package */ static final String ARG_NEGATIVE_BUTTON_CAPTION = "ARG_NEGATIVE_BUTTON_CAPTION";

    @Inject EventBus mEventBus;

    private TextView mTxtTitle;
    private TextView mTxtMessage;
    private Button mBtnPositive;
    private Button mBtnNegative;

    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        getControllerComponent().inject(this);
    }

    // THIS CODE MAKES THE SCREEN DIM, BUT THE ACTUAL LAYOUT IS NOT SHOWN

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

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

        initSubViews(rootView);

        populateSubViews();

        return rootView;

    }

    private void initSubViews(View rootView) {
        mTxtTitle = (TextView) rootView.findViewById(R.id.txt_dialog_title);
        mTxtMessage = (TextView) rootView.findViewById(R.id.txt_dialog_message);
        mBtnPositive = (Button) rootView.findViewById(R.id.btn_dialog_positive);
        mBtnNegative = (Button) rootView.findViewById(R.id.btn_dialog_negative);

        mBtnPositive.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                mEventBus.post(new PromptDialogDismissedEvent(getDialogTag(), PromptDialogDismissedEvent.BUTTON_POSITIVE));
            }
        });

        mBtnNegative.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                mEventBus.post(new PromptDialogDismissedEvent(getDialogTag(), PromptDialogDismissedEvent.BUTTON_NEGATIVE));
            }
        });
    }

    private void populateSubViews() {
        String title = getArguments().getString(ARG_TITLE);
        String message = getArguments().getString(ARG_MESSAGE);
        String positiveButtonCaption = getArguments().getString(ARG_POSITIVE_BUTTON_CAPTION);
        String negativeButtonCaption = getArguments().getString(ARG_NEGATIVE_BUTTON_CAPTION);

        mTxtTitle.setText(TextUtils.isEmpty(title) ? "" : title);
        mTxtMessage.setText(TextUtils.isEmpty(message) ? "" : message);
        mBtnPositive.setText(positiveButtonCaption);
        mBtnNegative.setText(negativeButtonCaption);
    }


    // THE BELOW CODE WORKS FINE (the dialog is shown and responsive)

//    @Override
//    public @NonNull Dialog onCreateDialog(Bundle savedInstanceState) {
//        String title = getArguments().getString(ARG_TITLE);
//        String message = getArguments().getString(ARG_MESSAGE);
//        String positiveButtonCaption = getArguments().getString(ARG_POSITIVE_BUTTON_CAPTION);
//        String negativeButtonCaption = getArguments().getString(ARG_NEGATIVE_BUTTON_CAPTION);
//
//        Dialog dialog = new AlertDialog.Builder(getActivity())
//                .setTitle(TextUtils.isEmpty(title) ? "" : title)
//                .setMessage(TextUtils.isEmpty(message) ? "" : message)
//                .setPositiveButton(TextUtils.isEmpty(positiveButtonCaption) ? "" : positiveButtonCaption,
//                        new DialogInterface.OnClickListener() {
//                            @Override
//                            public void onClick(DialogInterface dialog, int which) {
//                                mEventBus.post(new PromptDialogDismissedEvent(getDialogTag(), PromptDialogDismissedEvent.BUTTON_POSITIVE));
//                            }
//                        })
//                .setNegativeButton(TextUtils.isEmpty(negativeButtonCaption) ? "" : negativeButtonCaption,
//                        new DialogInterface.OnClickListener() {
//                            @Override
//                            public void onClick(DialogInterface dialog, int which) {
//                                mEventBus.post(new PromptDialogDismissedEvent(getDialogTag(), PromptDialogDismissedEvent.BUTTON_NEGATIVE));
//                            }
//                        })
//                .create();
//
//        return dialog;
//    }

}

Макет пользовательского диалога:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              xmlns:tools="http://schemas.android.com/tools"
              android:orientation="vertical"
              android:layout_width="match_parent"
              android:layout_height="match_parent">

    <TextView
        android:id="@+id/txt_dialog_title"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="15dp"
        android:layout_gravity="center_horizontal"
        style="@style/AppTheme.TextView.DialogTitle"
        tools:text="Dialog title"/>

    <View
        android:layout_width="match_parent"
        android:layout_height="1dp"
        android:layout_marginTop="10dp"
        android:background="@color/gray"/>

    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:padding="@dimen/fragment_default_padding">

        <TextView
            android:id="@+id/txt_dialog_message"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginTop="10dp"
            style="@style/AppTheme.TextView.DialogMessage"
            tools:text="Dialog message very very very very very very very very very very very very long"/>

        <Button
            android:id="@+id/btn_dialog_positive"
            android:layout_width="wrap_content"
            android:layout_height="@dimen/button_height"
            android:layout_marginTop="15dp"
            android:layout_below="@id/txt_dialog_message"
            android:layout_alignParentEnd="true"
            android:layout_alignParentRight="true"
            style="@style/AppTheme.GreenButton"
            tools:text="Positive"/>

        <Button
            android:id="@+id/btn_dialog_negative"
            android:layout_width="wrap_content"
            android:layout_height="@dimen/button_height"
            android:layout_marginTop="15dp"
            android:layout_below="@id/txt_dialog_message"
            android:layout_toStartOf="@+id/btn_dialog_positive"
            android:layout_toLeftOf="@id/btn_dialog_positive"
            android:layout_marginEnd="15dp"
            android:layout_marginRight="15dp"
            style="@style/AppTheme.GrayButton"
            tools:text="Negative"/>

    </RelativeLayout>

</LinearLayout>

person Vasiliy    schedule 18.01.2017    source источник
comment
stackoverflow.com/a/54629068/7074112   -  person Peter    schedule 04.08.2020


Ответы (4)


Я закончил с этим обходным путем, но я все еще хочу понять WTF:

    // This is a workaround for the strange behavior of onCreateView (which doesn't show dialog's layout)
    @NonNull
    @Override
    public Dialog onCreateDialog(Bundle savedInstanceState) {
        AlertDialog.Builder dialogBuilder = new AlertDialog.Builder(getContext());
        LayoutInflater inflater = LayoutInflater.from(getContext());
        View dialogView = inflater.inflate(R.layout.dialog_info_prompt, null);
        dialogBuilder.setView(dialogView);

        initSubViews(dialogView);

        populateSubViews();

        setCancelable(false);

        return dialogBuilder.create();
    }

    private void initSubViews(View rootView) {
        mTxtTitle = (TextView) rootView.findViewById(R.id.txt_dialog_title);
        mTxtMessage = (TextView) rootView.findViewById(R.id.txt_dialog_message);
        mBtnPositive = (Button) rootView.findViewById(R.id.btn_dialog_positive);
        mBtnNegative = (Button) rootView.findViewById(R.id.btn_dialog_negative);

        mBtnPositive.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                dismiss();
                mEventBus.post(new PromptDialogDismissedEvent(getDialogTag(), PromptDialogDismissedEvent.BUTTON_POSITIVE));
            }
        });

        mBtnNegative.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                dismiss();
                mEventBus.post(new PromptDialogDismissedEvent(getDialogTag(), PromptDialogDismissedEvent.BUTTON_NEGATIVE));
            }
        });
    }

    private void populateSubViews() {
        String title = getArguments().getString(ARG_TITLE);
        String message = getArguments().getString(ARG_MESSAGE);
        String positiveButtonCaption = getArguments().getString(ARG_POSITIVE_BUTTON_CAPTION);
        String negativeButtonCaption = getArguments().getString(ARG_NEGATIVE_BUTTON_CAPTION);

        mTxtTitle.setText(TextUtils.isEmpty(title) ? "" : title);
        mTxtMessage.setText(TextUtils.isEmpty(message) ? "" : message);
        mBtnPositive.setText(positiveButtonCaption);
        mBtnNegative.setText(negativeButtonCaption);
    }
person Vasiliy    schedule 23.01.2017
comment
Это очень раздражающая проблема - похоже, это действительно ошибка. - person cfl; 22.02.2017

Может быть, немного поздно, но я наткнулся на этот пост, потому что у меня была такая же проблема в моем приложении. Я до сих пор не знаю, почему это происходит, но я следовал этому Статья среднего размера о том, как настроить DialogFragment, и кажется, что изменение размера окна диалога на onStart может решить эту проблему.

styles.xml

<style name="FullScreenDialogStyle" parent="Theme.AppCompat.Dialog">
    <item name="android:windowNoTitle">true</item>
    <item name="colorPrimaryDark">@color/primary_dark</item>
    <item name="colorPrimary">@color/primary</item>

    <!-- Set this to true if you want Full Screen without status bar -->
    <item name="android:windowFullscreen">false</item>

    <item name="android:windowIsFloating">true</item>

    <!-- This is important! Don't forget to set window background -->
    <item name="android:windowBackground">@color/background_dialog</item>
</style>

Диалоговый фрагмент

@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setStyle(DialogFragment.STYLE_NORMAL, R.style.FullScreenDialogStyle);
}

@Override
public void onStart() {
    super.onStart();
    Dialog dialog = getDialog();
    if (dialog == null || dialog.getWindow() == null)
        return;

    int width = 776; //Width of your dialog
    int height = 512; //Height of your dialog
    dialog.getWindow().setLayout(width, height);
}
person Verhelst    schedule 10.04.2019

я добавляю высоту корневого макета wrap_content, это сработало для меня.

person Mostafa Amer    schedule 13.05.2020

С опозданием на 5 лет, но я пришел сюда с похожей проблемой. Решение аналогично тому, которое было принято ОП.

Проблема для меня заключалась в том, что в моем onCreate() я создавал AlertDialog и раздувал макет, но не хватало одного шага.

    public Dialog onCreateDialog(Bundle savedInstanceState){
     AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());

     View alertView = LayoutInflater.from(getContext()).inflate(R.layout.formulario_identificar, null);
    //just loading the rest of the buttons and doing stuff
    Button button1 = alertView.findViewById(R.id.xeneroId);
     .....
     return builder.create();
    }

Проблема в том, что создается объект AlertDialog.Builder, а макет раздувается... но макет не привязан к билдеру, который я возвращаю в builder.create() - другими словами, я возвращение строителя без макета, так сказать.

Решение было простым для меня:

 public Dialog onCreateDialog(Bundle savedInstanceState){
     AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());

     View alertView = LayoutInflater.from(getContext()).inflate(R.layout.formulario_identificar, null);
    
    *****builder.setView(alertView);******

    //just loading the rest of the buttons and doing stuff
    Button button1 = alertView.findViewById(R.id.xeneroId);
     .....
     return builder.create();
    }

Я прикрепил View (раздутый макет) к моему AlertDialog, так что проблема была решена.

person Manuel    schedule 09.06.2021