Add buttons to open next/previous chapter in the reader. Try entire app with hardware acceleration

This commit is contained in:
inorichi 2016-01-06 20:55:52 +01:00
parent 73aa93773c
commit d0089e3f5e
26 changed files with 159 additions and 64 deletions

View file

@ -12,6 +12,7 @@
android:allowBackup="true" android:allowBackup="true"
android:icon="@mipmap/ic_launcher" android:icon="@mipmap/ic_launcher"
android:label="@string/app_name" android:label="@string/app_name"
android:hardwareAccelerated="true"
android:theme="@style/AppTheme" > android:theme="@style/AppTheme" >
<activity <activity
android:name=".ui.main.MainActivity"> android:name=".ui.main.MainActivity">
@ -27,8 +28,7 @@
</activity> </activity>
<activity <activity
android:name=".ui.reader.ReaderActivity" android:name=".ui.reader.ReaderActivity"
android:parentActivityName=".ui.manga.MangaActivity" android:parentActivityName=".ui.manga.MangaActivity">
android:hardwareAccelerated="true">
<meta-data <meta-data
android:name="android.support.PARENT_ACTIVITY" android:name="android.support.PARENT_ACTIVITY"
android:value=".ui.manga.MangaActivity" /> android:value=".ui.manga.MangaActivity" />

View file

