Add UserLoader back + misc regression fixes

This commit is contained in:
Chris Josten 2021-08-18 00:04:27 +02:00
parent bb1e3ea21d
commit e04ec364c1
15 changed files with 406 additions and 21 deletions

View File

@ -9,6 +9,7 @@ set(JellyfinQt_SOURCES
src/model/item.cpp
src/model/playlist.cpp
src/model/shuffle.cpp
src/model/user.cpp
src/support/jsonconv.cpp
src/support/loader.cpp
@ -20,6 +21,8 @@ set(JellyfinQt_SOURCES
src/viewmodel/playbackmanager.cpp
src/viewmodel/playlist.cpp
src/viewmodel/userdata.cpp
src/viewmodel/usermodel.cpp
src/viewmodel/user.cpp
src/apiclient.cpp
src/apimodel.cpp
src/credentialmanager.cpp
@ -36,6 +39,7 @@ set(JellyfinQt_HEADERS
include/JellyfinQt/model/item.h
include/JellyfinQt/model/playlist.h
include/JellyfinQt/model/shuffle.h
include/JellyfinQt/model/user.h
include/JellyfinQt/support/jsonconv.h
include/JellyfinQt/support/jsonconvimpl.h
include/JellyfinQt/support/loader.h
@ -48,6 +52,8 @@ set(JellyfinQt_HEADERS
include/JellyfinQt/viewmodel/playbackmanager.h
include/JellyfinQt/viewmodel/playlist.h
include/JellyfinQt/viewmodel/userdata.h
include/JellyfinQt/viewmodel/usermodel.h
include/JellyfinQt/viewmodel/user.h
include/JellyfinQt/apiclient.h
include/JellyfinQt/apimodel.h
include/JellyfinQt/credentialmanager.h

View File

@ -252,6 +252,8 @@ extern template int extractTotalRecordCount(const QList<DTO::BaseItemDto> &resul
extern template void setRequestLimit(Loader::GetLatestMediaParams &params, int limit);
extern template bool setRequestStartIndex(Loader::GetLatestMediaParams &params, int offset);
extern template QList<DTO::UserDto> extractRecords(const QList<DTO::UserDto> &result);
extern template int extractTotalRecordCount(const QList<DTO::UserDto> &result);
#endif
/**

View File

@ -37,6 +37,8 @@
#include "viewmodel/modelstatus.h"
#include "viewmodel/playbackmanager.h"
#include "viewmodel/userdata.h"
#include "viewmodel/usermodel.h"
#include "viewmodel/user.h"
namespace Jellyfin {

View File

@ -0,0 +1,39 @@
/*
* Sailfin: a Jellyfin client written using Qt
* Copyright (C) 2021 Chris Josten and the Sailfin Contributors.
*
* 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
*/
#ifndef JELLYFIN_MODEL_USER_H
#define JELLYFIN_MODEL_USER_H
#include "../dto/userdto.h"
namespace Jellyfin {
namespace Model {
class User : public DTO::UserDto {
public:
User();
User(const DTO::UserDto &data, ApiClient *apiClient = nullptr);
bool sameAs(const DTO::UserDto &other);
};
}
} // NS Jellyfin
#endif // JELLYFIN_MODEL_USER_H

View File

@ -323,7 +323,6 @@ public:
QVariant data(const QModelIndex &index, int role) const override;
QSharedPointer<Model::Item> itemAt(int index);
};
#undef JFRN
} // NS Jellyfin

View File

@ -0,0 +1,94 @@
/*
* Sailfin: a Jellyfin client written using Qt
* Copyright (C) 2021 Chris Josten and the Sailfin Contributors.
*
* 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
*/
#ifndef JELLYFIN_VIEWMODEL_USER_H
#define JELLYFIN_VIEWMODEL_USER_H
#include <QObject>
#include <QSharedPointer>
#include <QString>
#include "../loader/http/getitem.h"
#include "../loader/requesttypes.h"
#include "../model/user.h"
#include "loader.h"
namespace Jellyfin {
namespace ViewModel {
class User : public QObject {
Q_OBJECT
public:
explicit User(QObject *parent = nullptr, QSharedPointer<Model::User> data = QSharedPointer<Model::User>::create());
Q_PROPERTY(QString name READ name NOTIFY nameChanged)
Q_PROPERTY(QString serverId READ serverId NOTIFY serverIdChanged)
Q_PROPERTY(QString serverName READ serverName NOTIFY serverNameChanged)
Q_PROPERTY(QString userId READ userId NOTIFY userIdChanged)
Q_PROPERTY(QString primaryImageTag READ primaryImageTag NOTIFY primaryImageTagChanged)
Q_PROPERTY(bool hasPassword READ hasPassword NOTIFY hasPasswordChanged)
Q_PROPERTY(bool hasConfiguredPassword READ hasConfiguredPassword NOTIFY hasConfiguredPasswordChanged)
Q_PROPERTY(bool hasConfiguredEasyPassword READ hasConfiguredEasyPassword NOTIFY hasConfiguredEasyPasswordChanged)
QString name() const { return m_data->name(); }
QString serverId() const { return m_data->serverId(); }
QString serverName() const { return m_data->serverName(); }
QString userId() const { return m_data->jellyfinId(); }
QString primaryImageTag() const { return m_data->primaryImageTag(); }
bool hasPassword() const { return m_data->hasPassword(); }
bool hasConfiguredPassword() const { return m_data->hasConfiguredPassword(); }
bool hasConfiguredEasyPassword() const { return m_data->hasConfiguredEasyPassword(); }
QSharedPointer<Model::User> data() const { return m_data; }
void setData(QSharedPointer<Model::User> newData);
signals:
void nameChanged(QString newName);
void serverIdChanged(QString newServerId);
void serverNameChanged(QString newServerName);
void userIdChanged(QString newUserId);
void primaryImageTagChanged(QString newPrimaryImageTag);
void hasPasswordChanged(bool newHasPassword);
void hasConfiguredPasswordChanged(bool newHasConfiguredPassword);
void hasConfiguredEasyPasswordChanged(bool newHasConfiguredEasyPasswordChanged);
private:
QSharedPointer<Model::User> m_data;
};
using UserLoaderBase = Loader<ViewModel::User, DTO::UserDto, Jellyfin::Loader::GetUserByIdParams>;
class UserLoader : public UserLoaderBase {
Q_OBJECT
public:
explicit UserLoader(QObject *parent = nullptr);
Q_PROPERTY(QString userId READ userId WRITE setUserId NOTIFY userIdChanged)
QString userId() const { return m_parameters.userId(); }
void setUserId(QString newUserId);
virtual bool canReload() const override;
signals:
void userIdChanged(const QString &newUserId) const;
};
}
} // NS Jellyfin
#endif // JELLYFIN_VIEWMODEL_USER_H

View File

@ -0,0 +1,75 @@
/*
* Sailfin: a Jellyfin client written using Qt
* Copyright (C) 2021 Chris Josten and the Sailfin Contributors.
*
* 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
*/
#ifndef JELLYFIN_VIEWMODEL_USERMODEL_H
#define JELLYFIN_VIEWMODEL_USERMODEL_H
#include <QAbstractListModel>
#include <QByteArray>
#include <QHash>
#include "../loader/http/getpublicusers.h"
#include "../loader/requesttypes.h"
#include "../model/user.h"
#include "../apimodel.h"
#include "propertyhelper.h"
namespace Jellyfin {
namespace ViewModel {
// Jellyfin Forward Read/Write Property
#define FWDPROP(type, propName, propSetName) JF_FWD_RW_PROP(type, propName, propSetName, this->m_parameters)
class UserModel : public ApiModel<Model::User> {
Q_OBJECT
public:
enum RoleNames {
userId = Qt::UserRole + 1,
name,
hasPassword,
primaryImageTag,
};
explicit UserModel (QObject *parent = nullptr);
virtual QHash<int, QByteArray> roleNames() const override {
return {
{ RoleNames::userId, "userId" },
{ RoleNames::name, "name" },
{ RoleNames::hasPassword, "hasPassword" },
{ RoleNames::primaryImageTag, "primaryImageTag" },
};
}
QVariant data(const QModelIndex &index, int role) const override;
};
using PublicUsersLoaderBase = LoaderModelLoader<Model::User, DTO::UserDto, QList<DTO::UserDto>, Jellyfin::Loader::GetPublicUsersParams>;
class PublicUsersLoader : public PublicUsersLoaderBase {
Q_OBJECT
public:
explicit PublicUsersLoader(QObject *parent = nullptr);
bool canReload() const override;
};
#undef FWDPROP
} // NS ViewModel
} // NS Jellyfin
#endif // JELLYFIN_VIEWMODEL_USERMODEL_H

View File

@ -141,6 +141,15 @@ bool setRequestStartIndex(Loader::GetLatestMediaParams &params, int offset) {
return false;
}
template<>
QList<DTO::UserDto> extractRecords(const QList<DTO::UserDto> &result) {
return result;
}
template<>
int extractTotalRecordCount(const QList<DTO::UserDto> &result) {
return result.size();
}
void registerModels(const char *URI) {
Q_UNUSED(URI)

View File

@ -24,6 +24,7 @@ void registerTypes(const char *uri) {
qmlRegisterType<ServerDiscoveryModel>(uri, 1, 0, "ServerDiscoveryModel");
qmlRegisterType<ViewModel::PlaybackManager>(uri, 1, 0, "PlaybackManager");
qmlRegisterUncreatableType<ViewModel::Item>(uri, 1, 0, "Item", "Acquire one via ItemLoader or exposed properties");
qmlRegisterUncreatableType<ViewModel::User>(uri, 1, 0, "User", "Acquire one via UserLoader or exposed properties");
qmlRegisterUncreatableType<EventBus>(uri, 1, 0, "EventBus", "Obtain one via your ApiClient");
qmlRegisterUncreatableType<WebSocket>(uri, 1, 0, "WebSocket", "Obtain one via your ApiClient");
qmlRegisterUncreatableType<ViewModel::UserData>(uri, 1, 0, "UserData", "Obtain one via an Item");
@ -32,16 +33,19 @@ void registerTypes(const char *uri) {
qmlRegisterUncreatableType<BaseApiModel>(uri, 1, 0, "BaseApiModel", "Please use one of its subclasses");
qmlRegisterUncreatableType<BaseModelLoader>(uri, 1, 0, "BaseModelLoader", "Please use one of its subclasses");
qmlRegisterType<ViewModel::ItemModel>(uri, 1, 0, "ItemModel");
qmlRegisterType<ViewModel::UserModel>(uri, 1, 0, "UserModel");
// Loaders
qmlRegisterUncreatableType<ViewModel::LoaderBase>(uri, 1, 0, "LoaderBase", "Use one of its subclasses");
qmlRegisterType<ViewModel::ItemLoader>(uri, 1, 0, "ItemLoader");
qmlRegisterType<ViewModel::UserLoader>(uri, 1, 0, "UserLoader");
qmlRegisterType<ViewModel::LatestMediaLoader>(uri, 1, 0, "LatestMediaLoader");
qmlRegisterType<ViewModel::UserItemsLoader>(uri, 1, 0, "UserItemsLoader");
qmlRegisterType<ViewModel::UserViewsLoader>(uri, 1, 0, "UsersViewsLoader");
qmlRegisterType<ViewModel::ResumeItemsLoader>(uri, 1, 0, "ResumeItemsLoader");
qmlRegisterType<ViewModel::ShowSeasonsLoader>(uri, 1, 0, "ShowSeasonsLoader");
qmlRegisterType<ViewModel::ShowEpisodesLoader>(uri, 1, 0, "ShowEpisodesLoader");
qmlRegisterType<ViewModel::PublicUsersLoader>(uri, 1, 0, "PublicUsersLoader");
// Enumerations
qmlRegisterUncreatableType<Jellyfin::DTO::GeneralCommandTypeClass>(uri, 1, 0, "GeneralCommandType", "Is an enum");

38
core/src/model/user.cpp Normal file
View File

@ -0,0 +1,38 @@
/*
* Sailfin: a Jellyfin client written using Qt
* Copyright (C) 2021 Chris Josten and the Sailfin Contributors.
*
* 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
*/
#include "JellyfinQt/model/user.h"
namespace Jellyfin {
namespace Model {
User::User() {}
User::User(const DTO::UserDto &other, ApiClient *apiClient)
: DTO::UserDto(other) {
Q_UNUSED(apiClient)
}
bool User::sameAs(const DTO::UserDto &other) {
return this->jellyfinId() == other.jellyfinId();
}
} // NS Model
} // NS Jellyfin

View File

@ -0,0 +1,55 @@
/*
* Sailfin: a Jellyfin client written using Qt
* Copyright (C) 2021 Chris Josten and the Sailfin Contributors.
*
* 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
*/
#include "JellyfinQt/viewmodel/user.h"
#include "JellyfinQt/loader/http/getuserbyid.h"
namespace Jellyfin {
namespace ViewModel {
// User
User::User(QObject *parent, QSharedPointer<Model::User> data)
: QObject(parent),
m_data(data) {
}
void User::setData(QSharedPointer<Model::User> newData) {
m_data = newData;
}
// UserLoader
UserLoader::UserLoader(QObject *parent)
: UserLoaderBase(new Jellyfin::Loader::HTTP::GetUserByIdLoader(), parent) {}
void UserLoader::setUserId(QString newUserId) {
m_parameters.setUserId(newUserId);
reloadIfNeeded();
emit userIdChanged(newUserId);
}
bool UserLoader::canReload() const {
return UserLoaderBase::canReload()
&& !m_parameters.userId().isEmpty();
}
} // NS ViewModel
} // NS Jellyfin

View File

@ -0,0 +1,56 @@
/*
* Sailfin: a Jellyfin client written using Qt
* Copyright (C) 2021 Chris Josten and the Sailfin Contributors.
*
* 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
*/
#include "JellyfinQt/viewmodel/usermodel.h"
namespace Jellyfin {
namespace ViewModel {
UserModel::UserModel(QObject *parent)
: ApiModel<Model::User>(parent) {}
QVariant UserModel::data(const QModelIndex &index, int role) const {
if (role <= Qt::UserRole || !index.isValid()) return QVariant();
int row = index.row();
if (row < 0 || row >= m_array.size()) return QVariant();
QSharedPointer<Model::User> item = m_array[row];
switch(role) {
case userId:
return QVariant(item->jellyfinId());
case name:
return QVariant(item->name());
case hasPassword:
return QVariant(item->hasPassword());
case primaryImageTag:
return QVariant(item->primaryImageTag());
default:
return QVariant();
}
}
PublicUsersLoader::PublicUsersLoader(QObject *parent)
: PublicUsersLoaderBase(new Loader::HTTP::GetPublicUsersLoader(), parent) {}
bool PublicUsersLoader::canReload() const {
return m_apiClient != nullptr && !m_apiClient->baseUrl().isNull();
}
} // NS ViewModel
} // NS Jellyfin

View File

@ -185,8 +185,8 @@ Page {
if (force || (appWindow.apiClient.authenticated && !_modelsLoaded)) {
_modelsLoaded = true;
mediaLibraryModel.reload()
//userResumeModel.reload()
//showNextUpModel.reload()
userResumeModel.reload()
showNextUpModel.reload()
}
}
@ -225,7 +225,7 @@ Page {
value: model.imageBlurHashes["Primary"][model.imageTags["Primary"]]
}
landscape: !isPortrait
progress: (typeof model.userData !== "undefined") ? model.userData.playedPercentage / 100 : 0.0
progress: (typeof model.userDataPlayedProgress !== 0.0) ? model.userDataPlayedPercentage / 100 : 0.0
onClicked: {
pageStack.push(Utils.getPageUrl(model.mediaType, model.type, model.isFolder), {"itemId": model.jellyfinId, "itemData": model.qtObject})

View File

@ -28,6 +28,8 @@ Page {
id: settingsPage
allowedOrientations: Orientation.All
property alias loggedInUser: userLoader.data
SilicaFlickable {
anchors.fill: parent
contentHeight: content.height
@ -58,9 +60,10 @@ Page {
rightMargin: Theme.horizontalPageMargin
}
height: user.implicitHeight + server.implicitHeight + Theme.paddingMedium
QtObject {
id: loggedInUser
//apiClient: ApiClient
J.UserLoader{
id: userLoader
apiClient: appWindow.apiClient
userId: appWindow.apiClient.userId
}
RemoteImage {
id: userIcon
@ -81,7 +84,7 @@ Page {
bottom: parent.verticalCenter
right: parent.right
}
text: loggedInUser.status == User.Ready ? loggedInUser.name : apiClient.userId
text: userLoader.status === J.UserLoader.Ready ? loggedInUser.name : apiClient.userId
color: Theme.highlightColor
}

View File

@ -33,6 +33,8 @@ Dialog {
property string loginMessage
property Page firstPage
property QtObject /*User*/ selectedUser: null
property bool userSelected: false
property bool selectedUserHasPassword: true
property string error
@ -66,13 +68,13 @@ Dialog {
}
}
QtObject { id: userModel; }
/*PublicUserModel {
J.UserModel {
id: userModel
apiClient: appWindow.apiClient
Component.onCompleted: reload();
}*/
loader: J.PublicUsersLoader {
id: userLoader
apiClient: appWindow.apiClient
}
}
DialogHeader {
id: dialogHeader
@ -100,14 +102,15 @@ Dialog {
width: parent.width
Repeater {
id: userRepeater
model: 0 //userModel
model: userModel
delegate: UserGridDelegate {
name: model.name
image: model.primaryImageTag ? "%1/Users/%2/Images/Primary?tag=%3".arg(apiClient.baseUrl).arg(model.jellyfinId).arg(model.primaryImageTag) : ""
image: model.primaryImageTag ? "%1/Users/%2/Images/Primary?tag=%3".arg(apiClient.baseUrl).arg(model.userId).arg(model.primaryImageTag) : ""
highlighted: model.name === username.text
onHighlightedChanged: {
if (highlighted) {
selectedUser = model.qtObject
userSelected = true
selectedUserHasPassword = model.hasPassword
}
}
onClicked: {
@ -138,7 +141,7 @@ Dialog {
// Wil be executed before the onHighlightChanged of the UserDelegate
// This is done to update the UI after an user is selected and this field has
// been changed, to not let the UI be in an invalid state (e.g. no user selected, password field disabled)
selectedUser = null
userSelected = false
}
}
@ -199,7 +202,7 @@ Dialog {
},
State {
name: "users"
when: userRepeater.count != 0 && selectedUser === null
when: userRepeater.count != 0 && !userSelected
PropertyChanges {
target: userList
visible: true
@ -207,7 +210,7 @@ Dialog {
},
State {
name: "selectedUserPassword"
when: selectedUser !== null && selectedUser.hasPassword
when: userSelected && selectedUserHasPassword
extend: "users"
PropertyChanges {
target: password
@ -216,7 +219,7 @@ Dialog {
},
State {
name: "selectedUserNoPassword"
when: selectedUser !== null && !selectedUser.hasPassword
when: userSelected && !selectedUserHasPassword
extend: "users"
PropertyChanges {
target: password