ReaderNavigationOverlayView.kt 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. package eu.kanade.tachiyomi.ui.reader
  2. import android.content.Context
  3. import android.graphics.Canvas
  4. import android.graphics.Color
  5. import android.graphics.Paint
  6. import android.util.AttributeSet
  7. import android.view.MotionEvent
  8. import android.view.View
  9. import android.view.ViewPropertyAnimator
  10. import androidx.core.content.ContextCompat
  11. import androidx.core.view.isVisible
  12. import eu.kanade.tachiyomi.ui.reader.viewer.ViewerNavigation
  13. import kotlin.math.abs
  14. class ReaderNavigationOverlayView(context: Context, attributeSet: AttributeSet) : View(context, attributeSet) {
  15. private var viewPropertyAnimator: ViewPropertyAnimator? = null
  16. private var navigation: ViewerNavigation? = null
  17. fun setNavigation(navigation: ViewerNavigation, tappingEnabled: Boolean, showOnStart: Boolean) {
  18. val firstLaunch = this.navigation == null
  19. this.navigation = navigation
  20. invalidate()
  21. if (isVisible || (!showOnStart && firstLaunch) || !tappingEnabled) {
  22. return
  23. }
  24. viewPropertyAnimator = animate()
  25. .alpha(1f)
  26. .setDuration(FADE_DURATION)
  27. .withStartAction {
  28. isVisible = true
  29. }
  30. .withEndAction {
  31. viewPropertyAnimator = null
  32. }
  33. viewPropertyAnimator?.start()
  34. }
  35. private val regionPaint = Paint()
  36. private val textPaint = Paint().apply {
  37. textAlign = Paint.Align.CENTER
  38. color = Color.WHITE
  39. textSize = 64f
  40. }
  41. private val textBorderPaint = Paint().apply {
  42. textAlign = Paint.Align.CENTER
  43. color = Color.BLACK
  44. textSize = 64f
  45. style = Paint.Style.STROKE
  46. strokeWidth = 8f
  47. }
  48. override fun onDraw(canvas: Canvas?) {
  49. if (navigation == null) return
  50. navigation?.regions?.forEach { region ->
  51. val rect = region.rectF
  52. canvas?.save()
  53. // Scale rect from 1f,1f to screen width and height
  54. canvas?.scale(width.toFloat(), height.toFloat())
  55. regionPaint.color = ContextCompat.getColor(context, region.type.colorRes)
  56. canvas?.drawRect(rect, regionPaint)
  57. canvas?.restore()
  58. // Don't want scale anymore because it messes with drawText
  59. canvas?.save()
  60. // Translate origin to rect start (left, top)
  61. canvas?.translate((width * rect.left), (height * rect.top))
  62. // Calculate center of rect width on screen
  63. val x = width * (abs(rect.left - rect.right) / 2)
  64. // Calculate center of rect height on screen
  65. val y = height * (abs(rect.top - rect.bottom) / 2)
  66. canvas?.drawText(context.getString(region.type.nameRes), x, y, textBorderPaint)
  67. canvas?.drawText(context.getString(region.type.nameRes), x, y, textPaint)
  68. canvas?.restore()
  69. }
  70. }
  71. override fun performClick(): Boolean {
  72. super.performClick()
  73. if (viewPropertyAnimator == null && isVisible) {
  74. viewPropertyAnimator = animate()
  75. .alpha(0f)
  76. .setDuration(FADE_DURATION)
  77. .withEndAction {
  78. isVisible = false
  79. viewPropertyAnimator = null
  80. }
  81. viewPropertyAnimator?.start()
  82. }
  83. return true
  84. }
  85. override fun onTouchEvent(event: MotionEvent?): Boolean {
  86. // Hide overlay if user start tapping or swiping
  87. performClick()
  88. return super.onTouchEvent(event)
  89. }
  90. }
  91. private const val FADE_DURATION = 1000L