我试图在 RecyclerView 中显示三个(至少我遇到问题的情况)项目,其中 StaggeredGridLayoutManager 有两列。第一项跨越两行。下面是它的样子:
现在,我将项目“项目 2”移到顶部。这是我在适配器中调用的代码(这是我编写的一个示例,用于演示我在更复杂的项目中遇到的问题):
private int findById(int id) {
for (int i = 0; i < items.size(); ++i) {
if (items.get(i).title.equals("Item " + id)) {
return i;
}
}
return -1;
}
// Moving the item "Item 2" with id = 2 and position = 0
public void moveItem(int id, int position) {
final int idx = findById(id);
final Item item = items.get(idx);
if (position != idx) {
items.remove(idx);
items.add(position, item);
notifyItemMoved(idx, position);
//notifyDataSetChanged();
}
}
之后,数组就好了:[Item 2, Item 1, Item 3]。但是,观点还差得很远:
如果我触摸 RecyclerView(如果没有足够的项目可以滚动,足以触发过度滚动效果),第 2 项向左移动,我希望首先看到它(使用一个不错的动画):
您可能在代码中看到,我尝试通过调用 notifyDataSetChanged() 来替换 notifyItemMoved(idx, position)。它可以工作,但变化不是动画。
我写了一个完整的示例来演示这一点,并将其放在 GitHub .它几乎是最小的(有移动项目和切换它们的跨度的选项)。
我看不出我做错了什么。这是 StaggeredGridLayoutManager 的错误吗?我想避免 notifyDataSetChanged() 因为我想保持动画的一致性。
编辑:经过一番挖掘,没有必要使用完全跨度的项目来显示问题。我删除了全跨度。当我尝试将第 2 项移动到位置 0 时,它没有移动:第 1 项紧随其后,第 3 项移动到右侧,所以我有:空单元格、第 2 项、新行,第 1 项,第 3 项。滚动后我的布局仍然正确。
更有趣的是我没有 GridLayoutManager 的问题。我需要一个完整的项目,所以它不是一个解决方案,但我想这确实是 StaggeredGridLayoutManager...
最佳答案
我没有完整的答案,但我可以向您指出解决方法和错误报告(我认为这是相关的)。
更新布局使其看起来像您的第二个屏幕截图的技巧是调用 invalidateSpanAssignments()在 StaggeredGridLayoutManger (sglm) 在您调用 notifyItemMoved() 之后. “挑战”是如果您在 nIM() 之后立即调用它,它不会运行。如果您将调用延迟几毫秒,它会的。因此,在您引用的 MainActivity 代码中,我已将您的 sglm一个私有(private)字段:
private StaggeredGridLayoutManager sglm;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
adapter = new Adapter();
recyclerView = (RecyclerView) findViewById(R.id.recycler_view);
sglm = new StaggeredGridLayoutManager(2, StaggeredGridLayoutManager.VERTICAL);
recyclerView.setLayoutManager(sglm);
recyclerView.setItemAnimator(new DefaultItemAnimator());
recyclerView.setAdapter(adapter);
}
在 switch block 中向下,在处理程序中引用它:
case R.id.move_sec_top:
adapter.moveItem(2, 0);
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
sglm.invalidateSpanAssignments();
}
}, 100);
return true;
结果是您的动画仍在运行,布局以您想要的方式结束。这是一个真正的kludge,但它确实有效。我相信这与我在以下链接中发现并报告的错误相同:
https://code.google.com/p/android/issues/detail?id=93156
虽然我的“症状”和要求的调用不同,但根本问题似乎是相同的。
祝你好运!
编辑:无需延迟发布,只需发布即可:
case R.id.move_sec_top:
adapter.moveItem(2, 0);
new Handler().post(new Runnable() {
@Override
public void run() {
sglm.invalidateSpanAssignments();
}
});
return true;
我最初的理论是调用被阻塞,直到布局传递结束,但我相信情况并非如此。相反,我现在认为如果您调用 invalidateSpanAssignments()立即,它实际上执行得太快了(在布局更改完成之前)。因此,上面的帖子(没有延迟)只是将调用添加到渲染队列的末尾,它发生在布局之后。
关于android - RecyclerView StaggeredGridLayoutManager 重新排序问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27800575/