mirror of
https://github.com/Nheko-Reborn/nheko.git
synced 2024-11-26 04:58:49 +03:00
Move rest of controls to separate file
This commit is contained in:
parent
f6fcae124f
commit
c1c9c71b08
5 changed files with 302 additions and 228 deletions
|
@ -11,6 +11,8 @@ import QtQuick.Layouts 1.2
|
||||||
import im.nheko 1.0
|
import im.nheko 1.0
|
||||||
|
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
|
id: content
|
||||||
|
|
||||||
required property double proportionalHeight
|
required property double proportionalHeight
|
||||||
required property int type
|
required property int type
|
||||||
required property int originalWidth
|
required property int originalWidth
|
||||||
|
@ -22,52 +24,51 @@ ColumnLayout {
|
||||||
|
|
||||||
function durationToString(duration) {
|
function durationToString(duration) {
|
||||||
function maybeZeroPrepend(time) {
|
function maybeZeroPrepend(time) {
|
||||||
return (time < 10) ? "0" + time.toString() :
|
return (time < 10) ? "0" + time.toString() : time.toString();
|
||||||
time.toString()
|
|
||||||
}
|
}
|
||||||
var totalSeconds = Math.floor(duration / 1000)
|
|
||||||
var seconds = totalSeconds % 60
|
var totalSeconds = Math.floor(duration / 1000);
|
||||||
var minutes = (Math.floor(totalSeconds / 60)) % 60
|
var seconds = totalSeconds % 60;
|
||||||
var hours = (Math.floor(totalSeconds / (60 * 24))) % 24
|
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
|
// Always show minutes and don't prepend zero into the leftmost element
|
||||||
var ss = maybeZeroPrepend(seconds)
|
var ss = maybeZeroPrepend(seconds);
|
||||||
var mm = (hours > 0) ? maybeZeroPrepend(minutes) : minutes.toString()
|
var mm = (hours > 0) ? maybeZeroPrepend(minutes) : minutes.toString();
|
||||||
var hh = hours.toString()
|
var hh = hours.toString();
|
||||||
|
if (hours < 1)
|
||||||
|
return mm + ":" + ss;
|
||||||
|
|
||||||
if (hours < 1) {
|
return hh + ":" + mm + ":" + ss;
|
||||||
return mm + ":" + ss
|
|
||||||
}
|
}
|
||||||
return hh + ":" + mm + ":" + ss
|
|
||||||
}
|
|
||||||
|
|
||||||
id: content
|
|
||||||
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|
||||||
MxcMedia {
|
MxcMedia {
|
||||||
id: mxcmedia
|
id: mxcmedia
|
||||||
|
|
||||||
// TODO: Show error in overlay or so?
|
// TODO: Show error in overlay or so?
|
||||||
onError: console.log(error)
|
onError: console.log(error)
|
||||||
roomm: room
|
roomm: room
|
||||||
// desiredVolume is a float from 0.0 -> 1.0, MediaPlayer volume is an int from 0 to 100
|
// desiredVolume is a float from 0.0 -> 1.0, MediaPlayer volume is an int from 0 to 100
|
||||||
// this value automatically gets clamped for us between these two values.
|
// this value automatically gets clamped for us between these two values.
|
||||||
volume: volumeSlider.desiredVolume * 100
|
volume: mediaControls.desiredVolume * 100
|
||||||
muted: volumeSlider.muted
|
muted: mediaControls.muted
|
||||||
}
|
}
|
||||||
|
|
||||||
Rectangle {
|
Rectangle {
|
||||||
id: videoContainer
|
id: videoContainer
|
||||||
visible: type == MtxEvent.VideoMessage
|
|
||||||
//property double tempWidth: Math.min(parent ? parent.width : undefined, model.data.width < 1 ? 400 : /////model.data.width)
|
//property double tempWidth: Math.min(parent ? parent.width : undefined, model.data.width < 1 ? 400 : /////model.data.width)
|
||||||
// property double tempWidth: (model.data.width < 1) ? 400 : model.data.width
|
// property double tempWidth: (model.data.width < 1) ? 400 : model.data.width
|
||||||
// property double tempHeight: tempWidth * model.data.proportionalHeight
|
// property double tempHeight: tempWidth * model.data.proportionalHeight
|
||||||
//property double tempWidth: Math.min(parent ? parent.width : undefined, originalWidth < 1 ? 400 : originalWidth)
|
//property double tempWidth: Math.min(parent ? parent.width : undefined, originalWidth < 1 ? 400 : originalWidth)
|
||||||
property double tempWidth: Math.min(parent ? parent.width: undefined, originalWidth < 1 ? 400 : originalWidth)
|
property double tempWidth: Math.min(parent ? parent.width : undefined, originalWidth < 1 ? 400 : originalWidth)
|
||||||
property double tempHeight: tempWidth * proportionalHeight
|
property double tempHeight: tempWidth * proportionalHeight
|
||||||
|
|
||||||
property double divisor: isReply ? 4 : 2
|
property double divisor: isReply ? 4 : 2
|
||||||
property bool tooHigh: tempHeight > timelineRoot.height / divisor
|
property bool tooHigh: tempHeight > timelineRoot.height / divisor
|
||||||
color: Nheko.colors.window
|
|
||||||
|
|
||||||
|
visible: type == MtxEvent.VideoMessage
|
||||||
|
color: Nheko.colors.window
|
||||||
Layout.preferredHeight: tooHigh ? timelineRoot.height / divisor : tempHeight
|
Layout.preferredHeight: tooHigh ? timelineRoot.height / divisor : tempHeight
|
||||||
Layout.preferredWidth: tooHigh ? (timelineRoot.height / divisor) / proportionalHeight : tempWidth
|
Layout.preferredWidth: tooHigh ? (timelineRoot.height / divisor) / proportionalHeight : tempWidth
|
||||||
Layout.maximumWidth: Layout.preferredWidth
|
Layout.maximumWidth: Layout.preferredWidth
|
||||||
|
@ -77,102 +78,115 @@ ColumnLayout {
|
||||||
source: thumbnailUrl.replace("mxc://", "image://MxcImage/")
|
source: thumbnailUrl.replace("mxc://", "image://MxcImage/")
|
||||||
asynchronous: true
|
asynchronous: true
|
||||||
fillMode: Image.PreserveAspectFit
|
fillMode: Image.PreserveAspectFit
|
||||||
|
|
||||||
// Button and window colored overlay to cache media
|
// Button and window colored overlay to cache media
|
||||||
Item {
|
Item {
|
||||||
// Display over video controls
|
// Display over video controls
|
||||||
z: videoOutput.z + 1
|
z: videoOutput.z + 1
|
||||||
visible: !mxcmedia.loaded
|
visible: !mxcmedia.loaded
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
|
|
||||||
//color: Nheko.colors.window
|
//color: Nheko.colors.window
|
||||||
//opacity: 0.5
|
//opacity: 0.5
|
||||||
Image {
|
Image {
|
||||||
property color buttonColor: (cacheVideoArea.containsMouse) ? Nheko.colors.highlight :
|
property color buttonColor: (cacheVideoArea.containsMouse) ? Nheko.colors.highlight : Nheko.colors.text
|
||||||
Nheko.colors.text
|
|
||||||
|
|
||||||
anchors.verticalCenter: parent.verticalCenter
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
anchors.horizontalCenter: parent.horizontalCenter
|
anchors.horizontalCenter: parent.horizontalCenter
|
||||||
source: "image://colorimage/:/icons/icons/ui/arrow-pointing-down.png?"+buttonColor
|
source: "image://colorimage/:/icons/icons/ui/arrow-pointing-down.png?" + buttonColor
|
||||||
}
|
}
|
||||||
|
|
||||||
MouseArea {
|
MouseArea {
|
||||||
id: cacheVideoArea
|
id: cacheVideoArea
|
||||||
|
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
hoverEnabled: true
|
hoverEnabled: true
|
||||||
enabled: !mxcmedia.loaded
|
enabled: !mxcmedia.loaded
|
||||||
onClicked: mxcmedia.eventId = eventId
|
onClicked: mxcmedia.eventId = eventId
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
VideoOutput {
|
VideoOutput {
|
||||||
id: videoOutput
|
id: videoOutput
|
||||||
|
|
||||||
clip: true
|
clip: true
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
fillMode: VideoOutput.PreserveAspectFit
|
fillMode: VideoOutput.PreserveAspectFit
|
||||||
source: mxcmedia
|
source: mxcmedia
|
||||||
flushMode: VideoOutput.FirstFrame
|
flushMode: VideoOutput.FirstFrame
|
||||||
|
|
||||||
// TODO: once we can use Qt 5.12, use HoverHandler
|
MediaControls {
|
||||||
MouseArea {
|
id: mediaControls
|
||||||
id: playerMouseArea
|
|
||||||
// Toggle play state on clicks
|
|
||||||
onClicked: {
|
|
||||||
if (controlRect.shouldShowControls &&
|
|
||||||
!controlRect.contains(mapToItem(controlRect, mouseX, mouseY))) {
|
|
||||||
(mxcmedia.state == MediaPlayer.PlayingState) ?
|
|
||||||
mxcmedia.pause() :
|
|
||||||
mxcmedia.play()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Rectangle {
|
|
||||||
id: controlRect
|
|
||||||
property int controlHeight: 25
|
|
||||||
property bool shouldShowControls: playerMouseArea.shouldShowControls ||
|
|
||||||
volumeSlider.controlsVisible
|
|
||||||
|
|
||||||
anchors.bottom: playerMouseArea.bottom
|
anchors.fill: parent
|
||||||
// Window color with 128/255 alpha
|
x: videoOutput.contentRect.x
|
||||||
color: {
|
y: videoOutput.contentRect.y
|
||||||
var wc = Nheko.colors.alternateBase
|
width: videoOutput.contentRect.width
|
||||||
return Qt.rgba(wc.r, wc.g, wc.b, 0.5)
|
height: videoOutput.contentRect.height
|
||||||
|
positionValue: mxcmedia.position
|
||||||
|
duration: mxcmedia.duration
|
||||||
|
mediaLoaded: mxcmedia.loaded
|
||||||
|
mediaState: mxcmedia.state
|
||||||
|
volumeOrientation: Qt.Vertical
|
||||||
|
onPositionChanged: mxcmedia.position = position
|
||||||
|
onActivated: mxcmedia.state == MediaPlayer.PlayingState ? mxcmedia.pause() : mxcmedia.play()
|
||||||
}
|
}
|
||||||
height: 40
|
|
||||||
width: playerMouseArea.width
|
|
||||||
opacity: shouldShowControls ? 1 : 0
|
|
||||||
// Fade controls in/out
|
|
||||||
Behavior on opacity {
|
|
||||||
OpacityAnimator {
|
|
||||||
duration: 100
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
// Audio player
|
||||||
|
|
||||||
|
// TODO: share code with the video player
|
||||||
|
Rectangle {
|
||||||
|
id: audioControlRect
|
||||||
|
|
||||||
|
property int controlHeight: 25
|
||||||
|
|
||||||
|
visible: type != MtxEvent.VideoMessage
|
||||||
|
Layout.preferredHeight: 40
|
||||||
|
|
||||||
RowLayout {
|
RowLayout {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
width: parent.width
|
width: parent.width
|
||||||
|
|
||||||
// Play/pause button
|
// Play/pause button
|
||||||
Image {
|
Image {
|
||||||
id: playbackStateImage
|
id: audioPlaybackStateImage
|
||||||
|
|
||||||
|
property color controlColor: (audioPlaybackStateArea.containsMouse) ? Nheko.colors.highlight : Nheko.colors.text
|
||||||
|
|
||||||
fillMode: Image.PreserveAspectFit
|
fillMode: Image.PreserveAspectFit
|
||||||
Layout.preferredHeight: controlRect.controlHeight
|
Layout.preferredHeight: controlRect.controlHeight
|
||||||
Layout.alignment: Qt.AlignVCenter
|
Layout.alignment: Qt.AlignVCenter
|
||||||
property color controlColor: (playbackStateArea.containsMouse) ?
|
source: {
|
||||||
Nheko.colors.highlight : Nheko.colors.text
|
if (!mxcmedia.loaded)
|
||||||
|
return "image://colorimage/:/icons/icons/ui/arrow-pointing-down.png?" + controlColor;
|
||||||
|
|
||||||
|
return (mxcmedia.state == MediaPlayer.PlayingState) ? "image://colorimage/:/icons/icons/ui/pause-symbol.png?" + controlColor : "image://colorimage/:/icons/icons/ui/play-sign.png?" + controlColor;
|
||||||
|
}
|
||||||
|
|
||||||
source: (mxcmedia.state == MediaPlayer.PlayingState) ?
|
|
||||||
"image://colorimage/:/icons/icons/ui/pause-symbol.png?"+controlColor :
|
|
||||||
"image://colorimage/:/icons/icons/ui/play-sign.png?"+controlColor
|
|
||||||
MouseArea {
|
MouseArea {
|
||||||
id: playbackStateArea
|
id: audioPlaybackStateArea
|
||||||
|
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
hoverEnabled: true
|
hoverEnabled: true
|
||||||
onClicked: {
|
onClicked: {
|
||||||
(mxcmedia.state == MediaPlayer.PlayingState) ?
|
if (!mxcmedia.loaded) {
|
||||||
mxcmedia.pause() :
|
mxcmedia.eventId = eventId;
|
||||||
mxcmedia.play()
|
return ;
|
||||||
|
}
|
||||||
|
(mxcmedia.state == MediaPlayer.PlayingState) ? mxcmedia.pause() : mxcmedia.play();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Label {
|
Label {
|
||||||
text: (!mxcmedia.loaded) ? "-/-" : (durationToString(mxcmedia.position) + "/" + durationToString(mxcmedia.duration))
|
text: (!mxcmedia.loaded) ? "-/-" : durationToString(mxcmedia.position) + "/" + durationToString(mxcmedia.duration)
|
||||||
color: Nheko.colors.text
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Slider {
|
Slider {
|
||||||
|
@ -185,109 +199,23 @@ ColumnLayout {
|
||||||
to: mxcmedia.duration
|
to: mxcmedia.duration
|
||||||
}
|
}
|
||||||
|
|
||||||
VolumeControl {
|
|
||||||
id: volumeSlider
|
|
||||||
orientation: Qt.Vertical
|
|
||||||
Layout.rightMargin: 5
|
|
||||||
Layout.preferredHeight: controlRect.controlHeight
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
// This breaks separation of concerns but this same thing doesn't work when called from controlRect...
|
|
||||||
property bool shouldShowControls: (containsMouse && controlHideTimer.running) ||
|
|
||||||
(mxcmedia.state != MediaPlayer.PlayingState) ||
|
|
||||||
controlRect.contains(mapToItem(controlRect, mouseX, mouseY))
|
|
||||||
|
|
||||||
// For hiding controls on stationary cursor
|
|
||||||
Timer {
|
|
||||||
id: controlHideTimer
|
|
||||||
interval: 1500 //ms
|
|
||||||
repeat: false
|
|
||||||
}
|
|
||||||
|
|
||||||
hoverEnabled: true
|
|
||||||
onPositionChanged: controlHideTimer.start()
|
|
||||||
|
|
||||||
x: videoOutput.contentRect.x
|
|
||||||
y: videoOutput.contentRect.y
|
|
||||||
width: videoOutput.contentRect.width
|
|
||||||
height: videoOutput.contentRect.height
|
|
||||||
propagateComposedEvents: true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Audio player
|
|
||||||
// TODO: share code with the video player
|
|
||||||
Rectangle {
|
|
||||||
id: audioControlRect
|
|
||||||
|
|
||||||
visible: type != MtxEvent.VideoMessage
|
|
||||||
property int controlHeight: 25
|
|
||||||
Layout.preferredHeight: 40
|
|
||||||
RowLayout {
|
|
||||||
anchors.fill: parent
|
|
||||||
width: parent.width
|
|
||||||
// Play/pause button
|
|
||||||
Image {
|
|
||||||
id: audioPlaybackStateImage
|
|
||||||
fillMode: Image.PreserveAspectFit
|
|
||||||
Layout.preferredHeight: controlRect.controlHeight
|
|
||||||
Layout.alignment: Qt.AlignVCenter
|
|
||||||
property color controlColor: (audioPlaybackStateArea.containsMouse) ?
|
|
||||||
Nheko.colors.highlight : Nheko.colors.text
|
|
||||||
|
|
||||||
source: {
|
|
||||||
if (!mxcmedia.loaded)
|
|
||||||
return "image://colorimage/:/icons/icons/ui/arrow-pointing-down.png?"+controlColor
|
|
||||||
return (mxcmedia.state == MediaPlayer.PlayingState) ?
|
|
||||||
"image://colorimage/:/icons/icons/ui/pause-symbol.png?"+controlColor :
|
|
||||||
"image://colorimage/:/icons/icons/ui/play-sign.png?"+controlColor
|
|
||||||
}
|
|
||||||
MouseArea {
|
|
||||||
id: audioPlaybackStateArea
|
|
||||||
|
|
||||||
anchors.fill: parent
|
|
||||||
hoverEnabled: true
|
|
||||||
onClicked: {
|
|
||||||
if (!mxcmedia.loaded) {
|
|
||||||
mxcmedia.eventId = eventId
|
|
||||||
return
|
|
||||||
}
|
|
||||||
(mxcmedia.state == MediaPlayer.PlayingState) ?
|
|
||||||
mxcmedia.pause() :
|
|
||||||
mxcmedia.play()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Label {
|
|
||||||
text: (!mxcmedia.loaded) ? "-/-" :
|
|
||||||
durationToString(mxcmedia.position) + "/" + durationToString(mxcmedia.duration)
|
|
||||||
}
|
|
||||||
|
|
||||||
Slider {
|
|
||||||
Layout.fillWidth: true
|
|
||||||
Layout.minimumWidth: 50
|
|
||||||
height: controlRect.controlHeight
|
|
||||||
value: mxcmedia.position
|
|
||||||
onMoved: mxcmedia.seek(value)
|
|
||||||
from: 0
|
|
||||||
to: mxcmedia.duration
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Label {
|
Label {
|
||||||
id: fileInfoLabel
|
id: fileInfoLabel
|
||||||
|
|
||||||
background: Rectangle {
|
|
||||||
color: Nheko.colors.base
|
|
||||||
}
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
text: body + " [" + filesize + "]"
|
text: body + " [" + filesize + "]"
|
||||||
textFormat: Text.PlainText
|
textFormat: Text.PlainText
|
||||||
elide: Text.ElideRight
|
elide: Text.ElideRight
|
||||||
color: Nheko.colors.text
|
color: Nheko.colors.text
|
||||||
|
|
||||||
|
background: Rectangle {
|
||||||
|
color: Nheko.colors.base
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
141
resources/qml/ui/media/MediaControls.qml
Normal file
141
resources/qml/ui/media/MediaControls.qml
Normal file
|
@ -0,0 +1,141 @@
|
||||||
|
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
||||||
|
//
|
||||||
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
import QtMultimedia 5.15
|
||||||
|
import QtQuick 2.15
|
||||||
|
import QtQuick.Controls 2.15
|
||||||
|
import QtQuick.Layouts 1.2
|
||||||
|
import im.nheko 1.0
|
||||||
|
|
||||||
|
Item {
|
||||||
|
id: control
|
||||||
|
|
||||||
|
property alias desiredVolume: volumeSlider.desiredVolume
|
||||||
|
property alias muted: volumeSlider.muted
|
||||||
|
property alias volumeOrientation: volumeSlider.orientation
|
||||||
|
property var mediaState
|
||||||
|
property bool mediaLoaded: false
|
||||||
|
property var duration
|
||||||
|
property var positionValue: 0
|
||||||
|
property var position
|
||||||
|
property int controlHeight: 25
|
||||||
|
property bool shouldShowControls: playerMouseArea.shouldShowControls || volumeSlider.controlsVisible
|
||||||
|
|
||||||
|
signal activated(real mouseX, real mouseY)
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
MouseArea {
|
||||||
|
id: playerMouseArea
|
||||||
|
|
||||||
|
property bool shouldShowControls: (containsMouse && controlHideTimer.running) || (control.mediaState != MediaPlayer.PlayingState) || controlRect.contains(mapToItem(controlRect, mouseX, mouseY))
|
||||||
|
|
||||||
|
onClicked: control.activated(mouseX, mouseY)
|
||||||
|
hoverEnabled: true
|
||||||
|
onPositionChanged: controlHideTimer.start()
|
||||||
|
onExited: controlHideTimer.start()
|
||||||
|
onEntered: controlHideTimer.start()
|
||||||
|
anchors.fill: control
|
||||||
|
propagateComposedEvents: true
|
||||||
|
}
|
||||||
|
|
||||||
|
Rectangle {
|
||||||
|
id: controlRect
|
||||||
|
|
||||||
|
// Window color with 128/255 alpha
|
||||||
|
color: {
|
||||||
|
var wc = Nheko.colors.alternateBase;
|
||||||
|
return Qt.rgba(wc.r, wc.g, wc.b, 0.5);
|
||||||
|
}
|
||||||
|
anchors.bottom: control.bottom
|
||||||
|
anchors.left: control.left
|
||||||
|
anchors.right: control.right
|
||||||
|
height: 40
|
||||||
|
opacity: control.shouldShowControls ? 1 : 0
|
||||||
|
|
||||||
|
RowLayout {
|
||||||
|
anchors.fill: parent
|
||||||
|
width: parent.width
|
||||||
|
|
||||||
|
// Play/pause button
|
||||||
|
Image {
|
||||||
|
id: playbackStateImage
|
||||||
|
|
||||||
|
property color controlColor: (playbackStateArea.containsMouse) ? Nheko.colors.highlight : Nheko.colors.text
|
||||||
|
|
||||||
|
fillMode: Image.PreserveAspectFit
|
||||||
|
Layout.preferredHeight: control.controlHeight
|
||||||
|
Layout.alignment: Qt.AlignVCenter
|
||||||
|
source: (control.mediaState == MediaPlayer.PlayingState) ? "image://colorimage/:/icons/icons/ui/pause-symbol.png?" + controlColor : "image://colorimage/:/icons/icons/ui/play-sign.png?" + controlColor
|
||||||
|
|
||||||
|
MouseArea {
|
||||||
|
id: playbackStateArea
|
||||||
|
|
||||||
|
anchors.fill: parent
|
||||||
|
hoverEnabled: true
|
||||||
|
onClicked: control.activated(mouseX, mouseY)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
Label {
|
||||||
|
text: (!control.mediaLoaded) ? "-/-" : (durationToString(control.positionValue) + "/" + durationToString(control.duration))
|
||||||
|
color: Nheko.colors.text
|
||||||
|
}
|
||||||
|
|
||||||
|
Slider {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.minimumWidth: 50
|
||||||
|
height: control.controlHeight
|
||||||
|
value: control.positionValue
|
||||||
|
onMoved: control.position = value
|
||||||
|
from: 0
|
||||||
|
to: control.duration
|
||||||
|
}
|
||||||
|
|
||||||
|
VolumeControl {
|
||||||
|
id: volumeSlider
|
||||||
|
|
||||||
|
Layout.rightMargin: 5
|
||||||
|
Layout.preferredHeight: control.controlHeight
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fade controls in/out
|
||||||
|
Behavior on opacity {
|
||||||
|
OpacityAnimator {
|
||||||
|
duration: 100
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// For hiding controls on stationary cursor
|
||||||
|
Timer {
|
||||||
|
id: controlHideTimer
|
||||||
|
|
||||||
|
interval: 1500 //ms
|
||||||
|
repeat: false
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -5,101 +5,104 @@
|
||||||
import QtMultimedia 5.15
|
import QtMultimedia 5.15
|
||||||
import QtQuick 2.15
|
import QtQuick 2.15
|
||||||
import QtQuick.Controls 2.15
|
import QtQuick.Controls 2.15
|
||||||
|
|
||||||
import im.nheko 1.0
|
import im.nheko 1.0
|
||||||
|
|
||||||
// Volume slider activator
|
// Volume slider activator
|
||||||
Image {
|
Image {
|
||||||
|
// TODO: add icons for different volume levels
|
||||||
|
id: volumeImage
|
||||||
|
|
||||||
property alias desiredVolume: volumeSlider.desiredVolume
|
property alias desiredVolume: volumeSlider.desiredVolume
|
||||||
property alias orientation: volumeSlider.orientation
|
property alias orientation: volumeSlider.orientation
|
||||||
property alias controlsVisible: volumeSliderRect.visible
|
property alias controlsVisible: volumeSliderRect.visible
|
||||||
property bool muted: false
|
property bool muted: false
|
||||||
property color controlColor: (volumeImageArea.containsMouse) ?
|
property color controlColor: (volumeImageArea.containsMouse) ? Nheko.colors.highlight : Nheko.colors.text
|
||||||
Nheko.colors.highlight : Nheko.colors.text
|
|
||||||
|
|
||||||
// TODO: add icons for different volume levels
|
|
||||||
id: volumeImage
|
|
||||||
source: (desiredVolume > 0 && !muted) ?
|
|
||||||
"image://colorimage/:/icons/icons/ui/volume-up.png?"+ controlColor :
|
|
||||||
"image://colorimage/:/icons/icons/ui/volume-off-indicator.png?"+ controlColor
|
|
||||||
|
|
||||||
|
source: (desiredVolume > 0 && !muted) ? "image://colorimage/:/icons/icons/ui/volume-up.png?" + controlColor : "image://colorimage/:/icons/icons/ui/volume-off-indicator.png?" + controlColor
|
||||||
fillMode: Image.PreserveAspectFit
|
fillMode: Image.PreserveAspectFit
|
||||||
|
|
||||||
MouseArea {
|
MouseArea {
|
||||||
id: volumeImageArea
|
id: volumeImageArea
|
||||||
|
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
hoverEnabled: true
|
hoverEnabled: true
|
||||||
onExited: volumeSliderHideTimer.start()
|
onExited: volumeSliderHideTimer.start()
|
||||||
onPositionChanged: volumeSliderHideTimer.start()
|
onPositionChanged: volumeSliderHideTimer.start()
|
||||||
onClicked: volumeImage.muted = !volumeImage.muted
|
onClicked: volumeImage.muted = !volumeImage.muted
|
||||||
|
|
||||||
// For hiding volume slider after a while
|
// For hiding volume slider after a while
|
||||||
Timer {
|
Timer {
|
||||||
id: volumeSliderHideTimer
|
id: volumeSliderHideTimer
|
||||||
|
|
||||||
interval: 1500
|
interval: 1500
|
||||||
repeat: false
|
repeat: false
|
||||||
running: false
|
running: false
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Rectangle {
|
Rectangle {
|
||||||
id: volumeSliderRect
|
id: volumeSliderRect
|
||||||
|
|
||||||
opacity: (visible) ? 1 : 0
|
opacity: (visible) ? 1 : 0
|
||||||
Behavior on opacity {
|
|
||||||
OpacityAnimator {
|
|
||||||
duration: 100
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// TODO: figure out a better way to put the slider popup above controlRect
|
|
||||||
anchors.bottom: volumeImage.top
|
anchors.bottom: volumeImage.top
|
||||||
anchors.bottomMargin: 10
|
anchors.bottomMargin: 10
|
||||||
anchors.horizontalCenter: volumeImage.horizontalCenter
|
anchors.horizontalCenter: volumeImage.horizontalCenter
|
||||||
color: {
|
color: {
|
||||||
var wc = Nheko.colors.window
|
var wc = Nheko.colors.window;
|
||||||
return Qt.rgba(wc.r, wc.g, wc.b, 0.5)
|
return Qt.rgba(wc.r, wc.g, wc.b, 0.5);
|
||||||
}
|
}
|
||||||
/* TODO: base width on the slider width (some issue with it not having a geometry
|
/* TODO: base width on the slider width (some issue with it not having a geometry
|
||||||
when using the width here?) */
|
when using the width here?) */
|
||||||
width: volumeImage.width * 0.7
|
width: volumeImage.width * 0.7
|
||||||
radius: volumeSlider.width / 2
|
radius: volumeSlider.width / 2
|
||||||
height: controlRect.height * 2 //100
|
height: controlRect.height * 2 //100
|
||||||
visible: volumeImageArea.containsMouse ||
|
visible: volumeImageArea.containsMouse || volumeSliderHideTimer.running || volumeSliderRectMouseArea.containsMouse
|
||||||
volumeSliderHideTimer.running ||
|
|
||||||
volumeSliderRectMouseArea.containsMouse
|
|
||||||
Slider {
|
Slider {
|
||||||
// TODO: the slider is slightly off-center on the left for some reason...
|
// TODO: the slider is slightly off-center on the left for some reason...
|
||||||
id: volumeSlider
|
id: volumeSlider
|
||||||
|
|
||||||
value: 1.0
|
|
||||||
// Desired value to avoid loop onMoved -> media.volume -> value -> onMoved...
|
// Desired value to avoid loop onMoved -> media.volume -> value -> onMoved...
|
||||||
property real desiredVolume: QtMultimedia.convertVolume(volumeSlider.value,
|
property real desiredVolume: QtMultimedia.convertVolume(volumeSlider.value, QtMultimedia.LogarithmicVolumeScale, QtMultimedia.LinearVolumeScale)
|
||||||
QtMultimedia.LogarithmicVolumeScale,
|
|
||||||
QtMultimedia.LinearVolumeScale)
|
|
||||||
|
|
||||||
anchors.fill: parent
|
value: 1
|
||||||
|
anchors.fill: volumeSliderRect
|
||||||
anchors.bottomMargin: volumeSliderRect.height * 0.1
|
anchors.bottomMargin: volumeSliderRect.height * 0.1
|
||||||
anchors.topMargin: volumeSliderRect.height * 0.1
|
anchors.topMargin: volumeSliderRect.height * 0.1
|
||||||
anchors.horizontalCenter: volumeSliderRect.horizontalCenter
|
anchors.horizontalCenter: volumeSliderRect.horizontalCenter
|
||||||
orientation: Qt.Vertical
|
orientation: Qt.Vertical
|
||||||
onDesiredVolumeChanged: {
|
onDesiredVolumeChanged: {
|
||||||
volumeImage.muted = !(desiredVolume > 0.0)
|
volumeImage.muted = !(desiredVolume > 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
// Used for resetting the timer on mouse moves on volumeSliderRect
|
// Used for resetting the timer on mouse moves on volumeSliderRect
|
||||||
|
|
||||||
MouseArea {
|
MouseArea {
|
||||||
id: volumeSliderRectMouseArea
|
id: volumeSliderRectMouseArea
|
||||||
|
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
hoverEnabled: true
|
hoverEnabled: true
|
||||||
propagateComposedEvents: true
|
propagateComposedEvents: true
|
||||||
onExited: volumeSliderHideTimer.start()
|
onExited: volumeSliderHideTimer.start()
|
||||||
|
|
||||||
onClicked: mouse.accepted = false
|
onClicked: mouse.accepted = false
|
||||||
onPressed: mouse.accepted = false
|
onPressed: mouse.accepted = false
|
||||||
onReleased: mouse.accepted = false
|
onReleased: mouse.accepted = false
|
||||||
onPressAndHold: mouse.accepted = false
|
onPressAndHold: mouse.accepted = false
|
||||||
onPositionChanged: {
|
onPositionChanged: {
|
||||||
mouse.accepted = false
|
mouse.accepted = false;
|
||||||
volumeSliderHideTimer.start()
|
volumeSliderHideTimer.start();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Behavior on opacity {
|
||||||
|
OpacityAnimator {
|
||||||
|
duration: 100
|
||||||
}
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
// TODO: figure out a better way to put the slider popup above controlRect
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -1,2 +1,3 @@
|
||||||
module im.nheko.UI.Media
|
module im.nheko.UI.Media
|
||||||
VolumeSlider 1.0 VolumeSlider.qml
|
VolumeSlider 1.0 VolumeSlider.qml
|
||||||
|
MediaControls 1.0 MediaControls.qml
|
|
@ -183,6 +183,7 @@
|
||||||
<file>qml/ui/Ripple.qml</file>
|
<file>qml/ui/Ripple.qml</file>
|
||||||
<file>qml/ui/Spinner.qml</file>
|
<file>qml/ui/Spinner.qml</file>
|
||||||
<file>qml/ui/animations/BlinkAnimation.qml</file>
|
<file>qml/ui/animations/BlinkAnimation.qml</file>
|
||||||
|
<file>qml/ui/media/MediaControls.qml</file>
|
||||||
<file>qml/ui/media/VolumeControl.qml</file>
|
<file>qml/ui/media/VolumeControl.qml</file>
|
||||||
<file>qml/voip/ActiveCallBar.qml</file>
|
<file>qml/voip/ActiveCallBar.qml</file>
|
||||||
<file>qml/voip/CallDevices.qml</file>
|
<file>qml/voip/CallDevices.qml</file>
|
||||||
|
|
Loading…
Reference in a new issue