Android 调用notifyItemMoved()后,无法解释在RecyclerView中强制向下滚动

Android 调用notifyItemMoved()后,无法解释在RecyclerView中强制向下滚动,android,android-recyclerview,Android,Android Recyclerview,在我的应用程序中,我有一个使用RecyclerView实现的项目列表。在某些情况下,项目可能会移动到列表的最后,我需要将其设置为动画。因此,首先我将项目移动到数据源的末尾(在我的例子中是ArrayList),并调用适配器的notifyItemMoved(oldPosition,dataSet.size()-1)方法。除了移动到末尾的元素(实际上,它是列表的最末尾还是稍低一点并不重要)是列表中的最顶端还是第一个部分可见的元素外,其他一切都正常。在这种情况下,不仅移动的项目被设置动画,而且整个列表也

在我的应用程序中,我有一个使用RecyclerView实现的项目列表。在某些情况下,项目可能会移动到列表的最后,我需要将其设置为动画。因此,首先我将项目移动到数据源的末尾(在我的例子中是ArrayList),并调用适配器的
notifyItemMoved(oldPosition,dataSet.size()-1)
方法。除了移动到末尾的元素(实际上,它是列表的最末尾还是稍低一点并不重要)是列表中的最顶端还是第一个部分可见的元素外,其他一切都正常。在这种情况下,不仅移动的项目被设置动画,而且整个列表也随之向下滚动

我认为我的代码可能有点混乱,所以我创建了一个带有RecyclerView组件的干净的测试应用程序,结果是一样的

我认为这是
RecyclerView
的一个缺陷,因为它仍然是一个非常绿色的组件,Goggle仍在继续开发它,尽管我很确定我不是唯一一个遇到这种问题的人,所以我希望有人能解决这个问题

下面是我的
活动
回收视图.Adapter
的代码片段和一个布局

活动

public class RecyclerViewActivity extends Activity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_recycler_view);

        RecyclerView recyclerView = (RecyclerView) findViewById(R.id.recycler_view);
        recyclerView.setHasFixedSize(true);
        RecyclerView.LayoutManager layoutManager = new LinearLayoutManager(this);
        recyclerView.setLayoutManager(layoutManager);

        RecyclerView.Adapter adapter = new RecyclerViewAdapter(recyclerView);
        recyclerView.setAdapter(adapter);
        recyclerView.addItemDecoration(new    SimpleDividerItemDecoration(RecyclerViewActivity.this));
    }
}
适配器

public class RecyclerViewAdapter extends RecyclerView.Adapter<RecyclerViewAdapter.ViewHolder> {
    private ArrayList<String> mDataSet;
    private RecyclerView mRecyclerView;

    public RecyclerViewAdapter(RecyclerView recyclerView) {
        mRecyclerView = recyclerView;

        mDataSet = new ArrayList<String>();
        mDataSet.add("San Francisco");
        mDataSet.add("Los Angeles");
        mDataSet.add("Seattle");
        mDataSet.add("Portland");
        mDataSet.add("Sacramento");
        mDataSet.add("San Diego");
        mDataSet.add("Chicago");
        mDataSet.add("Boston");
        mDataSet.add("New York");
        mDataSet.add("New Jersey");
        mDataSet.add("Washington");
        mDataSet.add("Miami");
        mDataSet.add("New Orlean");
    }

    public RecyclerViewAdapter.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.adapter_recycler_view_item, parent, false);
        view.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                int itemPosition = mRecyclerView.getChildAdapterPosition(v);
                String item = mDataSet.get(itemPosition);
                mDataSet.remove(itemPosition);
                mDataSet.add(item);
                notifyItemMoved(itemPosition, mDataSet.size() - 1);

            }
        });
        return new ViewHolder(view);
    }

    public void onBindViewHolder(ViewHolder holder, int position) {
        holder.itemName.setText(mDataSet.get(position));
    }

    @Override
    public int getItemCount() {
        return mDataSet.size();
    }

    public static class ViewHolder extends RecyclerView.ViewHolder {
        protected TextView itemName;

        public ViewHolder(View v) {
            super(v);

            itemName = (TextView) v.findViewById(R.id.recycler_view_item_text);
        }
    }
}
公共类RecycleServiceAdapter扩展了RecyclerView.Adapter{
私有ArrayList数据集;
私人回收视图mRecyclerView;
公共回收服务适配器(回收服务视图回收服务视图){
mRecyclerView=回收视图;
mDataSet=newarraylist();
mDataSet.add(“旧金山”);
mDataSet.add(“洛杉矶”);
mDataSet.add(“西雅图”);
mDataSet.add(“波特兰”);
mDataSet.add(“萨克拉门托”);
mDataSet.add(“圣地亚哥”);
mDataSet.add(“芝加哥”);
mDataSet.add(“波士顿”);
mDataSet.add(“纽约”);
mDataSet.add(“新泽西”);
mDataSet.add(“华盛顿”);
mDataSet.add(“迈阿密”);
mDataSet.add(“新奥尔良”);
}
public RecycleServiceAdapter.ViewHolder onCreateViewHolder(视图组父级,int-viewType){
View View=LayoutInflater.from(parent.getContext()).flate(R.layout.adapter\u recycler\u View\u item,parent,false);
view.setOnClickListener(新的view.OnClickListener(){
@凌驾
公共void onClick(视图v){
int itemPosition=mRecyclerView.getChildAdapterPosition(v);
String item=mDataSet.get(itemPosition);
mDataSet.remove(itemPosition);
mDataSet.add(项);
notifyItemMoved(itemPosition,mDataSet.size()-1);
}
});
返回新的ViewHolder(视图);
}
公共无效onBindViewHolder(ViewHolder,int位置){
holder.itemName.setText(mDataSet.get(position));
}
@凌驾
public int getItemCount(){
返回mDataSet.size();
}
公共静态类ViewHolder扩展了RecyclerView.ViewHolder{
受保护的TextView项目名称;
公共视图持有者(视图v){
超级(五);
itemName=(TextView)v.findViewById(R.id.recycler\u view\u item\u text);
}
}
}
布局

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:orientation="vertical"
              android:layout_width="match_parent"
              android:layout_height="match_parent">

    <android.support.v7.widget.RecyclerView
            android:id="@+id/recycler_view"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:scrollbars="vertical"/>

</LinearLayout>

1) 波特兰-好的

2) 西雅图-好的

3) 洛杉矶-好的

4)旧金山-不OK -整个列表向下滚动该项。

此外,在视频中还可以看到一些元素的相同行为示例,这些元素不在列表的顶部,但在列表的顶部可见,部分悬停在列表的顶部边框处。

我的临时解决方案:

// mLayoutManager is LinearLayoutManager from RecyclerView
int visible = mLayoutManager.findFirstVisibleItemPosition();
int offset = mLayoutManager.getChildAt(visible).getTop();

... remove, add and call notifyItemMoved

mLayoutManager.scrollToPositionWithOffset(visible, offset);

这是一个android bug嘿,兄弟,你解决了这个问题吗?@bill good question:-)也有同样的问题@愚蠢的最后,我使用“scrollToPosition(0)”方法临时解决了这个问题。@Bill没有,我没有找到任何合适的解决方案。你是怎么修理的?