- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我怎样才能得到我的 ItemDecoration
当我从我的适配器中删除给定 View 然后调用 notifyItemRemoved()
时,“更新”其他 View 的项目偏移量?
在我的特殊情况下,我有一个 ItemDecoration
为每个项目提供少量的顶部偏移量,并为最后一个项目提供大量的底部偏移量。当我删除最后一项时,成为新的最后一项 的 View 没有任何底部偏移。
之前和之后:
如果我随后滚动列表,当我返回到列表底部时,我的"new"最后一项具有正确的偏移量。只要“新的”最后一项在屏幕上保持可见,这只是一个问题。
如果我改变我的 notifyItemRemoved()
调用 notifyDataSetChanged()
,新的最后一项将应用正确的项目偏移量,但我失去了从 notifyItemRemoved()
获得的内置动画.
如果我继续使用 notifyItemRemoved()
并且另外调用notifyItemChanged()
在前一个项目上,我会得到正确的项目偏移量,但动画有点卡顿;倒数第二个项目(在删除之前)似乎淡出然后再次出现(在此屏幕截图中,写着“项目 19”的卡片刚刚被删除):
我知道我可以通过对我的 <RecyclerView>
应用底部填充来在我的列表末尾创建一个大空间。标记并指定 android:clipToPadding="false"
, 但由于各种原因,这种解决方案是 Not Acceptable 。
我最初是在我正在开发的一个更大的应用程序中遇到这个问题的,但这里有一个演示该问题的小应用程序。
主要 Activity .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>
项目.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>
最佳答案
我通过更改 removeItemAt()
解决了这个问题方法如下:
private void removeItemAt(int position) {
if (list.size() > 0) {
list.remove(position);
notifyItemRemoved(position);
if (position != 0) {
notifyItemChanged(position - 1, Boolean.FALSE);
}
}
}
notifyItemChanged()
的第二个参数是关键。它实际上可以是任何对象,但我选择了 Boolean.FALSE
因为它是一个“众所周知”的对象,并且因为它传达了一点 Intent :我不希望动画在我正在更改的项目上运行。
原来还有第二个onBindViewHolder()
RecyclerView.Adapter
中定义的方法我以前从未见过。来自源代码:
public void onBindViewHolder(VH holder, int position, List<Object> payloads) {
onBindViewHolder(holder, position);
}
摘自该方法的文档:
The
payloads
parameter is a merge list fromnotifyItemChanged(int, Object)
ornotifyItemRangeChanged(int, int, Object)
. If thepayloads
list is not empty, theViewHolder
is currently bound to old data andAdapter
may run an efficient partial update using the payload info. If the payload is empty,Adapter
must run a full bind.
换句话说,通过在 notifyItemChanged()
中传递某些东西(任何东西)作为有效负载,我们告诉系统我们只想执行“部分更新”(在可能的情况下)。
所以,当然,我们已经指示系统执行部分更新...但是这如何停止通过调用 notifyItemChanged(position - 1)
引起的闪烁呢? ?它与 RecyclerView.ItemAnimator
有关附加到所有 RecyclerView
默认情况下:DefaultItemAnimator
.
DefaultItemAnimator
的源代码包括这个方法:
@Override
public boolean canReuseUpdatedViewHolder(@NonNull ViewHolder viewHolder,
@NonNull List<Object> payloads) {
return !payloads.isEmpty() || super.canReuseUpdatedViewHolder(viewHolder, payloads);
}
您会注意到此方法还需要一个 List<Object> payloads
范围。这与其他 onBindViewHolder()
中使用的有效负载列表相同。上面提到的调用。
因为我们传递了一个 payload 参数,这个方法将返回 true
.由于动画师现在被告知可以重用已经存在的 ViewHolder
对于我们的“已更改”项目,它不会将其拆除并创建一个新项目(或重复使用回收的项目)...这会停止运行已更改项目上的默认淡入淡出动画!
关于android - 从 RecyclerView.Adapter 中删除项目后,RecyclerView.ItemDecoration 不会更新,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47342034/
我正在制作一个 android 应用程序,它允许用户在 editText 中输入关键字,当他们点击提交时,下面的 recyclerview 将显示来自 API 请求的结果。 我的 recyclerVi
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 这个问题似乎不是关于 a specific programming problem, a softwar
我想在我的 Svelte Kit 应用程序中使用 @sveltejs/adapter-static(想把它变成一个 SPA)。 我使用 npm i @sveltejs/adapter-static 安
我有一个简单的表单,在提交重定向到 AuthController 内的 processAction 之后,在这个 action 中我想要创建一个简单的 table bar。 已编辑: 引用Zend f
当我使用 ListView 时,我扩展了 ArrayAdapter 并将数据(一个列表)存储在 ArrayAdapter 中,因此当数据更改时,我调用: adapter.add(item).; 还有另
我有一个使用 Webpack 捆绑的 commonjs 浏览器应用程序( typescript )。它使用 webrtc,所以我想使用 webrtc-adapter package from npm
在 “DesignPatterns: Elements of Reusable Object-Oriented Software” 一书中谈到 C++ 实现 Adapter 模式时,它是这样的这个:
只需要帮助理解 ZF2 中的一些简单数据库查询。在 ZF1 中,我有这样简单的方法: public function recordset() { // listing of all records $
当使用Spring的Java配置时,你一定会看到像WebMvcConfigurerAdapter这样的类& HandlerInterceptorAdapter它们实现单个接口(interface),并
我将 doctrine2 与 ZF2 一起使用,我的一些库与 Zend\Db\Adapter\Adapter 一起使用,其他的与 doctrine2 一起使用。现在,他们两次连接到数据库。是否可以在原
过去几天我一直在工作灯上工作,现在正在尝试制作连接到数据库的应用程序。以下是截图: 1) Worklight\server\conf\worklight.properties 2) mySQLAdap
几年前,我在 this tutorial 之后开始学习 Zend 框架。 .在那里,它显示映射器是使用 Zend\Db\Adapter\Adapter 创建的。类来获取数据库连接,这就是我使用数据库的
我正在学习 RecyclerView 并在 developer 中的站点 Adapter 类扩展 RecyclerView.Adapter 。实现显示: public class MyAdapter
查看文档:http://docs.spring.io/spring-integration/reference/html/ip.html#tcp-adapters我知道适配器仅用于单向通信。但是,这里
当我移动到另一个 fragment 时,我如何恢复分页适配器项状态?我试过下面的文章,但没有用。 https://medium.com/@florina.muntenescu private fun
我正在使用 capistrano 并收到此错误消息: Please install the pg adapter: `gem install activerecord-pg-adapter` (can
我正在尝试在本地主机上安装 Magento 2,当我想连接数据库时出现错误。 错误是: “Magento\Framework\DB\Adapter\Pdo\MysqlFactory”生成的源类“\Ma
我的 Rails 服务器通过正常的开发脚本运行良好,我计划使用 Netbeans for Rails。当我使用 Netbeans 启动我现有的项目并重新启动我的 Webrick 服务器时,它显示错误
运行 Ruby on Rails (RoR) 应用程序或使用 ActiveRecord 框架的 Ruby 代码,您会收到错误消息: Please install the postgresql adap
我在运行 Redmine2.3-stable 时遇到这个错误,随后无法成功安装 mysql2 适配器。 这是错误: Please install the mysql2 adapter: `ge
我是一名优秀的程序员,十分优秀!