2020-09-27 18:38:33 +00:00
|
|
|
/*
|
|
|
|
Sailfin: a Jellyfin client written using Qt
|
|
|
|
Copyright (C) 2020 Chris Josten
|
|
|
|
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU Lesser General Public
|
|
|
|
License as published by the Free Software Foundation; either
|
|
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
This library 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
|
|
|
|
Lesser General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
|
|
License along with this library; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
2020-09-25 12:46:39 +00:00
|
|
|
import QtQuick 2.6
|
|
|
|
import QtMultimedia 5.6
|
|
|
|
import Sailfish.Silica 1.0
|
|
|
|
|
|
|
|
import nl.netsoj.chris.Jellyfin 1.0
|
|
|
|
|
|
|
|
import "videoplayer"
|
2020-10-01 19:45:34 +00:00
|
|
|
import "../"
|
2020-09-25 12:46:39 +00:00
|
|
|
|
2020-09-25 13:21:08 +00:00
|
|
|
/**
|
|
|
|
* A videoPlayer for Jellyfin videos
|
|
|
|
*/
|
|
|
|
|
|
|
|
SilicaItem {
|
2020-09-25 12:46:39 +00:00
|
|
|
id: playerRoot
|
|
|
|
property string itemId
|
|
|
|
property string title
|
|
|
|
property int progress
|
|
|
|
readonly property bool landscape: videoOutput.contentRect.width > videoOutput.contentRect.height
|
|
|
|
property MediaPlayer player
|
2020-09-27 15:59:11 +00:00
|
|
|
readonly property bool hudVisible: !hud.hidden || player.error !== MediaPlayer.NoError
|
2020-09-25 15:14:44 +00:00
|
|
|
property alias audioTrack: mediaSource.audioIndex
|
|
|
|
property alias subtitleTrack: mediaSource.subtitleIndex
|
2020-10-09 00:33:08 +00:00
|
|
|
property real startTicks: 0
|
2020-09-25 12:46:39 +00:00
|
|
|
|
2020-09-25 13:21:08 +00:00
|
|
|
// Blackground to prevent the ambience from leaking through
|
|
|
|
Rectangle {
|
|
|
|
anchors.fill: parent
|
2020-10-10 15:28:13 +00:00
|
|
|
color: Theme.overlayBackgroundColor
|
2020-09-25 13:21:08 +00:00
|
|
|
}
|
|
|
|
|
2020-10-08 01:00:08 +00:00
|
|
|
PlaybackManager {
|
2020-09-25 12:46:39 +00:00
|
|
|
id: mediaSource
|
|
|
|
apiClient: ApiClient
|
|
|
|
itemId: playerRoot.itemId
|
|
|
|
autoOpen: true
|
|
|
|
onStreamUrlChanged: {
|
|
|
|
if (mediaSource.streamUrl != "") {
|
|
|
|
player.source = streamUrl
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-01 19:45:34 +00:00
|
|
|
Connections {
|
|
|
|
target: player
|
|
|
|
onPlaybackStateChanged: mediaSource.state = player.playbackState
|
|
|
|
onPositionChanged: mediaSource.position = Utils.msToTicks(player.position)
|
|
|
|
}
|
|
|
|
|
2020-09-25 12:46:39 +00:00
|
|
|
|
|
|
|
VideoOutput {
|
|
|
|
id: videoOutput
|
|
|
|
source: player
|
|
|
|
anchors.fill: parent
|
|
|
|
}
|
|
|
|
|
|
|
|
VideoHud {
|
|
|
|
id: hud
|
|
|
|
anchors.fill: parent
|
|
|
|
player: playerRoot.player
|
|
|
|
title: videoPlayer.title
|
|
|
|
|
|
|
|
Label {
|
|
|
|
anchors.fill: parent
|
|
|
|
anchors.margins: Theme.horizontalPageMargin
|
|
|
|
text: itemId + "\n" + mediaSource.streamUrl + "\n"
|
|
|
|
+ player.status + "\n"
|
|
|
|
+ player.bufferProgress + "\n"
|
|
|
|
+ player.metaData.videoCodec + "@" + player.metaData.videoFrameRate + "(" + player.metaData.videoBitRate + ")" + "\n"
|
|
|
|
+ player.metaData.audioCodec + "(" + player.metaData.audioBitRate + ")" + "\n"
|
|
|
|
+ player.errorString + "\n"
|
|
|
|
font.pixelSize: Theme.fontSizeExtraSmall
|
|
|
|
wrapMode: "WordWrap"
|
|
|
|
visible: false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-27 15:59:11 +00:00
|
|
|
VideoError {
|
|
|
|
anchors.fill: videoOutput
|
|
|
|
player: playerRoot.player
|
|
|
|
}
|
|
|
|
|
2020-09-25 12:46:39 +00:00
|
|
|
function stop() {
|
|
|
|
player.stop()
|
2020-10-01 19:45:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Connections {
|
2020-10-01 20:15:17 +00:00
|
|
|
property bool enabled: true
|
2020-10-01 19:45:34 +00:00
|
|
|
id: playerReadyToSeek
|
|
|
|
target: player
|
|
|
|
onPlaybackStateChanged: {
|
2020-10-01 20:15:17 +00:00
|
|
|
if (!enabled) return;
|
2020-10-01 19:45:34 +00:00
|
|
|
if (startTicks > 0 && player.playbackState == MediaPlayer.PlayingState) {
|
|
|
|
console.log("Seeking to " + Utils.ticksToMs(startTicks))
|
|
|
|
player.seek(Utils.ticksToMs(startTicks))
|
|
|
|
playerReadyToSeek.enabled = false // Only seek the first time this property changes
|
|
|
|
}
|
|
|
|
}
|
2020-09-25 12:46:39 +00:00
|
|
|
}
|
|
|
|
}
|