Browse Source

Improvements for downloads fragment

inorichi 9 năm trước cách đây
mục cha
commit
4aee1ca8a3

+ 6 - 4
app/src/main/java/eu/kanade/mangafeed/data/download/DownloadManager.java

@@ -84,7 +84,7 @@ public class DownloadManager {
                     if (finished) {
                         DownloadService.stop(context);
                     }
-                }, e -> Timber.e(e.fillInStackTrace(), e.getMessage()));
+                }, e -> Timber.e(e.getCause(), e.getMessage()));
 
         isRunning = true;
     }
@@ -238,10 +238,12 @@ public class DownloadManager {
                     try {
                         DiskUtils.saveBufferedSourceToDirectory(resp.body().source(), directory, filename);
                     } catch (Exception e) {
+                        Timber.e(e.getCause(), e.getMessage());
                         return Observable.error(e);
                     }
                     return Observable.just(page);
-                });
+                })
+                .retry(2);
     }
 
     // Public method to get the image from the filesystem. It does NOT provide any way to download the image
@@ -310,7 +312,7 @@ public class DownloadManager {
                 pages = gson.fromJson(reader, collectionType);
             }
         } catch (FileNotFoundException e) {
-            Timber.e(e.fillInStackTrace(), e.getMessage());
+            Timber.e(e.getCause(), e.getMessage());
         } finally {
             if (reader != null) try { reader.close(); } catch (IOException e) { /* Do nothing */ }
         }
@@ -333,7 +335,7 @@ public class DownloadManager {
             out.write(gson.toJson(pages).getBytes());
             out.flush();
         } catch (IOException e) {
-            Timber.e(e.fillInStackTrace(), e.getMessage());
+            Timber.e(e.getCause(), e.getMessage());
         } finally {
             if (out != null) try { out.close(); } catch (IOException e) { /* Do nothing */ }
         }

+ 38 - 5
app/src/main/java/eu/kanade/mangafeed/ui/download/DownloadAdapter.java

@@ -1,17 +1,50 @@
 package eu.kanade.mangafeed.ui.download;
 
 import android.content.Context;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.view.ViewGroup;
 
+import java.util.ArrayList;
+import java.util.List;
+
+import eu.davidea.flexibleadapter.FlexibleAdapter;
+import eu.kanade.mangafeed.R;
 import eu.kanade.mangafeed.data.download.model.Download;
-import uk.co.ribot.easyadapter.EasyRecyclerAdapter;
 
