[英]RecyclerView and AdapterView, fails to show items
我正在做一些简单的示例,以作为指导,等等。我有一个带有自己的适配器的RecyclerView。 这些项目是数据模型,带有文本和图像。 ViewHolder,除了包含相应的视图外,还添加了一个布尔值以控制图像是否可见。
当我单击时,例如在第一项中,图像消失(或者如果再次单击,则显示)。
问题是,如果单击(例如第一个元素),图像将按预期消失,但是在滚动并且回收者加载不可见的新元素时,一个元素会突然与隐藏图像一起出现。
稍微调试一下,我看到在加载元素时,取决于布尔值会显示为true,理论上应该为false。
我不知道发生了什么,因为列表中的元素是不同的。
PS:就像我说的那样,代码真的很容易,所以不要指望伟大的事情。
主要活动:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
recyclerView = (RecyclerView) findViewById(R.id.recycler);
List<Person> personList = new ArrayList<>();
personList.add(new Person ("User1"));
personList.add(new Person ("User2"));
personList.add(new Person ("User3"));
personList.add(new Person ("User4"));
personList.add(new Person ("User5"));
personList.add(new Person ("User6"));
personList.add(new Person ("User7"));
personList.add(new Person ("User8"));
personList.add(new Person ("User9"));
personList.add(new Person ("User10"));
personList.add(new Person ("User11"));
personList.add(new Person ("User12"));
personList.add(new Person ("User13"));
personList.add(new Person ("User14"));
personList.add(new Person ("User15"));
personList.add(new Person ("User16"));
personList.add(new Person ("User17"));
personList.add(new Person ("User18"));
personList.add(new Person ("User19"));
personList.add(new Person ("User20"));
personList.add(new Person ("User22"));
personList.add(new Person ("User23"));
recyclerView.setLayoutManager(new LinearLayoutManager(this));
CustomImageAdapter adapter = new CustomImageAdapter(personList);
recyclerView.setAdapter(adapter);
适配器:
public class CustomImageAdapter extends RecyclerView.Adapter<CustomImageAdapter.ViewHolder> {
private List<Person> personList;
public CustomImageAdapter(List<Person> personList) {
this.personList = personList;
}
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View itemView = LayoutInflater.from(parent.getContext())
.inflate(R.layout.image_person, parent, false);
return new ViewHolder(itemView);
}
@Override
public void onBindViewHolder(final ViewHolder holder, int position) {
Person person = personList.get(position);
holder.name.setText(person.getName());
holder.itemView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (!holder.imageHide) {
holder.image.setVisibility(View.INVISIBLE);
} else {
holder.image.setVisibility(View.VISIBLE);
}
holder.imageHide = !holder.imageHide;
}
});
}
@Override
public int getItemCount() {
return personList.size();
}
public static class ViewHolder extends RecyclerView.ViewHolder {
TextView name;
ImageView image;
boolean imageHide = false;
public ViewHolder(View itemView) {
super(itemView);
name = (TextView) itemView.findViewById(R.id.name);
image = (ImageView) itemView.findViewById(R.id.image);
imageHide = false;
}
}
}
数据模型:(仅是一个隐秘的峰值)
public class Person {
private String name;
private String image;
public Person(String name) {
this.name = name;
}
....
}
布局:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:padding="10dp">
<TextView
android:id="@+id/name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="TextView" />
<ImageView
android:id="@+id/image"
android:layout_width="30dp"
android:layout_height="30dp"
app:srcCompat="@mipmap/ic_launcher"/>
</LinearLayout>
几个图像作为示例:
ViewHolder类旨在用于快速访问“ View”相关项目,例如具有这种性质的textviews或imageViews。
当内容需要基于对象模型是动态的时,则对象模型需要驱动可见性模式。 在viewholder类内没有嵌套的布尔值。
这样想吧。 回收者视图回收(n)个视图。
--View 1(显示人1)在Viewholder 1中可见
--View 2(显示人2)布尔值在Viewholder 1中可见
--View 3(显示人3)布尔值在Viewholder 1中可见
点击查看1
-查看1-保存布尔值false“图像不可见”
点击后当前视图
--View 1(显示人1)在Viewholder 1中不可见的布尔值
--View 2(显示人2)布尔值在Viewholder 1中可见
--View 3(显示人3)布尔值在Viewholder 1中可见
现在您滚动回收站
视图1不在屏幕上,并在下一个屏幕上回收,因为“人4”现在在观察者模式中保留了布尔值
--View 2(显示人2)布尔值在Viewholder 2中可见
--View 3(显示人员3)在Viewholder 3中可见
--View 1(显示第4个人)布尔值在Viewholder 1中不可见
因此,要纠正此问题:只需像这样修改代码:
public class Person {
private String name;
private String image;
private boolean isVisible;
public boolean getIsVisible(){
return isVisible;
}
public void setIsVisible(boolean value){
isVisible = value;
}
public Person(String name) {
this.name = name;
}
....
}
然后像这样修改您的适配器:
@Override
public void onBindViewHolder(final ViewHolder holder, int position) {
final Person person = personList.get(position);
holder.name.setText(person.getName());
holder.image.setVisibility(person.getIsVisibility() ? VISIBLE : INVISIBLE);
onItemClick(holder.root, person, position);
}
/*///////////////////////////////////////////////////////////////
// CLICK LISTENERS
*////////////////////////////////////////////////////////////////
private void onItemClick(final LinearLayout root, final Person model, final int position){
root.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
person.setIsVisible(!person.getIsVisible());
root.findViewById(YOUR_IMAGE_ID).setVisibility(person.getIsVisible() ? VISIBLE : INVISIBLE);
//or if you prefer to not findViewById, you can just update person boolean and call
//notifyDataSetChanged();
}
});
}
然后,如果您选择执行findViewById路由,则将根ID添加到您的XML文件和viewholder类中(建议使用更有效的方法,然后进行通知)
您可能需要考虑的另一件事是解释如何处理适配器外部的喀哒声。 通常,如果我不进行数据绑定,则倾向于使用接口。 将其添加到适配器的底部。
public interface ItemSelectedListener {
void personList_onItemClick(View view, int position, final Person person);
void personList_onItemLongClick(View view, int position, final Person person);
}
然后在构造函数中,需要ItemSelectedListener并将其存储在适配器类中。 然后,您可以修改onClick处理程序来执行
private void onItemClick(final LinearLayout root, final Person person, final int position){
root.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if(mItemSelectedListener != null){
mItemSelectedListener.personList_onClick(v, position, person);
}
}
});
}
当然,请重复上述操作以进行长按。
private void onItemLongClick(final LinearLayout root, final Person person, final int position){
root.setOnLongClickListener(new View.OnLongClickListener() {
@Override
public boolean onLongClick(View v) {
if(mItemSelectedListener != null){
mItemSelectedListener.personList_onItemLongClick(v, position, model);
}
return false;
}
});
}
并且当然也可以将用于长时间单击的侦听器添加到您的bindview中
@Override
public void onBindViewHolder(final ViewHolder holder, int position) {
final Person person = personList.get(position);
holder.name.setText(person.getName());
holder.image.setVisibility(person.getIsVisibility() ? VISIBLE : INVISIBLE);
onItemClick(holder.root, person, position);
onItemLongClick(holder.root, person, position);
}
您的代码看起来还不错,但可能与RecyclerViews的工作方式有关。 我认为这是发生在您身上的,因为您仅更改了持有人的价值,而持有人正在被回收以保存不同的数据项。
您是否可以尝试将boolean imageHide
到Person类,然后在onClick中更改person类中的该布尔值并设置图像的可见性。 然后在onBind中使用一些逻辑来检查该人是否应该隐藏或可见图像。
您需要在onBindViewHolder上添加一个验证,如下所示:
holder.image.setVisibility(holder.imageHide ?View.VISIBLE:View.INVISIBLE);
问候。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.