@ -9,6 +9,8 @@ import android.os.Bundle;
import android.support.annotation.NonNull; import android.support.annotation.NonNull;
import android.support.v4.content.ContextCompat; import android.support.v4.content.ContextCompat;
import android.support.v7.widget.Toolbar; import android.support.v7.widget.Toolbar;
import android.view.Menu;
import android.view.MenuItem;
import android.view.Surface; import android.view.Surface;
import android.view.View; import android.view.View;
import android.view.WindowManager; import android.view.WindowManager;
@ -111,6 +113,16 @@ public class ReaderActivity extends BaseRxActivity<ReaderPresenter> {
super.onDestroy(); super.onDestroy();
} }
@Override
public boolean onCreateOptionsMenu(Menu menu) {
return readerMenu.onCreateOptionsMenu(menu);
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
return readerMenu.onOptionsItemSelected(item) || super.onOptionsItemSelected(item);
}
@Override @Override
protected void onSaveInstanceState(@NonNull Bundle outState) { protected void onSaveInstanceState(@NonNull Bundle outState) {
Icepick.saveInstanceState(readerMenu, outState); Icepick.saveInstanceState(readerMenu, outState);
@ -131,6 +143,10 @@ public class ReaderActivity extends BaseRxActivity<ReaderPresenter> {
readerMenu.onChapterReady(pages.size(), manga, chapter, currentPage); readerMenu.onChapterReady(pages.size(), manga, chapter, currentPage);
} }
public void onAdjacentChapters(Chapter previous, Chapter next) {
readerMenu.onAdjacentChapters(previous, next);
}
private BaseReader createViewer(Manga manga) { private BaseReader createViewer(Manga manga) {
int mangaViewer = manga.viewer == 0 ? preferences.getDefaultViewer() : manga.viewer; int mangaViewer = manga.viewer == 0 ? preferences.getDefaultViewer() : manga.viewer;
@ -160,6 +176,21 @@ public class ReaderActivity extends BaseRxActivity<ReaderPresenter> {
readerMenu.toggle(); readerMenu.toggle();
} }
public void requestNextChapter() {
getPresenter().setCurrentPage(viewer != null ? viewer.getCurrentPage() : 0);
if (!getPresenter().loadNextChapter()) {
ToastUtil.showShort(this, R.string.no_next_chapter);
}
}
public void requestPreviousChapter() {
getPresenter().setCurrentPage(viewer != null ? viewer.getCurrentPage() : 0);
if (!getPresenter().loadPreviousChapter()) {
ToastUtil.showShort(this, R.string.no_previous_chapter);
}
}
private void initializeSettings() { private void initializeSettings() {
subscriptions.add(preferences.showPageNumber() subscriptions.add(preferences.showPageNumber()
.asObservable() .asObservable()

View file

@ -5,6 +5,8 @@ import android.content.Context;
import android.support.v7.app.AlertDialog; import android.support.v7.app.AlertDialog;
import android.support.v7.widget.Toolbar; import android.support.v7.widget.Toolbar;
import android.view.Gravity; import android.view.Gravity;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View; import android.view.View;
import android.view.WindowManager; import android.view.WindowManager;
import android.view.WindowManager.LayoutParams; import android.view.WindowManager.LayoutParams;
@ -42,6 +44,11 @@ public class ReaderMenu {
@Bind(R.id.reader_extra_settings) ImageButton extraSettings; @Bind(R.id.reader_extra_settings) ImageButton extraSettings;
@Bind(R.id.reader_brightness) ImageButton brightnessSettings; @Bind(R.id.reader_brightness) ImageButton brightnessSettings;
private MenuItem nextChapterBtn;
private MenuItem prevChapterBtn;
private Chapter prevChapter;
private Chapter nextChapter;
private ReaderActivity activity; private ReaderActivity activity;
private PreferencesHelper preferences; private PreferencesHelper preferences;
@ -106,6 +113,25 @@ public class ReaderMenu {
showing = false; showing = false;
} }
public boolean onCreateOptionsMenu(Menu menu) {
activity.getMenuInflater().inflate(R.menu.reader, menu);
nextChapterBtn = menu.findItem(R.id.action_next_chapter);
prevChapterBtn = menu.findItem(R.id.action_previous_chapter);
setAdjacentChaptersVisibility();
return true;
}
public boolean onOptionsItemSelected(MenuItem item) {
if (item == prevChapterBtn) {
activity.requestPreviousChapter();
} else if (item == nextChapterBtn) {
activity.requestNextChapter();
} else {
return false;
}
return true;
}
public void onChapterReady(int numPages, Manga manga, Chapter chapter, int currentPageIndex) { public void onChapterReady(int numPages, Manga manga, Chapter chapter, int currentPageIndex) {
if (manga.viewer == ReaderActivity.RIGHT_TO_LEFT && !inverted) { if (manga.viewer == ReaderActivity.RIGHT_TO_LEFT && !inverted) {
// Invert the seekbar and textview fields for the right to left reader // Invert the seekbar and textview fields for the right to left reader
@ -136,6 +162,17 @@ public class ReaderMenu {
seekBar.setProgress(pageIndex); seekBar.setProgress(pageIndex);
} }
public void onAdjacentChapters(Chapter previous, Chapter next) {
prevChapter = previous;
nextChapter = next;
setAdjacentChaptersVisibility();
}
private void setAdjacentChaptersVisibility() {
if (prevChapterBtn != null) prevChapterBtn.setVisible(prevChapter != null);
if (nextChapterBtn != null) nextChapterBtn.setVisible(nextChapter != null);
}
private void initializeOptions() { private void initializeOptions() {
// Orientation changes // Orientation changes
add(preferences.lockOrientation().asObservable() add(preferences.lockOrientation().asObservable()

View file

@ -2,6 +2,7 @@ package eu.kanade.mangafeed.ui.reader;
import android.os.Bundle; import android.os.Bundle;
import android.support.annotation.NonNull; import android.support.annotation.NonNull;
import android.util.Pair;
import java.io.File; import java.io.File;
import java.util.List; import java.util.List;
@ -53,8 +54,9 @@ public class ReaderPresenter extends BasePresenter<ReaderActivity> {
private static final int GET_PAGE_LIST = 1; private static final int GET_PAGE_LIST = 1;
private static final int GET_PAGE_IMAGES = 2; private static final int GET_PAGE_IMAGES = 2;
private static final int RETRY_IMAGES = 3; private static final int GET_ADJACENT_CHAPTERS = 3;
private static final int PRELOAD_NEXT_CHAPTER = 4; private static final int RETRY_IMAGES = 4;
private static final int PRELOAD_NEXT_CHAPTER = 5;
@Override @Override
protected void onCreate(Bundle savedState) { protected void onCreate(Bundle savedState) {
@ -71,12 +73,16 @@ public class ReaderPresenter extends BasePresenter<ReaderActivity> {
(view, pages) -> {}, (view, pages) -> {},
(view, error) -> Timber.e("An error occurred while preloading a chapter")); (view, error) -> Timber.e("An error occurred while preloading a chapter"));
restartableReplay(GET_PAGE_IMAGES, restartableLatestCache(GET_PAGE_IMAGES,
() -> getPageImagesObservable() this::getPageImagesObservable,
.doOnCompleted(this::preloadNextChapter),
(view, page) -> {}, (view, page) -> {},
(view, error) -> Timber.e("An error occurred while downloading an image")); (view, error) -> Timber.e("An error occurred while downloading an image"));
restartableLatestCache(GET_ADJACENT_CHAPTERS,
this::getAdjacentChaptersObservable,
(view, pair) -> view.onAdjacentChapters(pair.first, pair.second),
(view, error) -> Timber.e("An error occurred while getting adjacent chapters"));
restartableLatestCache(RETRY_IMAGES, restartableLatestCache(RETRY_IMAGES,
this::getRetryPageObservable, this::getRetryPageObservable,
(view, page) -> {}, (view, page) -> {},
@ -86,7 +92,7 @@ public class ReaderPresenter extends BasePresenter<ReaderActivity> {
() -> getPageListObservable() () -> getPageListObservable()
.doOnNext(pages -> pageList = pages) .doOnNext(pages -> pageList = pages)
.doOnCompleted(() -> { .doOnCompleted(() -> {
getAdjacentChapters(); start(GET_ADJACENT_CHAPTERS);
start(GET_PAGE_IMAGES); start(GET_PAGE_IMAGES);
start(RETRY_IMAGES); start(RETRY_IMAGES);
}), }),
@ -117,6 +123,7 @@ public class ReaderPresenter extends BasePresenter<ReaderActivity> {
// These are started by GET_PAGE_LIST, so we don't let them restart itselves // These are started by GET_PAGE_LIST, so we don't let them restart itselves
stop(GET_PAGE_IMAGES); stop(GET_PAGE_IMAGES);
stop(GET_ADJACENT_CHAPTERS);
stop(RETRY_IMAGES); stop(RETRY_IMAGES);
stop(PRELOAD_NEXT_CHAPTER); stop(PRELOAD_NEXT_CHAPTER);
} }
@ -153,9 +160,19 @@ public class ReaderPresenter extends BasePresenter<ReaderActivity> {
pageObservable = Observable.from(pageList) pageObservable = Observable.from(pageList)
.flatMap(page -> downloadManager.getDownloadedImage(page, chapterDir)); .flatMap(page -> downloadManager.getDownloadedImage(page, chapterDir));
} }
return Observable.defer(() -> pageObservable) return pageObservable.subscribeOn(Schedulers.io())
.subscribeOn(Schedulers.io()) .doOnCompleted(this::preloadNextChapter);
.onBackpressureBuffer() }
private Observable<Pair<Chapter, Chapter>> getAdjacentChaptersObservable() {
return Observable.zip(
db.getPreviousChapter(chapter).createObservable().take(1),
db.getNextChapter(chapter).createObservable().take(1),
Pair::create)
.doOnNext(pair -> {
previousChapter = pair.first;
nextChapter = pair.second;
})
.observeOn(AndroidSchedulers.mainThread()); .observeOn(AndroidSchedulers.mainThread());
} }
@ -275,28 +292,22 @@ public class ReaderPresenter extends BasePresenter<ReaderActivity> {
this.currentPage = currentPage; this.currentPage = currentPage;
} }
private void getAdjacentChapters() { public boolean loadNextChapter() {
add(db.getNextChapter(chapter).createObservable()
.take(1)
.subscribe(result -> nextChapter = result));
add(db.getPreviousChapter(chapter).createObservable()
.take(1)
.subscribe(result -> previousChapter = result));
}
public void loadNextChapter() {
if (hasNextChapter()) { if (hasNextChapter()) {
onChapterLeft(); onChapterLeft();
loadChapter(nextChapter); loadChapter(nextChapter);
return true;
} }
return false;
} }
public void loadPreviousChapter() { public boolean loadPreviousChapter() {
if (hasPreviousChapter()) { if (hasPreviousChapter()) {
onChapterLeft(); onChapterLeft();
loadChapter(previousChapter); loadChapter(previousChapter);
return true;
} }
return false;
} }
public boolean hasNextChapter() { public boolean hasNextChapter() {

View file

@ -4,12 +4,9 @@ import android.view.MotionEvent;
import java.util.List; import java.util.List;
import eu.kanade.mangafeed.R;
import eu.kanade.mangafeed.data.source.model.Page; import eu.kanade.mangafeed.data.source.model.Page;
import eu.kanade.mangafeed.ui.base.fragment.BaseFragment; import eu.kanade.mangafeed.ui.base.fragment.BaseFragment;
import eu.kanade.mangafeed.ui.reader.ReaderActivity; import eu.kanade.mangafeed.ui.reader.ReaderActivity;
import eu.kanade.mangafeed.ui.reader.ReaderPresenter;
import eu.kanade.mangafeed.util.ToastUtil;
public abstract class BaseReader extends BaseFragment { public abstract class BaseReader extends BaseFragment {
@ -32,27 +29,6 @@ public abstract class BaseReader extends BaseFragment {
return page; return page;
} }
public void requestNextChapter() {
ReaderPresenter presenter = getReaderActivity().getPresenter();
if (presenter.hasNextChapter()) {
presenter.setCurrentPage(getCurrentPage());
presenter.loadNextChapter();
} else {
ToastUtil.showShort(getActivity(), R.string.no_next_chapter);
}
}
public void requestPreviousChapter() {
ReaderPresenter presenter = getReaderActivity().getPresenter();
if (presenter.hasPreviousChapter()) {
presenter.setCurrentPage(getCurrentPage());
presenter.loadPreviousChapter();
} else {
ToastUtil.showShort(getActivity(), R.string.no_previous_chapter);
}
}
public void onPageChanged(int position) { public void onPageChanged(int position) {
currentPage = getPageForPosition(position); currentPage = getPageForPosition(position);
updatePageNumber(); updatePageNumber();

View file

@ -70,10 +70,12 @@ public abstract class PagerReader extends BaseReader {
@Override @Override
public void onPageListReady(List<Page> pages, int currentPage) { public void onPageListReady(List<Page> pages, int currentPage) {
this.pages = pages; if (this.pages != pages) {
this.currentPage = currentPage; this.pages = pages;
if (isResumed()) { this.currentPage = currentPage;
setPages(); if (isResumed()) {
setPages();
}
} }
} }

View file

@ -4,12 +4,12 @@ public class LeftToRightReader extends HorizontalReader {
@Override @Override
public void onFirstPageOut() { public void onFirstPageOut() {
requestPreviousChapter(); getReaderActivity().requestPreviousChapter();
} }
@Override @Override
public void onLastPageOut() { public void onLastPageOut() {
requestNextChapter(); getReaderActivity().requestNextChapter();
} }
} }

View file

@ -27,12 +27,12 @@ public class RightToLeftReader extends HorizontalReader {
@Override @Override
public void onFirstPageOut() { public void onFirstPageOut() {
requestNextChapter(); getReaderActivity().requestNextChapter();
} }
@Override @Override
public void onLastPageOut() { public void onLastPageOut() {
requestPreviousChapter(); getReaderActivity().requestPreviousChapter();
} }
} }

View file

@ -18,12 +18,12 @@ public class VerticalReader extends PagerReader {
@Override @Override
public void onFirstPageOut() { public void onFirstPageOut() {
requestPreviousChapter(); getReaderActivity().requestPreviousChapter();
} }
@Override @Override
public void onLastPageOut() { public void onLastPageOut() {
requestNextChapter(); getReaderActivity().requestNextChapter();
} }
} }

View file

@ -58,6 +58,13 @@ public class WebtoonAdapter extends RecyclerView.Adapter<WebtoonAdapter.ImageHol
notifyItemInserted(page.getPageNumber()); notifyItemInserted(page.getPageNumber());
} }
public void clear() {
if (pages != null) {
pages.clear();
notifyDataSetChanged();
}
}
public static class ImageHolder extends RecyclerView.ViewHolder { public static class ImageHolder extends RecyclerView.ViewHolder {
@Bind(R.id.page_image_view) SubsamplingScaleImageView imageView; @Bind(R.id.page_image_view) SubsamplingScaleImageView imageView;

View file

@ -73,9 +73,13 @@ public class WebtoonReader extends BaseReader {
@Override @Override
public void onPause() { public void onPause() {
unsubscribeStatus();
super.onPause();
}
private void unsubscribeStatus() {
if (subscription != null && !subscription.isUnsubscribed()) if (subscription != null && !subscription.isUnsubscribed())
subscription.unsubscribe(); subscription.unsubscribe();
super.onPause();
} }
@Override @Override
@ -85,14 +89,18 @@ public class WebtoonReader extends BaseReader {
@Override @Override
public void onPageListReady(List<Page> pages, int currentPage) { public void onPageListReady(List<Page> pages, int currentPage) {
this.pages = pages; if (this.pages != pages) {
if (isResumed()) { this.pages = pages;
setPages(); if (isResumed()) {
setPages();
}
} }
} }
private void setPages() { private void setPages() {
if (pages != null) { if (pages != null) {
unsubscribeStatus();
adapter.clear();
observeStatus(0); observeStatus(0);
} }
} }
@ -112,8 +120,8 @@ public class WebtoonReader extends BaseReader {
PublishSubject<Integer> statusSubject = PublishSubject.create(); PublishSubject<Integer> statusSubject = PublishSubject.create();
page.setStatusSubject(statusSubject); page.setStatusSubject(statusSubject);
if (subscription != null && !subscription.isUnsubscribed()) // Unsubscribe from the previous page
subscription.unsubscribe(); unsubscribeStatus();
subscription = statusSubject subscription = statusSubject
.startWith(page.getStatus()) .startWith(page.getStatus())

View file

@ -4,5 +4,6 @@
android:fromAlpha="0.0" android:fromAlpha="0.0"
android:toAlpha="1.0" android:toAlpha="1.0"
android:interpolator="@android:anim/accelerate_interpolator" android:interpolator="@android:anim/accelerate_interpolator"
android:duration="300" /> android:duration="300"
android:fillAfter="true"/>
</set> </set>

Binary file not shown.

After

Width:  |  Height:  |  Size: 195 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 209 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 179 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 177 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 150 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 152 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 233 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 257 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 309 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 335 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 391 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 423 B

View file

@ -0,0 +1,19 @@
<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<item
android:id="@+id/action_previous_chapter"
android:title="@string/action_previous_chapter"
android:icon="@drawable/ic_skip_previous_white_24dp"
android:visible="false"
app:showAsAction="always" />
<item
android:id="@+id/action_next_chapter"
android:title="@string/action_next_chapter"
android:icon="@drawable/ic_skip_next_white_24dp"
android:visible="false"
app:showAsAction="always" />
</menu>

View file

@ -34,6 +34,9 @@
<string name="action_next_unread">Next unread</string> <string name="action_next_unread">Next unread</string>
<string name="action_start">Start</string> <string name="action_start">Start</string>
<string name="action_stop">Stop</string> <string name="action_stop">Stop</string>
<string name="action_previous_chapter">Previous chapter</string>
<string name="action_next_chapter">Next chapter</string>
<!-- Buttons --> <!-- Buttons -->
<string name="button_ok">OK</string> <string name="button_ok">OK</string>