Add VIS vehicle signal support
[apps/mediaplayer.git] / app / MediaPlayer.qml
index d37e097..95bc22b 100644 (file)
  * limitations under the License.
  */
 
-import QtQuick 2.6
-import QtQuick.Layouts 1.1
-import QtQuick.Controls 2.0
-import QtMultimedia 5.6
+import QtQuick 2.11
+import QtQuick.Layouts 1.11
+import QtQuick.Controls 2.4
+import QtQuick.Window 2.13
+
 import AGL.Demo.Controls 1.0
-import MediaPlayer 1.0
-import 'api' as API
 
 ApplicationWindow {
     id: root
 
-    API.LightMediaScanner {
-        id: binding
-        url: bindingAddress
+    width: container.width * container.scale
+    height: container.height * container.scale
+
+    Item {
+        id: player
+
+        property string title: ""
+        property string album: ""
+        property string artist: ""
+
+        property bool av_connected: false
+
+        property int duration: 0
+        property int position: 0
+
+        property string status: "stopped"
+
+        function time2str(value) {
+            return Qt.formatTime(new Date(value), (value > 3600000) ? 'hh:mm:ss' : 'mm:ss')
+        }
     }
 
-    API.BluetoothManager {
-        id: bluetooth
-        url: bindingAddress
+    Component.onCompleted : {
+        // Let the mediaplayer backend know we're ready for metadata events
+        mediaplayer.start()
+
+        VehicleSignals.connect()
     }
 
-    MediaPlayer {
-        id: player
-        audioRole: MediaPlayer.MusicRole
-        autoLoad: true
-        playlist: playlist
+    Connections {
+        target: mediaplayer
 
-        function time2str(value) {
-            return Qt.formatTime(new Date(value), 'mm:ss')
+        onMetadataChanged: {
+            var track = metadata.track
+
+            if ('status' in metadata) {
+                player.status = metadata.status
+            }
+
+            if ('connected' in metadata) {
+                player.av_connected = metadata.connected
+            }
+
+            if (track) {
+                if ('image' in track)
+                     return
+                player.title = track.title
+                player.album = track.album
+                player.artist = track.artist
+
+                if ('duration' in track)
+                     player.duration = track.duration
+
+                if ('index' in track) {
+                     playlistview.currentIndex = track.index
+                }
+            }
+
+            if ('position' in metadata) {
+                player.position = metadata.position
+            }
+        }
+    }
+
+    Connections {
+        target: VehicleSignals
+
+        onConnected: {
+           VehicleSignals.authorize()
+        }
+
+        onAuthorized: {
+           VehicleSignals.subscribe("Vehicle.Cabin.Infotainment.Media.Action")
+       }
+
+        onSignalNotification: {
+            if (path === "Vehicle.Cabin.Infotainment.Media.Action") {
+                if (value == "SkipForward") {
+                    mediaplayer.next()
+                } else if (value == "SkipBackward") {
+                    mediaplayer.previous()
+                } else if (value == "NextSource") {
+                    if (player.av_connected)
+                        mediaplayer.connect()
+                    else
+                        mediaplayer.disconnect()
+                }
+            }
         }
     }
 
     Timer {
         id: timer
         interval: 250
-        running: (bluetooth.av_connected && bluetooth.state == "playing")
+        running: player.av_connected && player.status == "playing"
         repeat: true
         onTriggered: {
-            bluetooth.position = bluetooth.position + 250
-            slider.value = bluetooth.position
+            player.position = player.position + 250
         }
     }
 
-    Playlist {
-        id: playlist
-        playbackMode: random.checked ? Playlist.Random : loop.checked ? Playlist.Loop : Playlist.Sequential
-    }
-
+    Item {
+        id: container
+        anchors.centerIn: parent
+        width: Window.width
+       height: Window.height
+        //scale: screenInfo.scale_factor()
+        scale: 1
 
     ColumnLayout {
         anchors.fill: parent
@@ -71,14 +141,13 @@ ApplicationWindow {
             Layout.preferredHeight: 1080
             clip: true
             Image {
-                id: albumart
                 anchors.left: parent.left
                 anchors.right: parent.right
                 anchors.bottom: parent.bottom
                 height: sourceSize.height * width / sourceSize.width
                 fillMode: Image.PreserveAspectCrop
-                source: player.metaData.coverArtImage ? player.metaData.coverArtImage : ''
-                visible: bluetooth.connected == false
+                source: AlbumArt
+                visible: player.av_connected === false
             }
 
             Item {
@@ -100,17 +169,18 @@ ApplicationWindow {
                         Layout.fillHeight: true
                         Row {
                             spacing: 20
-                            ToggleButton {
-                                id: random
-                                visible: bluetooth.connected == false
-                                offImage: './images/AGL_MediaPlayer_Shuffle_Inactive.svg'
-                                onImage: './images/AGL_MediaPlayer_Shuffle_Active.svg'
-                            }
+                            //ToggleButton {
+                            //    id: random
+                            //    visible: bluetooth.connected == false
+                            //    offImage: './images/AGL_MediaPlayer_Shuffle_Inactive.svg'
+                            //    onImage: './images/AGL_MediaPlayer_Shuffle_Active.svg'
+                            //}
                             ToggleButton {
                                 id: loop
-                                visible: bluetooth.connected == false
+                                visible: player.av_connected === false
                                 offImage: './images/AGL_MediaPlayer_Loop_Inactive.svg'
                                 onImage: './images/AGL_MediaPlayer_Loop_Active.svg'
+                                onClicked: { mediaplayer.loop(checked ? "playlist" : "off") }
                             }
                         }
                         ColumnLayout {
@@ -118,13 +188,13 @@ ApplicationWindow {
                             Label {
                                 id: title
                                 Layout.alignment: Layout.Center
-                                text: bluetooth.av_connected ? bluetooth.title : (player.metaData.title ? player.metaData.title : '')
+                                text: player.title
                                 horizontalAlignment: Label.AlignHCenter
                                 verticalAlignment: Label.AlignVCenter
                             }
                             Label {
                                 Layout.alignment: Layout.Center
-                                text: bluetooth.av_connected ? bluetooth.artist : (player.metaData.contributingArtist ? player.metaData.contributingArtist : '')
+                                text: player.artist
                                 horizontalAlignment: Label.AlignHCenter
                                 verticalAlignment: Label.AlignVCenter
                                 font.pixelSize: title.font.pixelSize * 0.6
@@ -134,30 +204,24 @@ ApplicationWindow {
                     Slider {
                         id: slider
                         Layout.fillWidth: true
-                        to: bluetooth.av_connected ? bluetooth.duration : player.duration
-                        enabled: bluetooth.av_connected == false
-                        value: bluetooth.av_connected ? bluetooth.position : player.position
-                        function getPosition() {
-                            if (bluetooth.av_connected) {
-                                return player.time2str(bluetooth.position)
-                            }
-                            return player.time2str(player.position)
-                        }
+                        to: player.duration
+                        enabled: player.av_connected === false
+                        value: player.position
                         Label {
                             id: position
                             anchors.left: parent.left
                             anchors.bottom: parent.top
                             font.pixelSize: 32
-                            text: slider.getPosition()
+                            text: player.time2str(player.position)
                         }
                         Label {
                             id: duration
                             anchors.right: parent.right
                             anchors.bottom: parent.top
                             font.pixelSize: 32
-                            text: bluetooth.av_connected ? player.time2str(bluetooth.duration) : player.time2str(player.duration)
+                            text: player.time2str(player.duration)
                         }
-                        onPressedChanged: player.seek(value)
+                        onPressedChanged: mediaplayer.seek(value)
                     }
                     RowLayout {
                         Layout.fillHeight: true
@@ -172,70 +236,53 @@ ApplicationWindow {
                             id: previous
                             offImage: './images/AGL_MediaPlayer_BackArrow.svg'
                             onClicked: {
-                                if (bluetooth.av_connected) {
-                                    bluetooth.sendMediaCommand("Previous")
-                                    bluetooth.position = 0
-                                } else {
-                                    playlist.previous()
-                                }
+                                mediaplayer.previous()
                             }
                         }
                         ImageButton {
                             id: play
-                            offImage: './images/AGL_MediaPlayer_Player_Play.svg'
-                            onClicked: {
-                                if (bluetooth.av_connected) {
-                                    bluetooth.sendMediaCommand("Play")
-                                } else {
-                                    player.play()
-                                }
-                            }
                             states: [
                                 State {
-                                    when: player.playbackState === MediaPlayer.PlayingState
+                                    when: player.status == "playing"
                                     PropertyChanges {
                                         target: play
                                         offImage: './images/AGL_MediaPlayer_Player_Pause.svg'
-                                        onClicked: player.pause()
+                                        onClicked: {
+                                            mediaplayer.pause()
+                                        }
                                     }
                                 },
                                 State {
-                                    when: bluetooth.av_connected && bluetooth.state == "playing"
+                                    when: player.status != "playing"
                                     PropertyChanges {
                                         target: play
-                                        offImage: './images/AGL_MediaPlayer_Player_Pause.svg'
-                                        onClicked: bluetooth.sendMediaCommand("Pause")
+                                        offImage: './images/AGL_MediaPlayer_Player_Play.svg'
+                                        onClicked: mediaplayer.play()
                                     }
                                 }
-
                             ]
                         }
                         ImageButton {
                             id: forward
                             offImage: './images/AGL_MediaPlayer_ForwardArrow.svg'
                             onClicked: {
-                                if (bluetooth.av_connected) {
-                                    bluetooth.sendMediaCommand("Next")
-                                } else {
-                                    playlist.next()
-                                }
+                                mediaplayer.next()
                             }
                         }
 
                         Item { Layout.fillWidth: true }
  
                         ToggleButton {
-                              enabled: bluetooth.connected
-                              checked: bluetooth.av_connected
-                              offImage: './images/AGL_MediaPlayer_Bluetooth_Inactive.svg'
-                              onImage: './images/AGL_MediaPlayer_Bluetooth_Active.svg'
-
+                              visible: true
+                              checked: player.av_connected
                               onClicked: {
-                                  if (bluetooth.av_connected) {
-                                      bluetooth.disconnect_profiles()
-                                  } else {
-                                      bluetooth.connect_profiles()
-                                  }
+                                if (checked)
+                                        mediaplayer.connect()
+                                else
+                                        mediaplayer.disconnect()
+                              }
+                              contentItem: Image {
+                                source: player.av_connected ? './images/AGL_MediaPlayer_Bluetooth_Active.svg' : './images/AGL_MediaPlayer_Bluetooth_Inactive.svg'
                               }
                         }
                     }
@@ -247,23 +294,18 @@ ApplicationWindow {
             Layout.fillHeight: true
             Layout.preferredHeight: 407
 
-           PlaylistWithMetadata {
-               id: playlistmodel
-                source: playlist
-            }
-
             ListView {
                 anchors.fill: parent
                 id: playlistview
-                visible: bluetooth.av_connected == false
+                visible: player.av_connected === false
                 clip: true
                 header: Label {
                     x: 50
                     text: 'PLAYLIST'
                     opacity: 0.5
                 }
-                model: playlistmodel
-                currentIndex: playlist.currentIndex
+                model: MediaplayerModel
+                currentIndex: -1
 
                 delegate: MouseArea {
                     id: delegate
@@ -273,12 +315,6 @@ ApplicationWindow {
                         anchors.fill: parent
                         anchors.leftMargin: 50
                         anchors.rightMargin: 50
-                        Image {
-                            source: model.coverArt
-                            fillMode: Image.PreserveAspectFit
-                            Layout.preferredWidth: delegate.height
-                            Layout.preferredHeight: delegate.height
-                        }
                         ColumnLayout {
                             Layout.fillWidth: true
                             Label {
@@ -288,19 +324,18 @@ ApplicationWindow {
                             Label {
                                 Layout.fillWidth: true
                                 text: model.artist
-                                color: '#66FF99'
+                                color: '#00ADDC'
                                 font.pixelSize: 32
                             }
                         }
-                        Label {
-                            text: player.time2str(model.duration)
-                            color: '#66FF99'
-                            font.pixelSize: 32
-                        }
+                        //Label {
+                        //    text: player.time2str(model.duration)
+                        //    color: '#00ADDC'
+                        //    font.pixelSize: 32
+                        //}
                     }
                     onClicked: {
-                        playlist.currentIndex = model.index
-                        player.play()
+                        mediaplayer.picktrack(playlistview.model[index].index)
                     }
                 }
 
@@ -312,3 +347,4 @@ ApplicationWindow {
         }
     }
 }
+}