【问题标题】:Picasso image misplacement issue when using with RecyclerView与 RecyclerView 一起使用时毕加索图像错位问题
【发布时间】:2019-01-08 13:27:44
【问题描述】:

在我的应用程序中,我展示了从网络服务获取的文章。这些文章有标题、图像和其他字段。为了显示图像,我使用Picasso

问题 - 滚动 RecyclerView 时,我看到图像放错了位置。有些项目有重复的图像。我知道是 RecyclerView 重用了 xml 项目布局,因此它会产生问题。但我认为必须有一些解决这个问题的办法。我已经用谷歌搜索了这个并在stackoverflow上找到了一些帖子,但这没有帮助。例如 -

Picasso loads pictures to the wrong imageview in a list adapter

我尝试了以下方法,但遇到了同样的问题-

public class ArticleAdapter extends RecyclerView.Adapter<ArticleAdapter.ArticleViewHolder> {

    protected Activity mActivity;
    protected List<Article> mItems = new ArrayList<>();
    private static OnEntityClickCallback mCallback;

    public ArticleAdapter(Activity pActivity) {
        mActivity = pActivity;
    }

    public void setItemClickCallback(OnEntityClickCallback pCallback) {
        this.mCallback = pCallback;
    }

    @Override
public ArticleViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
    View articleView = LayoutInflater.from(parent.getContext()).inflate(R.layout.layout_article_item, parent, false);
    return new ArticleViewHolder(articleView);
}

    public void addItem(List<Article> moreItems) {
        mItems.addAll(moreItems);
        notifyDataSetChanged();
    }

    public void removeItems(){
        mItems.clear();
        notifyDataSetChanged();
    }


    @Override
    public void onBindViewHolder(ArticleViewHolder holder, int position) {
        holder.bind(mItems.get(position));
    }

    @Override
    public int getItemCount() {
        return mItems.size();
    }


    static class ArticleViewHolder extends RecyclerView.ViewHolder {

        private TextView tvTitle;
        TextView tvAuthor;
        TextView tvPostdate;
        ImageView imgImage;
        RatingBar ratingBar;

        public ArticleViewHolder(View view) {
            super(view);
            this.tvTitle = (TextView) view.findViewById(R.id.tv_title);
            this.tvAuthor = (TextView) view.findViewById(R.id.tv_author);
            this.tvPostdate = (TextView) view.findViewById(R.id.tv_date);
            this.imgImage = (ImageView) view.findViewById(R.id.img_image);
            this.ratingBar = (RatingBar) view.findViewById(R.id.ratingBar);
        }

       public void bind(final Article article) {

        tvTitle.setText(article.getTitle());
        tvPostdate.setText(article.getPostdate());
        tvAuthor.setText(article.getAuthor());

        // Canceling the older request
        Picasso.with(imgImage.getContext()).cancelRequest(imgImage);
         // Creating a new request.
        if (article.getImage() != null) {
                Picasso.with(imgImage.getContext()).load(article.getImageUrl())
                .placeholder(R.drawable.noimage)
                .error(R.drawable.noimage)
                .into(imgImage);
        }

    this.itemView.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            if (mCallback != null)
                mCallback.onEntitySelected(article);
        }
    });
}

    }
}

更新 - 我的布局文件的一部分

 <ImageView
                    android:id="@+id/img_image"
                    android:layout_width="match_parent"
                    android:layout_height="@dimen/cardview_image_thumbnail_height"
                    android:layout_alignParentStart="true"
                    android:layout_alignParentLeft="true"
                    android:scaleType="centerCrop"
                    android:layout_margin="@dimen/cardview_padding_btw_widgets"
                    android:src="@drawable/noimage"
                    />

任何帮助将不胜感激。谢谢。

【问题讨论】:

  • 只有图片放错了还是文字也放错了?
  • @Soham 只有图片放错了位置。
  • 能否请您发布您的适配器类。
  • @Soham 添加了适配器类。请检查。

标签: android android-recyclerview picasso


【解决方案1】:

外部设置drawable,因为你得到的imageview已经附加了一些drawable。 Imageview 不会重新创建,因此 ImageView 中的前一个图像将可见,直到您将其删除。

    if (article.getImage() != null) {
            Picasso.with(imgImage.getContext()).load(article.getImageUrl())
                    .placeholder(R.drawable.noimage)
                    .error(R.drawable.noimage)
                    .into(imgImage);
    }else{
//         imgImage.setImageDrawable(null);
         imgImage.setImageDrawable(R.drawable.some_drawable);
    }

【讨论】:

  • 我在这里没有看到任何变化。你能详细说明一下吗
  • 检查这一行 else{ imgImage.setImageDrawable(null); }。实际上发生的事情是您的图像视图被回收并且它具有上次的图像。因此,如果 imageUrl 为空,您应该删除该图像视图中存在的任何可绘制对象。告诉我这是否有效
  • 好的。但我在 xml 布局中有一个默认图像。如果图像 url 为空,我希望显示该图像。
  • 在这种情况下这样做 else{ imgImage.setImageDrawable(R.drawable.some_drawable); }
  • 实际上你必须在外部设置drawable,因为你得到的imageview已经附加了一些drawable。 imageview 不会重新创建,因此在外部设置之前,xml 布局中的默认图像不会可见
【解决方案2】:

另外,它对我有用:

@Override
public long getItemId(int position) {
    return position;
}

@Override
public int getItemViewType(int position) {
   return position;
}

这些方法是访问列表数据的更简洁的方法。

【讨论】:

    【解决方案3】:

    在 onBindViewHolder 中使用 setTag 它将保存数据并且您的图像不会错位。

    @Override
        public void onBindViewHolder(ArticleViewHolder holder, int position) {
            holder.bind(mItems.get(position));
            holder.imgImage.setTag(mItems.get(position));
        }
    

    也许试试你的方法也许会奏效

    @Override
        public void onBindViewHolder(ArticleViewHolder holder, int position) {
            holder.bind.setTag(mItems.get(position));
        }
    

    【讨论】:

      【解决方案4】:

      您的视图被重复使用。

      致电Picasso.with(imgImage.getContext()).cancelRequest(imgImage); 取消任何现有的ImageView 请求。

      然后您可以在 article.getImage() != null 时加载您的图片,并在您的文章图片为空时提供后备,例如 imgImage.setImageDrawable(R.drawable.some_drawable);

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 2021-11-04
        • 1970-01-01
        • 1970-01-01
        • 2021-02-21
        • 2016-02-02
        • 1970-01-01
        • 2020-11-14
        • 1970-01-01
        相关资源
        最近更新 更多