"Подтолкнуть" вид списка, но не панель, когда отображается мягкий ввод

У меня есть такой макет на странице View Pager:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@android:color/black">
    <LinearLayout
        android:id="@+id/bar"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginTop="24dp"
        android:layout_alignParentTop="true"
        android:orientation="horizontal"
        android:gravity="center"
        android:background="@drawable/background_gray">
        <TextView
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:textSize="16sp"
            android:paddingTop="15dp"
            android:paddingBottom="15dp"
            android:paddingLeft="3dp" />
    </LinearLayout>
    <RelativeLayout
        android:id="@+id/container_chat_box"
        android:background="@drawable/conversation_background_gray"
        android:layout_alignParentBottom="true"
        android:layout_marginBottom="50dp">
        <EditText
            android:id="@+id/chat_box"
            style="@style/ConversationChatBox" />
    </RelativeLayout>
    <ListView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_below="@+id/bar"
        android:background="@android:color/white"
        android:stackFromBottom="true"
        android:transcriptMode="normal"
        android:layout_above="@id/chat_box"
        android:paddingBottom="15dp"
        android:clipToPadding="false"
        android:divider="@null"
        android:dividerHeight="0dp"
        android:listSelector="@android:color/transparent" />
</RelativeLayout>

Это идея о том, как это выглядит:

----------
[   bar   ]
[list view]
[list view]
[list view]
[list view]
[edit text]

Когда я нажимаю на редактируемый текст, я бы хотел, чтобы панель оставалась там, где она есть (вверху экрана), но представление списка нужно сдвинуть вверх, чтобы оно выглядело так:

------------
[   bar   ]
[list view]
[list view]
[edit text]
[keyboard ]

Что я получаю (планка поднимается и исчезает):

------------
[list view]
[list view]
[list view]
[edit text]
[keyboard ]

У меня есть это на моем AndroidManifest:

android:windowSoftInputMode="adjustPan"

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

Есть ли способ удержать панель на месте, пока просмотр списка нажат? Заранее спасибо.

1 ответ

Решение

Я смог решить проблему, хотя пока тестировал только на нескольких телефонах.

Вот решение:

1) Измените режим мягкого ввода на android:windowSoftInputMode="AdjustResize" и добавьте android:fitsSystemWindows="true" в корневой вид макета.

2) Создан класс CustomInsetsLayout, который расширяет RelativeLayout (код ниже), чтобы я мог контролировать, что происходит с макетом при изменении размера экрана.

3) В этот момент у меня была нежелательная черная полоса в нижней части макета. Эта панель была нижней границей поля, которая должна была быть скрыта под панелью навигации, но флаги FULLSCREEN, отвечающие за это, не работают с AdjustResize. Поэтому мне пришлось добавить код в CustomInsetsLayout, чтобы удалить нижнее поле из корневого представления при изменении размера экрана.

Вот так выглядит класс CustomInsetsLayout (ссылки и примечания в описании класса):

package com.playchat.ui.full;

import android.content.Context;
import android.graphics.Rect;
import android.os.Build;
import android.util.AttributeSet;
import android.view.WindowInsets;
import android.widget.RelativeLayout;

/**
 * http://stackru.com/questions/21092888/windowsoftinputmode-adjustresize-not-working-with-translucent-action-navbar
 *
 * @author Kevin
 *         Date Created: 3/7/14
 *
 * https://code.google.com/p/android/issues/detail?id=63777
 *
 * When using a translucent status bar on API 19+, the window will not
 * resize to make room for input methods (i.e.
 * {@link android.view.WindowManager.LayoutParams#SOFT_INPUT_ADJUST_RESIZE} and
 * {@link android.view.WindowManager.LayoutParams#SOFT_INPUT_ADJUST_PAN} are
 * ignored).
 *
 * To work around this; override {@link #fitSystemWindows(android.graphics.Rect)},
 * capture and override the system insets, and then call through to FrameLayout's
 * implementation.
 *
 * For reasons yet unknown, modifying the bottom inset causes this workaround to
 * fail. Modifying the top, left, and right insets works as expected.
 */
public class CustomInsetsLayout extends RelativeLayout {
    private int[] mInsets = new int[4];

    public CustomInsetsLayout(Context context) {
        super(context);
    }

    public CustomInsetsLayout(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public CustomInsetsLayout(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
    }

    public final int[] getInsets() {
        return mInsets;
    }

    @Override
    protected final boolean fitSystemWindows(Rect insets) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            // Intentionally do not modify the bottom inset. For some reason,
            // if the bottom inset is modified, window resizing stops working.
            // TODO: Figure out why.
            mInsets[0] = insets.left;
            mInsets[1] = insets.top;
            mInsets[2] = insets.right;

            insets.left = 0;
            insets.top = 0;
            insets.right = 0;

            if (insets.bottom > 0) {
                // Remove bottom margin from the root view
                RelativeLayout.LayoutParams params = (RelativeLayout.LayoutParams) getLayoutParams();
                params.bottomMargin = 0;
                setLayoutParams(params);

            }
        }

        return super.fitSystemWindows(insets);
    }

    @Override
    public final WindowInsets onApplyWindowInsets(WindowInsets insets) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT_WATCH) {
            if (insets.getSystemWindowInsetBottom() > 0) {
                // Remove bottom margin from the root view
                RelativeLayout.LayoutParams params = (RelativeLayout.LayoutParams) getLayoutParams();
                params.bottomMargin = 0;
                setLayoutParams(params);

            }
            mInsets[0] = insets.getSystemWindowInsetLeft();
            mInsets[1] = insets.getSystemWindowInsetTop();
            mInsets[2] = insets.getSystemWindowInsetRight();
            return super.onApplyWindowInsets(insets.replaceSystemWindowInsets(0, 0, 0,
                insets.getSystemWindowInsetBottom()));

        } else {
            return insets;
        }
    }
}
Другие вопросы по тегам