|
@@ -1,16 +1,16 @@
|
|
|
package eu.kanade.tachiyomi.ui.reader
|
|
|
|
|
|
-import android.app.Dialog
|
|
|
import android.content.Context
|
|
|
import android.content.Intent
|
|
|
import android.content.pm.ActivityInfo
|
|
|
import android.content.res.Configuration
|
|
|
import android.graphics.Color
|
|
|
import android.os.Build
|
|
|
+import android.os.Build.VERSION_CODES.KITKAT
|
|
|
import android.os.Bundle
|
|
|
import android.support.v4.content.ContextCompat
|
|
|
import android.view.*
|
|
|
-import android.view.WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
|
|
|
+import android.view.WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS
|
|
|
import android.view.animation.Animation
|
|
|
import android.view.animation.AnimationUtils
|
|
|
import android.widget.SeekBar
|
|
@@ -33,11 +33,13 @@ import eu.kanade.tachiyomi.util.toast
|
|
|
import eu.kanade.tachiyomi.widget.SimpleAnimationListener
|
|
|
import eu.kanade.tachiyomi.widget.SimpleSeekBarListener
|
|
|
import kotlinx.android.synthetic.main.activity_reader.*
|
|
|
-import kotlinx.android.synthetic.main.reader_menu.*
|
|
|
+import me.zhanghai.android.systemuihelper.SystemUiHelper
|
|
|
+import me.zhanghai.android.systemuihelper.SystemUiHelper.*
|
|
|
import nucleus.factory.RequiresPresenter
|
|
|
import rx.Subscription
|
|
|
import rx.subscriptions.CompositeSubscription
|
|
|
import timber.log.Timber
|
|
|
+import uy.kohesive.injekt.injectLazy
|
|
|
import java.text.DecimalFormat
|
|
|
|
|
|
@RequiresPresenter(ReaderPresenter::class)
|
|
@@ -62,8 +64,6 @@ class ReaderActivity : BaseRxActivity<ReaderPresenter>() {
|
|
|
|
|
|
private var viewer: BaseReader? = null
|
|
|
|
|
|
- private var uiFlags: Int = 0
|
|
|
-
|
|
|
lateinit var subscriptions: CompositeSubscription
|
|
|
private set
|
|
|
|
|
@@ -77,16 +77,13 @@ class ReaderActivity : BaseRxActivity<ReaderPresenter>() {
|
|
|
|
|
|
private val decimalFormat = DecimalFormat("#.###")
|
|
|
|
|
|
- private var popupMenu: ReaderPopupMenu? = null
|
|
|
-
|
|
|
- private var nextChapterBtn: MenuItem? = null
|
|
|
+ private val volumeKeysEnabled by lazy { preferences.readWithVolumeKeys().getOrDefault() }
|
|
|
|
|
|
- private var prevChapterBtn: MenuItem? = null
|
|
|
+ val preferences by injectLazy<PreferencesHelper>()
|
|
|
|
|
|
- private val volumeKeysEnabled by lazy { preferences.readWithVolumeKeys().getOrDefault() }
|
|
|
+ private var systemUi: SystemUiHelper? = null
|
|
|
|
|
|
- val preferences: PreferencesHelper
|
|
|
- get() = presenter.prefs
|
|
|
+ private var menuVisible = false
|
|
|
|
|
|
override fun onCreate(savedState: Bundle?) {
|
|
|
super.onCreate(savedState)
|
|
@@ -100,47 +97,56 @@ class ReaderActivity : BaseRxActivity<ReaderPresenter>() {
|
|
|
setupToolbar(toolbar)
|
|
|
subscriptions = CompositeSubscription()
|
|
|
|
|
|
- initializeMenu()
|
|
|
initializeSettings()
|
|
|
+ initializeBottomMenu()
|
|
|
|
|
|
if (savedState != null) {
|
|
|
- setMenuVisibility(savedState.getBoolean(MENU_VISIBLE), animate = false)
|
|
|
+ menuVisible = savedState.getBoolean(MENU_VISIBLE)
|
|
|
}
|
|
|
|
|
|
+ setMenuVisibility(menuVisible)
|
|
|
+
|
|
|
maxBitmapSize = GLUtil.getMaxTextureSize()
|
|
|
- }
|
|
|
|
|
|
- override fun onResume() {
|
|
|
- super.onResume()
|
|
|
- setSystemUiVisibility()
|
|
|
+ left_chapter.setOnClickListener {
|
|
|
+ if (viewer != null) {
|
|
|
+ if (viewer is RightToLeftReader)
|
|
|
+ requestNextChapter()
|
|
|
+ else
|
|
|
+ requestPreviousChapter()
|
|
|
+ }
|
|
|
+ }
|
|
|
+ right_chapter.setOnClickListener {
|
|
|
+ if (viewer != null) {
|
|
|
+ if (viewer is RightToLeftReader)
|
|
|
+ requestPreviousChapter()
|
|
|
+ else
|
|
|
+ requestNextChapter()
|
|
|
+ }
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
override fun onDestroy() {
|
|
|
subscriptions.unsubscribe()
|
|
|
- popupMenu?.dismiss()
|
|
|
viewer = null
|
|
|
super.onDestroy()
|
|
|
}
|
|
|
|
|
|
override fun onCreateOptionsMenu(menu: Menu): Boolean {
|
|
|
menuInflater.inflate(R.menu.reader, menu)
|
|
|
- nextChapterBtn = menu.findItem(R.id.action_next_chapter)
|
|
|
- prevChapterBtn = menu.findItem(R.id.action_previous_chapter)
|
|
|
- setAdjacentChaptersVisibility()
|
|
|
return true
|
|
|
}
|
|
|
|
|
|
override fun onOptionsItemSelected(item: MenuItem): Boolean {
|
|
|
when (item.itemId) {
|
|
|
- R.id.action_previous_chapter -> requestPreviousChapter()
|
|
|
- R.id.action_next_chapter -> requestNextChapter()
|
|
|
+ R.id.action_settings -> ReaderSettingsDialog().show(supportFragmentManager, "settings")
|
|
|
else -> return super.onOptionsItemSelected(item)
|
|
|
}
|
|
|
return true
|
|
|
}
|
|
|
|
|
|
override fun onSaveInstanceState(outState: Bundle) {
|
|
|
- outState.putBoolean(MENU_VISIBLE, reader_menu.visibility == View.VISIBLE)
|
|
|
+ outState.putBoolean(MENU_VISIBLE, menuVisible)
|
|
|
super.onSaveInstanceState(outState)
|
|
|
}
|
|
|
|
|
@@ -148,7 +154,7 @@ class ReaderActivity : BaseRxActivity<ReaderPresenter>() {
|
|
|
val chapterToUpdate = presenter.getMangaSyncChapterToUpdate()
|
|
|
|
|
|
if (chapterToUpdate > 0) {
|
|
|
- if (presenter.prefs.askUpdateMangaSync()) {
|
|
|
+ if (preferences.askUpdateMangaSync()) {
|
|
|
MaterialDialog.Builder(this)
|
|
|
.content(getString(R.string.confirm_update_manga_sync, chapterToUpdate))
|
|
|
.positiveText(android.R.string.yes)
|
|
@@ -165,13 +171,6 @@ class ReaderActivity : BaseRxActivity<ReaderPresenter>() {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- override fun onWindowFocusChanged(hasFocus: Boolean) {
|
|
|
- super.onWindowFocusChanged(hasFocus)
|
|
|
- if (hasFocus) {
|
|
|
- setSystemUiVisibility()
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
override fun dispatchKeyEvent(event: KeyEvent): Boolean {
|
|
|
if (!isFinishing) {
|
|
|
when (event.keyCode) {
|
|
@@ -270,16 +269,19 @@ class ReaderActivity : BaseRxActivity<ReaderPresenter>() {
|
|
|
viewer?.onPageListAppendReady(chapter)
|
|
|
}
|
|
|
|
|
|
- @Suppress("UNUSED_PARAMETER")
|
|
|
fun onAdjacentChapters(previous: Chapter?, next: Chapter?) {
|
|
|
- setAdjacentChaptersVisibility()
|
|
|
- }
|
|
|
+ val isInverted = viewer is RightToLeftReader
|
|
|
|
|
|
- private fun setAdjacentChaptersVisibility() {
|
|
|
- prevChapterBtn?.isVisible = presenter.hasPreviousChapter()
|
|
|
- nextChapterBtn?.isVisible = presenter.hasNextChapter()
|
|
|
- }
|
|
|
+ // Chapters are inverted for the right to left reader
|
|
|
+ val hasRightChapter = (if (isInverted) previous else next) != null
|
|
|
+ val hasLeftChapter = (if (isInverted) next else previous) != null
|
|
|
+
|
|
|
+ right_chapter.isEnabled = hasRightChapter
|
|
|
+ right_chapter.alpha = if (hasRightChapter) 1f else 0.4f
|
|
|
|
|
|
+ left_chapter.isEnabled = hasLeftChapter
|
|
|
+ left_chapter.alpha = if (hasLeftChapter) 1f else 0.4f
|
|
|
+ }
|
|
|
|
|
|
private fun getOrCreateViewer(manga: Manga): BaseReader {
|
|
|
val mangaViewer = if (manga.viewer == 0) preferences.defaultViewer() else manga.viewer
|
|
@@ -323,7 +325,7 @@ class ReaderActivity : BaseRxActivity<ReaderPresenter>() {
|
|
|
}
|
|
|
|
|
|
fun toggleMenu() {
|
|
|
- setMenuVisibility(reader_menu.visibility == View.GONE)
|
|
|
+ setMenuVisibility(!menuVisible)
|
|
|
}
|
|
|
|
|
|
fun requestNextChapter() {
|
|
@@ -338,34 +340,7 @@ class ReaderActivity : BaseRxActivity<ReaderPresenter>() {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- private fun setMenuVisibility(visible: Boolean, animate: Boolean = true) {
|
|
|
- if (visible) {
|
|
|
- reader_menu.visibility = View.VISIBLE
|
|
|
-
|
|
|
- if (animate) {
|
|
|
- val toolbarAnimation = AnimationUtils.loadAnimation(this, R.anim.enter_from_top)
|
|
|
- toolbar.startAnimation(toolbarAnimation)
|
|
|
-
|
|
|
- val bottomMenuAnimation = AnimationUtils.loadAnimation(this, R.anim.enter_from_bottom)
|
|
|
- reader_menu_bottom.startAnimation(bottomMenuAnimation)
|
|
|
- }
|
|
|
- } else {
|
|
|
- val toolbarAnimation = AnimationUtils.loadAnimation(this, R.anim.exit_to_top)
|
|
|
- toolbarAnimation.setAnimationListener(object : SimpleAnimationListener() {
|
|
|
- override fun onAnimationEnd(animation: Animation) {
|
|
|
- reader_menu.visibility = View.GONE
|
|
|
- }
|
|
|
- })
|
|
|
- toolbar.startAnimation(toolbarAnimation)
|
|
|
-
|
|
|
- val bottomMenuAnimation = AnimationUtils.loadAnimation(this, R.anim.exit_to_bottom)
|
|
|
- reader_menu_bottom.startAnimation(bottomMenuAnimation)
|
|
|
-
|
|
|
- popupMenu?.dismiss()
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- private fun initializeMenu() {
|
|
|
+ private fun initializeBottomMenu() {
|
|
|
// Intercept all events in this layout
|
|
|
reader_menu_bottom.setOnTouchListener { v, event -> true }
|
|
|
|
|
@@ -376,94 +351,17 @@ class ReaderActivity : BaseRxActivity<ReaderPresenter>() {
|
|
|
}
|
|
|
}
|
|
|
})
|
|
|
-
|
|
|
- lock_orientation.setOnClickListener { v ->
|
|
|
- showImmersiveDialog(MaterialDialog.Builder(this)
|
|
|
- .title(R.string.pref_rotation_type)
|
|
|
- .items(R.array.rotation_type)
|
|
|
- .itemsCallbackSingleChoice(preferences.rotation().getOrDefault() - 1,
|
|
|
- { d, itemView, which, text ->
|
|
|
- preferences.rotation().set(which + 1)
|
|
|
- true
|
|
|
- })
|
|
|
- .build())
|
|
|
- }
|
|
|
-
|
|
|
- reader_zoom_selector.setOnClickListener { v ->
|
|
|
- showImmersiveDialog(MaterialDialog.Builder(this)
|
|
|
- .title(R.string.pref_zoom_start)
|
|
|
- .items(R.array.zoom_start)
|
|
|
- .itemsCallbackSingleChoice(preferences.zoomStart().getOrDefault() - 1,
|
|
|
- { d, itemView, which, text ->
|
|
|
- preferences.zoomStart().set(which + 1)
|
|
|
- true
|
|
|
- })
|
|
|
- .build())
|
|
|
- }
|
|
|
-
|
|
|
- reader_scale_type_selector.setOnClickListener { v ->
|
|
|
- showImmersiveDialog(MaterialDialog.Builder(this)
|
|
|
- .title(R.string.pref_image_scale_type)
|
|
|
- .items(R.array.image_scale_type)
|
|
|
- .itemsCallbackSingleChoice(preferences.imageScaleType().getOrDefault() - 1,
|
|
|
- { d, itemView, which, text ->
|
|
|
- preferences.imageScaleType().set(which + 1)
|
|
|
- true
|
|
|
- })
|
|
|
- .build())
|
|
|
- }
|
|
|
-
|
|
|
- reader_selector.setOnClickListener { v ->
|
|
|
- showImmersiveDialog(MaterialDialog.Builder(this)
|
|
|
- .title(R.string.pref_viewer_type)
|
|
|
- .items(R.array.viewers_selector)
|
|
|
- .itemsCallbackSingleChoice(presenter.manga.viewer,
|
|
|
- { d, itemView, which, text ->
|
|
|
- presenter.updateMangaViewer(which)
|
|
|
- recreate()
|
|
|
- true
|
|
|
- })
|
|
|
- .build())
|
|
|
- }
|
|
|
-
|
|
|
- val popupView = layoutInflater.inflate(R.layout.reader_popup, null)
|
|
|
- popupMenu = ReaderPopupMenu(this, popupView)
|
|
|
-
|
|
|
- reader_extra_settings.setOnClickListener {
|
|
|
- popupMenu?.let {
|
|
|
- if (!it.isShowing)
|
|
|
- it.showAtLocation(reader_extra_settings,
|
|
|
- Gravity.BOTTOM or Gravity.RIGHT, 0, reader_menu_bottom.height)
|
|
|
- else
|
|
|
- it.dismiss()
|
|
|
- }
|
|
|
-
|
|
|
- }
|
|
|
-
|
|
|
}
|
|
|
|
|
|
private fun initializeSettings() {
|
|
|
- subscriptions += preferences.showPageNumber().asObservable()
|
|
|
- .subscribe { setPageNumberVisibility(it) }
|
|
|
-
|
|
|
subscriptions += preferences.rotation().asObservable()
|
|
|
- .subscribe {
|
|
|
- setRotation(it)
|
|
|
+ .subscribe { setRotation(it) }
|
|
|
|
|
|
- val isPortrait = resources.configuration.orientation == Configuration.ORIENTATION_PORTRAIT
|
|
|
-
|
|
|
- val resourceId = if (it == 1)
|
|
|
- R.drawable.ic_screen_rotation_white_24dp
|
|
|
- else if (isPortrait)
|
|
|
- R.drawable.ic_screen_lock_portrait_white_24dp
|
|
|
- else
|
|
|
- R.drawable.ic_screen_lock_landscape_white_24dp
|
|
|
-
|
|
|
- lock_orientation.setImageResource(resourceId)
|
|
|
- }
|
|
|
+ subscriptions += preferences.showPageNumber().asObservable()
|
|
|
+ .subscribe { setPageNumberVisibility(it) }
|
|
|
|
|
|
- subscriptions += preferences.hideStatusBar().asObservable()
|
|
|
- .subscribe { setStatusBarVisibility(it) }
|
|
|
+ subscriptions += preferences.fullscreen().asObservable()
|
|
|
+ .subscribe { setFullscreen(it) }
|
|
|
|
|
|
subscriptions += preferences.keepScreenOn().asObservable()
|
|
|
.subscribe { setKeepScreenOn(it) }
|
|
@@ -496,6 +394,16 @@ class ReaderActivity : BaseRxActivity<ReaderPresenter>() {
|
|
|
page_number.visibility = if (visible) View.VISIBLE else View.INVISIBLE
|
|
|
}
|
|
|
|
|
|
+ private fun setFullscreen(enabled: Boolean) {
|
|
|
+ systemUi = if (enabled) {
|
|
|
+ val level = if (Build.VERSION.SDK_INT >= KITKAT) LEVEL_IMMERSIVE else LEVEL_HIDE_STATUS_BAR
|
|
|
+ val flags = FLAG_IMMERSIVE_STICKY or FLAG_LAYOUT_IN_SCREEN_OLDER_DEVICES
|
|
|
+ SystemUiHelper(this, level, flags)
|
|
|
+ } else {
|
|
|
+ null
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
private fun setKeepScreenOn(enabled: Boolean) {
|
|
|
if (enabled) {
|
|
|
window.addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON)
|
|
@@ -523,29 +431,6 @@ class ReaderActivity : BaseRxActivity<ReaderPresenter>() {
|
|
|
window.attributes = window.attributes.apply { screenBrightness = value }
|
|
|
}
|
|
|
|
|
|
- private fun setStatusBarVisibility(hidden: Boolean) {
|
|
|
- createUiHideFlags(hidden)
|
|
|
- setSystemUiVisibility()
|
|
|
- }
|
|
|
-
|
|
|
- private fun createUiHideFlags(statusBarHidden: Boolean) {
|
|
|
- uiFlags = 0
|
|
|
- uiFlags = uiFlags or View.SYSTEM_UI_FLAG_HIDE_NAVIGATION
|
|
|
- if (statusBarHidden) {
|
|
|
- uiFlags = uiFlags or View.SYSTEM_UI_FLAG_LAYOUT_STABLE or
|
|
|
- View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION or
|
|
|
- View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN or
|
|
|
- View.SYSTEM_UI_FLAG_FULLSCREEN
|
|
|
- }
|
|
|
- if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
|
|
|
- uiFlags = uiFlags or View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- fun setSystemUiVisibility() {
|
|
|
- window.decorView.systemUiVisibility = uiFlags
|
|
|
- }
|
|
|
-
|
|
|
private fun applyTheme(theme: Int) {
|
|
|
readerTheme = theme
|
|
|
val rootView = window.decorView.rootView
|
|
@@ -560,12 +445,38 @@ class ReaderActivity : BaseRxActivity<ReaderPresenter>() {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- private fun showImmersiveDialog(dialog: Dialog) {
|
|
|
- // Hack to not leave immersive mode
|
|
|
- dialog.window.setFlags(FLAG_NOT_FOCUSABLE, FLAG_NOT_FOCUSABLE)
|
|
|
- dialog.show()
|
|
|
- dialog.window.decorView.systemUiVisibility = window.decorView.systemUiVisibility
|
|
|
- dialog.window.clearFlags(FLAG_NOT_FOCUSABLE)
|
|
|
+ private fun setMenuVisibility(visible: Boolean) {
|
|
|
+ menuVisible = visible
|
|
|
+ if (visible) {
|
|
|
+ systemUi?.show()
|
|
|
+ reader_menu.visibility = View.VISIBLE
|
|
|
+
|
|
|
+ val toolbarAnimation = AnimationUtils.loadAnimation(this, R.anim.enter_from_top)
|
|
|
+ toolbarAnimation.setAnimationListener(object : SimpleAnimationListener() {
|
|
|
+ override fun onAnimationStart(animation: Animation) {
|
|
|
+ // Fix status bar being translucent the first time it's opened.
|
|
|
+ if (Build.VERSION.SDK_INT >= 21) {
|
|
|
+ window.addFlags(FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS)
|
|
|
+ }
|
|
|
+ }
|
|
|
+ })
|
|
|
+ toolbar.startAnimation(toolbarAnimation)
|
|
|
+
|
|
|
+ val bottomMenuAnimation = AnimationUtils.loadAnimation(this, R.anim.enter_from_bottom)
|
|
|
+ reader_menu_bottom.startAnimation(bottomMenuAnimation)
|
|
|
+ } else {
|
|
|
+ systemUi?.hide()
|
|
|
+ val toolbarAnimation = AnimationUtils.loadAnimation(this, R.anim.exit_to_top)
|
|
|
+ toolbarAnimation.setAnimationListener(object : SimpleAnimationListener() {
|
|
|
+ override fun onAnimationEnd(animation: Animation) {
|
|
|
+ reader_menu.visibility = View.GONE
|
|
|
+ }
|
|
|
+ })
|
|
|
+ toolbar.startAnimation(toolbarAnimation)
|
|
|
+
|
|
|
+ val bottomMenuAnimation = AnimationUtils.loadAnimation(this, R.anim.exit_to_bottom)
|
|
|
+ reader_menu_bottom.startAnimation(bottomMenuAnimation)
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
}
|