Android:只有我的 ListView 的最后一行对 OnClickListener 有反应
Android: Only the last row of my ListView is reacting to an OnClickListener
我有一个带 View Holder 的 ListView
。当我按下 ListView
的其中一行中的一个按钮时,它应该会在该位置展开该行并在我再次按下时折叠。我在布局上使用 show/hide 来执行此操作。但是,点击并没有展开正确的行。
示例:
I was clicked
I am a row that was not clicked
I am a row that was not clicked
I am a row that was not clicked // yet my layout was shown and hidden
每当我滚动时都会发生这种情况,只有我 ListView
中可见的最后一行会对我按下的任何行做出反应。
代码
public class PetAdapter extends BaseAdapter {
...
...
@Override
public int getCount() {
return mPetList.size();
}
@Override
public int getViewTypeCount() {
return 2;
}
@Override
public int getItemViewType(int position) {
if (mPetList.get(position).getPetType().equals(ConstantValues.CAT)) {
return 0;
} else {
return 1;
}
}
@Override
public Object getItem(int position) {
return mPetList.get(position);
}
@Override
public long getItemId(int position) {
return Integer.parseInt(mPetList.get(position).getId());
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
Pet pet;
View view = convertView;
switch (getItemViewType(position)) {
case 0:
pet = mPetList.get(position);
mRowType = ConstantValues.CAT;
view = createCatRow(pet, convertView, parent);
break;
case 1:
pet = mPetList.get(position);
mRowType = ConstantValues.DOG;
view = createDogRow(pet, convertView, parent);
break;
}
return view;
}
膨胀行
public View createCatRow(Pet pet, View convertView, ViewGroup parent) {
if (convertView != null) {
mCatHolder = (CatViewHolder) convertView.getTag();
} else {
convertView = LayoutInflater.from(mContext).inflate(R.layout.cat_row, parent, false);
mCatHolder= new CatViewHolder(convertView);
convertView.setTag(mCatHolder);
}
setCatText(pet);
setCatListeners(pet);
return convertView;
}
设置文本
public void setCatText(Pet pet) {
mCatHolder.catName.setText(pet.getName());
}
设置监听器
public void setCatListeners(Pet pet) {
mCatHolder.catName.setOnClickListener(getCatNameListener(pet));
}
public View.OnClickListener getCatNameListener(final Pet pet) {
return new View.OnClickListener() {
@Override
public void onClick(View v) {
expandCard(pet);
}
};
}
设置可见性
public void expandCard(Pet pet) {
if (mCatHolder.expanded.getVisibility() == View.GONE) {
mCatHolder.collapsed.setVisibility(View.GONE);
mCatHolder.expanded.setVisibility(View.VISIBLE);
} else {
mCatHolder.collapsed.setVisibility(View.VISIBLE);
mCatHolder.expanded.setVisibility(View.GONE);
}
}
视图持有人
public static class CatViewHolder {
@Bind(R.id.expanded) RelativeLayout expanded;
@Bind(R.id.collapsed ) RelativeLayout collapsed;
@Bind(R.id.cat_name) TextView catName;
public CatViewHolder(View view) {
ButterKnife.bind(this, view);
}
}
请帮助我理解为什么会发生这种行为。为什么最后一行对第一行的 OnClick 事件做出反应。谢谢
在RecyclerView
中,视图被回收。所以第一行的视图被重新用于后面的行,它仍然是 OnClickListener
。您必须在 every onBindViewHolder()
.
中设置侦听器
您在适配器 class 中持有对视图持有者的引用,然后将 onClicklistener 应用于视图。这意味着 viewholder 变量将引用最后呈现的视图,并将 onclicklistener 设置为最后呈现的视图。
您需要在您的 getview 方法中使 viewholder 引用一个局部变量,并将其传递给所有需要该 viewholder 的函数。
我有一个带 View Holder 的 ListView
。当我按下 ListView
的其中一行中的一个按钮时,它应该会在该位置展开该行并在我再次按下时折叠。我在布局上使用 show/hide 来执行此操作。但是,点击并没有展开正确的行。
示例:
I was clicked
I am a row that was not clicked
I am a row that was not clicked
I am a row that was not clicked // yet my layout was shown and hidden
每当我滚动时都会发生这种情况,只有我 ListView
中可见的最后一行会对我按下的任何行做出反应。
代码
public class PetAdapter extends BaseAdapter {
...
...
@Override
public int getCount() {
return mPetList.size();
}
@Override
public int getViewTypeCount() {
return 2;
}
@Override
public int getItemViewType(int position) {
if (mPetList.get(position).getPetType().equals(ConstantValues.CAT)) {
return 0;
} else {
return 1;
}
}
@Override
public Object getItem(int position) {
return mPetList.get(position);
}
@Override
public long getItemId(int position) {
return Integer.parseInt(mPetList.get(position).getId());
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
Pet pet;
View view = convertView;
switch (getItemViewType(position)) {
case 0:
pet = mPetList.get(position);
mRowType = ConstantValues.CAT;
view = createCatRow(pet, convertView, parent);
break;
case 1:
pet = mPetList.get(position);
mRowType = ConstantValues.DOG;
view = createDogRow(pet, convertView, parent);
break;
}
return view;
}
膨胀行
public View createCatRow(Pet pet, View convertView, ViewGroup parent) {
if (convertView != null) {
mCatHolder = (CatViewHolder) convertView.getTag();
} else {
convertView = LayoutInflater.from(mContext).inflate(R.layout.cat_row, parent, false);
mCatHolder= new CatViewHolder(convertView);
convertView.setTag(mCatHolder);
}
setCatText(pet);
setCatListeners(pet);
return convertView;
}
设置文本
public void setCatText(Pet pet) {
mCatHolder.catName.setText(pet.getName());
}
设置监听器
public void setCatListeners(Pet pet) {
mCatHolder.catName.setOnClickListener(getCatNameListener(pet));
}
public View.OnClickListener getCatNameListener(final Pet pet) {
return new View.OnClickListener() {
@Override
public void onClick(View v) {
expandCard(pet);
}
};
}
设置可见性
public void expandCard(Pet pet) {
if (mCatHolder.expanded.getVisibility() == View.GONE) {
mCatHolder.collapsed.setVisibility(View.GONE);
mCatHolder.expanded.setVisibility(View.VISIBLE);
} else {
mCatHolder.collapsed.setVisibility(View.VISIBLE);
mCatHolder.expanded.setVisibility(View.GONE);
}
}
视图持有人
public static class CatViewHolder {
@Bind(R.id.expanded) RelativeLayout expanded;
@Bind(R.id.collapsed ) RelativeLayout collapsed;
@Bind(R.id.cat_name) TextView catName;
public CatViewHolder(View view) {
ButterKnife.bind(this, view);
}
}
请帮助我理解为什么会发生这种行为。为什么最后一行对第一行的 OnClick 事件做出反应。谢谢
在RecyclerView
中,视图被回收。所以第一行的视图被重新用于后面的行,它仍然是 OnClickListener
。您必须在 every onBindViewHolder()
.
您在适配器 class 中持有对视图持有者的引用,然后将 onClicklistener 应用于视图。这意味着 viewholder 变量将引用最后呈现的视图,并将 onclicklistener 设置为最后呈现的视图。
您需要在您的 getview 方法中使 viewholder 引用一个局部变量,并将其传递给所有需要该 viewholder 的函数。