From 0bf9a1f5b4555e92653e3aeb74382e08c12779ac Mon Sep 17 00:00:00 2001 From: Jonne Pihlanen Date: Tue, 30 Jul 2019 23:06:34 +0300 Subject: [PATCH 1/7] Add serializer & instantiate it for the app. --- harbour-nayttamo.pro | 6 ++-- src/harbour-nayttamo.cpp | 10 ++++++ src/serializer.cpp | 73 ++++++++++++++++++++++++++++++++++++++++ src/serializer.h | 28 +++++++++++++++ 4 files changed, 115 insertions(+), 2 deletions(-) create mode 100644 src/serializer.cpp create mode 100644 src/serializer.h diff --git a/harbour-nayttamo.pro b/harbour-nayttamo.pro index bb5d8dc..65cae6c 100644 --- a/harbour-nayttamo.pro +++ b/harbour-nayttamo.pro @@ -15,7 +15,8 @@ TARGET = harbour-nayttamo CONFIG += sailfishapp libcrypto SOURCES += src/harbour-nayttamo.cpp \ - src/urldecrypt.cpp + src/urldecrypt.cpp \ + src/serializer.cpp OTHER_FILES += \ qml/cover/CoverPage.qml \ @@ -73,6 +74,7 @@ REQUIRED = $$find(DEFINES, "APP_ID") $$find(DEFINES, "APP_KEY") $$find(DEFINES, } HEADERS += \ - src/urldecrypt.h + src/urldecrypt.h \ + src/serializer.h unix: PKGCONFIG += libcrypto diff --git a/src/harbour-nayttamo.cpp b/src/harbour-nayttamo.cpp index ea4e905..d227088 100644 --- a/src/harbour-nayttamo.cpp +++ b/src/harbour-nayttamo.cpp @@ -40,6 +40,7 @@ #include #include "urldecrypt.h" +#include "serializer.h" int main(int argc, char *argv[]) { @@ -55,6 +56,15 @@ int main(int argc, char *argv[]) view->rootContext()->setContextProperty("urlDecrypt", &urlDecrypt); + auto worker = new QThread; + + QScopedPointer serializer(new Serializer); + view->rootContext()->setContextProperty("serializer", serializer.data()); + serializer->moveToThread(worker); + + QObject::connect(worker, SIGNAL(finished()), worker, SLOT(deleteLater())); + worker->start(); + view->engine()->addImportPath(SailfishApp::pathTo("qml/components").toString()); view->setSource(SailfishApp::pathTo("qml/main.qml")); diff --git a/src/serializer.cpp b/src/serializer.cpp new file mode 100644 index 0000000..e3fb178 --- /dev/null +++ b/src/serializer.cpp @@ -0,0 +1,73 @@ +#include "serializer.h" + +#include +#include +#include +#include +#include +#include +#include + +Serializer::Serializer(QObject *parent) : QObject(parent) +{ + this->ensureDir(); + this->readState(); + qDebug() << this->m_state; +} + +Serializer::~Serializer() +{ + this->writeState(); +} + +bool Serializer::readState() +{ + QFile file(this->getPath()); + + if (!file.open(QIODevice::ReadWrite)) { + this->m_state = QJsonObject(); + return false; + } + + this->m_state = (QJsonDocument::fromJson(file.readAll())).object(); + + return true; +} + +bool Serializer::writeState() const +{ + QFile file(this->getPath()); + + if (!file.open(QIODevice::ReadWrite)) { + return false; + } + + QJsonDocument saveDoc(m_state); + file.write(saveDoc.toJson()); + + return true; +} + +void Serializer::setState(QJsonObject state) +{ + m_state = state; +} + +QJsonObject Serializer::getState() +{ + return m_state; +} + +void Serializer::ensureDir() const +{ + QDir dir(QStandardPaths::writableLocation(QStandardPaths::DataLocation)); + + if (!dir.exists()) { + dir.mkpath("."); + } +} + +QString Serializer::getPath() const +{ + return QStandardPaths::writableLocation(QStandardPaths::DataLocation) + QDir::separator() + "state.json"; +} diff --git a/src/serializer.h b/src/serializer.h new file mode 100644 index 0000000..93c05c7 --- /dev/null +++ b/src/serializer.h @@ -0,0 +1,28 @@ +#ifndef SERIALIZER_H +#define SERIALIZER_H + +#include +#include + +class Serializer : public QObject +{ + Q_OBJECT +public: + explicit Serializer(QObject *parent = 0); + ~Serializer(); + + Q_INVOKABLE void setState(QJsonObject state); + Q_INVOKABLE QJsonObject getState(); + +signals: +public slots: +private: + bool readState(); + bool writeState() const; + QString getPath() const; + void ensureDir() const; + + QJsonObject m_state; +}; + +#endif // SERIALIZER_H From ec8d5c5b10940f4748c2646543fd09bd7e7b680f Mon Sep 17 00:00:00 2001 From: Jonne Pihlanen Date: Tue, 30 Jul 2019 23:09:17 +0300 Subject: [PATCH 2/7] Create some state handling functions & prototype the resume functionality. --- qml/main.qml | 21 +++++++++++++++++++++ qml/pages/PlayerPage.qml | 5 +++++ 2 files changed, 26 insertions(+) diff --git a/qml/main.qml b/qml/main.qml index 95b18dc..bf57c52 100644 --- a/qml/main.qml +++ b/qml/main.qml @@ -1,6 +1,7 @@ import QtQuick 2.2 import Sailfish.Silica 1.0 import QtMultimedia 5.5 + import "pages" ApplicationWindow @@ -11,6 +12,26 @@ ApplicationWindow property string coverTitle: "" property string coverSubTitle: "" + property var state: ({ + startedPrograms: {} + }) + onStateChanged: serializer.setState(state) + + Component.onCompleted: { + state = serializer.getState() + } + + function setState(obj, path, value) { + if (!Array.isArray(path)) path = path.split('.') + if (path.length === 1) return obj[path[0]] = value + if (!obj[path[0]]) obj[path[0]] = {} + return setState(obj[path[0]], path.slice(1), value) + } + + function insertStartedProgram(program) { + setState(state, 'startedPrograms.' + program.id, program.progress) + } + function updateCover(newCoverMode, newCoverTitle, newCoverSubtitle) { coverMode = newCoverMode ? newCoverMode : "" coverTitle = newCoverTitle ? newCoverTitle : "" diff --git a/qml/pages/PlayerPage.qml b/qml/pages/PlayerPage.qml index 7da8b66..d331511 100644 --- a/qml/pages/PlayerPage.qml +++ b/qml/pages/PlayerPage.qml @@ -33,6 +33,10 @@ Page { if (visible) updateCover(qsTr("Now playing"), program.title, program.itemTitle) } + Component.onDestruction: { + appWindow.insertStartedProgram({ id: program.id, progress: mediaPlayer.position }) + } + function initialize() { YleApi.getMediaUrl(program.id, program.mediaId) .then(function(response) { @@ -41,6 +45,7 @@ Page { subtitlesText.getSubtitles(subtitlesUrl) } mediaPlayer.source = response.url + if (appWindow.state.startedPrograms[program.id]) mediaPlayer.seek(appWindow.state.startedPrograms[program.id]) mediaPlayer.play() YleApi.reportUsage(program.id, program.mediaId) }) From 787aaf7a1a929d901b29941f764ffb92d537959d Mon Sep 17 00:00:00 2001 From: Jonne Pihlanen Date: Mon, 19 Aug 2019 22:51:49 +0300 Subject: [PATCH 3/7] Saving to file works correctly now, initializes state with null so it can be detected easily if the first onChange is from that. --- qml/main.qml | 44 +++++++++++++++++++++++++++++++++----------- src/serializer.cpp | 14 +++++++++++--- src/serializer.h | 1 + 3 files changed, 45 insertions(+), 14 deletions(-) diff --git a/qml/main.qml b/qml/main.qml index bf57c52..bf47409 100644 --- a/qml/main.qml +++ b/qml/main.qml @@ -12,24 +12,47 @@ ApplicationWindow property string coverTitle: "" property string coverSubTitle: "" - property var state: ({ - startedPrograms: {} - }) - onStateChanged: serializer.setState(state) + property var state: null + + onStateChanged: { + if (state === null) { + console.log('initialization, do not set the state') + return + } + serializer.setState(state) + } Component.onCompleted: { state = serializer.getState() } - function setState(obj, path, value) { - if (!Array.isArray(path)) path = path.split('.') - if (path.length === 1) return obj[path[0]] = value - if (!obj[path[0]]) obj[path[0]] = {} - return setState(obj[path[0]], path.slice(1), value) + function setState(path, value) { + function setStateRecursive(obj, path, value) { + if (!Array.isArray(path)) { + path = path.split('.') + } + + if (path.length === 1) { + obj[path[0]] = value + return + } + + if (!obj[path[0]]) { + obj[path[0]] = {} + } + + return setStateRecursive(obj[path[0]], path.slice(1), value) + } + + var newState = JSON.parse(JSON.stringify(state)) + console.log('start', JSON.stringify(newState)) + setStateRecursive(newState, path, value) + console.log("done", JSON.stringify(newState)) + state = newState } function insertStartedProgram(program) { - setState(state, 'startedPrograms.' + program.id, program.progress) + setState('startedPrograms.' + program.id, program.progress) } function updateCover(newCoverMode, newCoverTitle, newCoverSubtitle) { @@ -46,4 +69,3 @@ ApplicationWindow cover: Qt.resolvedUrl("cover/CoverPage.qml") allowedOrientations: defaultAllowedOrientations } - diff --git a/src/serializer.cpp b/src/serializer.cpp index e3fb178..645691d 100644 --- a/src/serializer.cpp +++ b/src/serializer.cpp @@ -12,7 +12,6 @@ Serializer::Serializer(QObject *parent) : QObject(parent) { this->ensureDir(); this->readState(); - qDebug() << this->m_state; } Serializer::~Serializer() @@ -20,16 +19,25 @@ Serializer::~Serializer() this->writeState(); } +QJsonObject Serializer::getInitialState() const +{ + return QJsonObject + { + {"startedPrograms", QJsonObject()} + }; +} + bool Serializer::readState() { QFile file(this->getPath()); if (!file.open(QIODevice::ReadWrite)) { - this->m_state = QJsonObject(); return false; } - this->m_state = (QJsonDocument::fromJson(file.readAll())).object(); + auto json = (QJsonDocument::fromJson(file.readAll())).object(); + + this->m_state = json.empty() ? this->getInitialState() : json; return true; } diff --git a/src/serializer.h b/src/serializer.h index 93c05c7..5707462 100644 --- a/src/serializer.h +++ b/src/serializer.h @@ -21,6 +21,7 @@ public slots: bool writeState() const; QString getPath() const; void ensureDir() const; + QJsonObject getInitialState() const; QJsonObject m_state; }; From 69e78cbf56780c50173d3502860aec394d3b4c2e Mon Sep 17 00:00:00 2001 From: Jonne Pihlanen Date: Tue, 20 Aug 2019 21:06:02 +0300 Subject: [PATCH 4/7] formatting. --- src/serializer.cpp | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/src/serializer.cpp b/src/serializer.cpp index 645691d..7f1c58f 100644 --- a/src/serializer.cpp +++ b/src/serializer.cpp @@ -31,7 +31,8 @@ bool Serializer::readState() { QFile file(this->getPath()); - if (!file.open(QIODevice::ReadWrite)) { + if (!file.open(QIODevice::ReadWrite)) + { return false; } @@ -46,7 +47,8 @@ bool Serializer::writeState() const { QFile file(this->getPath()); - if (!file.open(QIODevice::ReadWrite)) { + if (!file.open(QIODevice::ReadWrite)) + { return false; } @@ -70,7 +72,8 @@ void Serializer::ensureDir() const { QDir dir(QStandardPaths::writableLocation(QStandardPaths::DataLocation)); - if (!dir.exists()) { + if (!dir.exists()) + { dir.mkpath("."); } } From dcbd584e689629453cf5e762a064cc47db00b7b1 Mon Sep 17 00:00:00 2001 From: Jonne Pihlanen Date: Tue, 20 Aug 2019 21:07:28 +0300 Subject: [PATCH 5/7] Add safety check which ensure that serializer is available when calling setState. --- qml/main.qml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/qml/main.qml b/qml/main.qml index bf47409..2a44150 100644 --- a/qml/main.qml +++ b/qml/main.qml @@ -19,7 +19,7 @@ ApplicationWindow console.log('initialization, do not set the state') return } - serializer.setState(state) + serializer && serializer.setState(state) } Component.onCompleted: { From e10e6c5cb2106c6ba13bcd4d23605197832cb364 Mon Sep 17 00:00:00 2001 From: Jonne Pihlanen Date: Tue, 20 Aug 2019 21:08:06 +0300 Subject: [PATCH 6/7] Store state of program when pressing pause. --- qml/pages/PlayerPage.qml | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/qml/pages/PlayerPage.qml b/qml/pages/PlayerPage.qml index d331511..e6eef72 100644 --- a/qml/pages/PlayerPage.qml +++ b/qml/pages/PlayerPage.qml @@ -127,6 +127,10 @@ Page { if (subtitlesUrl && subtitlesText) { subtitlesText.getSubtitles(subtitlesUrl) } + + if (mediaPlayer.playbackState === MediaPlayer.PausedState) { + appWindow.insertStartedProgram({ id: program.id, progress: mediaPlayer.position }) + } } } } From 56b3da3d05ae61bccc53f8ecce59514a248f8be7 Mon Sep 17 00:00:00 2001 From: Jonne Pihlanen Date: Sat, 7 Nov 2020 22:27:55 +0200 Subject: [PATCH 7/7] feature: start implementing a dialog for asking whether user wants to continue on a program that is in progress. --- harbour-nayttamo.pro | 1 + qml/components/ContinueWatchingDialog.qml | 5 +++ qml/pages/PlayerPage.qml | 45 ++++++++++++++++------- 3 files changed, 37 insertions(+), 14 deletions(-) create mode 100644 qml/components/ContinueWatchingDialog.qml diff --git a/harbour-nayttamo.pro b/harbour-nayttamo.pro index 2988f2a..511c41a 100644 --- a/harbour-nayttamo.pro +++ b/harbour-nayttamo.pro @@ -19,6 +19,7 @@ SOURCES += src/harbour-nayttamo.cpp \ src/serializer.cpp DISTFILES += \ + qml/components/ContinueWatchingDialog.qml \ rpm/harbour-nayttamo.changes.in \ rpm/harbour-nayttamo.spec \ rpm/harbour-nayttamo.yaml \ diff --git a/qml/components/ContinueWatchingDialog.qml b/qml/components/ContinueWatchingDialog.qml new file mode 100644 index 0000000..9c36e13 --- /dev/null +++ b/qml/components/ContinueWatchingDialog.qml @@ -0,0 +1,5 @@ +import QtQuick 2.0 + +Item { + +} diff --git a/qml/pages/PlayerPage.qml b/qml/pages/PlayerPage.qml index a2e2049..960f65b 100644 --- a/qml/pages/PlayerPage.qml +++ b/qml/pages/PlayerPage.qml @@ -40,20 +40,36 @@ Page { function initialize() { YleApi.getMediaUrl(program.id, program.mediaId) - .then(function(response) { - if (response.subtitlesUrl && subtitlesText) { - subtitlesUrl = response.subtitlesUrl - subtitlesText.getSubtitles(subtitlesUrl) - } - mediaPlayer.source = response.url - if (appWindow.state.startedPrograms[program.id]) mediaPlayer.seek(appWindow.state.startedPrograms[program.id]) + .then(function(response) { + if (response.subtitlesUrl && subtitlesText) { + subtitlesUrl = response.subtitlesUrl + subtitlesText.getSubtitles(subtitlesUrl) + } + mediaPlayer.source = response.url + if (appWindow.state.startedPrograms[program.id]) { + var position = appWindow.state.startedPrograms[program.id] + var dialog = pageStack.push(Qt.resolvedUrl("ContinueWatchingDialog.qml"), + {"name": program.name, "position": position }) + dialog.accepted.connect(function() { + mediaPlayer.seek(position) + mediaPlayer.play() + YleApi.reportUsage(program.id, program.mediaId) + }) + + dialog.rejected.connect(function() { + mediaPlayer.play() + YleApi.reportUsage(program.id, program.mediaId) + }) + + } else { mediaPlayer.play() YleApi.reportUsage(program.id, program.mediaId) - }) - .catch(function(error) { - console.log("mediaUrlError", JSON.stringify(error)) - errorState = true; - }) + } + }) + .catch(function(error) { + console.log("mediaUrlError", JSON.stringify(error)) + errorState = true; + }) } // The effective value will be restricted by ApplicationWindow.allowedOrientations @@ -83,6 +99,7 @@ Page { source: mediaPlayer Component.onCompleted: initialize() + Component.onDestruction: { mediaPlayer.stop() mediaPlayer.source = "" @@ -122,8 +139,8 @@ Page { anchors.fill: parent onClicked: { mediaPlayer.playbackState === MediaPlayer.PlayingState - ? mediaPlayer.pause() - : mediaPlayer.play() + ? mediaPlayer.pause() + : mediaPlayer.play() if (subtitlesUrl && subtitlesText) { subtitlesText.getSubtitles(subtitlesUrl)