nheko/resources/qml/ui/media/MediaControls.qml

245 lines
7.3 KiB
QML
Raw Normal View History

// SPDX-FileCopyrightText: Nheko Contributors
2021-11-10 04:17:00 +01:00
//
// SPDX-License-Identifier: GPL-3.0-or-later
2021-11-11 06:16:25 +01:00
import "../"
2021-11-11 19:18:45 +01:00
import "../../"
2021-11-10 04:17:00 +01:00
import QtMultimedia 5.15
import QtQuick 2.15
import QtQuick.Controls 2.15
2021-11-11 06:16:25 +01:00
import QtQuick.Layouts 1.15
2021-11-10 04:17:00 +01:00
import im.nheko 1.0
2021-11-11 19:18:45 +01:00
Rectangle {
2021-11-10 04:17:00 +01:00
id: control
property alias desiredVolume: volumeSlider.desiredVolume
2021-11-11 19:18:45 +01:00
property bool muted: false
2021-11-11 06:16:25 +01:00
property bool playingVideo: false
2021-11-10 04:17:00 +01:00
property var mediaState
property bool mediaLoaded: false
property var duration
property var positionValue: 0
property var position
property bool shouldShowControls: !playingVideo || playerMouseArea.shouldShowControls || volumeSlider.state == "shown"
2021-11-10 04:17:00 +01:00
2021-11-11 19:18:45 +01:00
signal playPauseActivated()
signal loadActivated()
2021-11-10 04:17:00 +01:00
function showControls() {
controlHideTimer.restart();
}
2021-11-10 04:17:00 +01:00
function durationToString(duration) {
function maybeZeroPrepend(time) {
return (time < 10) ? "0" + time.toString() : time.toString();
}
var totalSeconds = Math.floor(duration / 1000);
var seconds = totalSeconds % 60;
var minutes = (Math.floor(totalSeconds / 60)) % 60;
var hours = (Math.floor(totalSeconds / (60 * 24))) % 24;
// Always show minutes and don't prepend zero into the leftmost element
var ss = maybeZeroPrepend(seconds);
var mm = (hours > 0) ? maybeZeroPrepend(minutes) : minutes.toString();
var hh = hours.toString();
if (hours < 1)
return mm + ":" + ss;
return hh + ":" + mm + ":" + ss;
}
2021-11-11 21:32:38 +01:00
color: {
var wc = palette.alternateBase;
2021-11-11 21:32:38 +01:00
return Qt.rgba(wc.r, wc.g, wc.b, 0.5);
}
opacity: control.shouldShowControls ? 1 : 0
height: controlLayout.implicitHeight
HoverHandler {
2021-11-10 04:17:00 +01:00
id: playerMouseArea
property bool shouldShowControls: hovered || controlHideTimer.running || control.mediaState != MediaPlayer.PlayingState
2021-11-10 04:17:00 +01:00
2021-11-11 21:32:38 +01:00
onHoveredChanged: showControls()
2021-11-10 04:17:00 +01:00
}
2021-11-11 06:16:25 +01:00
ColumnLayout {
id: controlLayout
2021-11-11 21:32:38 +01:00
enabled: control.shouldShowControls
2021-11-11 19:18:45 +01:00
spacing: 0
2021-11-10 04:17:00 +01:00
anchors.bottom: control.bottom
anchors.left: control.left
anchors.right: control.right
2021-11-11 06:16:25 +01:00
NhekoSlider {
Layout.fillWidth: true
2021-11-11 19:18:45 +01:00
Layout.leftMargin: Nheko.paddingSmall
Layout.rightMargin: Nheko.paddingSmall
enabled: control.mediaLoaded
2021-11-11 06:16:25 +01:00
value: control.positionValue
onMoved: control.position = value
from: 0
to: control.duration
alwaysShowSlider: false
}
2021-11-11 19:18:45 +01:00
RowLayout {
Layout.margins: Nheko.paddingSmall
spacing: Nheko.paddingSmall
Layout.fillWidth: true
// Cache/Play/pause button
ImageButton {
id: playbackStateImage
2021-11-11 21:32:38 +01:00
Layout.alignment: Qt.AlignLeft
buttonTextColor: palette.text
2021-11-11 19:18:45 +01:00
Layout.preferredHeight: 24
Layout.preferredWidth: 24
image: {
if (control.mediaLoaded) {
if (control.mediaState == MediaPlayer.PlayingState)
2021-11-14 02:23:10 +01:00
return ":/icons/icons/ui/pause-symbol.svg";
2021-11-11 19:18:45 +01:00
else
2021-11-14 02:23:10 +01:00
return ":/icons/icons/ui/play-sign.svg";
2021-11-11 19:18:45 +01:00
} else {
2021-11-14 02:23:10 +01:00
return ":/icons/icons/ui/download.svg";
2021-11-11 19:18:45 +01:00
}
}
2021-11-11 21:32:38 +01:00
onClicked: control.mediaLoaded ? control.playPauseActivated() : control.loadActivated()
2021-11-11 06:16:25 +01:00
}
2021-11-11 19:18:45 +01:00
ImageButton {
id: volumeButton
2021-11-11 06:16:25 +01:00
2021-11-11 21:32:38 +01:00
Layout.alignment: Qt.AlignLeft
buttonTextColor: palette.text
2021-11-11 19:18:45 +01:00
Layout.preferredHeight: 24
Layout.preferredWidth: 24
image: {
2021-11-11 21:32:38 +01:00
if (control.muted || control.desiredVolume <= 0)
2021-11-14 02:23:10 +01:00
return ":/icons/icons/ui/volume-off-indicator.svg";
2021-11-11 21:32:38 +01:00
else
2021-11-14 02:23:10 +01:00
return ":/icons/icons/ui/volume-up.svg";
2021-11-11 19:18:45 +01:00
}
onClicked: control.muted = !control.muted
}
2021-11-11 06:16:25 +01:00
2021-11-11 19:18:45 +01:00
NhekoSlider {
2021-11-11 21:32:38 +01:00
id: volumeSlider
2021-11-10 04:17:00 +01:00
2021-11-11 21:32:38 +01:00
property real desiredVolume: QtMultimedia.convertVolume(volumeSlider.value, QtMultimedia.LogarithmicVolumeScale, QtMultimedia.LinearVolumeScale)
2021-11-10 04:17:00 +01:00
2021-11-11 21:32:38 +01:00
state: ""
2021-11-11 19:18:45 +01:00
Layout.alignment: Qt.AlignLeft
Layout.preferredWidth: 0
opacity: 0
orientation: Qt.Horizontal
value: 1
onDesiredVolumeChanged: {
control.muted = !(desiredVolume > 0);
2021-11-11 06:16:25 +01:00
}
2021-11-11 19:18:45 +01:00
transitions: [
Transition {
from: ""
to: "shown"
SequentialAnimation {
2021-11-11 21:32:38 +01:00
PauseAnimation {
duration: 50
}
2021-11-11 19:18:45 +01:00
NumberAnimation {
duration: 100
properties: "opacity"
2021-11-11 21:32:38 +01:00
easing.type: Easing.InQuad
2021-11-11 19:18:45 +01:00
}
2021-11-11 21:32:38 +01:00
2021-11-11 19:18:45 +01:00
}
NumberAnimation {
properties: "Layout.preferredWidth"
duration: 150
}
2021-11-11 21:32:38 +01:00
2021-11-11 19:18:45 +01:00
},
Transition {
from: "shown"
to: ""
SequentialAnimation {
2021-11-11 21:32:38 +01:00
PauseAnimation {
duration: 100
}
2021-11-11 19:18:45 +01:00
ParallelAnimation {
NumberAnimation {
duration: 100
properties: "opacity"
easing.type: Easing.InQuad
}
NumberAnimation {
properties: "Layout.preferredWidth"
duration: 150
}
2021-11-11 21:32:38 +01:00
2021-11-11 19:18:45 +01:00
}
2021-11-11 21:32:38 +01:00
2021-11-11 19:18:45 +01:00
}
2021-11-10 04:17:00 +01:00
2021-11-11 19:18:45 +01:00
}
]
2021-11-11 21:32:38 +01:00
states: State {
name: "shown"
when: Settings.mobileMode || volumeButton.hovered || volumeSlider.hovered || volumeSlider.pressed
PropertyChanges {
target: volumeSlider
Layout.preferredWidth: 100
}
PropertyChanges {
target: volumeSlider
opacity: 1
}
}
2021-11-11 19:18:45 +01:00
}
Label {
Layout.alignment: Qt.AlignRight
2022-03-21 00:48:27 +01:00
text: (!control.mediaLoaded ? "-- " : durationToString(control.positionValue)) + " / " + durationToString(control.duration)
color: palette.text
2021-11-11 19:18:45 +01:00
}
Item {
Layout.fillWidth: true
2021-11-10 04:17:00 +01:00
}
}
}
2021-11-11 19:18:45 +01:00
2021-11-10 04:17:00 +01:00
// For hiding controls on stationary cursor
Timer {
id: controlHideTimer
interval: 1500 //ms
repeat: false
}
2021-11-11 21:32:38 +01:00
// Fade controls in/out
Behavior on opacity {
OpacityAnimator {
duration: 100
}
}
2021-11-10 04:17:00 +01:00
}