【发布时间】:2017-11-17 01:43:46
【问题描述】:
我的问题
当我从适配器中删除给定视图然后调用notifyItemRemoved() 时,如何让我的ItemDecoration“更新”other 视图的项目偏移量?
在我的特殊情况下,我有一个 ItemDecoration,它为每个项目提供少量 top 偏移量,并为 bottom 偏移量>仅最后一项。当我删除最后一项时,成为新的最后一项的视图没有任何底部偏移。
之前和之后:
如果我随后滚动列表,当我返回列表底部时,我的“新”最后一项具有正确的偏移量。只要“新”最后一个项目在屏幕上仍然可见,这只是一个问题。
我尝试过的
如果我将notifyItemRemoved() 调用更改为notifyDataSetChanged(),新的最后一项将应用正确的项偏移,但我会丢失从notifyItemRemoved() 获得的内置动画。
如果我继续使用notifyItemRemoved() 并且另外在前一个项目上调用notifyItemChanged(),那么我将获得正确的项目偏移量,但动画有些卡顿;倒数第二个项目(在移除之前)似乎淡出然后再次出现(在此屏幕截图中,“项目 19”的卡片刚刚被移除):
我知道我可以通过将底部填充应用于我的<RecyclerView> 标记并指定android:clipToPadding="false",在列表末尾创建一个大空间,但由于各种原因,这种解决方案是不可接受的。
重现
我最初在我正在开发的一个更大的应用程序中遇到了这个问题,但这里有一个小应用程序演示了这个问题。
MainActivity.java:
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
final MyAdapter adapter = new MyAdapter(20);
Button button = findViewById(R.id.button);
button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
adapter.removeItemAt(adapter.getItemCount() - 1);
}
});
RecyclerView recycler = findViewById(R.id.recycler);
recycler.setLayoutManager(new LinearLayoutManager(this));
recycler.addItemDecoration(new MyItemDecoration());
recycler.setAdapter(adapter);
}
private static class MyItemDecoration extends RecyclerView.ItemDecoration {
@Override
public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
int viewPosition = parent.getChildViewHolder(view).getLayoutPosition();
int lastPosition = state.getItemCount() - 1;
outRect.top = 20;
outRect.bottom = (viewPosition == lastPosition) ? 200 : 0;
}
}
private static class MyAdapter extends RecyclerView.Adapter<MyViewHolder> {
private final List<String> list;
private MyAdapter(int count) {
this.list = new ArrayList<>();
for (int i = 0; i < count; ++i) {
list.add("" + i);
}
}
@Override
public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
LayoutInflater inflater = LayoutInflater.from(parent.getContext());
View itemView = inflater.inflate(R.layout.item, parent, false);
return new MyViewHolder(itemView);
}
@Override
public void onBindViewHolder(MyViewHolder holder, int position) {
holder.text.setText("item: " + list.get(position));
}
@Override
public int getItemCount() {
return list.size();
}
private void removeItemAt(int position) {
if (list.size() > 0) {
list.remove(position);
notifyItemRemoved(position);
}
}
}
private static class MyViewHolder extends RecyclerView.ViewHolder {
private final TextView text;
public MyViewHolder(View itemView) {
super(itemView);
this.text = itemView.findViewById(R.id.text);
}
}
}
activity_main.xml:
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:background="#eee">
<Button
android:id="@+id/button"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="REMOVE LAST ITEM"/>
<android.support.v7.widget.RecyclerView
android:id="@+id/recycler"
android:layout_width="match_parent"
android:layout_height="0dp"
android:layout_weight="1"/>
</LinearLayout>
item.xml:
<?xml version="1.0" encoding="utf-8"?>
<android.support.v7.widget.CardView
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_margin="4dp">
<TextView
android:id="@+id/text"
android:layout_width="match_parent"
android:layout_height="64dp"
android:layout_margin="8dp"
android:gravity="center"
android:textColor="#000"
android:textSize="16sp"/>
</android.support.v7.widget.CardView>
【问题讨论】:
标签: android android-recyclerview