Как определить видоискатель слева или справа в RecyclerView с помощью StaggeredGridLayoutManager - PullRequest
0 голосов
/ 04 мая 2018

Я прочитал много похожих статей, но до сих пор не нашел ответа, как узнать, отображается или нет держатель вида в RecyclerView с StaggeredGridLayoutManager.

Положение: У меня есть RecyclerView, StaggeredGrid и я хочу сделать заполнение как

8 dp [left view] 8 dp [right view] 8 dp

Так как я не могу сделать это в XML, я должен добавить несколько полей -

Для левого обзора: левое поле 8dp, правое поле 4dp

Для правого обзора: левое поле 4dp, правое поле 8dp

Обычно представления размещаются так:

[0][1]
[2][3]
[4][5]

Самое простое решение - попытаться определить его по позиции:

override fun onBindViewHolder(ViewHolder holder, int position) {
    ...
        val params = holder.cardView.layoutParams as FrameLayout.LayoutParams

        if (position % 2 == 0) {
            params.leftMargin = pxFromDp(context, 8f).toInt()
            params.rightMargin = pxFromDp(context, 4f).toInt()
        }
        if (position % 2 == 1) {
            params.rightMargin = pxFromDp(context, 8f).toInt()
            params.leftMargin = pxFromDp(context, 4f).toInt()
        }
        params.bottomMargin = pxFromDp(context, 2f).toInt()
        params.topMargin = pxFromDp(context, 6f).toInt()

        holder.cardView.layoutParams = params
    ...
}

И это работает, но если вид 2 имеет меньшую высоту, чем вид 1, они помещаются

[0][1]
[3][2]
[5][4]

Так что не работает.

Как узнать, есть ли левый или правый держатель для просмотра?

1 Ответ

0 голосов
/ 04 мая 2018

Единственная опция, которая действительно помогла, это использование пользовательских RecyclerView.ItemDecoration()

class StaggeredGridDecoration(val margin: Int) : RecyclerView.ItemDecoration() {
    override fun getItemOffsets(outRect: Rect, view: View, parent: RecyclerView, state: RecyclerView.State?) {
        super.getItemOffsets(outRect, view, parent, state)
        val position = parent.getChildAdapterPosition(view)
        val spanIndex = (view.layoutParams as StaggeredGridLayoutManager.LayoutParams).spanIndex
        val type = parent.adapter.getItemViewType(position)
        val halfOfMargin = margin / 2
        when (type) {
            1 -> {//use here 0 if you didnt create custom types
                var top = 0
                val bottom = pxFromDp(parent.context, margin.toFloat()).toInt()
                //for first 2 elements I need additional margin top
                if (position < 3) {
                    top = pxFromDp(parent.context, margin.toFloat()).toInt()
                }
                if (spanIndex == 0) {
                    //settings for left column
                    val left = pxFromDp(parent.context, margin.toFloat()).toInt()
                    val right = pxFromDp(parent.context, halfOfMargin.toFloat()).toInt()
                    setMargins(view, left, right, top, bottom)
                } else {
                    //settings for right column
                    val left = pxFromDp(parent.context, halfOfMargin.toFloat()).toInt()
                    val right = pxFromDp(parent.context, margin.toFloat()).toInt()
                    setMargins(view, left, right, top, bottom)
                }
            }
        }
    }

    private fun setMargins(view: View, left: Int, right: Int, top: Int, bottom: Int) {
        val cardView: CardView = view.findViewById(R.id.cardView)
        val params = cardView.layoutParams as FrameLayout.LayoutParams

        params.rightMargin = right
        params.leftMargin = left
        params.bottomMargin = bottom
        params.topMargin = top

        cardView.layoutParams = params
    }

}

И только что добавил это

recyclerView.addItemDecoration(new StaggeredGridDecoration(8));
...