18

我正在构建一个类似于聊天的 Android 应用程序,类似于环聊。为此,我使用带有stackFromBottom=trueand的垂直 ListView transcriptMode="normal"

该列表从较早的消息(顶部)到较年轻的消息(底部)排序。正常状态下,ListView 滚动到底部,显示最年轻的消息。

一旦用户滚动到列表顶部,ListView 使用反向无限滚动适配器加载更多(旧)消息。加载较旧的消息后,它们将被添加到列表的顶部。

当旧消息添加到列表顶部时,所需的行为是 ListView 将其滚动位置保持在列表底部。也就是说,当旧消息添加到顶部时,滚动视图应该显示与添加旧消息之前相同的消息。

不幸的是,这不起作用。ListView不是将滚动位置保持在底部,而是将滚动位置保持在列表的顶部。也就是说,在将较旧的消息添加到列表后,列表会显示最顶部(即最旧)的消息。

有没有一种简单的方法来告诉 ListView 在向其顶部添加新项目时将其滚动位置保持在底部而不是顶部?

更新

出于演示目的,我尽可能地减少了用例和代码。下面的示例创建一个简单的字符串数组列表。单击一个项目将在列表顶部添加另一个项目。长按一个项目将在列表底部添加另一个项目。

在列表底部添加项目时一切正常(长按)。在列表顶部添加项目时(简单单击),则有 3 种情况:

  • 如果列表滚动到底部,那么一切正常。将项目添加到顶部后,列表仍然滚动到底部。
  • 如果列表没有滚动到底部(也没有滚动到顶部),则列表将保持其滚动位置(从顶部)。这使得列表看起来“跳”了一个项目。在这种情况下,我希望列表不要“跳起来”,即我希望它从底部保持滚动位置。
  • 如果列表滚动到顶部,则会发生与情况 2 相同的情况。首选行为与情况 2 相同。

(最后2个案例其实是一样的,我把它们分开是因为案例2可以更好地说明问题。)

代码

public class MyListActivity extends Activity {
  int topIndex = 1;

  int bottomIndex = 20;

  protected final void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.my_list_activity);

    final ListView list = (ListView) findViewById(R.id.list);
    list.setTranscriptMode(ListView.TRANSCRIPT_MODE_NORMAL);
    list.setStackFromBottom(true);

    final ArrayAdapter<String> adapter = new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1, android.R.id.text1);
    for (int i = topIndex; i <= bottomIndex; i++) {
      adapter.add(Integer.toString(i));
    }
    list.setAdapter(adapter);

    list.setOnItemClickListener(new OnItemClickListener() {
      @Override
      public void onItemClick(AdapterView<?> arg0, View arg1, int arg2, long arg3) {
        adapter.insert(Integer.toString(--topIndex), 0);
      }
    });

    list.setOnItemLongClickListener(new OnItemLongClickListener() {
      @Override
      public boolean onItemLongClick(AdapterView<?> arg0, View arg1, int arg2, long arg3) {
        adapter.add(Integer.toString(++bottomIndex));
        return true;
      }
    });
  }
}
4

4 回答 4

12

我已经想通了。无论您在哪里获取新数据,在更改适配器中的数据之前,我都会调用以下命令:

int firstVisibleItem = list.getFirstVisiblePosition();
int oldCount = adapter.getCount();
View view = list.getChildAt(0);
int pos = (view == null ? 0 :  view.getBottom());

然后我打电话:

adapter.setData(data);

list.setSelectionFromTop(firstVisibleItem + adapter.getCount() - oldCount + 1, pos);
于 2015-02-05T02:39:24.507 回答
3

首先获取 firstVisiblePosition (或倒置后的最后一个。你自己尝试检查一下)然后获取添加到 List 的 View 的顶部,然后使用 setSe;ectionFromTop() 方法,你可以滚动到所需的位置。

代码:

int index = mList.getFirstVisiblePosition();
View v = mList.getChildAt(0); //or upside down (list.size - 1)
int top = (v == null) ? 0 : v.getTop(); //returns the top of the view its Y co-ordinates. [Doc][1]
mList.setSelectionFromTop(index, top);

资源。

于 2014-02-26T20:06:46.157 回答
0

编辑以考虑其他信息:

我想我找到了你的问题。我通过使用独立列表作为 XML 中的唯一视图复制了您描述的用例:

<ListView
    android:id="@+id/list"
    android:layout_width="match_parent"
    android:layout_height="match_parent" >
</ListView>

我通过将布局嵌套在父布局中来解决问题以遵循您想要的用例:

<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent" >

    <ListView
        android:id="@+id/list"
        android:layout_width="match_parent"
        android:layout_height="match_parent" >
    </ListView>

</FrameLayout>
于 2014-02-26T20:15:34.487 回答
-2

在适配器中插入记录后,您可以尝试使用

   yourListView.setSelection(yourAdapter.getCount() - 1);
于 2014-07-11T13:44:51.613 回答