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

时间:2018-03-14 09:41:51

标签: android firebase firebase-realtime-database android-recyclerview firebaseui

我有一个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));
            }
        }
    }
});

请帮忙。

1 个答案:

答案 0 :(得分:0)

如评论中所述,此请求不应在RecyclerView.Adapter中完成,并且<{1}}中绝对不是

你应该将它抽象为一个单独的类,onBindViewHolder或框架自己的Presenter,这就是说:

一旦提出请求并且新数据进入,您应该更新适配器的数据ViewModel并通知这些更改。

报告点击(最终会触发请求)应该通过您可以拥有model的界面来完成:

Adapter

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

所以你的适配器

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

然后您的适配器还应该在适配器的数据上有一个更新项,并通知此更改。由于您的模型已更改,因此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())); } 会正确更新UI。

当然,您必须实现这些UI更改(数据下载/更新/等)

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