RecyclerView - 获取特定位置的视图
Posted
技术标签:
【中文标题】RecyclerView - 获取特定位置的视图【英文标题】:RecyclerView - Get view at particular position 【发布时间】:2016-02-20 10:38:10 【问题描述】:我有一个RecyclerView
和ImageView
的活动。我正在使用RecyclerView
水平显示图像列表。当我单击RecyclerView
中的图像时,活动中的ImageView
应该显示图像的更大图片。到目前为止一切正常。
现在活动中有两个ImageButtons
:imageButton_left
和imageButton_right
。当我单击imageButton_left
时,ImageView
中的图像应向左转,RecyclerView
中的缩略图应反映此更改。 imageButton_right
的情况类似。
我可以旋转ImageView
。但是,如何旋转RecyclerView
中的缩略图?如何获得ViewHolder
的ImageView
?
代码:
活动 XML:
<?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_
android:layout_
android:orientation="vertical">
<android.support.v7.widget.RecyclerView
android:id="@+id/recyclerview"
android:layout_
android:layout_
android:layout_margin="10dp" />
<LinearLayout
android:layout_
android:layout_
android:layout_margin="10dp"
android:orientation="vertical">
<ImageView
android:id="@+id/original_image"
android:layout_
android:layout_
android:scaleType="fitXY"
android:src="@drawable/image_not_available_2" />
<LinearLayout
android:layout_
android:layout_
android:layout_marginTop="10dp"
android:gravity="center_horizontal"
android:orientation="horizontal">
<ImageButton
android:id="@+id/imageButton_left"
android:layout_
android:layout_
android:layout_marginRight="20dp"
android:background="@drawable/rotate_left_icon" />
<ImageButton
android:id="@+id/imageButton_right"
android:layout_
android:layout_
android:background="@drawable/rotate_right_icon" />
</LinearLayout>
</LinearLayout>
</LinearLayout>
我的活动代码:
public class SecondActivity extends AppCompatActivity implements IRecyclerViewClickListener
RecyclerView mRecyclerView;
LinearLayoutManager mLayoutManager;
RecyclerViewAdapter mRecyclerViewAdapter;
List<String> urls = new ArrayList<String>();
ImageView mOriginalImageView;
ImageButton mLeftRotate, mRightRotate;
@Override
protected void onCreate(Bundle savedInstanceState)
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_second);
urls.clear();
mRecyclerView = (RecyclerView) findViewById(R.id.recyclerview);
mLayoutManager = new LinearLayoutManager(this, android.support.v7.widget.LinearLayoutManager.HORIZONTAL, false);
mLayoutManager.setOrientation(android.support.v7.widget.LinearLayoutManager.HORIZONTAL);
mRecyclerView.setLayoutManager(mLayoutManager);
mRecyclerViewAdapter = new RecyclerViewAdapter(this, urls);
mRecyclerView.setAdapter(mRecyclerViewAdapter);
mOriginalImageView = (ImageView) findViewById(R.id.original_image);
mLeftRotate = (ImageButton) findViewById(R.id.imageButton_left);
mLeftRotate.setOnClickListener(new View.OnClickListener()
@Override
public void onClick(View v)
mOriginalImageView.setRotation(mOriginalImageView.getRotation() - 90);
);
mRightRotate = (ImageButton) findViewById(R.id.imageButton_right);
mRightRotate.setOnClickListener(new View.OnClickListener()
@Override
public void onClick(View v)
mOriginalImageView.setRotation(mOriginalImageView.getRotation() + 90);
);
Intent intent = getIntent();
if (intent != null)
String portfolio = intent.getStringExtra("portfolio");
try
JSONArray jsonArray = new JSONArray(portfolio);
for (int i = 0; i < jsonArray.length(); i++)
JSONObject jsonObject = jsonArray.getJSONObject(i);
String url = jsonObject.getString("url");
urls.add(url);
Log.d(Const.DEBUG, "URLs: " + urls.toString());
mRecyclerViewAdapter.notifyDataSetChanged();
catch (Exception e)
e.printStackTrace();
@Override
public void onItemClick(int position)
Picasso.with(this).load(urls.get(position)).into(mOriginalImageView);
我的 RecyclerView 自定义适配器:
public class RecyclerViewAdapter extends RecyclerView.Adapter<RecyclerViewAdapter.ViewHolder>
Context context;
List<String> mUrls = new ArrayList<String>();
IRecyclerViewClickListener mIRecyclerViewClickListener;
public int position;
public int getPosition()
return position;
public void setPosition(int position)
this.position = position;
public RecyclerViewAdapter(Context context, List<String> urls)
this.context = context;
this.mUrls.clear();
this.mUrls = urls;
Log.d(Const.DEBUG, "Urls Size: " + urls.size());
Log.d(Const.DEBUG, urls.toString());
if (context instanceof IRecyclerViewClickListener)
mIRecyclerViewClickListener = (IRecyclerViewClickListener) context;
else
Log.d(Const.DEBUG, "Implement IRecyclerViewClickListener in Activity");
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType)
View view = LayoutInflater.from(context).inflate(R.layout.item_horizontal_recyclerview, parent, false);
ViewHolder holder = new ViewHolder(view);
return holder;
@Override
public void onBindViewHolder(ViewHolder holder, int position)
Picasso.with(context).load(mUrls.get(position)).into(holder.mImageView);
@Override
public int getItemCount()
return mUrls.size();
public void rotateThumbnail()
public class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener
public ImageView mImageView;
public View v;
public ViewHolder(View v)
super(v);
v.setTag(getAdapterPosition());
v.setOnClickListener(this);
this.mImageView = (ImageView) v.findViewById(R.id.image);
@Override
public void onClick(View v)
this.v = v;
mIRecyclerViewClickListener.onItemClick(getAdapterPosition());
【问题讨论】:
您应该在 viewHolder 构造函数中创建它,添加一个属性来告诉您是否需要旋转图像,然后只需更改数据上的该属性并调用 notifyDataSetChanged 以使 recyclerView 重新加载跨度> 任何答案都能解决您的问题吗? ViewHolder 对于性能问题应该是静态的。外部类的所有变量都应该通过构造函数传递。 作为@Scott 回答的补充点击here 【参考方案1】:你可以同时使用
recyclerViewInstance.findViewHolderForAdapterPosition(adapterPosition)
和
recyclerViewInstance.findViewHolderForLayoutPosition(layoutPosition)
。
确保 RecyclerView 视图使用两种类型的位置
适配器位置:项目在适配器中的位置。这是从适配器的角度来看的位置。
布局位置:项目在最新布局计算中的位置。这是从 LayoutManager 的角度来看的位置。
你应该使用getAdapterPosition()
来表示findViewHolderForAdapterPosition(adapterPosition)
和getLayoutPosition()
来表示findViewHolderForLayoutPosition(layoutPosition)
。
取一个成员变量来保存recyclerview适配器中先前选择的项目位置和其他成员变量来检查用户是否是第一次点击。
底部附有示例代码和屏幕截图以获取更多信息。
public class MainActivity extends AppCompatActivity
private RecyclerView mRecyclerList = null;
private RecyclerAdapter adapter = null;
@Override
protected void onCreate(Bundle savedInstanceState)
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mRecyclerList = (RecyclerView) findViewById(R.id.recyclerList);
@Override
protected void onStart()
RecyclerView.LayoutManager layoutManager = null;
String[] daysArray = new String[15];
String[] datesArray = new String[15];
super.onStart();
for (int i = 0; i < daysArray.length; i++)
daysArray[i] = "Sunday";
datesArray[i] = "12 Feb 2017";
adapter = new RecyclerAdapter(mRecyclerList, daysArray, datesArray);
layoutManager = new LinearLayoutManager(MainActivity.this);
mRecyclerList.setAdapter(adapter);
mRecyclerList.setLayoutManager(layoutManager);
public class RecyclerAdapter extends RecyclerView.Adapter<RecyclerAdapter.MyCardViewHolder>
private final String TAG = "RecyclerAdapter";
private Context mContext = null;
private TextView mDaysTxt = null, mDateTxt = null;
private LinearLayout mDateContainerLayout = null;
private String[] daysArray = null, datesArray = null;
private RecyclerView mRecyclerList = null;
private int previousPosition = 0;
private boolean flagFirstItemSelected = false;
public RecyclerAdapter(RecyclerView mRecyclerList, String[] daysArray, String[] datesArray)
this.mRecyclerList = mRecyclerList;
this.daysArray = daysArray;
this.datesArray = datesArray;
@Override
public MyCardViewHolder onCreateViewHolder(ViewGroup parent, int viewType)
LayoutInflater layoutInflater = null;
View view = null;
MyCardViewHolder cardViewHolder = null;
mContext = parent.getContext();
layoutInflater = LayoutInflater.from(mContext);
view = layoutInflater.inflate(R.layout.date_card_row, parent, false);
cardViewHolder = new MyCardViewHolder(view);
return cardViewHolder;
@Override
public void onBindViewHolder(MyCardViewHolder holder, final int position)
mDaysTxt = holder.mDaysTxt;
mDateTxt = holder.mDateTxt;
mDateContainerLayout = holder.mDateContainerLayout;
mDaysTxt.setText(daysArray[position]);
mDateTxt.setText(datesArray[position]);
if (!flagFirstItemSelected)
mDateContainerLayout.setBackgroundColor(Color.GREEN);
flagFirstItemSelected = true;
else
mDateContainerLayout.setBackground(null);
@Override
public int getItemCount()
return daysArray.length;
class MyCardViewHolder extends RecyclerView.ViewHolder
TextView mDaysTxt = null, mDateTxt = null;
LinearLayout mDateContainerLayout = null;
LinearLayout linearLayout = null;
View view = null;
MyCardViewHolder myCardViewHolder = null;
public MyCardViewHolder(View itemView)
super(itemView);
mDaysTxt = (TextView) itemView.findViewById(R.id.daysTxt);
mDateTxt = (TextView) itemView.findViewById(R.id.dateTxt);
mDateContainerLayout = (LinearLayout) itemView.findViewById(R.id.dateContainerLayout);
mDateContainerLayout.setOnClickListener(new View.OnClickListener()
@Override
public void onClick(View v)
LinearLayout linearLayout = null;
View view = null;
if (getAdapterPosition() == previousPosition)
view = mRecyclerList.findViewHolderForAdapterPosition(previousPosition).itemView;
linearLayout = (LinearLayout) view.findViewById(R.id.dateContainerLayout);
linearLayout.setBackgroundColor(Color.GREEN);
previousPosition = getAdapterPosition();
else
view = mRecyclerList.findViewHolderForAdapterPosition(previousPosition).itemView;
linearLayout = (LinearLayout) view.findViewById(R.id.dateContainerLayout);
linearLayout.setBackground(null);
view = mRecyclerList.findViewHolderForAdapterPosition(getAdapterPosition()).itemView;
linearLayout = (LinearLayout) view.findViewById(R.id.dateContainerLayout);
linearLayout.setBackgroundColor(Color.GREEN);
previousPosition = getAdapterPosition();
);
【讨论】:
我认为这个讨论已经筋疲力尽了!感谢您进一步澄清此问题。 当我尝试更改前一项的颜色时,我在 else 块中得到 NullPointer 异常,所以我添加了空检查,现在它永远不会进入该块,如果它是有 NullPointer 异常,所以以前的项目颜色永远不会改变。现在我的 Recycler 视图中有多个彩色项目的项目。我该如何解决这个问题【参考方案2】:recyclerView.post(new Runnable()
@Override
public void run()
for (int i = 0; i < 5; i++)
View viewTemp = recyclerView.getLayoutManager().findViewByPosition(i);
CardView cardView = (CardView) viewTemp.findViewById(R.id.cardView);
//TO DO
);
【讨论】:
您好@peter-linh,感谢您为 *** 做出贡献,请详细说明您的答案。看看***.com/help/how-to-answer,看看如何写出好的答案。【参考方案3】:RecyclerView - 获取特定位置的视图。 //int是=位置; 这行代码对我有用。
@Override
public void onBindViewHolder(@NonNull @org.jetbrains.annotations.NotNull QuizAdapter.MyViewHolder holder, int position)
final QuizModel tmodel = modellist.get(position);
holder.tv_question.setText(tmodel.getQuestion());
holder.tv_option_One.setText(tmodel.getOptionA());
holder.tv_option_two.setText(tmodel.getOptionB());
holder.tv_option_three.setText(tmodel.getOptionC());
holder.tv_option_four.setText(tmodel.getOptionD());
holder.tv_option_five.setText(tmodel.getOptionE());
holder.ll.setOnClickListener(new View.OnClickListener()
@Override
public void onClick(View v)
int is = position;
Toast.makeText(v.getContext(), "count" + is, Toast.LENGTH_LONG).show();
Intent i = new Intent(context, DetailsQuizActivity.class);
i.putExtra("question", tmodel.getQuestion());
i.putExtra("option_a", tmodel.getOptionA());
i.putExtra("option_b", tmodel.getOptionB());
i.putExtra("option_c", tmodel.getOptionC());
i.putExtra("option_d", tmodel.getOptionD());
i.putExtra("option_e", tmodel.getOptionE());
i.putExtra("answer", tmodel.getOptionAnswer());
i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(i);
);
【讨论】:
【参考方案4】:你可以在下面使用
mRecyclerview.getChildAt(pos);
【讨论】:
【参考方案5】:如果你们每次尝试获取具有任何 int 位置的视图时都为 null,请尝试向适配器添加一个新的构造函数参数,例如:
class RecyclerViewTableroAdapter(
private val fichas: Array<MFicha?>,
private val activity: View.OnClickListener,
private val indicesGanadores:MutableList<Int>
) : RecyclerView.Adapter<RecyclerViewTableroAdapter.ViewHolder>()
//CODE
如果我的游戏获胜,我添加了 indicesGanadores 来为我的卡片视图背景着色。
override fun onBindViewHolder(holder: ViewHolder, position: Int)
//CODE
if(indicesGanadores.contains(position))
holder.cardViewFicha.setCardBackgroundColor((activity as MainActivity).resources.getColor(R.color.DarkGreen))
//MORE CODE
如果我不必为背景着色,我只需发送一个空的可变列表,如下所示:
binding.recyclerViewMain.adapter = RecyclerViewTableroAdapter(fichasTablero, this@MainActivity, mutableListOf<Int>())
编码愉快!...
【讨论】:
【参考方案6】:要从 recyclerView 的特定位置获取视图,您需要在将返回 RecyclerView.ViewHolder 的 recyclerView 对象上调用 findViewHolderForAdapterPosition(position)
从返回的带有 itemView 的 RecyclerView.ViewHolder 对象中,您可以访问该特定位置的所有视图
RecyclerView.ViewHolder rv_view = recyclerView.findViewHolderForAdapterPosition(position);
ImageView iv_wish = rv_view.itemView.findViewById(R.id.iv_item);
【讨论】:
【参考方案7】:从回收器视图列表中获取特定视图或在回收器视图的编辑文本中显示错误。
private void popupErrorMessageAtPosition(int itemPosition)
RecyclerView.ViewHolder viewHolder = recyclerView.findViewHolderForAdapterPosition(itemPosition);
View view = viewHolder.itemView;
EditText etDesc = (EditText) view.findViewById(R.id.et_description);
etDesc.setError("Error message here !");
【讨论】:
【参考方案8】:您可以简单地使用回收器视图的“findViewHolderForAdapterPosition”方法,然后您将获得一个 viewHolder 对象,然后将该视图转换为您的适配器视图,这样您就可以直接访问您的视图的视图
以下是kotlin的示例代码
val viewHolder = recyclerView.findViewHolderForAdapterPosition(position)
val textview=(viewHolder as YourViewHolder).yourTextView
【讨论】:
【参考方案9】:请记住,您必须等到适配器将添加到列表中,然后您才能尝试按位置获取视图
final int i = 0;
recyclerView.setAdapter(adapter);
recyclerView.post(new Runnable()
@Override
public void run()
View view = recyclerView.getLayoutManager().findViewByPosition(i);
);
【讨论】:
如果您添加多个项目,这并不总是有效(即:如果您有一个添加它们的按钮并快速按下它)。您应该延迟添加可运行文件。【参考方案10】:您可以制作 ViewHolder 的 ArrayList :
ArrayList<MyViewHolder> myViewHolders = new ArrayList<>();
ArrayList<MyViewHolder> myViewHolders2 = new ArrayList<>();
并且,所有的 ViewHolder(s) 都存储在列表中,例如:
@Override
public void onBindViewHolder(@NonNull final MyViewHolder holder, final int position)
final String str = arrayList.get(position);
myViewHolders.add(position,holder);
并根据您的要求在 ArrayList 中添加/删除其他 ViewHolder。
【讨论】:
感谢您,我花了 2 天的时间努力寻找可回收的支架。 简单的方法。谢谢【参考方案11】:您可以使用以下方法获取 recyclerview 上特定位置的视图
int position = 2;
RecyclerView.ViewHolder viewHolder = recyclerview.findViewHolderForItemId(position);
View view = viewHolder.itemView;
ImageView imageView = (ImageView)view.findViewById(R.id.imageView);
【讨论】:
如果我这样做,那么这个错误会出现在 itemView' on a null object reference。我该怎么办?【参考方案12】:你也可以这样做,当你想在点击一个 recyclerview 位置项后修改视图时,这会有所帮助
@Override
public void onClick(View view, int position)
View v = rv_notifications.getChildViewHolder(view).itemView;
TextView content = v.findViewById(R.id.tv_content);
content.setText("Helloo");
【讨论】:
【参考方案13】:这就是你要找的东西。
我也有这个问题。和你一样,答案很难找到。但是有一种简单的方法可以从特定位置获取 ViewHolder(您可能会在适配器中做很多事情)。
myRecyclerView.findViewHolderForAdapterPosition(pos);
注意:如果视图已被回收,这将返回 null
。感谢 Michael 迅速发现我的重要遗漏。
【讨论】:
这正是我在 LayoutManager 中寻找的内容 这可行,但是,如果您尝试引用的ViewHolder
被“回收”,它将返回 null
。
@Michael,好收获!这是对所有人的警告,首先测试 null (是的,就像 Android 中的其他所有内容一样)!
我们可以从适配器内部执行此操作吗?
@Mauker :您可以在任何可以访问 RecyclerView 实例的地方使用它。【参考方案14】:
如果您想要 View
,请确保访问 ViewHolder 的 itemView
属性,如下所示:myRecyclerView.findViewHolderForAdapterPosition(pos).itemView;
【讨论】:
【参考方案15】:我想您正在使用LinearLayoutManager
来显示列表。它有一个很好的方法叫做findViewByPosition
找到代表给定适配器位置的视图。
您只需要您感兴趣的项目的适配器位置。
edit:正如 Paul Woitaschek 在 cmets 中所指出的,findViewByPosition
是 LayoutManager
的一种方法,因此它适用于所有 LayoutManager(即 StaggeredGridLayoutManager
等)
【讨论】:
这实际上是LayoutManager
的一个方法,所以它适用于所有 LayoutManager 子类。
看了源码。当不在预布局中时,LinearLayoutManger 通过支持数组上的 directAccess 进行查找,而 recyclerView 方法总是进行遍历。所以这种方法可能更有效。
嘿@stanO 你能帮我解决这个问题吗***.com/questions/49952965/…
Returns: View--表示给定位置的子视图,如果位置没有布局则为null...recyclerview --> 大多数东西都返回null
请记住,您必须等到适配器将添加到列表中,有关详细信息,请参阅下面的答案以上是关于RecyclerView - 获取特定位置的视图的主要内容,如果未能解决你的问题,请参考以下文章
获取在 RecyclerView 中单击的特定项目的 ID [重复]
RecyclerView - 如何平滑滚动到嵌套滚动视图内某个位置的项目顶部?