add volume circle
This commit is contained in:
parent
6ff8f7c1e9
commit
c27f2685c8
2 changed files with 117 additions and 0 deletions
|
@ -0,0 +1,115 @@
|
|||
/* NewPlayer
|
||||
*
|
||||
* @author Christian Schabesberger
|
||||
*
|
||||
* Copyright (C) NewPipe e.V. 2024 <code(at)newpipe-ev.de>
|
||||
*
|
||||
* NewPlayer is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* NewPlayer is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with NewPlayer. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
package net.newpipe.newplayer.ui.videoplayer.gesture_ui
|
||||
|
||||
import android.util.Log
|
||||
import androidx.compose.foundation.Canvas
|
||||
import androidx.compose.foundation.layout.Box
|
||||
import androidx.compose.foundation.layout.padding
|
||||
import androidx.compose.foundation.layout.size
|
||||
import androidx.compose.foundation.shape.CircleShape
|
||||
import androidx.compose.material.icons.Icons
|
||||
import androidx.compose.material.icons.automirrored.filled.VolumeDown
|
||||
import androidx.compose.material.icons.automirrored.filled.VolumeMute
|
||||
import androidx.compose.material.icons.automirrored.filled.VolumeOff
|
||||
import androidx.compose.material.icons.automirrored.filled.VolumeUp
|
||||
import androidx.compose.material.icons.filled.BrightnessHigh
|
||||
import androidx.compose.material.icons.filled.BrightnessLow
|
||||
import androidx.compose.material.icons.filled.BrightnessMedium
|
||||
import androidx.compose.material3.Icon
|
||||
import androidx.compose.material3.Surface
|
||||
import androidx.compose.ui.graphics.drawscope.Stroke
|
||||
import androidx.compose.runtime.Composable
|
||||
import androidx.compose.ui.Alignment
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.draw.shadow
|
||||
import androidx.compose.ui.geometry.Offset
|
||||
import androidx.compose.ui.geometry.Size
|
||||
import androidx.compose.ui.graphics.Color
|
||||
import androidx.compose.ui.graphics.StrokeCap
|
||||
import androidx.compose.ui.res.stringResource
|
||||
import androidx.compose.ui.tooling.preview.Preview
|
||||
import androidx.compose.ui.unit.dp
|
||||
import net.newpipe.newplayer.R
|
||||
import net.newpipe.newplayer.ui.theme.VideoPlayerTheme
|
||||
|
||||
private const val TAG = "VolumeCircle"
|
||||
|
||||
private const val LINE_STROKE_WIDTH = 4
|
||||
private const val CIRCLE_SIZE = 100
|
||||
|
||||
@Composable
|
||||
fun VolumeCircle(modifier: Modifier = Modifier, volumeFraction: Float, isBrightnes: Boolean = false) {
|
||||
assert(0f < volumeFraction && volumeFraction < 1f) {
|
||||
Log.e(TAG, "Volume fraction must be in ragne [0;1]. It was $volumeFraction")
|
||||
}
|
||||
|
||||
Box(modifier.shadow(elevation = 1.dp, shape = CircleShape).padding(2.dp)){
|
||||
Canvas(Modifier.size(CIRCLE_SIZE.dp)) {
|
||||
val arcSize = (CIRCLE_SIZE - LINE_STROKE_WIDTH).dp.toPx();
|
||||
drawCircle(color = Color.Black.copy(alpha = 0.3f), radius = (CIRCLE_SIZE / 2).dp.toPx())
|
||||
drawArc(
|
||||
topLeft = Offset(
|
||||
(LINE_STROKE_WIDTH / 2).dp.toPx(), (LINE_STROKE_WIDTH / 2).dp.toPx()
|
||||
),
|
||||
size = Size(arcSize, arcSize),
|
||||
startAngle = -90f,
|
||||
sweepAngle = 360f * volumeFraction,
|
||||
useCenter = false,
|
||||
color = Color.White,
|
||||
style = Stroke(width = 4.dp.toPx(), cap = StrokeCap.Round)
|
||||
)
|
||||
}
|
||||
|
||||
Icon(
|
||||
modifier = Modifier.align(Alignment.Center).size(60.dp),
|
||||
imageVector = (if (isBrightnes) getBrightnesIcon(volumeFraction = volumeFraction)
|
||||
else getVolumeIcon(volumeFraction = volumeFraction)),
|
||||
contentDescription = stringResource(
|
||||
id = if (isBrightnes) R.string.brightness_indicator
|
||||
else R.string.volume_indicator
|
||||
)
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
@Composable
|
||||
private fun getVolumeIcon(volumeFraction: Float) =
|
||||
if (volumeFraction == 0f) Icons.AutoMirrored.Filled.VolumeOff
|
||||
else if (volumeFraction < 0.3) Icons.AutoMirrored.Filled.VolumeMute
|
||||
else if (volumeFraction < 0.6) Icons.AutoMirrored.Filled.VolumeDown
|
||||
else Icons.AutoMirrored.Filled.VolumeUp
|
||||
|
||||
@Composable
|
||||
private fun getBrightnesIcon(volumeFraction: Float) =
|
||||
if (volumeFraction < 0.3) Icons.Filled.BrightnessLow
|
||||
else if (volumeFraction < 0.6) Icons.Filled.BrightnessMedium
|
||||
else Icons.Filled.BrightnessHigh
|
||||
|
||||
@Preview(device = "spec:width=1080px,height=600px,dpi=440,orientation=landscape")
|
||||
@Composable
|
||||
fun VolumeCirclePreview() {
|
||||
VideoPlayerTheme {
|
||||
Surface(color = Color.White) {
|
||||
VolumeCircle(volumeFraction = 0.3f, isBrightnes = false)
|
||||
}
|
||||
}
|
||||
}
|
|
@ -38,4 +38,6 @@
|
|||
<string name="fast_seeking_backward">Fast seeking backward by %d seconds.</string>
|
||||
<string name="fast_seeking_forward">Fast seeking forward by %d seconds.</string>
|
||||
<string name="seconds">Seconds</string>
|
||||
<string name="volume_indicator">Volume indicator</string>
|
||||
<string name="brightness_indicator">Brightness indicator</string>
|
||||
</resources>
|
Loading…
Reference in a new issue