Move rest of controls to separate file

This commit is contained in:
Joseph Donofry 2021-11-09 22:17:00 -05:00
parent f6fcae124f
commit c1c9c71b08
No known key found for this signature in database
GPG key ID: E8A1D78EF044B0CB
5 changed files with 302 additions and 228 deletions

View file

@ -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
} }
}
} }

View 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
}
}

View file

@ -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
}
} }

View file

@ -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

View file

@ -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>