keywords: GridLayoutManager, ItemDecoration, RecyclerView, item, width.
我在使用 RecyclerView 配合 GridLayoutManger 来排版 item 的时候,ui 上出现了一点棘手的问题。我需要让 item 的水平方向的间隔出现在中间,而不是 GridLayoutManager 默认的样子。即,如果一行 3 个 item,那么第一个和第三个需要分别与 RecyclerView 的左边和 RecyclerView 的右边对齐,让留白出现在第一个、第二个 item 之间以及第二个、第三个 item 之间。但是,GridLayoutManager 默认行为是每一个 item 右边都有一个小留白。第一个 item 与 RecyclerView 的左边对齐,第三个 item 与 RecyclerView 的右边也有一个留白(本文的目标就是想去掉这个留白)。
一图以蔽之:
默认是这样的,注意空白的位置:
我们想达到的效果是这样的:
解决方法自然是利用 ItemDecoration 。不过这其中有些小地方还挺有意思的。
首先,不妨假设 item 的宽度一样,并且 itemWidth * itemCountInOneRow 小于 RecyclerView 的可用宽度。那么在这种情况下,GridLayoutManager 在排版 item 的时候,会将 RecyclerView 的宽度除以 itemCountInOneRow 作为其分配给每一个 item 的可用宽度。然后我们就得到了图1的效果。
要想完成我们的需求,需要借助 ItemDecoration。我们需要计算出,所有留白的总宽度,让后除以 (itemCountInOneRow - 1)。这里的(itemCountInOneRow - 1)就是间隔的数目。得到的结果就是平均每个 item 需要保留的留白的宽度。然后,我们需要根据 item 的位置,指定其留白的具体位置(top、right、bottom、left)。具体逻辑如下面代码片段所示:
//(language: kotlin)
//(可以理解成 java 的 switch,效果是一样的)
val indexInRow = index % columnCount
when (indexInRow) {
0 -> {
outRect?.right = spaceForOneItem
}
columnCount - 1 -> {
outRect?.left = spaceForOneItem
}
else -> {
outRect?.left = spaceForOneItem / 2
outRect?.right = spaceForOneItem / 2
}
}
这里需要注意,每一个 item 的留白是相同的,计算方式跟刚才我们描述的一样。不要试图去让每个 item 的留白宽度不同。我们需要做的只是去指定留白的位置。**我们需要保证的是,第一个 item 的 outRect.right
= 最后一个 item 的 outRect.left
= 中间的 item 的 outRect.left + outRect.right
**