我正在构建一个类似于 Google Hangouts 聊天界面的界面。新消息将添加到列表底部。向上滚动到列表顶部将触发加载以前的消息历史记录。当历史来自网络时,这些消息将被添加到列表的顶部,并且不应从触发加载时用户停止的位置触发任何类型的滚动。换句话说,“加载指示器”显示在列表顶部:
然后将其替换为任何加载的历史记录。
我已经完成了所有这些工作......除了我不得不诉诸反思来完成的一件事。在将项目添加到附加到 ListView 的适配器时,有很多问题和答案仅涉及保存和恢复滚动位置。我的问题是,当我执行以下操作时(简化但应该不言自明):
public void addNewItems(List<Item> items) {
final int positionToSave = listView.getFirstVisiblePosition();
adapter.addAll(items);
listView.post(new Runnable() {
@Override
public void run() {
listView.setSelection(positionToSave);
}
});
}
然后用户会看到快速闪到 ListView 顶部,然后快速闪回到正确的位置。问题比较明显,也被很多人发现:setSelection()
一直不爽,直到notifyDataSetChanged()
重画了ListView
。所以我们必须给post()
视图给它一个绘制的机会。但这看起来很可怕。
我已经通过使用反射“修复”了它。我讨厌它。在其核心,我想要完成的是重置第一个位置,而ListView
无需经历绘制周期的繁琐,直到我设置了位置。为此,ListView 有一个有用的字段:mFirstPosition
。天哪,这正是我需要调整的!不幸的是,它是包私有的。同样不幸的是,似乎没有任何方法可以以编程方式设置它或以任何不涉及无效循环的方式影响它......产生丑陋的行为。
因此,以失败为后备的反思:
try {
Field field = AdapterView.class.getDeclaredField("mFirstPosition");
field.setAccessible(true);
field.setInt(listView, positionToSave);
}
catch (Exception e) { // CATCH ALL THE EXCEPTIONS </meme>
e.printStackTrace();
listView.post(new Runnable() {
@Override
public void run() {
listView.setSelection(positionToSave);
}
});
}
}
它有效吗?是的。可怕吗?是的。将来会起作用吗?谁知道?有没有更好的办法?那是我的问题。
我如何在没有反思的情况下做到这一点?
答案可能是“自己编写ListView
可以处理的”。我只想问你是否看过ListView
.
编辑:根据 Luksprog 的评论/答案,没有反思的工作解决方案。
Luksprog推荐了一个OnPreDrawListener()
. 迷人!我以前搞过 ViewTreeObservers,但从来没有一个。经过一番折腾,以下类型的东西似乎工作得非常完美。
public void addNewItems(List<Item> items) {
final int positionToSave = listView.getFirstVisiblePosition();
adapter.addAll(items);
listView.post(new Runnable() {
@Override
public void run() {
listView.setSelection(positionToSave);
}
});
listView.getViewTreeObserver().addOnPreDrawListener(new OnPreDrawListener() {
@Override
public boolean onPreDraw() {
if(listView.getFirstVisiblePosition() == positionToSave) {
listView.getViewTreeObserver().removeOnPreDrawListener(this);
return true;
}
else {
return false;
}
}
});
}
很酷。