如何使用 Firebase 和 RecyclerView 管理多个下载

How to manage multiple download with Firebase and RecyclerView

我有 RecyclerViewFirebaseRecyclerAdapter 物品。每个项目都引用了 Firebase 存储中的一些文件。每个项目都有下载按钮。我使用 storageReference.getFile(location) 从 firebase 存储下载文件。 当我点击下载按钮并等待下载完成时,选中的图标(绿色图标)在特定位置正确显示。

我的问题是当我同时下载多个项目时(当我点击多个项目时),绿色图标没有显示在每个项目的适当位置。

如何同时管理每个项目的多个下载,以便在特定项目下载完成后在特定位置显示绿色图标?

下载按钮后面OnBindViewHolder中的代码。

holder.imgDocumentChecker.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(final View view) {
        holder.imgDocumentChecker.setVisibility(View.GONE);
        holder.txtProgress.setVisibility(View.VISIBLE);
        holder.progressBar.setVisibility(View.VISIBLE);
        final Document item = getItem(holder.getAdapterPosition());
        final File parent = fileHelper.parentFolderGenerator(item);
        counter = 0;
        if (item.getFirebaseStorageFiles() != null && !item.getFirebaseStorageFiles().isEmpty()) {
            if (parent.listFiles() == null) {
                final String[] files = item.getFirebaseStorageFiles().split("____");
                if (files != null && files.length > 0) {
                    for (int i = 0; i < files.length; i++) {
                        String fileName = files[i].substring(files[i].lastIndexOf("/") + 1);
                        String storageFileUrl = Constants.ROOT_FIREBASE_STORAGE_URL + item.getDocFolder() + File.separator + files[i];
                        if (!parent.exists()) {
                            parent.mkdirs();
                        }

                        File localFile = new File(parent, fileName);
                        StorageReference storageReference = storage.getReferenceFromUrl(storageFileUrl);
                        storageReference.getFile(localFile)
                          .addOnSuccessListener(new OnSuccessListener<FileDownloadTask.TaskSnapshot>() {
                            @Override
                            public void onSuccess(FileDownloadTask.TaskSnapshot taskSnapshot) {
                                counter++;
                                int valeur = (100 / files.length) * counter;
                                holder.progressBar.setProgress(valeur);
                                holder.txtProgress.setText(valeur + "");
                                Log.d("progressvalues:",  "valeur:"+valeur+"|counter:"+counter+"|lengh:"+files.length);

                                if (counter == files.length) {
                                    holder.txtProgress.setVisibility(View.GONE);
                                    holder.progressBar.setVisibility(View.GONE);
                                    holder.imgDocumentChecker.setVisibility(View.VISIBLE);
                                    holder.imgDocumentChecker.setImageResource(R.drawable.done_ic);
                                }
                            }
                        }).addOnProgressListener(new OnProgressListener<FileDownloadTask.TaskSnapshot>() {
                            @Override
                            public void onProgress(FileDownloadTask.TaskSnapshot taskSnapshot) {
                                double progress = (100.0 * taskSnapshot.getBytesTransferred()) / taskSnapshot.getTotalByteCount();
                            }
                        }).addOnFailureListener(new OnFailureListener() {
                            @Override
                            public void onFailure(@NonNull Exception e) {
                                holder.txtProgress.setVisibility(View.GONE);
                                holder.progressBar.setVisibility(View.GONE);
                            }
                        });
                    }
                }
            } else {
                startDocumentContentActivity(item, getRef(position));
            }
        }
    }
});

请帮忙。

如评论中所述,此请求不应该在 RecyclerView.Adapter 中完成,并且 绝对不应该onBindViewHolder 中完成。

您应该将其抽象为单独的 class、Presenter 或框架自己的 ViewModel,也就是说:

发出该请求并收到新数据后,您应该更新适配器的数据 model 并通知这些更改。

报告点击(最终会触发请求)应该通过一个界面来完成,您可以在该界面上使用 Adapter:

public interface Listener {
  void onItemClicked(YourModelPojo item);
}

你的 Activity/Fragment/Presenter 无论你想做什么抽象都可以实现这个 Listener 接口。

所以你的适配器

holder.imgDocumentChecker.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(final View view) {
        // Update the UI the request is being made and report to listener
        listener.onItemClicked(data.get(getAdapterPosition()));
}

那么你的适配器也应该有一个关于适配器数据的更新项并通知这个变化。由于您的模型已更改,因此您的 bindViewHolder 会正确更新 UI。

当然,您必须实施这些 UI 更改(数据 downloading/updated/etc。)


我写了一些你可以依赖的文章(开源代码):