为什么 findFirstVisibleItemPosition 不按预期工作?

3

我已经苦恼了很长时间,我试图用recyclerview替换我的listview,我的listview看起来像这样:

<ListView
            android:id="@+id/browser_sub_list_view"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_alignParentBottom="true"
            android:layout_alignParentTop="true"
            android:clipToPadding="false"
            android:drawSelectorOnTop="true"
            android:paddingTop="340dp"/>

现在当我调用

view.getFirstVisiblePosition(); 

在这个列表视图中,我返回了第一个项目的位置,该位置为零,并且由于我给出了顶部填充,所以它将一直返回零,直到第一个项目超出屏幕范围为止。
但是当我尝试使用RecyclerView做同样的事情时,即使RecyclerView的第一个项目可见,它仍会给我不同的第一个可见项目的值。这是因为RecyclerView现在将toppadding视为屏幕边界,现在我不知道如何解决这个问题。请帮帮我。
以下是我的代码。
MainActivity.Java
public class MainActivity extends AppCompatActivity {
    private RecyclerView mRecyclerView;
    private TheAdapter adapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mRecyclerView = (RecyclerView) findViewById(R.id.recyclerview);
        mRecyclerView.setLayoutManager(new LinearLayoutManager(this));

        ArrayList<String> items = new ArrayList<>();
        for (int i = 0; i < 300; i++) {
            items.add("This is item:-" + i);
        }

        adapter = new TheAdapter(items);
        mRecyclerView.setAdapter(adapter);
        mRecyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
            @Override
            public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
                super.onScrollStateChanged(recyclerView, newState);
            }

            @Override
            public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
                LinearLayoutManager linearLayoutManager = (LinearLayoutManager) recyclerView.getLayoutManager();
                Log.d("DISTANCE", "" + linearLayoutManager.findFirstVisibleItemPosition());
            }
        });
    }
}

activity_main.xml

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
                xmlns:tools="http://schemas.android.com/tools"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                tools:context=".MainActivity">

    <android.support.v7.widget.RecyclerView
        android:id="@+id/recyclerview"
        android:layout_width="match_parent"
        android:clipToPadding="false"
        android:paddingTop="340dp"
        android:layout_height="match_parent"
        />

</RelativeLayout>

这是我的适配器类

class TheAdapter extends RecyclerView.Adapter<TheAdapter.ItemHolder> {

    public TheAdapter(ArrayList<String> data) {
        mData = data;
    }

    private ArrayList<String> mData;

    @Override
    public ItemHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.list_item, parent, false);

        return new ItemHolder(view);
    }

    @Override
    public void onBindViewHolder(ItemHolder holder, int position) {
        holder.mTextView.setText(mData.get(position));
    }

    @Override
    public int getItemCount() {

        return mData == null ? 0 : mData.size();
    }

    public class ItemHolder extends RecyclerView.ViewHolder {
        private TextView mTextView;

        public ItemHolder(View itemView) {
            super(itemView);
            mTextView = (TextView) itemView.findViewById(R.id.textTitle);
        }
    }
}

这真的太疯狂了,而且这种行为很不符合直觉。你有参考资料证明这是正确的行为吗?看起来 android:clipToPadding="false" 并没有被 findFirstVisibleItemPosition() 所遵守。 - Cheticamp
抱歉,我没有任何参考资料,但从技术上讲,如果第一个项目仍然可见,那么为什么要跳转并进入第二个项目?在listview中,android:clipToPadding="false"是被认可的。也许我应该发布示例项目。 - George
1个回答

3

在找到更好的解决方案之前,您可以尝试以下代码。 int pos 是由findFirstVisibleItemPosition()报告的位置。该方法将返回RecyclerView中最顶部可见的位置。

public int findRealFirstVisibleItemPosition(int pos) {
    View view;
    final LinearLayoutManager linearLayoutManager =
            (LinearLayoutManager) mRecyclerView.getLayoutManager();

    while (pos > 0) {
        view = linearLayoutManager.findViewByPosition(pos - 1);
        if (view == null) {
            break;
        }
        pos = pos - 1;
    }

    return pos;
}

我记得,大的填充会导致这种滚动视图的问题。


如果被接受,这是一个很好的完美答案,但是我不知道如何处理具有两列的GridLayoutManger,因为它以0,2,4的顺序返回结果,我将位置除以2,几乎完美,但是如果您有任何针对GridLayoutManger的解决方案,请也更新一下。 - George

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接