diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml
index 99c68eb9b..73870258b 100644
--- a/app/src/main/AndroidManifest.xml
+++ b/app/src/main/AndroidManifest.xml
@@ -42,6 +42,17 @@
android:label="@string/title_activity_settings"
android:parentActivityName=".ui.activity.MainActivity" >
+
+
+
+
+
+
+
+
diff --git a/app/src/main/java/eu/kanade/mangafeed/data/helpers/DatabaseHelper.java b/app/src/main/java/eu/kanade/mangafeed/data/helpers/DatabaseHelper.java
index beb7ad8f5..76ecfadea 100644
--- a/app/src/main/java/eu/kanade/mangafeed/data/helpers/DatabaseHelper.java
+++ b/app/src/main/java/eu/kanade/mangafeed/data/helpers/DatabaseHelper.java
@@ -103,6 +103,11 @@ public class DatabaseHelper implements MangaManager, ChapterManager {
return mMangaManager.getMangasWithUnread();
}
+ @Override
+ public Observable> getFavoriteMangas() {
+ return mMangaManager.getFavoriteMangas();
+ }
+
@Override
public Observable> getManga(String url) {
return mMangaManager.getManga(url);
diff --git a/app/src/main/java/eu/kanade/mangafeed/data/helpers/SourceManager.java b/app/src/main/java/eu/kanade/mangafeed/data/helpers/SourceManager.java
index 0444cc167..25dbc3a0a 100644
--- a/app/src/main/java/eu/kanade/mangafeed/data/helpers/SourceManager.java
+++ b/app/src/main/java/eu/kanade/mangafeed/data/helpers/SourceManager.java
@@ -6,7 +6,6 @@ import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
-import eu.kanade.mangafeed.data.caches.CacheManager;
import eu.kanade.mangafeed.sources.Batoto;
import eu.kanade.mangafeed.sources.MangaHere;
import eu.kanade.mangafeed.sources.base.Source;
@@ -17,8 +16,6 @@ public class SourceManager {
public static final int MANGAHERE = 2;
private HashMap mSourcesMap;
- private NetworkHelper mNetworkHelper;
- private CacheManager mCacheManager;
private Context context;
public SourceManager(Context context) {
diff --git a/app/src/main/java/eu/kanade/mangafeed/data/managers/MangaManager.java b/app/src/main/java/eu/kanade/mangafeed/data/managers/MangaManager.java
index 1ff99c231..6aa717da0 100644
--- a/app/src/main/java/eu/kanade/mangafeed/data/managers/MangaManager.java
+++ b/app/src/main/java/eu/kanade/mangafeed/data/managers/MangaManager.java
@@ -16,6 +16,8 @@ public interface MangaManager {
Observable> getMangasWithUnread();
+ Observable> getFavoriteMangas();
+
Observable> getManga(String url);
Observable> getManga(long id);
diff --git a/app/src/main/java/eu/kanade/mangafeed/data/managers/MangaManagerImpl.java b/app/src/main/java/eu/kanade/mangafeed/data/managers/MangaManagerImpl.java
index 5d546c304..a4ae052fd 100644
--- a/app/src/main/java/eu/kanade/mangafeed/data/managers/MangaManagerImpl.java
+++ b/app/src/main/java/eu/kanade/mangafeed/data/managers/MangaManagerImpl.java
@@ -55,6 +55,19 @@ public class MangaManagerImpl extends BaseManager implements MangaManager {
.createObservable();
}
+ @Override
+ public Observable> getFavoriteMangas() {
+ return db.get()
+ .listOfObjects(Manga.class)
+ .withQuery(Query.builder()
+ .table(MangasTable.TABLE)
+ .where(MangasTable.COLUMN_FAVORITE + "=?")
+ .whereArgs(1)
+ .build())
+ .prepare()
+ .createObservable();
+ }
+
public Observable> getManga(String url) {
return db.get()
.listOfObjects(Manga.class)
diff --git a/app/src/main/java/eu/kanade/mangafeed/data/services/LibraryUpdateService.java b/app/src/main/java/eu/kanade/mangafeed/data/services/LibraryUpdateService.java
new file mode 100644
index 000000000..a692e32f5
--- /dev/null
+++ b/app/src/main/java/eu/kanade/mangafeed/data/services/LibraryUpdateService.java
@@ -0,0 +1,168 @@
+package eu.kanade.mangafeed.data.services;
+
+import android.app.Service;
+import android.content.BroadcastReceiver;
+import android.content.Context;
+import android.content.Intent;
+import android.os.IBinder;
+
+import java.util.ArrayList;
+import java.util.List;
+import java.util.concurrent.atomic.AtomicInteger;
+
+import javax.inject.Inject;
+
+import eu.kanade.mangafeed.App;
+import eu.kanade.mangafeed.BuildConfig;
+import eu.kanade.mangafeed.R;
+import eu.kanade.mangafeed.data.helpers.DatabaseHelper;
+import eu.kanade.mangafeed.data.helpers.SourceManager;
+import eu.kanade.mangafeed.data.models.Manga;
+import eu.kanade.mangafeed.util.AndroidComponentUtil;
+import eu.kanade.mangafeed.util.NetworkUtil;
+import eu.kanade.mangafeed.util.NotificationUtil;
+import eu.kanade.mangafeed.util.PostResult;
+import rx.Observable;
+import rx.Subscription;
+import timber.log.Timber;
+
+public class LibraryUpdateService extends Service {
+
+ @Inject DatabaseHelper db;
+ @Inject SourceManager sourceManager;
+
+ private Subscription updateSubscription;
+ private Subscription favoriteMangasSubscription;
+
+ public static final int UPDATE_NOTIFICATION_ID = 1;
+
+ public static Intent getStartIntent(Context context) {
+ return new Intent(context, LibraryUpdateService.class);
+ }
+
+ public static boolean isRunning(Context context) {
+ return AndroidComponentUtil.isServiceRunning(context, LibraryUpdateService.class);
+ }
+
+ @Override
+ public void onCreate() {
+ super.onCreate();
+ App.get(this).getComponent().inject(this);
+ }
+
+ @Override
+ public void onDestroy() {
+ if (updateSubscription != null)
+ updateSubscription.unsubscribe();
+ super.onDestroy();
+ }
+
+ @Override
+ public int onStartCommand(Intent intent, int flags, final int startId) {
+ Timber.i("Starting sync...");
+
+ if (!NetworkUtil.isNetworkConnected(this)) {
+ Timber.i("Sync canceled, connection not available");
+ AndroidComponentUtil.toggleComponent(this, SyncOnConnectionAvailable.class, true);
+ stopSelf(startId);
+ return START_NOT_STICKY;
+ }
+
+ if (favoriteMangasSubscription != null && !favoriteMangasSubscription.isUnsubscribed())
+ favoriteMangasSubscription.unsubscribe();
+
+ favoriteMangasSubscription = db.getFavoriteMangas()
+ .subscribe(mangas -> {
+ // Don't receive further db updates
+ favoriteMangasSubscription.unsubscribe();
+ this.startUpdating(mangas, startId);
+ });
+
+ return START_STICKY;
+ }
+
+ private void startUpdating(final List mangas, final int startId) {
+ if (updateSubscription != null && !updateSubscription.isUnsubscribed())
+ updateSubscription.unsubscribe();
+
+ final AtomicInteger count = new AtomicInteger(0);
+
+ List updates = new ArrayList<>();
+
+ updateSubscription = Observable.from(mangas)
+ .doOnNext(manga -> {
+ NotificationUtil.create(this, UPDATE_NOTIFICATION_ID,
+ getString(R.string.notification_progress, count.incrementAndGet(), mangas.size()),
+ manga.title);
+ })
+ .concatMap(manga -> sourceManager.get(manga.source)
+ .pullChaptersFromNetwork(manga.url)
+ .flatMap(chapters -> db.insertOrRemoveChapters(manga, chapters))
+ .filter(result -> result.getNumberOfRowsInserted() > 0)
+ .flatMap(result -> Observable.just(new MangaUpdate(manga, result)))
+ )
+ .subscribe(update -> {
+ updates.add(update);
+ }, error -> {
+ Timber.e("Error syncing");
+ stopSelf(startId);
+ }, () -> {
+ NotificationUtil.createBigText(this, UPDATE_NOTIFICATION_ID,
+ getString(R.string.notification_completed), getUpdatedMangas(updates));
+ stopSelf(startId);
+ });
+ }
+
+ private String getUpdatedMangas(List updates) {
+ final StringBuilder result = new StringBuilder();
+ if (updates.isEmpty()) {
+ result.append(getString(R.string.notification_no_new_chapters)).append("\n");
+ } else {
+ result.append(getString(R.string.notification_new_chapters));
+
+ for (MangaUpdate update : updates) {
+ result.append("\n").append(update.getManga().title);
+ }
+ }
+
+ return result.toString();
+ }
+
+ @Override
+ public IBinder onBind(Intent intent) {
+ return null;
+ }
+
+ public static class SyncOnConnectionAvailable extends BroadcastReceiver {
+
+ @Override
+ public void onReceive(Context context, Intent intent) {
+ if (NetworkUtil.isNetworkConnected(context)) {
+ if (BuildConfig.DEBUG) {
+ Timber.i("Connection is now available, triggering sync...");
+ }
+ AndroidComponentUtil.toggleComponent(context, this.getClass(), false);
+ context.startService(getStartIntent(context));
+ }
+ }
+ }
+
+ private static class MangaUpdate {
+ private Manga manga;
+ private PostResult result;
+
+ public MangaUpdate(Manga manga, PostResult result) {
+ this.manga = manga;
+ this.result = result;
+ }
+
+ public Manga getManga() {
+ return manga;
+ }
+
+ public PostResult getResult() {
+ return result;
+ }
+ }
+
+}
diff --git a/app/src/main/java/eu/kanade/mangafeed/injection/component/AppComponent.java b/app/src/main/java/eu/kanade/mangafeed/injection/component/AppComponent.java
index 0a93bb755..90a117d18 100644
--- a/app/src/main/java/eu/kanade/mangafeed/injection/component/AppComponent.java
+++ b/app/src/main/java/eu/kanade/mangafeed/injection/component/AppComponent.java
@@ -5,6 +5,7 @@ import android.app.Application;
import javax.inject.Singleton;
import dagger.Component;
+import eu.kanade.mangafeed.data.services.LibraryUpdateService;
import eu.kanade.mangafeed.injection.module.AppModule;
import eu.kanade.mangafeed.injection.module.DataModule;
import eu.kanade.mangafeed.presenter.CataloguePresenter;
@@ -40,6 +41,8 @@ public interface AppComponent {
void inject(Source source);
+ void inject(LibraryUpdateService libraryUpdateService);
+
Application application();
}
diff --git a/app/src/main/java/eu/kanade/mangafeed/ui/fragment/LibraryFragment.java b/app/src/main/java/eu/kanade/mangafeed/ui/fragment/LibraryFragment.java
index 6d6ee81c6..e1191bf94 100644
--- a/app/src/main/java/eu/kanade/mangafeed/ui/fragment/LibraryFragment.java
+++ b/app/src/main/java/eu/kanade/mangafeed/ui/fragment/LibraryFragment.java
@@ -19,6 +19,7 @@ import butterknife.ButterKnife;
import butterknife.OnItemClick;
import eu.kanade.mangafeed.R;
import eu.kanade.mangafeed.data.models.Manga;
+import eu.kanade.mangafeed.data.services.LibraryUpdateService;
import eu.kanade.mangafeed.presenter.LibraryPresenter;
import eu.kanade.mangafeed.ui.activity.MainActivity;
import eu.kanade.mangafeed.ui.activity.MangaDetailActivity;
@@ -68,6 +69,21 @@ public class LibraryFragment extends BaseRxFragment {
initializeSearch(menu);
}
+ @Override
+ public boolean onOptionsItemSelected(MenuItem item) {
+ switch (item.getItemId()) {
+ case R.id.action_refresh:
+ if (!LibraryUpdateService.isRunning(activity)) {
+ Intent intent = LibraryUpdateService.getStartIntent(activity);
+ activity.startService(intent);
+ }
+
+ return true;
+ }
+
+ return super.onOptionsItemSelected(item);
+ }
+
private void initializeSearch(Menu menu) {
final SearchView sv = (SearchView) menu.findItem(R.id.action_search).getActionView();
sv.setOnQueryTextListener(new SearchView.OnQueryTextListener() {
diff --git a/app/src/main/java/eu/kanade/mangafeed/util/AndroidComponentUtil.java b/app/src/main/java/eu/kanade/mangafeed/util/AndroidComponentUtil.java
new file mode 100644
index 000000000..3789160fe
--- /dev/null
+++ b/app/src/main/java/eu/kanade/mangafeed/util/AndroidComponentUtil.java
@@ -0,0 +1,33 @@
+package eu.kanade.mangafeed.util;
+
+import android.app.ActivityManager;
+import android.app.ActivityManager.RunningServiceInfo;
+import android.content.ComponentName;
+import android.content.Context;
+import android.content.pm.PackageManager;
+
+import timber.log.Timber;
+
+public class AndroidComponentUtil {
+
+ public static void toggleComponent(Context context, Class componentClass, boolean enable) {
+ Timber.i((enable ? "Enabling " : "Disabling ") + componentClass.getSimpleName());
+ ComponentName componentName = new ComponentName(context, componentClass);
+ PackageManager pm = context.getPackageManager();
+ pm.setComponentEnabledSetting(componentName,
+ enable ? PackageManager.COMPONENT_ENABLED_STATE_ENABLED :
+ PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
+ PackageManager.DONT_KILL_APP);
+ }
+
+ public static boolean isServiceRunning(Context context, Class serviceClass) {
+ ActivityManager manager =
+ (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
+ for (RunningServiceInfo service : manager.getRunningServices(Integer.MAX_VALUE)) {
+ if (serviceClass.getName().equals(service.service.getClassName())) {
+ return true;
+ }
+ }
+ return false;
+ }
+}
diff --git a/app/src/main/java/eu/kanade/mangafeed/util/NotificationUtil.java b/app/src/main/java/eu/kanade/mangafeed/util/NotificationUtil.java
new file mode 100644
index 000000000..3f20bf729
--- /dev/null
+++ b/app/src/main/java/eu/kanade/mangafeed/util/NotificationUtil.java
@@ -0,0 +1,44 @@
+package eu.kanade.mangafeed.util;
+
+import android.app.NotificationManager;
+import android.content.Context;
+import android.support.v4.app.NotificationCompat;
+
+import eu.kanade.mangafeed.R;
+
+public class NotificationUtil {
+
+ public static void create(Context context, int nId, String title, String body, int iconRes) {
+ NotificationCompat.Builder mBuilder = new NotificationCompat.Builder(context)
+ .setSmallIcon(iconRes == -1 ? R.drawable.ic_action_refresh : iconRes)
+ .setContentTitle(title)
+ .setContentText(body);
+
+
+ NotificationManager mNotificationManager =
+ (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
+
+ mNotificationManager.notify(nId, mBuilder.build());
+ }
+
+ public static void createBigText(Context context, int nId, String title, String body, int iconRes) {
+ NotificationCompat.Builder mBuilder = new NotificationCompat.Builder(context)
+ .setSmallIcon(iconRes == -1 ? R.drawable.ic_action_refresh : iconRes)
+ .setContentTitle(title)
+ .setStyle(new NotificationCompat.BigTextStyle().bigText(body));
+
+ NotificationManager mNotificationManager =
+ (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
+
+ mNotificationManager.notify(nId, mBuilder.build());
+ }
+
+ public static void create(Context context, int nId, String title, String body) {
+ create(context, nId, title, body, -1);
+ }
+
+ public static void createBigText(Context context, int nId, String title, String body) {
+ createBigText(context, nId, title, body, -1);
+ }
+
+}
diff --git a/app/src/main/java/eu/kanade/mangafeed/util/PostResult.java b/app/src/main/java/eu/kanade/mangafeed/util/PostResult.java
index d64c3fd86..f76bacdd1 100644
--- a/app/src/main/java/eu/kanade/mangafeed/util/PostResult.java
+++ b/app/src/main/java/eu/kanade/mangafeed/util/PostResult.java
@@ -19,17 +19,14 @@ public class PostResult {
this.numberOfRowsDeleted = numberOfRowsDeleted;
}
- @Nullable
public Integer getNumberOfRowsUpdated() {
return numberOfRowsUpdated;
}
- @Nullable
public Integer getNumberOfRowsInserted() {
return numberOfRowsInserted;
}
- @Nullable
public Integer getNumberOfRowsDeleted() {
return numberOfRowsDeleted;
}
diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml
index ba03e1378..747235e06 100644
--- a/app/src/main/res/values/strings.xml
+++ b/app/src/main/res/values/strings.xml
@@ -82,4 +82,10 @@
Mark as unread
Selected chapters: %1$d
+ Update progress: %1$d/%2$d
+ Update completed
+ No new chapters found
+ Found new chapters for:
+
+