Получить дочернюю высоту представления в RecyclerView ItemDecoration.getItemOffsets

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

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

Проблема, с которой я сталкиваюсь, заключается в том, что view.getHeight() не всегда возвращает правильную высоту. Обычно в первый раз, когда getItemOffsets() вызывается, высота намного меньше, чем должна быть (например, 30px против 300px). Это происходит, даже если я даю дочерним представлениям фиксированную высоту в XML-макете. Я предполагаю, что это как-то связано с циклом измерения / макета, поскольку я не уверен, как получить правильные размеры вида в ItemDecoration.

Как правильно получить программную высоту дочерних представлений в getItemOffsets()?

ItemDecoration:

public class MinHeightPaddingItemDecoration extends RecyclerView.ItemDecoration {
    private static final String LOG_TAG = MinHeightPaddingItemDecoration.class.getSimpleName();

    private int mMinHeight;

    public MinHeightPaddingItemDecoration(int minHeight) {
        super();
        mMinHeight = minHeight;
    }

    @Override
    public void getItemOffsets(Rect outRect, View view, RecyclerView recyclerView, RecyclerView.State state) {
        int itemCount = state.getItemCount();
        int lastPosition = itemCount - 1;
        int itemPosition = recyclerView.getChildAdapterPosition(view);
        int layoutPosition = recyclerView.getChildLayoutPosition(view);

        // If this view isnt on screen then do nothing
        if (layoutPosition != RecyclerView.NO_POSITION && itemPosition == lastPosition) {

            // NOTE: view.getHeight() doesn't always return the correct height, even if the layout is given a fixed height in the XML
            int childHeight = view.getHeight();

            int totalChildHeight = childHeight * itemCount;

            int minHeight = getMinHeight();
            if (totalChildHeight < minHeight) {
                outRect.bottom = minHeight - totalChildHeight;
            }
        }
    }

    private int getMinHeight() {
        return mMinHeight;
    }
}

recycler_view_item.xml

<?xml version="1.0" encoding="utf-8"?>
<layout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools">
    <data>
        <variable name="controller" type="my.ViewController"/>
    </data>

    <android.support.v7.widget.CardView
        android:layout_width="match_parent"
        android:layout_height="100dp"
        android:layout_gravity="center"
        android:clickable="true"
        android:onClick="@{() -> controller.doSomething()}">

        <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent">
                <TextView
                    android:layout_width="match_parent"
                    android:layout_height="wrap_content"
                    android:text="Child layout"
                    />
        </RelativeLayout>

    </android.support.v7.widget.CardView>
</layout>

1 ответ

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

// NOTE: view.getHeight() doesn't always return the correct height 
// NOTE: even if the layout is given a fixed height in the XML. 
// NOTE: Instead directly access the LayoutParams height value
int childHeight = view.getLayoutParams().height;

Я нашел своего рода хак, прочитав https://yoda.entelect.co.za/view/9627/how-to-android-recyclerview-item-decorations.

Перед вычислением заполнения на основе child.width я вручную измеряю View, если это необходимо:

override fun getItemOffsets(outRect: Rect, view: View, parent: RecyclerView, state: RecyclerView.State) {
    if(view.width == 0) fixLayoutSize(view, parent)
    calculatePadding(outRect, view, parent, state)
}

private fun fixLayoutSize(view: View, parent: ViewGroup) {
    if (view.layoutParams == null) {
        view.layoutParams = ViewGroup.LayoutParams(
            ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT
        )
    }
    val widthSpec = View.MeasureSpec.makeMeasureSpec(parent.width, View.MeasureSpec.EXACTLY)
    val heightSpec = View.MeasureSpec.makeMeasureSpec(parent.height, View.MeasureSpec.UNSPECIFIED)
    val childWidth = ViewGroup.getChildMeasureSpec(
        widthSpec, parent.paddingLeft + parent.paddingRight, view.layoutParams.width
    )
    val childHeight = ViewGroup.getChildMeasureSpec(
        heightSpec, parent.paddingTop + parent.paddingBottom, view.layoutParams.height
    )
    view.measure(childWidth, childHeight)
    view.layout(0, 0, view.measuredWidth, view.measuredHeight)
}
Другие вопросы по тегам