Fix gesture controls for portrait videos

Use `PlayerViewModel.isFullscreen` to check if video is playing in full screen instead of screen orientation.
Scale progress bar `progress` according to it's max size.
This commit is contained in:
Krunal Patel 2022-11-28 11:54:31 +05:30
parent b3824b2e43
commit fe02b0e30c
6 changed files with 61 additions and 25 deletions

View File

@ -9,6 +9,7 @@ import android.os.Build
import android.os.Bundle import android.os.Bundle
import android.view.View import android.view.View
import android.view.WindowManager import android.view.WindowManager
import androidx.activity.viewModels
import androidx.core.view.WindowCompat import androidx.core.view.WindowCompat
import androidx.core.view.WindowInsetsCompat import androidx.core.view.WindowInsetsCompat
import androidx.core.view.WindowInsetsControllerCompat import androidx.core.view.WindowInsetsControllerCompat
@ -16,6 +17,7 @@ import com.github.libretube.constants.IntentData
import com.github.libretube.databinding.ActivityOfflinePlayerBinding import com.github.libretube.databinding.ActivityOfflinePlayerBinding
import com.github.libretube.databinding.ExoStyledPlayerControlViewBinding import com.github.libretube.databinding.ExoStyledPlayerControlViewBinding
import com.github.libretube.ui.base.BaseActivity import com.github.libretube.ui.base.BaseActivity
import com.github.libretube.ui.models.PlayerViewModel
import com.github.libretube.util.DownloadHelper import com.github.libretube.util.DownloadHelper
import com.github.libretube.util.PlayerHelper import com.github.libretube.util.PlayerHelper
import com.google.android.exoplayer2.ExoPlayer import com.google.android.exoplayer2.ExoPlayer
@ -32,6 +34,7 @@ class OfflinePlayerActivity : BaseActivity() {
private lateinit var player: ExoPlayer private lateinit var player: ExoPlayer
private lateinit var playerView: StyledPlayerView private lateinit var playerView: StyledPlayerView
private lateinit var playerBinding: ExoStyledPlayerControlViewBinding private lateinit var playerBinding: ExoStyledPlayerControlViewBinding
private val playerViewModel: PlayerViewModel by viewModels()
override fun onCreate(savedInstanceState: Bundle?) { override fun onCreate(savedInstanceState: Bundle?) {
hideSystemBars() hideSystemBars()
@ -154,6 +157,16 @@ class OfflinePlayerActivity : BaseActivity() {
windowInsetsController.hide(WindowInsetsCompat.Type.systemBars()) windowInsetsController.hide(WindowInsetsCompat.Type.systemBars())
} }
override fun onResume() {
playerViewModel.isFullscreen.value = true
super.onResume()
}
override fun onPause() {
playerViewModel.isFullscreen.value = false
super.onPause()
}
override fun onDestroy() { override fun onDestroy() {
player.release() player.release()
super.onDestroy() super.onDestroy()

View File

@ -19,4 +19,6 @@ interface PlayerGestureOptions {
fun onZoom() fun onZoom()
fun onMinimize() fun onMinimize()
fun onFullscreenChange(isFullscreen: Boolean)
} }

View File

@ -85,11 +85,11 @@ internal class CustomExoPlayerView(
this.doubleTapOverlayBinding = doubleTapOverlayBinding this.doubleTapOverlayBinding = doubleTapOverlayBinding
this.trackSelector = trackSelector this.trackSelector = trackSelector
this.gestureViewBinding = playerGestureControlsViewBinding this.gestureViewBinding = playerGestureControlsViewBinding
this.playerGestureController = PlayerGestureController(context, this) this.playerGestureController = PlayerGestureController(context as BaseActivity, this)
this.brightnessHelper = BrightnessHelper(context as Activity) this.brightnessHelper = BrightnessHelper(context as Activity)
this.audioHelper = AudioHelper(context) this.audioHelper = AudioHelper(context)
// Set touch listner for tap and swipe gestures. // Set touch listener for tap and swipe gestures.
setOnTouchListener(playerGestureController) setOnTouchListener(playerGestureController)
initializeGestureProgress() initializeGestureProgress()
@ -461,13 +461,6 @@ internal class CustomExoPlayerView(
params.bottomMargin = offset.toInt() params.bottomMargin = offset.toInt()
it.layoutParams = params it.layoutParams = params
} }
if (PlayerHelper.swipeGestureEnabled && this::brightnessHelper.isInitialized) {
when (newConfig?.orientation) {
Configuration.ORIENTATION_LANDSCAPE -> brightnessHelper.restoreSavedBrightness()
else -> brightnessHelper.resetToSystemBrightness(false)
}
}
} }
override fun onSingleTap() { override fun onSingleTap() {
@ -497,14 +490,14 @@ internal class CustomExoPlayerView(
} }
override fun onSwipeLeftScreen(distanceY: Float) { override fun onSwipeLeftScreen(distanceY: Float) {
if (!PlayerHelper.swipeGestureEnabled || resources.configuration.orientation == Configuration.ORIENTATION_PORTRAIT) return if (!PlayerHelper.swipeGestureEnabled) return
if (isControllerFullyVisible) hideController() if (isControllerFullyVisible) hideController()
updateBrightness(distanceY) updateBrightness(distanceY)
} }
override fun onSwipeRightScreen(distanceY: Float) { override fun onSwipeRightScreen(distanceY: Float) {
if (!PlayerHelper.swipeGestureEnabled || resources.configuration.orientation == Configuration.ORIENTATION_PORTRAIT) return if (!PlayerHelper.swipeGestureEnabled) return
if (isControllerFullyVisible) hideController() if (isControllerFullyVisible) hideController()
updateVolume(distanceY) updateVolume(distanceY)
@ -522,4 +515,14 @@ internal class CustomExoPlayerView(
override fun onMinimize() { override fun onMinimize() {
resizeMode = AspectRatioFrameLayout.RESIZE_MODE_FIT resizeMode = AspectRatioFrameLayout.RESIZE_MODE_FIT
} }
override fun onFullscreenChange(isFullscreen: Boolean) {
if (PlayerHelper.swipeGestureEnabled && this::brightnessHelper.isInitialized) {
if (isFullscreen) {
brightnessHelper.restoreSavedBrightness()
} else {
brightnessHelper.resetToSystemBrightness(false)
}
}
}
} }

View File

@ -5,6 +5,7 @@ import android.util.AttributeSet
import android.view.LayoutInflater import android.view.LayoutInflater
import androidx.constraintlayout.widget.ConstraintLayout import androidx.constraintlayout.widget.ConstraintLayout
import com.github.libretube.databinding.PlayerGestureControlsViewBinding import com.github.libretube.databinding.PlayerGestureControlsViewBinding
import com.github.libretube.extensions.normalize
class PlayerGestureControlsView( class PlayerGestureControlsView(
context: Context, context: Context,
@ -20,7 +21,21 @@ class PlayerGestureControlsView(
override fun onSizeChanged(width: Int, height: Int, oldWidth: Int, oldHeight: Int) { override fun onSizeChanged(width: Int, height: Int, oldWidth: Int, oldHeight: Int) {
super.onSizeChanged(width, height, oldHeight, oldHeight) super.onSizeChanged(width, height, oldHeight, oldHeight)
binding.brightnessProgressBar.max = (height * 0.7).toInt() // Set new max value of progress bar corresponding to the new height and
binding.volumeProgressBar.max = (height * 0.7).toInt() // make progress accordingly, store oldProgress before changing it to avoid
// inconsistency when old progress > new max
binding.brightnessProgressBar.apply {
val oldMax = max
val oldProgress = progress
max = (height * 0.7).toInt()
progress = oldProgress.normalize(0, oldMax, 0, max)
}
binding.volumeProgressBar.apply {
val oldMax = max
val oldProgress = progress
max = (height * 0.7).toInt()
progress = oldProgress.normalize(0, oldMax, 0, max)
}
} }
} }

View File

@ -1,7 +1,6 @@
package com.github.libretube.util package com.github.libretube.util
import android.app.Activity import android.app.Activity
import android.os.Build
import android.view.WindowManager import android.view.WindowManager
import com.github.libretube.constants.PreferenceKeys import com.github.libretube.constants.PreferenceKeys
import com.github.libretube.extensions.normalize import com.github.libretube.extensions.normalize
@ -46,9 +45,6 @@ class BrightnessHelper(private val activity: Activity) {
* Set current screen brightness to saved brightness value. * Set current screen brightness to saved brightness value.
*/ */
fun restoreSavedBrightness() { fun restoreSavedBrightness() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N && activity.isInPictureInPictureMode) {
return
}
brightness = savedBrightness brightness = savedBrightness
} }

View File

@ -1,8 +1,6 @@
package com.github.libretube.util package com.github.libretube.util
import android.annotation.SuppressLint import android.annotation.SuppressLint
import android.content.Context
import android.content.res.Configuration
import android.content.res.Resources import android.content.res.Resources
import android.os.Handler import android.os.Handler
import android.os.Looper import android.os.Looper
@ -11,27 +9,36 @@ import android.view.GestureDetector
import android.view.MotionEvent import android.view.MotionEvent
import android.view.ScaleGestureDetector import android.view.ScaleGestureDetector
import android.view.View import android.view.View
import androidx.activity.viewModels
import com.github.libretube.ui.base.BaseActivity
import com.github.libretube.ui.interfaces.PlayerGestureOptions import com.github.libretube.ui.interfaces.PlayerGestureOptions
import com.github.libretube.ui.models.PlayerViewModel
import kotlin.math.abs import kotlin.math.abs
class PlayerGestureController(context: Context, private val listener: PlayerGestureOptions) : class PlayerGestureController(activity: BaseActivity, private val listener: PlayerGestureOptions) :
View.OnTouchListener { View.OnTouchListener {
// width and height should be obtained each time using getter to adopt layout size changes. // width and height should be obtained each time using getter to adopt layout size changes.
private val width get() = Resources.getSystem().displayMetrics.widthPixels private val width get() = Resources.getSystem().displayMetrics.widthPixels
private val height get() = Resources.getSystem().displayMetrics.heightPixels private val height get() = Resources.getSystem().displayMetrics.heightPixels
private val orientation get() = Resources.getSystem().configuration.orientation
private val elapsedTime get() = SystemClock.elapsedRealtime() private val elapsedTime get() = SystemClock.elapsedRealtime()
private val playerViewModel: PlayerViewModel by activity.viewModels()
private val handler: Handler = Handler(Looper.getMainLooper()) private val handler: Handler = Handler(Looper.getMainLooper())
private val gestureDetector: GestureDetector private val gestureDetector: GestureDetector
private val scaleGestureDetector: ScaleGestureDetector private val scaleGestureDetector: ScaleGestureDetector
private var isFullscreen = false
private var isMoving = false private var isMoving = false
var isEnabled = true var isEnabled = true
init { init {
gestureDetector = GestureDetector(context, GestureListener(), handler) gestureDetector = GestureDetector(activity, GestureListener(), handler)
scaleGestureDetector = ScaleGestureDetector(context, ScaleGestureListener(), handler) scaleGestureDetector = ScaleGestureDetector(activity, ScaleGestureListener(), handler)
playerViewModel.isFullscreen.observe(activity) {
isFullscreen = it
listener.onFullscreenChange(it)
}
} }
@SuppressLint("ClickableViewAccessibility") @SuppressLint("ClickableViewAccessibility")
@ -47,8 +54,8 @@ class PlayerGestureController(context: Context, private val listener: PlayerGest
gestureDetector.onTouchEvent(event) gestureDetector.onTouchEvent(event)
} catch (_: Exception) { } } catch (_: Exception) { }
// If orientation is landscape then allow `onScroll` to consume event and return true. // If video is playing in full-screen then allow `onScroll` to consume event and return true.
return orientation == Configuration.ORIENTATION_LANDSCAPE return isFullscreen
} }
private inner class ScaleGestureListener : ScaleGestureDetector.OnScaleGestureListener { private inner class ScaleGestureListener : ScaleGestureDetector.OnScaleGestureListener {