How do I maintain the Immersive Mode in Dialogs? How do I maintain the Immersive Mode in Dialogs? android android

How do I maintain the Immersive Mode in Dialogs?


After a lot of research into the issue there is a hacky fix for this, which involved tearing apart the Dialog class to find. The navigation bar is shown when the dialog window is added to the Window Manager even if you set the UI visibility before adding it to the manager. In the Android Immersive example it's commented that:

// * Uses semi-transparent bars for the nav and status bars// * This UI flag will *not* be cleared when the user interacts with the UI.// When the user swipes, the bars will temporarily appear for a few seconds and then// disappear again.

I believe that's what we're seeing here (that a user-interaction is being triggered when a new, focusable, window view is added to the manager).

How can we work around this? Make the Dialog non-focusable when we create it (so we don't trigger a user-interaction) and then make it focusable after it's displayed.

//Here's the magic..//Set the dialog to not focusable (makes navigation ignore us adding the window)dialog.getWindow().setFlags(WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE, WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE);//Show the dialog!dialog.show();//Set the dialog to immersivedialog.getWindow().getDecorView().setSystemUiVisibility(context.getWindow().getDecorView().getSystemUiVisibility());//Clear the not focusable flag from the windowdialog.getWindow().clearFlags(WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE);

Clearly this is not ideal but it seems to be an Android bug, they should check if the Window has immersive set.

I've updated my working test code (forgive the hacky messiness) to Github. I've tested on the Nexus 5 emulator, it will probably blow up with anything less than KitKat but its for proof-of-concept only.


For your information, thanks to @Beaver6813's answer, I've been able to get this working using DialogFragment.

in the onCreateView method of my DialogFragment, I've just added the following :

    getDialog().getWindow().setFlags(WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE, WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE);    getDialog().getWindow().getDecorView().setSystemUiVisibility(getActivity().getWindow().getDecorView().getSystemUiVisibility());    getDialog().setOnShowListener(new DialogInterface.OnShowListener() {        @Override        public void onShow(DialogInterface dialog) {            //Clear the not focusable flag from the window            getDialog().getWindow().clearFlags(WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE);            //Update the WindowManager with the new attributes (no nicer way I know of to do this)..            WindowManager wm = (WindowManager) getActivity().getSystemService(Context.WINDOW_SERVICE);            wm.updateViewLayout(getDialog().getWindow().getDecorView(), getDialog().getWindow().getAttributes());        }    });


If you want to use onCreateDialog(), try this class. It works pretty well for me...

public class ImmersiveDialogFragment extends DialogFragment {    @Override    public Dialog onCreateDialog(Bundle savedInstanceState) {        AlertDialog alertDialog = new AlertDialog.Builder(getActivity())                .setTitle("Example Dialog")                .setMessage("Some text.")                .create();        // Temporarily set the dialogs window to not focusable to prevent the short        // popup of the navigation bar.        alertDialog.getWindow().addFlags(WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE);        return alertDialog;    }    public void showImmersive(Activity activity) {        // Show the dialog.        show(activity.getFragmentManager(), null);        // It is necessary to call executePendingTransactions() on the FragmentManager        // before hiding the navigation bar, because otherwise getWindow() would raise a        // NullPointerException since the window was not yet created.        getFragmentManager().executePendingTransactions();        // Hide the navigation bar. It is important to do this after show() was called.        // If we would do this in onCreateDialog(), we would get a requestFeature()        // error.        getDialog().getWindow().getDecorView().setSystemUiVisibility(            getActivity().getWindow().getDecorView().getSystemUiVisibility()        );        // Make the dialogs window focusable again.        getDialog().getWindow().clearFlags(            WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE        );    }}

To show the dialog, do the following in your activity...

new ImmersiveDialogFragment().showImmersive(this);