-public class DownloadAdapter extends EasyRecyclerAdapter<Download> {
+public class DownloadAdapter extends FlexibleAdapter<DownloadHolder, Download> {
+
+    private Context context;
 
     public DownloadAdapter(Context context) {
-        super(context, DownloadHolder.class);
+        this.context = context;
+        mItems = new ArrayList<>();
+        setHasStableIds(true);
+    }
+
+    @Override
+    public DownloadHolder onCreateViewHolder(ViewGroup parent, int viewType) {
+        View v = LayoutInflater.from(context).inflate(R.layout.item_download, parent, false);
+        return new DownloadHolder(v);
+    }
+
+    @Override
+    public void onBindViewHolder(DownloadHolder holder, int position) {
+        final Download download = getItem(position);
+        holder.onSetValues(download);
+    }
+
+    @Override
+    public long getItemId(int position) {
+        return getItem(position).chapter.id;
     }
 
-    public int getPositionForItem(Download item) {
-        return getItems() != null && getItems().size() > 0 ? getItems().indexOf(item) : -1;
+    public void setItems(List<Download> downloads) {
+        mItems = downloads;
+        notifyDataSetChanged();
     }
+
+    @Override
+    public void updateDataSet(String param) {}
+
 }

+ 17 - 28
app/src/main/java/eu/kanade/mangafeed/ui/download/DownloadFragment.java

@@ -1,13 +1,12 @@
 package eu.kanade.mangafeed.ui.download;
 
 import android.os.Bundle;
+import android.support.annotation.Nullable;
 import android.support.v7.widget.LinearLayoutManager;
 import android.support.v7.widget.RecyclerView;
 import android.view.LayoutInflater;
 import android.view.View;
 import android.view.ViewGroup;
-import android.widget.ProgressBar;
-import android.widget.TextView;
 
 import java.util.List;
 
@@ -21,8 +20,7 @@ import nucleus.factory.RequiresPresenter;
 @RequiresPresenter(DownloadPresenter.class)
 public class DownloadFragment extends BaseRxFragment<DownloadPresenter> {
 
-    @Bind(R.id.download_list) RecyclerView downloadList;
-    private LinearLayoutManager downloadListLayout;
+    @Bind(R.id.download_list) RecyclerView recyclerView;
     private DownloadAdapter adapter;
 
     public static DownloadFragment newInstance() {
@@ -38,8 +36,8 @@ public class DownloadFragment extends BaseRxFragment<DownloadPresenter> {
 
         setToolbarTitle(R.string.label_download_queue);
 
-        downloadListLayout = new LinearLayoutManager(getActivity());
-        downloadList.setLayoutManager(downloadListLayout);
+        recyclerView.setLayoutManager(new LinearLayoutManager(getActivity()));
+        recyclerView.setHasFixedSize(true);
         createAdapter();
 
         return view;
@@ -47,39 +45,30 @@ public class DownloadFragment extends BaseRxFragment<DownloadPresenter> {
 
     private void createAdapter() {
         adapter = new DownloadAdapter(getActivity());
-        downloadList.setAdapter(adapter);
+        recyclerView.setAdapter(adapter);
     }
 
     public void onNextDownloads(List<Download> downloads) {
         adapter.setItems(downloads);
     }
 
-    private View getDownloadRow(Download download) {
-        int first = downloadListLayout.findFirstVisibleItemPosition();
-        int last = downloadListLayout.findLastVisibleItemPosition();
-        int pos = adapter.getPositionForItem(download);
-
-        if (pos != -1 && pos >= first && pos <= last) {
-            return downloadListLayout.getChildAt(pos - first);
-        }
-        return null;
-    }
-
     public void updateProgress(Download download) {
-        View row = getDownloadRow(download);
-        if (row != null) {
-            ProgressBar progress = (ProgressBar) row.findViewById(R.id.download_progress);
-            if (progress.getMax() == 1) progress.setMax(download.pages.size() * 100);
-            progress.setProgress(download.totalProgress);
+        DownloadHolder holder = getHolder(download);
+        if (holder != null) {
+            holder.setDownloadProgress(download);
         }
     }
 
     public void updateDownloadedPages(Download download) {
-        View row = getDownloadRow(download);
-        if (row != null) {
-            TextView progress = (TextView) row.findViewById(R.id.download_progress_text);
-            String progressText = download.downloadedImages + "/" + download.pages.size();
-            progress.setText(progressText);
+        DownloadHolder holder = getHolder(download);
+        if (holder != null) {
+            holder.setDownloadedPages(download);
         }
     }
+
+    @Nullable
+    private DownloadHolder getHolder(Download download) {
+        return (DownloadHolder) recyclerView.findViewHolderForItemId(download.chapter.id);
+    }
+
 }

+ 22 - 14
app/src/main/java/eu/kanade/mangafeed/ui/download/DownloadHolder.java

@@ -1,29 +1,27 @@
 package eu.kanade.mangafeed.ui.download;
 
+import android.support.v7.widget.RecyclerView;
 import android.view.View;
 import android.widget.ProgressBar;
 import android.widget.TextView;
 
+import butterknife.Bind;
+import butterknife.ButterKnife;
 import eu.kanade.mangafeed.R;
 import eu.kanade.mangafeed.data.download.model.Download;
-import uk.co.ribot.easyadapter.ItemViewHolder;
-import uk.co.ribot.easyadapter.PositionInfo;
-import uk.co.ribot.easyadapter.annotations.LayoutId;
-import uk.co.ribot.easyadapter.annotations.ViewId;
 
-@LayoutId(R.layout.item_download)
-public class DownloadHolder extends ItemViewHolder<Download> {
+public class DownloadHolder extends RecyclerView.ViewHolder {
 
-    @ViewId(R.id.download_title) TextView downloadTitle;
-    @ViewId(R.id.download_progress) ProgressBar downloadProgress;
-    @ViewId(R.id.download_progress_text) TextView downloadProgressText;
+    @Bind(R.id.download_title) TextView downloadTitle;
+    @Bind(R.id.download_progress) ProgressBar downloadProgress;
+    @Bind(R.id.download_progress_text) TextView downloadProgressText;
 
     public DownloadHolder(View view) {
         super(view);
+        ButterKnife.bind(this, view);
     }
 
-    @Override
-    public void onSetValues(Download download, PositionInfo positionInfo) {
+    public void onSetValues(Download download) {
         downloadTitle.setText(download.chapter.name);
 
         if (download.pages == null) {
@@ -32,10 +30,20 @@ public class DownloadHolder extends ItemViewHolder<Download> {
             downloadProgressText.setText("");
         } else {
             downloadProgress.setMax(download.pages.size() * 100);
-            downloadProgress.setProgress(download.totalProgress);
-            String progressText = download.downloadedImages + "/" + download.pages.size();
-            downloadProgressText.setText(progressText);
+            setDownloadProgress(download);
+            setDownloadedPages(download);
         }
     }
 
+    public void setDownloadedPages(Download download) {
+        String progressText = download.downloadedImages + "/" + download.pages.size();
+        downloadProgressText.setText(progressText);
+    }
+
+    public void setDownloadProgress(Download download) {
+        if (downloadProgress.getMax() == 1)
+            downloadProgress.setMax(download.pages.size() * 100);
+        downloadProgress.setProgress(download.totalProgress);
+    }
+
 }

+ 8 - 50
app/src/main/java/eu/kanade/mangafeed/ui/download/DownloadPresenter.java

@@ -16,7 +16,6 @@ import rx.Observable;
 import rx.Subscription;
 import rx.android.schedulers.AndroidSchedulers;
 import rx.schedulers.Schedulers;
-import rx.subjects.PublishSubject;
 import timber.log.Timber;
 
 public class DownloadPresenter extends BasePresenter<DownloadFragment> {
@@ -25,8 +24,8 @@ public class DownloadPresenter extends BasePresenter<DownloadFragment> {
 
     private DownloadQueue downloadQueue;
     private Subscription statusSubscription;
+    private Subscription pageProgressSubscription;
     private HashMap<Download, Subscription> progressSubscriptions;
-    private HashMap<Download, Subscription> pageStatusSubscriptions;
 
     public final static int GET_DOWNLOAD_QUEUE = 1;
 
@@ -36,7 +35,6 @@ public class DownloadPresenter extends BasePresenter<DownloadFragment> {
 
         downloadQueue = downloadManager.getQueue();
         progressSubscriptions = new HashMap<>();
-        pageStatusSubscriptions = new HashMap<>();
 
         restartableLatestCache(GET_DOWNLOAD_QUEUE,
                 () -> Observable.just(downloadQueue.get()),
@@ -57,6 +55,10 @@ public class DownloadPresenter extends BasePresenter<DownloadFragment> {
                 .subscribe(download -> {
                     processStatus(download, view);
                 }));
+
+        add(pageProgressSubscription = downloadQueue.getProgressObservable()
+                .observeOn(AndroidSchedulers.mainThread())
+                .subscribe(view::updateDownloadedPages));
     }
 
     @Override
@@ -69,17 +71,16 @@ public class DownloadPresenter extends BasePresenter<DownloadFragment> {
         switch (download.getStatus()) {
             case Download.DOWNLOADING:
                 observeProgress(download, view);
-                observePagesStatus(download, view);
+                // Initial update of the downloaded pages
+                view.updateDownloadedPages(download);
                 break;
             case Download.DOWNLOADED:
                 unsubscribeProgress(download);
-                unsubscribePagesStatus(download);
                 view.updateProgress(download);
                 view.updateDownloadedPages(download);
                 break;
             case Download.ERROR:
                 unsubscribeProgress(download);
-                unsubscribePagesStatus(download);
                 break;
         }
     }
@@ -104,62 +105,19 @@ public class DownloadPresenter extends BasePresenter<DownloadFragment> {
         progressSubscriptions.put(download, subscription);
     }
 
-    private void observePagesStatus(Download download, DownloadFragment view) {
-        // Initial update of the downloaded pages
-        view.updateDownloadedPages(download);
-
-        PublishSubject<Integer> pageStatusSubject = PublishSubject.create();
-        for (Page page : download.pages) {
-            if (page.getStatus() != Page.READY)
-                page.setStatusSubject(pageStatusSubject);
-        }
-
-        Subscription subscription = pageStatusSubject
-                .filter(status -> status == Page.READY)
-                .observeOn(AndroidSchedulers.mainThread())
-                .subscribe(status -> {
-                    view.updateDownloadedPages(download);
-                });
-
-        // Avoid leaking subscriptions
-        Subscription oldSubscription = pageStatusSubscriptions.remove(download);
-        if (oldSubscription != null) oldSubscription.unsubscribe();
-
-        pageStatusSubscriptions.put(download, subscription);
-    }
-
     private void unsubscribeProgress(Download download) {
         Subscription subscription = progressSubscriptions.remove(download);
         if (subscription != null)
             subscription.unsubscribe();
     }
 
-    private void unsubscribePagesStatus(Download download) {
-        if (download.pages != null) {
-            for (Page page : download.pages)
-                page.setStatusSubject(null);
-        }
-
-        Subscription subscription = pageStatusSubscriptions.remove(download);
-        if (subscription != null)
-            subscription.unsubscribe();
-    }
-
     private void destroySubscriptions() {
-        for (Download download : pageStatusSubscriptions.keySet()) {
-            for (Page page : download.pages)
-                page.setStatusSubject(null);
-        }
-        for (Subscription subscription : pageStatusSubscriptions.values()) {
-            subscription.unsubscribe();
-        }
-        pageStatusSubscriptions.clear();
-
         for (Subscription subscription : progressSubscriptions.values()) {
             subscription.unsubscribe();
         }
         progressSubscriptions.clear();
 
+        remove(pageProgressSubscription);
         remove(statusSubscription);
     }