Android: windowSoftInputMode = "adjustResize" не имеет никакого значения?

У меня есть диалог faux, который использует этот макет:

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout    xmlns:android="http://schemas.android.com/apk/res/android"
                android:layout_height="match_parent"
                android:layout_width="match_parent"
                android:id="@+id/containerPageConatiner">

    <FrameLayout    android:id="@+id/dialogHolder"
                    android:layout_height="wrap_content"
                    android:layout_width="wrap_content"
                    android:padding="15dp"
                    android:layout_gravity="center"
                    android:background="@drawable/panel_picture_frame_bg_focus_blue"/>    
</FrameLayout> 

Я размещаю фрагмент внутри <FrameLayout> в зависимости от открывающегося диалога. Активность, управляющая диалоговым окном, выглядит следующим образом:

<activity
    android:label="@string/app_name"
    android:name=".activity.DialogActivity"
    android:theme="@style/CustomTheme.Screen.Transparent" 
    android:windowSoftInputMode="adjustResize">

К сожалению, когда вы нажимаете на текст редактирования внутри диалогового окна, изменение размера не происходит. windowSoftInputMode буквально не имеет значения, поскольку функциональность такая же, как режим панорамирования.

Документация гласит: "Это, конечно, работает только для приложений, которые имеют область с изменяемым размером, которая может быть уменьшена, чтобы сделать достаточно места", но не скажите, что это означает, "область с изменяемой площадью" и заставляет меня думать, что каким-то образом у меня нет области с изменяемой размерностью?

Если кто-нибудь знает, что он может мне помочь?

ИЗМЕНИТЬ

Окружать диалоговое окно, как это ничем не изменяет:

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

    <View
        android:layout_width="0dp"
        android:layout_height="0dp"
        android:layout_weight="1" />

    <FrameLayout    
        android:id="@+id/dialogHolder"
        android:layout_height="wrap_content"
    android:layout_width="wrap_content"
        android:padding="15dp"
        android:layout_gravity="center"
        android:background="@drawable/panel_picture_frame_bg_focus_blue"/>

    <View
        android:layout_width="0dp"
        android:layout_height="0dp"
        android:layout_weight="1" />
</LinearLayout>

EDIT2

Scrollview как родительский не помогает:

<?xml version="1.0" encoding="utf-8"?>
<ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/containerPageConatiner"
    android:layout_width="match_parent"
    android:layout_height="match_parent" >

    <FrameLayout
            android:id="@+id/dialogHolder"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="center"
            android:padding="15dp" />

</ScrollView>

Ответ 1

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

В моей иерархии тем у меня было это свойство:

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

который был погребен на уровне Theme.Black.NoTitleBar.FullScreen - предка моей пользовательской темы.

Документация предполагает, что это "Флаг, указывающий, должно ли это окно заполнять весь экран" . Это звучит неплохо, если у вас есть приложение, которое занимает весь экран... Кроме того, он все еще занимает весь экран без флага.

На самом деле, как только вы это поняли, абсолютно никаких изменений в приложении совсем нет... кроме adjustResize теперь работает отлично.

Ответ 2

В то время как у меня тоже была такая же проблема в библиотеке, которую я создал. (MaterialDrawer)

Насколько я вижу, все предоставленные ответы не решают основной проблемы, они просто указывают на удаление полноэкранного флага (android:windowFullscreen), что не является решением для многих из них.

Вышеупомянутая "проблема" появляется только в версиях Android, начиная с API уровня 19 (KITKAT), потому что они изменили поведение. Чтобы быть правильной, это не проблема, она работает по назначению. См. Комментарий сотрудника Android (Android-Issue).


Итак, я начал копаться в источнике Android и пришел к следующему решению, используя ViewTreeObserver.OnGlobalLayoutListener и реагируя, если Клавиатура будет показана/скрыта. Если отображается клавиатура, я добавляю прописку в нижнюю часть представления контейнера, которая затем будет эмулировать то же, что и adjustResize.


Решение

Чтобы упростить использование, я обернул все это в простой класс-помощник KeyboardUtil.

/**
 * Created by mikepenz on 14.03.15.
 * This class implements a hack to change the layout padding on bottom if the keyboard is shown
 * to allow long lists with editTextViews
 * Basic idea for this solution found here: http://stackoverflow.com/a/9108219/325479
 */
