RecyclerView - 获取特定位置的视图

188

我有一个带有RecyclerViewImageView的活动。我使用RecyclerView水平展示图像列表。当我在RecyclerView中点击图像时,ImageView应该显示图像的更大版本。目前一切正常。

现在活动中有两个额外的ImageButtonsimageButton_leftimageButton_right。当我点击imageButton_left时,ImageView中的图像应该向左旋转,并且RecyclerView中的缩略图也应反映此更改。对于imageButton_right也是如此。

我可以旋转ImageView,但是如何旋转RecyclerView中的缩略图呢?如何获取ViewHolderImageView?

代码:

活动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_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <android.support.v7.widget.RecyclerView
        android:id="@+id/recyclerview"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="10dp" />


    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="10dp"
        android:orientation="vertical">

        <ImageView
            android:id="@+id/original_image"
            android:layout_width="200dp"
            android:layout_height="200dp"
            android:scaleType="fitXY"
            android:src="@drawable/image_not_available_2" />

        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginTop="10dp"
            android:gravity="center_horizontal"
            android:orientation="horizontal">


            <ImageButton
                android:id="@+id/imageButton_left"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_marginRight="20dp"
                android:background="@drawable/rotate_left_icon" />

            <ImageButton
                android:id="@+id/imageButton_right"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                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());
        }
    }


}

1
你应该在viewHolder的构造函数中添加一个属性,用于判断是否需要旋转图像,然后只需更改数据上的该属性并调用notifyDataSetChanged,以使recyclerView重新加载。 - Nanoc
任何答案解决了你的问题吗? - SMBiggs
ViewHolder 应该是静态的,以提高性能。所有来自外部类的变量都应通过构造函数传递。 - AppiDevo
作为对@Scott答案的补充,点击这里 - mut tony
16个回答

319

这就是你要找的答案

我也遇到了这个问题,和你一样,很难找到答案。但是有一种简单的方式可以从特定位置获取ViewHolder(在适配器中可能经常使用)。

myRecyclerView.findViewHolderForAdapterPosition(pos);

注意: 如果视图已经被回收,这将返回null。感谢Michael迅速捕捉到这个遗漏。此外,如果用户正在快速滚动屏幕,即使在获得此ViewHolder之后,视图也可能被回收,导致各种问题。因此,在使用此技术时要小心,特别是在进行相对较慢的工作(如更改视图外观)时。


3
这正是我在LayoutManager中寻找的。 - Ed Lee
18
如果您试图引用已经“回收”的ViewHolder,它将返回null。即使这样,此方法仍然有效。 - Michael
8
@Michael,干得好!这是对所有人的警示,在进行操作前先测试是否为空(就像在Android中的其他操作一样)! - SMBiggs
2
我们可以从适配器内部完成这个操作吗? - Mauker
2
@Mauker:您可以从任何可以访问RecyclerView实例的地方使用它。 - SMBiggs
显示剩余5条评论

182

我想你正在使用LinearLayoutManager来显示列表。它有一个很好的方法叫做findViewByPosition

查找表示给定适配器位置的视图。

你只需要知道你感兴趣的项的适配器位置即可。

编辑:如评论中所指出,findViewByPositionLayoutManager的一个方法,因此它可以与所有的LayoutManager(例如StaggeredGridLayoutManager等)一起使用。


11
这实际上是 LayoutManager 的一种方法,因此可以适用于所有 LayoutManager 的子类。 - Paul Woitaschek
查看了源代码。当不在预布局状态下时,LinearLayoutManger通过直接访问后备数组进行查找,而recyclerView方法始终进行遍历。因此,这种方法可能更有效。 - NameSpace
嘿@stanO,你能帮我解决这个问题吗?https://dev59.com/oqrka4cB1Zd3GeqPjMuw?noredirect=1#comment87861669_49952965 - user9550968
4
返回:View--表示给定位置的子视图,如果该位置没有布局,则返回null... RecyclerView --> 大多数情况下返回null - me_
1
请记住,您必须等待适配器添加到列表中,有关详细信息请查看我的下面的回答。 - Konstantin Konopko
@stanO,请问你能看到我的问题吗?https://stackoverflow.com/questions/68554233/how-to-get-view-with-position-in-recyclerview? - Nima Khalili

34
如果您想要获得 View,确保访问 ViewHolder 的 itemView 属性,如下所示:myRecyclerView.findViewHolderForAdapterPosition(pos).itemView;

16

您可以同时使用 recyclerViewInstance.findViewHolderForAdapterPosition(adapterPosition)recyclerViewInstance.findViewHolderForLayoutPosition(layoutPosition) 。请确保 RecyclerView 视图使用两种类型的位置。

适配器位置:适配器中项目的位置。这是从适配器的角度来看的位置。

布局位置:最新布局计算中项目的位置。这是从 LayoutManager 的角度来看的位置。对于 findViewHolderForAdapterPosition(adapterPosition),应使用 getAdapterPosition(),而对于 findViewHolderForLayoutPosition(layoutPosition),应使用 getLayoutPosition()

在 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();    
                }    
            }    
        });    
    }    
}       

} 第一个元素被选择 第二个元素被选择并且之前选中的元素变成未选中状态 第五个元素被选择并且之前选中的元素变成未选中状态


我以为这个讨论已经结束了!感谢您进一步澄清这个问题。 - SMBiggs
2
当我尝试更改前一个项目的颜色时,else块中出现了空指针异常,因此我添加了空值检查。现在如果出现NullPointer异常,它就永远不会进入该块,因此前一个项目的颜色永远不会被更改。现在我的Recycler视图有多种颜色的项目。我该如何解决这个问题? - Abhilash Reddy

12
你可以简单地使用Recycler View的"findViewHolderForAdapterPosition"方法,从中获取一个viewHolder对象,然后将该viewholder转换为你的适配器viewholder类型,这样你就可以直接访问你的viewholder视图了。
以下是Kotlin示例代码。
 val viewHolder = recyclerView.findViewHolderForAdapterPosition(position)
 val textview=(viewHolder as YourViewHolder).yourTextView

9
你可以创建一个 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。


谢谢,我为寻找可回收的支架已经苦苦挣扎了两天。 - nimi0112
简单的方法。谢谢。 - ashishdhiman2007

8
要从 recyclerView 的特定位置获取 View,需要在 recyclerView 对象上调用 findViewHolderForAdapterPosition(position) 方法,该方法会返回 RecyclerView.ViewHolder。
通过返回的 RecyclerView.ViewHolder 对象的 itemView,您可以访问该特定位置上的所有 View。
RecyclerView.ViewHolder rv_view = recyclerView.findViewHolderForAdapterPosition(position);

ImageView iv_wish = rv_view.itemView.findViewById(R.id.iv_item);

6
您可以使用以下方法获取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}。我该怎么办? - Ali Raza

6

记住,您必须等待适配器被添加到列表中,然后才能尝试按位置获取视图。

final int i = 0;
recyclerView.setAdapter(adapter);
recyclerView.post(new Runnable() {
    @Override
    public void run() {
        View view = recyclerView.getLayoutManager().findViewByPosition(i);
    }
});

2
如果您添加多个项目(即:如果您有一个按钮添加它们并快速按下它),则这并不总是有效。您应该添加带延迟的可运行项。 - Alex Burdusel

2
从Recycler View列表中获取特定视图或在Recycler View的EditText中显示错误。最初的回答是:
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 !");
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接