在 RecyclerView 中滚动时加载位图无法正常工作

Loading Bitmaps doesn't work correct, while scrolling in RecyclerView

我正在我的音乐应用程序中加载专辑插图。因为不能在Main-Thread上加载,所以用的是Threads,搞得一团糟!

有时图片加载的尺寸不正确或显示为棕色方块。如果我快速滚动,就会出现这些问题。如果我滚动缓慢,它会起作用!

我的MusicStore的重要方法-Class:

public Bitmap getAlbumArtwork(long AlbumID, int Height, int Width) {
    ParcelFileDescriptor pfd;

    Bitmap bCover = null;
    BitmapFactory.Options bOptions = new BitmapFactory.Options();
    bOptions.inJustDecodeBounds = true;

    try {
        Uri ArtworkUri = Uri.parse("content://media/external/audio/albumart");
        Uri uri = ContentUris.withAppendedId(ArtworkUri, AlbumID);
        pfd = mContext.getContentResolver().openFileDescriptor(uri, "r");

        if (pfd != null) {
            BitmapFactory.decodeFileDescriptor(pfd.getFileDescriptor(), null, bOptions);

            bOptions.inSampleSize = calculateInSampleSize(bOptions, Width, Height);

            bOptions.inJustDecodeBounds = false;

            bCover = BitmapFactory.decodeFileDescriptor(pfd.getFileDescriptor(), null, bOptions);

            pfd.close();
        }
    }
    catch (IOException ioe) {
        BitmapFactory.decodeResource(mContext.getResources(), R.drawable.standardartwork, bOptions);

        bOptions.inSampleSize = calculateInSampleSize(bOptions, Width, Height);

        bOptions.inJustDecodeBounds = false;

        bCover = BitmapFactory.decodeResource(mContext.getResources(), R.drawable.standardartwork, bOptions);
    }

    return bCover;
}

public void setAlbumArtwork(final long AlbumID, final ImageView ArtworkView) {
    Thread thArtwork = new Thread(new Runnable() {
        @Override
        public void run() {
            final Bitmap bArtwork = getAlbumArtwork(AlbumID, ArtworkView.getHeight() / 2, ArtworkView.getWidth() / 2);

            handler.postDelayed(new Runnable() {
                @Override
                public void run() {
                    ArtworkView.setImageBitmap(bArtwork);

                    threadList.remove(Thread.currentThread());
                }
            }, 50);
        }
    });

    threadList.add(thArtwork);

    thArtwork.start();
}

private int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) {
    final int height = options.outHeight;
    final int width = options.outWidth;

    int size = 1;

    if (height > reqHeight && width > reqWidth) {
        final int halfHeight = height / 2;
        final int halfWidth = width / 2;

        while ((halfHeight / size) > reqHeight && (halfWidth / size) > reqWidth) {
            size *= 2;
        }
    }

    return size;
}

还有我的 RecyclerViewAdapter:

public class SongRecyclerViewAdapter extends RecyclerView.Adapter<SongRecyclerViewAdapter.Holder> {
    private Context mContext;
    private Song[] sSongs;

    private MusicStore musicStore;

    public SongRecyclerViewAdapter(Context context, Song[] songs) {
        mContext = context;
        sSongs = songs;

        musicStore = new MusicStore(mContext);
    }

    @Override
    public Holder onCreateViewHolder(ViewGroup parent, int viewType) {
        View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.adapter_songview, parent, false);

        Holder holder = new Holder(view);

        return holder;
    }

    @Override
    public void onBindViewHolder(Holder holder, int position) {
        musicStore.setAlbumArtwork(sSongs[position].getAlbumID(), holder.imvSong);

        holder.txvSongTitle.setText(sSongs[position].getTitle());
        holder.txvSongInfo.setText(sSongs[position].getArtists());
    }

    @Override
    public void onViewDetachedFromWindow(Holder holder) {

    }

    @Override
    public int getItemCount() {
        return sSongs != null ? sSongs.length : 0;
    }

    public class Holder extends RecyclerView.ViewHolder {
        LinearLayout linearLayout;
        ImageView imvSong;
        TextView txvSongTitle;
        TextView txvSongInfo;

        public Holder(View layout) {
            super(layout);

            linearLayout = (LinearLayout) layout;

            imvSong = (ImageView) layout.findViewById(R.id.imvSong);
            txvSongTitle = (TextView) layout.findViewById(R.id.adap_txvSongtitle);
            txvSongInfo = (TextView) layout.findViewById(R.id.adap_txvSongInfo);
        }
    }
}

我绝对愿意接受任何其他想要正确加载位图的想法!

如果您愿意,可以尝试一下: https://play.google.com/store/apps/details?id=at.guger.musixs

谢谢!

http://square.github.io/picasso/

设置和文档非常简单。您也可以传入 Uri,Picasso 会解析它。

例子

Picasso.with(this).load(uri).into(imageView);

如果您还可以指定宽度和高度、占位符等内容。

.resize(width, height).placeholder(placeholderImg)