public class KeyboardUtil {
    private View decorView;
    private View contentView;

    public KeyboardUtil(Activity act, View contentView) {
        this.decorView = act.getWindow().getDecorView();
        this.contentView = contentView;

        //only required on newer android versions. it was working on API level 19 (Build.VERSION_CODES.KITKAT)
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            decorView.getViewTreeObserver().addOnGlobalLayoutListener(onGlobalLayoutListener);
        }
    }

    public void enable() {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            decorView.getViewTreeObserver().addOnGlobalLayoutListener(onGlobalLayoutListener);
        }
    }

    public void disable() {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            decorView.getViewTreeObserver().removeOnGlobalLayoutListener(onGlobalLayoutListener);
        }
    }


    //a small helper to allow showing the editText focus
    ViewTreeObserver.OnGlobalLayoutListener onGlobalLayoutListener = new ViewTreeObserver.OnGlobalLayoutListener() {
        @Override
        public void onGlobalLayout() {
            Rect r = new Rect();
            //r will be populated with the coordinates of your view that area still visible.
            decorView.getWindowVisibleDisplayFrame(r);

            //get screen height and calculate the difference with the useable area from the r
            int height = decorView.getContext().getResources().getDisplayMetrics().heightPixels;
            int diff = height - r.bottom;

            //if it could be a keyboard add the padding to the view
            if (diff != 0) {
                // if the use-able screen height differs from the total screen height we assume that it shows a keyboard now
                //check if the padding is 0 (if yes set the padding for the keyboard)
                if (contentView.getPaddingBottom() != diff) {
                    //set the padding of the contentView for the keyboard
                    contentView.setPadding(0, 0, 0, diff);
                }
            } else {
                //check if the padding is != 0 (if yes reset the padding)
                if (contentView.getPaddingBottom() != 0) {
                    //reset the padding of the contentView
                    contentView.setPadding(0, 0, 0, 0);
                }
            }
        }
    };


    /**
     * Helper to hide the keyboard
     *
     * @param act
     */
    public static void hideKeyboard(Activity act) {
        if (act != null && act.getCurrentFocus() != null) {
            InputMethodManager inputMethodManager = (InputMethodManager) act.getSystemService(Activity.INPUT_METHOD_SERVICE);
            inputMethodManager.hideSoftInputFromWindow(act.getCurrentFocus().getWindowToken(), 0);
        }
    }
}

Затем вы можете использовать его в своей деятельности или фрагменте, выполнив следующие действия:

//initialize the KeyboardUtil (you can do this global)
KeyboardUtil keyboardUtil = new KeyboardUtil(activity, getContent().getChildAt(0));

//enable it
keyboardUtil.enable();

//disable it
keyboardUtil.disable();

Весь класс util используется в вышеупомянутой библиотеке MaterialDrawer и может быть найден здесь KeyboardUtil. Это всегда будет содержать последнюю версию. (если есть улучшения)

Ответ 3

Кажется, что проблема связана с FrameLayout, так как она ведет себя так, что каждый ребенок, занимающий видимое пространство этого фрейма, поэтому не нуждается в изменении размера, чтобы соответствовать детям. Попробуйте использовать RelativeLayout. Он должен работать.

Ответ 4

Попробуйте поставить LinearLayout на ScrollView, который работал у меня один раз..

Ответ 5

Без использования ScrollView в качестве моего родителя я просто добавил android:fitsSystemWindows="true" в родительский вид (который был RelativeLayout и adjustResize в манифест для активности и работал.

Ответ 6

Мне нужно было установить

<item name="android:windowFullscreen">false</item> 

Несмотря на это, я никогда не устанавливал его в true, и приложение фактически не было полноэкранным.

Ответ 7

Вы можете настроить программно. Добавьте это в onCreate() вашей деятельности.

getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_ADJUST_RESIZE);

Ответ 8

В качестве оригинального плаката, который был обнаружен при присвоении Fullscreen Flag активности, атрибут android: windowFullscreen не будет работать, и поэтому ваше окно не будет изменяться, когда будет видна мягкая клавиатура, и она не будет прокручиваться.

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

Ответ 9

Убедитесь, что вы установили windowTranslucentStatus в false в своих стилях.