mirror of
https://github.com/HenkKalkwater/harbour-sailfin.git
synced 2024-11-24 18:15:16 +00:00
Merge pull request #28 from heartfin/25-music-player-more-specialised-folder-pages
Add more specialised pages
This commit is contained in:
commit
fb81913eab
|
@ -253,6 +253,8 @@ extern template void setRequestLimit(Loader::GetPublicUsersParams ¶ms, int l
|
||||||
extern template bool setRequestStartIndex(Loader::GetPublicUsersParams ¶ms, int offset);
|
extern template bool setRequestStartIndex(Loader::GetPublicUsersParams ¶ms, int offset);
|
||||||
extern template void setRequestLimit(Loader::GetNextUpParams ¶ms, int limit);
|
extern template void setRequestLimit(Loader::GetNextUpParams ¶ms, int limit);
|
||||||
extern template bool setRequestStartIndex(Loader::GetNextUpParams ¶ms, int offset);
|
extern template bool setRequestStartIndex(Loader::GetNextUpParams ¶ms, int offset);
|
||||||
|
extern template void setRequestLimit(Loader::GetAlbumArtistsParams ¶ms, int limit);
|
||||||
|
extern template bool setRequestStartIndex(Loader::GetAlbumArtistsParams ¶ms, int offset);
|
||||||
|
|
||||||
extern template QList<DTO::UserDto> extractRecords(const QList<DTO::UserDto> &result);
|
extern template QList<DTO::UserDto> extractRecords(const QList<DTO::UserDto> &result);
|
||||||
extern template int extractTotalRecordCount(const QList<DTO::UserDto> &result);
|
extern template int extractTotalRecordCount(const QList<DTO::UserDto> &result);
|
||||||
|
|
|
@ -57,6 +57,7 @@ extern template QString toString(const float &source, convertType<float>);
|
||||||
extern template QString toString(const double &source, convertType<double>);
|
extern template QString toString(const double &source, convertType<double>);
|
||||||
extern template QString toString(const bool &source, convertType<bool>);
|
extern template QString toString(const bool &source, convertType<bool>);
|
||||||
extern template QString toString(const QString &source, convertType<QString>);
|
extern template QString toString(const QString &source, convertType<QString>);
|
||||||
|
extern template QString toString(const QStringList &source, convertType<QStringList>);
|
||||||
|
|
||||||
} // NS Support
|
} // NS Support
|
||||||
} // NS Jellyfin
|
} // NS Jellyfin
|
||||||
|
|
|
@ -198,6 +198,7 @@ public:
|
||||||
if (m_reply->isRunning()) {
|
if (m_reply->isRunning()) {
|
||||||
m_reply->abort();
|
m_reply->abort();
|
||||||
m_reply->deleteLater();
|
m_reply->deleteLater();
|
||||||
|
m_reply = nullptr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -121,6 +121,15 @@ public:
|
||||||
Q_PROPERTY(QJsonObject imageTags READ imageTags NOTIFY imageTagsChanged)
|
Q_PROPERTY(QJsonObject imageTags READ imageTags NOTIFY imageTagsChanged)
|
||||||
Q_PROPERTY(QStringList backdropImageTags READ backdropImageTags NOTIFY backdropImageTagsChanged)
|
Q_PROPERTY(QStringList backdropImageTags READ backdropImageTags NOTIFY backdropImageTagsChanged)
|
||||||
Q_PROPERTY(QJsonObject imageBlurHashes READ imageBlurHashes NOTIFY imageBlurHashesChanged)
|
Q_PROPERTY(QJsonObject imageBlurHashes READ imageBlurHashes NOTIFY imageBlurHashesChanged)
|
||||||
|
Q_PROPERTY(int trailerCount READ trailerCount NOTIFY trailerCountChanged)
|
||||||
|
Q_PROPERTY(int movieCount READ movieCount NOTIFY movieCountChanged)
|
||||||
|
Q_PROPERTY(int seriesCount READ seriesCount NOTIFY seriesCountChanged)
|
||||||
|
Q_PROPERTY(int programCount READ programCount NOTIFY programCountChanged)
|
||||||
|
Q_PROPERTY(int episodeCount READ episodeCount NOTIFY episodeCountChanged)
|
||||||
|
Q_PROPERTY(int songCount READ songCount NOTIFY songCountChanged)
|
||||||
|
Q_PROPERTY(int albumCount READ albumCount NOTIFY albumCountChanged)
|
||||||
|
Q_PROPERTY(int artistCount READ artistCount NOTIFY artistCountChanged)
|
||||||
|
Q_PROPERTY(int musicVideoCount READ musicVideoCount NOTIFY musicVideoCountChanged)
|
||||||
Q_PROPERTY(QString mediaType READ mediaType READ mediaType NOTIFY mediaTypeChanged)
|
Q_PROPERTY(QString mediaType READ mediaType READ mediaType NOTIFY mediaTypeChanged)
|
||||||
Q_PROPERTY(int width READ width NOTIFY widthChanged)
|
Q_PROPERTY(int width READ width NOTIFY widthChanged)
|
||||||
Q_PROPERTY(int height READ height NOTIFY heightChanged)
|
Q_PROPERTY(int height READ height NOTIFY heightChanged)
|
||||||
|
@ -166,6 +175,17 @@ public:
|
||||||
QStringList backdropImageTags() const { return m_data->backdropImageTags(); }
|
QStringList backdropImageTags() const { return m_data->backdropImageTags(); }
|
||||||
QJsonObject imageBlurHashes() const { return m_data->imageBlurHashes(); }
|
QJsonObject imageBlurHashes() const { return m_data->imageBlurHashes(); }
|
||||||
QString mediaType() const { return m_data->mediaType(); }
|
QString mediaType() const { return m_data->mediaType(); }
|
||||||
|
|
||||||
|
int trailerCount() const { return m_data->trailerCount().value_or(0); }
|
||||||
|
int movieCount() const { return m_data->movieCount().value_or(0); }
|
||||||
|
int seriesCount() const { return m_data->seriesCount().value_or(0); }
|
||||||
|
int programCount() const { return m_data->programCount().value_or(0); }
|
||||||
|
int episodeCount() const { return m_data->episodeCount().value_or(0); }
|
||||||
|
int songCount() const { return m_data->songCount().value_or(0); }
|
||||||
|
int albumCount() const { return m_data->albumCount().value_or(0); }
|
||||||
|
int artistCount() const { return m_data->artistCount().value_or(0); }
|
||||||
|
int musicVideoCount() const { return m_data->musicVideoCount().value_or(0); }
|
||||||
|
|
||||||
int width() const { return m_data->width().value_or(0); }
|
int width() const { return m_data->width().value_or(0); }
|
||||||
int height() const { return m_data->height().value_or(0); }
|
int height() const { return m_data->height().value_or(0); }
|
||||||
|
|
||||||
|
@ -226,6 +246,15 @@ signals:
|
||||||
void imageTagsChanged();
|
void imageTagsChanged();
|
||||||
void backdropImageTagsChanged();
|
void backdropImageTagsChanged();
|
||||||
void imageBlurHashesChanged();
|
void imageBlurHashesChanged();
|
||||||
|
void trailerCountChanged(int newTrailerCount);
|
||||||
|
void movieCountChanged(int newMovieCount);
|
||||||
|
void seriesCountChanged(int newSeriesCount);
|
||||||
|
void programCountChanged(int newProgramCount);
|
||||||
|
void episodeCountChanged(int newEpisodeCount);
|
||||||
|
void songCountChanged(int newSongCount);
|
||||||
|
void albumCountChanged(int newAlbumCount);
|
||||||
|
void artistCountChanged(int newArtistCount);
|
||||||
|
void musicVideoCountChanged(int newMusicVideoCount);
|
||||||
void mediaTypeChanged(const QString &newMediaType);
|
void mediaTypeChanged(const QString &newMediaType);
|
||||||
void widthChanged(int newWidth);
|
void widthChanged(int newWidth);
|
||||||
void heightChanged(int newHeight);
|
void heightChanged(int newHeight);
|
||||||
|
|
|
@ -290,6 +290,42 @@ public:
|
||||||
FWDPROP(QString, seriesId, SeriesId)
|
FWDPROP(QString, seriesId, SeriesId)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
using AlbumArtistLoaderBase = AbstractUserParameterLoader<Model::Item, DTO::BaseItemDto, DTO::BaseItemDtoQueryResult, Jellyfin::Loader::GetAlbumArtistsParams>;
|
||||||
|
class AlbumArtistLoader : public AlbumArtistLoaderBase {
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
explicit AlbumArtistLoader(QObject *parent = nullptr);
|
||||||
|
|
||||||
|
FWDLISTPROP(Jellyfin::DTO::ImageTypeClass::Value, enableImageTypes, EnableImageTypes);
|
||||||
|
FWDPROP(bool, enableImages, EnableImages)
|
||||||
|
FWDPROP(bool, enableTotalRecordCount, EnableTotalRecordCount)
|
||||||
|
FWDPROP(bool, enableUserData, EnableUserData)
|
||||||
|
FWDPROP(QStringList, excludeItemTypes, ExcludeItemTypes)
|
||||||
|
FWDLISTPROP(Jellyfin::DTO::ItemFieldsClass::Value, fields, Fields)
|
||||||
|
FWDLISTPROP(Jellyfin::DTO::ItemFilterClass::Value, filters, Filters)
|
||||||
|
FWDPROP(QStringList, genreIds, GenreIds)
|
||||||
|
FWDPROP(QStringList, genres, Genres)
|
||||||
|
FWDPROP(qint32, imageTypeLimit, ImageTypeLimit)
|
||||||
|
FWDPROP(QStringList, includeItemTypes, IncludeItemTypes)
|
||||||
|
FWDPROP(bool, isFavorite, IsFavorite)
|
||||||
|
FWDPROP(int, limit, Limit)
|
||||||
|
FWDPROP(QStringList, mediaTypes, MediaTypes)
|
||||||
|
FWDPROP(double, minCommunityRating, MinCommunityRating)
|
||||||
|
FWDPROP(QString, nameLessThan, NameLessThan)
|
||||||
|
FWDPROP(QString, nameStartsWith, NameStartsWith)
|
||||||
|
FWDPROP(QString, nameStartsWithOrGreater, NameStartsWithOrGreater)
|
||||||
|
FWDPROP(QStringList, officialRatings, OfficialRatings)
|
||||||
|
FWDPROP(QString, parentId, ParentId)
|
||||||
|
FWDPROP(QStringList, personIds, PersonIds)
|
||||||
|
FWDPROP(QStringList, personTypes, PersonTypes)
|
||||||
|
FWDPROP(QString, searchTerm, SearchTerm)
|
||||||
|
FWDPROP(int, startIndex, StartIndex)
|
||||||
|
FWDPROP(QStringList, studioIds, StudioIds)
|
||||||
|
FWDPROP(QStringList, studios, Studios)
|
||||||
|
FWDPROP(QStringList, tags, Tags)
|
||||||
|
FWDPROP(QString, userId, UserId)
|
||||||
|
FWDLISTPROP(int, years, Years);
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Base class for each model that works with items.
|
* @brief Base class for each model that works with items.
|
||||||
|
|
|
@ -184,6 +184,16 @@ bool setRequestStartIndex(Loader::GetNextUpParams ¶ms, int offset) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<>
|
||||||
|
void setRequestLimit(Loader::GetAlbumArtistsParams ¶ms, int limit) {
|
||||||
|
params.setLimit(limit);
|
||||||
|
}
|
||||||
|
template<>
|
||||||
|
bool setRequestStartIndex(Loader::GetAlbumArtistsParams ¶ms, int offset) {
|
||||||
|
params.setStartIndex(offset);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
template<>
|
template<>
|
||||||
QList<DTO::UserDto> extractRecords(const QList<DTO::UserDto> &result) {
|
QList<DTO::UserDto> extractRecords(const QList<DTO::UserDto> &result) {
|
||||||
return result;
|
return result;
|
||||||
|
|
|
@ -77,6 +77,7 @@ void JellyfinPlugin::registerTypes(const char *uri) {
|
||||||
qmlRegisterType<ViewModel::ShowEpisodesLoader>(uri, 1, 0, "ShowEpisodesLoader");
|
qmlRegisterType<ViewModel::ShowEpisodesLoader>(uri, 1, 0, "ShowEpisodesLoader");
|
||||||
qmlRegisterType<ViewModel::NextUpLoader>(uri, 1, 0, "NextUpLoader");
|
qmlRegisterType<ViewModel::NextUpLoader>(uri, 1, 0, "NextUpLoader");
|
||||||
qmlRegisterType<ViewModel::PublicUsersLoader>(uri, 1, 0, "PublicUsersLoader");
|
qmlRegisterType<ViewModel::PublicUsersLoader>(uri, 1, 0, "PublicUsersLoader");
|
||||||
|
qmlRegisterType<ViewModel::AlbumArtistLoader>(uri, 1, 0, "AlbumArtistLoader");
|
||||||
|
|
||||||
// Enumerations
|
// Enumerations
|
||||||
qmlRegisterUncreatableType<Jellyfin::DTO::GeneralCommandTypeClass>(uri, 1, 0, "GeneralCommandType", "Is an enum");
|
qmlRegisterUncreatableType<Jellyfin::DTO::GeneralCommandTypeClass>(uri, 1, 0, "GeneralCommandType", "Is an enum");
|
||||||
|
|
|
@ -257,6 +257,10 @@ template <>
|
||||||
QString toString(const QString &source, convertType<QString>) {
|
QString toString(const QString &source, convertType<QString>) {
|
||||||
return source;
|
return source;
|
||||||
}
|
}
|
||||||
|
template <>
|
||||||
|
QString toString(const QStringList &source, convertType<QStringList>) {
|
||||||
|
return source.join(',');
|
||||||
|
}
|
||||||
|
|
||||||
} // NS Support
|
} // NS Support
|
||||||
} // NS Jellyfin
|
} // NS Jellyfin
|
||||||
|
|
|
@ -18,6 +18,7 @@
|
||||||
*/
|
*/
|
||||||
#include "JellyfinQt/viewmodel/itemmodel.h"
|
#include "JellyfinQt/viewmodel/itemmodel.h"
|
||||||
|
|
||||||
|
#include "JellyfinQt/loader/http/artists.h"
|
||||||
#include "JellyfinQt/loader/http/items.h"
|
#include "JellyfinQt/loader/http/items.h"
|
||||||
#include "JellyfinQt/loader/http/userlibrary.h"
|
#include "JellyfinQt/loader/http/userlibrary.h"
|
||||||
#include "JellyfinQt/loader/http/userviews.h"
|
#include "JellyfinQt/loader/http/userviews.h"
|
||||||
|
@ -57,6 +58,9 @@ ShowEpisodesLoader::ShowEpisodesLoader(QObject *parent)
|
||||||
NextUpLoader::NextUpLoader(QObject *parent)
|
NextUpLoader::NextUpLoader(QObject *parent)
|
||||||
: NextUpLoaderBase(new Jellyfin::Loader::HTTP::GetNextUpLoader(), parent) {}
|
: NextUpLoaderBase(new Jellyfin::Loader::HTTP::GetNextUpLoader(), parent) {}
|
||||||
|
|
||||||
|
AlbumArtistLoader::AlbumArtistLoader(QObject *parent)
|
||||||
|
: AlbumArtistLoaderBase(new Jellyfin::Loader::HTTP::GetAlbumArtistsLoader(), parent) {}
|
||||||
|
|
||||||
ItemModel::ItemModel(QObject *parent)
|
ItemModel::ItemModel(QObject *parent)
|
||||||
: ApiModel<Model::Item>(parent) {
|
: ApiModel<Model::Item>(parent) {
|
||||||
connect(this, &QAbstractItemModel::rowsInserted, this, &ItemModel::onInsertItems);
|
connect(this, &QAbstractItemModel::rowsInserted, this, &ItemModel::onInsertItems);
|
||||||
|
@ -128,7 +132,7 @@ QSharedPointer<Model::Item> ItemModel::itemAt(int index) {
|
||||||
|
|
||||||
void ItemModel::onInsertItems(const QModelIndex &parent, int start, int end) {
|
void ItemModel::onInsertItems(const QModelIndex &parent, int start, int end) {
|
||||||
if (parent.isValid()) return;
|
if (parent.isValid()) return;
|
||||||
qDebug() << "Connecting " << (end - start + 1) << "items!";
|
//qDebug() << "Connecting " << (end - start + 1) << "items!";
|
||||||
for (int i = start; i <= end; i++) {
|
for (int i = start; i <= end; i++) {
|
||||||
connect(itemAt(i).data(), &Model::Item::userDataChanged, this, &ItemModel::onUserDataUpdated);
|
connect(itemAt(i).data(), &Model::Item::userDataChanged, this, &ItemModel::onUserDataUpdated);
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,6 +11,10 @@
|
||||||
|
|
||||||
# * date Author's Name <author's email> version-release
|
# * date Author's Name <author's email> version-release
|
||||||
# - Summary of changes
|
# - Summary of changes
|
||||||
|
#
|
||||||
|
* Future version Chris Josten <chris@netsoj.nl> 0.??.??-1
|
||||||
|
- New features
|
||||||
|
- New layout for artist pages
|
||||||
|
|
||||||
* Wed Jul 20 2022 Chris Josten <chris@netsoj.nl> 0.4.2-1
|
* Wed Jul 20 2022 Chris Josten <chris@netsoj.nl> 0.4.2-1
|
||||||
- Bugfixes:
|
- Bugfixes:
|
||||||
|
|
|
@ -26,6 +26,7 @@ set(sailfin_QML_SOURCES
|
||||||
qml/components/videoplayer/VideoError.qml
|
qml/components/videoplayer/VideoError.qml
|
||||||
qml/components/videoplayer/VideoHud.qml
|
qml/components/videoplayer/VideoHud.qml
|
||||||
qml/components/IconListItem.qml
|
qml/components/IconListItem.qml
|
||||||
|
qml/components/ItemChildrenShowcase.qml
|
||||||
qml/components/JItem.qml
|
qml/components/JItem.qml
|
||||||
qml/components/LibraryItemDelegate.qml
|
qml/components/LibraryItemDelegate.qml
|
||||||
qml/components/MoreSection.qml
|
qml/components/MoreSection.qml
|
||||||
|
@ -53,6 +54,8 @@ set(sailfin_QML_SOURCES
|
||||||
qml/pages/itemdetails/EpisodePage.qml
|
qml/pages/itemdetails/EpisodePage.qml
|
||||||
qml/pages/itemdetails/FilmPage.qml
|
qml/pages/itemdetails/FilmPage.qml
|
||||||
qml/pages/itemdetails/MusicAlbumPage.qml
|
qml/pages/itemdetails/MusicAlbumPage.qml
|
||||||
|
qml/pages/itemdetails/MusicArtistPage.qml
|
||||||
|
qml/pages/itemdetails/MusicLibraryPage.qml
|
||||||
qml/pages/itemdetails/PhotoPage.qml
|
qml/pages/itemdetails/PhotoPage.qml
|
||||||
qml/pages/itemdetails/SeasonPage.qml
|
qml/pages/itemdetails/SeasonPage.qml
|
||||||
qml/pages/itemdetails/SeriesPage.qml
|
qml/pages/itemdetails/SeriesPage.qml
|
||||||
|
|
|
@ -23,6 +23,19 @@ import QtQuick 2.6
|
||||||
import Sailfish.Silica 1.0
|
import Sailfish.Silica 1.0
|
||||||
|
|
||||||
QtObject {
|
QtObject {
|
||||||
|
|
||||||
|
readonly property int gridColumns: {
|
||||||
|
switch(Screen.sizeCategory) {
|
||||||
|
case Screen.Small:
|
||||||
|
case Screen.Medium:
|
||||||
|
return 3
|
||||||
|
case Screen.Large:
|
||||||
|
return 5
|
||||||
|
case Screen.ExtraLarge:
|
||||||
|
return 7
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
readonly property real libraryDelegateWidth: {
|
readonly property real libraryDelegateWidth: {
|
||||||
switch(Screen.sizeCategory) {
|
switch(Screen.sizeCategory) {
|
||||||
case Screen.Small:
|
case Screen.Small:
|
||||||
|
|
|
@ -47,6 +47,32 @@ function ticksToText(ticks, showHours) {
|
||||||
return timeToText(ticks / 10000, showHours);
|
return timeToText(ticks / 10000, showHours);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function propsToQuery(options) {
|
||||||
|
var query = "";
|
||||||
|
for (var prop in options) {
|
||||||
|
if (options.hasOwnProperty(prop)) {
|
||||||
|
var value = options[prop];
|
||||||
|
if (prop === "maxWidth" || prop === "maxHeight") {
|
||||||
|
value = Math.floor(options[prop]);
|
||||||
|
}
|
||||||
|
query += "&" + prop + "=" + value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return query;
|
||||||
|
}
|
||||||
|
|
||||||
|
function randomBackdrop(baseUrl, item) {
|
||||||
|
var count = item.backdropImageTags.length;
|
||||||
|
if (count === 0) return -1
|
||||||
|
return Math.floor(Math.random() * count);
|
||||||
|
}
|
||||||
|
|
||||||
|
function itemBackdropUrl(baseUrl, item, idx, options) {
|
||||||
|
var extraQuery = propsToQuery(options)
|
||||||
|
return baseUrl + "/Items/" + item.jellyfinId + "/Images/Backdrop/" + idx + "?tag=" + item.backdropImageTags[idx] + extraQuery;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
function itemImageUrl(baseUrl, item, type, options) {
|
function itemImageUrl(baseUrl, item, type, options) {
|
||||||
if (item === null || !item.imageTags[type]) { return "" }
|
if (item === null || !item.imageTags[type]) { return "" }
|
||||||
return itemModelImageUrl(baseUrl, item.jellyfinId, item.imageTags[type], type, options)
|
return itemModelImageUrl(baseUrl, item.jellyfinId, item.imageTags[type], type, options)
|
||||||
|
@ -54,16 +80,7 @@ function itemImageUrl(baseUrl, item, type, options) {
|
||||||
|
|
||||||
function itemModelImageUrl(baseUrl, itemId, tag, type, options) {
|
function itemModelImageUrl(baseUrl, itemId, tag, type, options) {
|
||||||
if (tag === undefined) return ""
|
if (tag === undefined) return ""
|
||||||
var extraQuery = "";
|
var extraQuery = propsToQuery(options)
|
||||||
for (var prop in options) {
|
|
||||||
if (options.hasOwnProperty(prop)) {
|
|
||||||
var value = options[prop];
|
|
||||||
if (prop === "maxWidth" || prop === "maxHeight") {
|
|
||||||
value = Math.floor(options[prop]);
|
|
||||||
}
|
|
||||||
extraQuery += "&" + prop + "=" + value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return baseUrl + "/Items/" + itemId + "/Images/" + type + "?tag=" + tag + extraQuery
|
return baseUrl + "/Items/" + itemId + "/Images/" + type + "?tag=" + tag + extraQuery
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -88,11 +105,20 @@ function getPageUrl(mediaType, itemType, isFolder) {
|
||||||
return Qt.resolvedUrl("pages/itemdetails/SeasonPage.qml")
|
return Qt.resolvedUrl("pages/itemdetails/SeasonPage.qml")
|
||||||
case "episode":
|
case "episode":
|
||||||
return Qt.resolvedUrl("pages/itemdetails/EpisodePage.qml")
|
return Qt.resolvedUrl("pages/itemdetails/EpisodePage.qml")
|
||||||
|
case "musicartist":
|
||||||
|
return Qt.resolvedUrl("pages/itemdetails/MusicArtistPage.qml")
|
||||||
case "musicalbum":
|
case "musicalbum":
|
||||||
case "playlist":
|
case "playlist":
|
||||||
return Qt.resolvedUrl("pages/itemdetails/MusicAlbumPage.qml")
|
return Qt.resolvedUrl("pages/itemdetails/MusicAlbumPage.qml")
|
||||||
case "photo":
|
case "photo":
|
||||||
return Qt.resolvedUrl("pages/itemdetails/PhotoPage.qml")
|
return Qt.resolvedUrl("pages/itemdetails/PhotoPage.qml")
|
||||||
|
case "collectionfolder":
|
||||||
|
// TODO: support for other collection folders
|
||||||
|
switch(mediaType.toLowerCase()) {
|
||||||
|
case "music":
|
||||||
|
return Qt.resolvedUrl("pages/itemdetails/MusicLibraryPage.qml")
|
||||||
|
}
|
||||||
|
// FALLTRHOUGH
|
||||||
default:
|
default:
|
||||||
switch (mediaType ? mediaType.toLowerCase() : isFolder ? "folder" : "") {
|
switch (mediaType ? mediaType.toLowerCase() : isFolder ? "folder" : "") {
|
||||||
case "folder":
|
case "folder":
|
||||||
|
|
60
sailfish/qml/components/ItemChildrenShowcase.qml
Normal file
60
sailfish/qml/components/ItemChildrenShowcase.qml
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
import QtQuick 2.6
|
||||||
|
import Sailfish.Silica 1.0
|
||||||
|
|
||||||
|
import nl.netsoj.chris.Jellyfin 1.0 as J
|
||||||
|
|
||||||
|
import "../"
|
||||||
|
|
||||||
|
MoreSection {
|
||||||
|
id: header
|
||||||
|
busy: itemModel.loader.status === J.ModelStatus.Loading || extraBusy
|
||||||
|
property bool extraBusy: false
|
||||||
|
property alias loader: itemModel.loader
|
||||||
|
property string collectionType
|
||||||
|
property bool collapseWhenEmpty: true
|
||||||
|
|
||||||
|
J.ItemModel {
|
||||||
|
id: itemModel
|
||||||
|
}
|
||||||
|
|
||||||
|
SilicaListView {
|
||||||
|
readonly property bool isPortrait: Utils.usePortraitCover(collectionType)
|
||||||
|
id: list
|
||||||
|
clip: true
|
||||||
|
height: {
|
||||||
|
if (count > 0 || !collapseWhenEmpty) {
|
||||||
|
if (isPortrait) {
|
||||||
|
Constants.libraryDelegatePosterHeight
|
||||||
|
} else {
|
||||||
|
Constants.libraryDelegateHeight
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Behavior on height {
|
||||||
|
NumberAnimation { easing.type: Easing.OutQuad; duration: 300 }
|
||||||
|
}
|
||||||
|
model: itemModel
|
||||||
|
width: parent.width
|
||||||
|
orientation: ListView.Horizontal
|
||||||
|
leftMargin: Theme.horizontalPageMargin
|
||||||
|
rightMargin: Theme.horizontalPageMargin
|
||||||
|
spacing: Theme.paddingLarge
|
||||||
|
delegate: LibraryItemDelegate {
|
||||||
|
property string id: model.jellyfinId
|
||||||
|
title: model.name
|
||||||
|
poster: Utils.itemModelImageUrl(appWindow.apiClient.baseUrl, model.jellyfinId, model.imageTags["Primary"], "Primary", {"height": height})
|
||||||
|
Binding on blurhash {
|
||||||
|
when: poster != ""
|
||||||
|
value: model.imageBlurHashes["Primary"][model.imageTags["Primary"]]
|
||||||
|
}
|
||||||
|
landscape: !list.isPortrait
|
||||||
|
progress: (typeof model.userDataPlayedProgress !== 0.0) ? model.userDataPlayedPercentage / 100 : 0.0
|
||||||
|
|
||||||
|
onClicked: {
|
||||||
|
appWindow.navigateToItem(model.jellyfinId, model.mediaType, model.type, model.isFolder);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -90,8 +90,8 @@ SilicaItem {
|
||||||
id: blurhashImage
|
id: blurhashImage
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
fillMode: root.fillMode
|
fillMode: root.fillMode
|
||||||
sourceSize.height: 32
|
sourceSize.height: 16
|
||||||
sourceSize.width: 32 * aspectRatio
|
sourceSize.width: 16 * aspectRatio
|
||||||
source: blurhash.length > 0 ? "image://blurhash/" + encodeURIComponent(blurhash) : ""
|
source: blurhash.length > 0 ? "image://blurhash/" + encodeURIComponent(blurhash) : ""
|
||||||
opacity: 0
|
opacity: 0
|
||||||
}
|
}
|
||||||
|
|
|
@ -78,19 +78,10 @@ Page {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
MoreSection {
|
ItemChildrenShowcase {
|
||||||
//- Section header for films and TV shows that an user hasn't completed yet.
|
//- Section header for films and TV shows that an user hasn't completed yet.
|
||||||
text: qsTr("Resume watching")
|
text: qsTr("Resume watching")
|
||||||
clickable: false
|
clickable: false
|
||||||
busy: userResumeLoader.status === J.ModelStatus.Loading
|
|
||||||
Loader {
|
|
||||||
width: parent.width
|
|
||||||
sourceComponent: carrouselView
|
|
||||||
property alias itemModel: userResumeModel
|
|
||||||
property string collectionType: "series"
|
|
||||||
|
|
||||||
J.ItemModel {
|
|
||||||
id: userResumeModel
|
|
||||||
loader: J.ResumeItemsLoader {
|
loader: J.ResumeItemsLoader {
|
||||||
id: userResumeLoader
|
id: userResumeLoader
|
||||||
apiClient: appWindow.apiClient
|
apiClient: appWindow.apiClient
|
||||||
|
@ -98,53 +89,29 @@ Page {
|
||||||
//recursive: true
|
//recursive: true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
ItemChildrenShowcase {
|
||||||
}
|
|
||||||
MoreSection {
|
|
||||||
//- Section header for next episodes in a TV show that an user was watching.
|
//- Section header for next episodes in a TV show that an user was watching.
|
||||||
text: qsTr("Next up")
|
text: qsTr("Next up")
|
||||||
clickable: false
|
clickable: false
|
||||||
busy: showNextUpLoader.status === J.ModelStatus.Loading
|
|
||||||
|
|
||||||
Loader {
|
|
||||||
width: parent.width
|
|
||||||
sourceComponent: carrouselView
|
|
||||||
property alias itemModel: showNextUpModel
|
|
||||||
property string collectionType: "series"
|
|
||||||
|
|
||||||
J.ItemModel {
|
|
||||||
id: showNextUpModel
|
|
||||||
loader: J.NextUpLoader {
|
loader: J.NextUpLoader {
|
||||||
id: showNextUpLoader
|
id: showNextUpLoader
|
||||||
apiClient: appWindow.apiClient
|
apiClient: appWindow.apiClient
|
||||||
enableUserData: true
|
enableUserData: true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
Repeater {
|
Repeater {
|
||||||
model: mediaLibraryModel
|
model: mediaLibraryModel
|
||||||
MoreSection {
|
ItemChildrenShowcase {
|
||||||
text: model.name
|
text: model.name
|
||||||
busy: userItemModel.status !== J.UsersViewsLoader.Ready
|
onHeaderClicked: appWindow.navigateToItem(model.jellyfinId, model.collectionType, model.type, model.isFolder);
|
||||||
onHeaderClicked: appWindow.navigateToItem(model.jellyfinId, model.mediaType, model.type, model.isFolder);
|
collectionType: model.collectionType || ""
|
||||||
Loader {
|
|
||||||
width: parent.width
|
|
||||||
sourceComponent: carrouselView
|
|
||||||
property alias itemModel: userItemModel
|
|
||||||
property string collectionType: model.collectionType || ""
|
|
||||||
|
|
||||||
J.ItemModel {
|
|
||||||
id: userItemModel
|
|
||||||
loader: J.LatestMediaLoader {
|
loader: J.LatestMediaLoader {
|
||||||
apiClient: appWindow.apiClient
|
apiClient: appWindow.apiClient
|
||||||
parentId: jellyfinId
|
parentId: jellyfinId
|
||||||
}
|
}
|
||||||
}
|
|
||||||
Connections {
|
Connections {
|
||||||
target: mediaLibraryLoader
|
target: mediaLibraryLoader
|
||||||
onReady: userItemModel.reload()
|
onReady: loader.reload()
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -171,6 +138,7 @@ Page {
|
||||||
onStatusChanged: {
|
onStatusChanged: {
|
||||||
if (status == PageStatus.Active) {
|
if (status == PageStatus.Active) {
|
||||||
appWindow.itemData = null
|
appWindow.itemData = null
|
||||||
|
//appWindow.navigateToItem("14b92f36bfc877ae741079fef49a3d80", "MusicArtist", "MusicArtist", true)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -193,50 +161,6 @@ Page {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Component {
|
|
||||||
id: carrouselView
|
|
||||||
SilicaListView {
|
|
||||||
property bool isPortrait: Utils.usePortraitCover(collectionType)
|
|
||||||
id: list
|
|
||||||
clip: true
|
|
||||||
height: {
|
|
||||||
if (count > 0) {
|
|
||||||
if (isPortrait) {
|
|
||||||
Constants.libraryDelegatePosterHeight
|
|
||||||
} else {
|
|
||||||
Constants.libraryDelegateHeight
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
0
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Behavior on height {
|
|
||||||
NumberAnimation { easing.type: Easing.OutQuad; duration: 300 }
|
|
||||||
}
|
|
||||||
model: itemModel
|
|
||||||
width: parent.width
|
|
||||||
orientation: ListView.Horizontal
|
|
||||||
leftMargin: Theme.horizontalPageMargin
|
|
||||||
rightMargin: Theme.horizontalPageMargin
|
|
||||||
spacing: Theme.paddingLarge
|
|
||||||
delegate: LibraryItemDelegate {
|
|
||||||
property string id: model.jellyfinId
|
|
||||||
title: model.name
|
|
||||||
poster: Utils.itemModelImageUrl(appWindow.apiClient.baseUrl, model.jellyfinId, model.imageTags["Primary"], "Primary", {"height": height})
|
|
||||||
Binding on blurhash {
|
|
||||||
when: poster !== ""
|
|
||||||
value: model.imageBlurHashes["Primary"][model.imageTags["Primary"]]
|
|
||||||
}
|
|
||||||
landscape: !isPortrait
|
|
||||||
progress: (typeof model.userDataPlayedProgress !== 0.0) ? model.userDataPlayedPercentage / 100 : 0.0
|
|
||||||
|
|
||||||
onClicked: {
|
|
||||||
appWindow.navigateToItem(model.jellyfinId, model.mediaType, model.type, model.isFolder);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
state: "default"
|
state: "default"
|
||||||
states: [
|
states: [
|
||||||
State {
|
State {
|
||||||
|
|
|
@ -28,6 +28,10 @@ BaseDetailPage {
|
||||||
id: pageRoot
|
id: pageRoot
|
||||||
|
|
||||||
property bool _collectionModelLoaded: false
|
property bool _collectionModelLoaded: false
|
||||||
|
property bool allowSort: true
|
||||||
|
property var modelStatus: collectionModel.loader.modelStatus
|
||||||
|
property string pageTitle: itemData.name
|
||||||
|
property alias loader: collectionModel.loader
|
||||||
|
|
||||||
J.ItemModel {
|
J.ItemModel {
|
||||||
id: collectionModel
|
id: collectionModel
|
||||||
|
@ -35,16 +39,25 @@ BaseDetailPage {
|
||||||
id: collectionLoader
|
id: collectionLoader
|
||||||
apiClient: appWindow.apiClient
|
apiClient: appWindow.apiClient
|
||||||
parentId: itemData.jellyfinId
|
parentId: itemData.jellyfinId
|
||||||
autoReload: itemData.jellyfinId.length > 0 && (pageRoot.status == PageStatus.Active || _collectionModelLoaded)
|
|
||||||
//onParentIdChanged: if (parentId.length > 0) reload()
|
|
||||||
sortBy: "SortName"
|
sortBy: "SortName"
|
||||||
|
autoReload: itemData.jellyfinId.length > 0 && (pageRoot.status == PageStatus.Active || _collectionModelLoaded)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Binding {
|
||||||
|
target: collectionModel.loader
|
||||||
|
property: "autoReload"
|
||||||
|
value: (pageRoot.status == PageStatus.Active || pageRoot._collectionModelLoaded)
|
||||||
|
}
|
||||||
|
|
||||||
|
Connections {
|
||||||
|
target: collectionModel.loader
|
||||||
onStatusChanged: {
|
onStatusChanged: {
|
||||||
if (status === J.ModelStatus.Ready) {
|
if (status === J.ModelStatus.Ready) {
|
||||||
_collectionModelLoaded = true
|
_collectionModelLoaded = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
SilicaGridView {
|
SilicaGridView {
|
||||||
id: gridView
|
id: gridView
|
||||||
|
@ -56,16 +69,18 @@ BaseDetailPage {
|
||||||
visible: itemData.status !== J.ItemLoader.Error
|
visible: itemData.status !== J.ItemLoader.Error
|
||||||
|
|
||||||
header: PageHeader {
|
header: PageHeader {
|
||||||
title: itemData.name || qsTr("Loading")
|
title: pageTitle || qsTr("Loading")
|
||||||
}
|
}
|
||||||
PullDownMenu {
|
PullDownMenu {
|
||||||
id: downMenu
|
id: downMenu
|
||||||
|
visible: pageRoot.allowSort
|
||||||
MenuItem {
|
MenuItem {
|
||||||
|
id: sortMenuItem
|
||||||
//: Menu item for selecting the sort order of a collection
|
//: Menu item for selecting the sort order of a collection
|
||||||
text: qsTr("Sort by")
|
text: qsTr("Sort by")
|
||||||
onClicked: pageStack.push(sortPageComponent)
|
onClicked: pageStack.push(sortPageComponent)
|
||||||
}
|
}
|
||||||
busy: collectionLoader.status === J.ModelStatus.Loading
|
busy: modelStatus === J.ModelStatus.Loading
|
||||||
}
|
}
|
||||||
add: Transition {
|
add: Transition {
|
||||||
id: trans
|
id: trans
|
||||||
|
@ -169,9 +184,9 @@ BaseDetailPage {
|
||||||
onClicked: openMenu()
|
onClicked: openMenu()
|
||||||
|
|
||||||
function apply(field, order) {
|
function apply(field, order) {
|
||||||
collectionLoader.sortBy = field;
|
collectionModel.loader.sortBy = field;
|
||||||
collectionLoader.sortOrder = order;
|
collectionModel.loader.sortOrder = order;
|
||||||
collectionLoader.reload()
|
collectionModel.loader.reload()
|
||||||
pageStack.pop()
|
pageStack.pop()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
287
sailfish/qml/pages/itemdetails/MusicArtistPage.qml
Normal file
287
sailfish/qml/pages/itemdetails/MusicArtistPage.qml
Normal file
|
@ -0,0 +1,287 @@
|
||||||
|
/*
|
||||||
|
Sailfin: a Jellyfin client written using Qt
|
||||||
|
Copyright (C) 2022 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
|
||||||
|
*/
|
||||||
|
import QtQuick 2.6
|
||||||
|
import QtQuick.Layouts 1.1
|
||||||
|
import Sailfish.Silica 1.0
|
||||||
|
|
||||||
|
import nl.netsoj.chris.Jellyfin 1.0 as J
|
||||||
|
|
||||||
|
import "../../components"
|
||||||
|
import "../.."
|
||||||
|
|
||||||
|
BaseDetailPage {
|
||||||
|
id: albumPage
|
||||||
|
readonly property int _maxItems: 12
|
||||||
|
|
||||||
|
J.ItemModel {
|
||||||
|
id: albumsModel
|
||||||
|
loader: J.UserItemsLoader {
|
||||||
|
apiClient: appWindow.apiClient
|
||||||
|
sortBy: "PremiereDate,ProductionYear,SortName"
|
||||||
|
sortOrder: "Descending"
|
||||||
|
fields: [J.ItemFields.ItemCounts, J.ItemFields.PrimaryImageAspectRatio]
|
||||||
|
includeItemTypes: ["MusicAlbum"]
|
||||||
|
albumArtistIds: itemData.jellyfinId
|
||||||
|
recursive: true
|
||||||
|
autoReload: itemData.jellyfinId.length > 0
|
||||||
|
limit: _maxItems
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Component {
|
||||||
|
id: fullAlbumsModelComponent
|
||||||
|
J.UserItemsLoader {
|
||||||
|
apiClient: appWindow.apiClient
|
||||||
|
sortBy: "PremiereDate,ProductionYear,SortName"
|
||||||
|
sortOrder: "Descending"
|
||||||
|
fields: [J.ItemFields.ItemCounts, J.ItemFields.PrimaryImageAspectRatio]
|
||||||
|
includeItemTypes: ["MusicAlbum"]
|
||||||
|
albumArtistIds: itemData.jellyfinId
|
||||||
|
recursive: true
|
||||||
|
autoReload: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
J.ItemModel {
|
||||||
|
id: appearsOnModel
|
||||||
|
loader: J.UserItemsLoader {
|
||||||
|
apiClient: appWindow.apiClient
|
||||||
|
sortBy: "PremiereDate,ProductionYear,SortName"
|
||||||
|
sortOrder: "Descending"
|
||||||
|
fields: [J.ItemFields.ItemCounts, J.ItemFields.PrimaryImageAspectRatio]
|
||||||
|
includeItemTypes: ["MusicAlbum"]
|
||||||
|
contributingArtistIds: itemData.jellyfinId
|
||||||
|
recursive: true
|
||||||
|
autoReload: itemData.jellyfinId.length > 0
|
||||||
|
limit: _maxItems
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Component {
|
||||||
|
id: fullAppearsOnModelComponent
|
||||||
|
J.UserItemsLoader {
|
||||||
|
apiClient: appWindow.apiClient
|
||||||
|
sortBy: "PremiereDate,ProductionYear,SortName"
|
||||||
|
sortOrder: "Descending"
|
||||||
|
fields: [J.ItemFields.ItemCounts, J.ItemFields.PrimaryImageAspectRatio]
|
||||||
|
includeItemTypes: ["MusicAlbum"]
|
||||||
|
contributingArtistIds: itemData.jellyfinId
|
||||||
|
recursive: true
|
||||||
|
autoReload: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
SilicaFlickable {
|
||||||
|
anchors.fill: parent
|
||||||
|
contentHeight: content.height
|
||||||
|
|
||||||
|
Column {
|
||||||
|
id: content
|
||||||
|
width: parent.width
|
||||||
|
|
||||||
|
Item {
|
||||||
|
id: header
|
||||||
|
width: parent.width
|
||||||
|
height: backdrop.height + title.height
|
||||||
|
|
||||||
|
RemoteImage {
|
||||||
|
id: backdrop
|
||||||
|
anchors {
|
||||||
|
top: parent.top
|
||||||
|
left: parent.left
|
||||||
|
right: parent.right
|
||||||
|
}
|
||||||
|
height: width / 16 * 9
|
||||||
|
fillMode: Image.PreserveAspectCrop
|
||||||
|
source: Utils.itemBackdropUrl(apiClient.baseUrl, itemData, 0, {"maxWidth": parent.width})
|
||||||
|
blurhash: itemData.imageBlurHashes["Backdrop"][itemData.backdropImageTags[0]]
|
||||||
|
}
|
||||||
|
|
||||||
|
Shim {
|
||||||
|
anchors {
|
||||||
|
top: parent.top
|
||||||
|
left: parent.left
|
||||||
|
right: parent.right
|
||||||
|
}
|
||||||
|
height: Theme.itemSizeHuge
|
||||||
|
upsideDown: true
|
||||||
|
shimOpacity: Theme.opacityOverlay
|
||||||
|
}
|
||||||
|
|
||||||
|
RemoteImage {
|
||||||
|
id: artistImage
|
||||||
|
anchors {
|
||||||
|
right: parent.right
|
||||||
|
rightMargin: Theme.horizontalPageMargin
|
||||||
|
bottom: title.bottom
|
||||||
|
}
|
||||||
|
source: Utils.itemImageUrl(apiClient.baseUrl, itemData, "Primary", {"maxWidth": parent.width})
|
||||||
|
blurhash: itemData.imageBlurHashes["Primary"][itemData.imageTags["Primary"]]
|
||||||
|
width: Constants.libraryDelegateWidth
|
||||||
|
height: width / itemData.primaryImageAspectRatio
|
||||||
|
fallbackColor: Utils.colorFromString(itemData.name)
|
||||||
|
}
|
||||||
|
|
||||||
|
PageHeader {
|
||||||
|
id: title
|
||||||
|
title: itemData.name
|
||||||
|
description: qsTr("%1 songs | %2 albums")
|
||||||
|
.arg(itemData.songCount)
|
||||||
|
.arg(itemData.albumCount)
|
||||||
|
anchors {
|
||||||
|
top: backdrop.bottom
|
||||||
|
left: parent.left
|
||||||
|
right: artistImage.left
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Spacer
|
||||||
|
Item { height: Theme.paddingLarge; width: 1; visible: !aboutBackground.visible }
|
||||||
|
BackgroundItem {
|
||||||
|
property bool _expanded: false
|
||||||
|
id: aboutBackground
|
||||||
|
anchors {
|
||||||
|
left: parent.left
|
||||||
|
right: parent.right
|
||||||
|
}
|
||||||
|
height: about.height + Theme.paddingLarge
|
||||||
|
clip: true
|
||||||
|
onClicked: aboutBackground._expanded = !aboutBackground._expanded
|
||||||
|
visible: aboutLabel.text.length > 0
|
||||||
|
//Behavior on height { SmoothedAnimation { duration: 300; } }
|
||||||
|
|
||||||
|
Item {
|
||||||
|
id: about
|
||||||
|
anchors {
|
||||||
|
left: parent.left
|
||||||
|
leftMargin: Theme.horizontalPageMargin
|
||||||
|
right: parent.right
|
||||||
|
rightMargin: Theme.horizontalPageMargin
|
||||||
|
}
|
||||||
|
height: aboutLabel.height
|
||||||
|
Label {
|
||||||
|
id: aboutLabel
|
||||||
|
anchors {
|
||||||
|
left: parent.left
|
||||||
|
right: parent.right
|
||||||
|
}
|
||||||
|
topPadding: Theme.paddingLarge
|
||||||
|
bottomPadding: Theme.paddingLarge
|
||||||
|
color: aboutBackground.highlighted ? Theme.highlightColor : Theme.primaryColor
|
||||||
|
text: itemData.overview
|
||||||
|
wrapMode: Text.WordWrap
|
||||||
|
height: aboutBackground._expanded ? implicitHeight : Math.min(font.pixelSize * lineHeight * 8, implicitHeight)
|
||||||
|
Behavior on height { SmoothedAnimation { id: expandAnimation; duration: 300; } }
|
||||||
|
textFormat: Text.PlainText
|
||||||
|
}
|
||||||
|
OpacityRampEffect {
|
||||||
|
enabled: !aboutBackground._expanded || expandAnimation.running
|
||||||
|
offset: aboutBackground._expanded ? 1.0 : 0.5
|
||||||
|
sourceItem: aboutLabel
|
||||||
|
direction: OpacityRamp.TopToBottom
|
||||||
|
}
|
||||||
|
}
|
||||||
|
HighlightImage {
|
||||||
|
anchors {
|
||||||
|
right: parent.right
|
||||||
|
rightMargin: Theme.horizontalPageMargin
|
||||||
|
bottom: parent.bottom
|
||||||
|
bottomMargin: Theme.paddingMedium
|
||||||
|
}
|
||||||
|
|
||||||
|
source: "image://theme/icon-lock-more"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
MoreSection {
|
||||||
|
text: qsTr("Discography")
|
||||||
|
visible: albumRepeater.count > 0
|
||||||
|
onHeaderClicked: pageStack.push(Qt.resolvedUrl("CollectionPage.qml"), {
|
||||||
|
"loader": fullAlbumsModelComponent.createObject(albumPage),
|
||||||
|
"allowSort": false,
|
||||||
|
//: Page title for the page with an overview of all albums, eps and singles by a specific artist
|
||||||
|
"pageTitle": qsTr("Discography of %1").arg(itemData.name)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
GridLayout {
|
||||||
|
width: parent.width
|
||||||
|
columns: 3
|
||||||
|
columnSpacing: 0
|
||||||
|
rowSpacing: 0
|
||||||
|
anchors {
|
||||||
|
left: parent.left
|
||||||
|
right: parent.right
|
||||||
|
}
|
||||||
|
Repeater {
|
||||||
|
id: albumRepeater
|
||||||
|
model: albumsModel
|
||||||
|
|
||||||
|
LibraryItemDelegate {
|
||||||
|
readonly property int _multiplier: index === 0 ? 2 : 1
|
||||||
|
poster: Utils.itemModelImageUrl(appWindow.apiClient.baseUrl, model.jellyfinId, model.imageTags["Primary"], "Primary", {"height": height})
|
||||||
|
blurhash: model.imageBlurHashes["Primary"][model.imageTags["Primary"]]
|
||||||
|
title: model.name
|
||||||
|
Layout.preferredWidth: Constants.libraryDelegateWidth * _multiplier
|
||||||
|
Layout.preferredHeight: Constants.libraryDelegateHeight * _multiplier
|
||||||
|
Layout.rowSpan: _multiplier
|
||||||
|
Layout.columnSpan: _multiplier
|
||||||
|
onClicked: appWindow.navigateToItem(model.jellyfinId, model.mediaType, model.type, model.isFolder)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
MoreSection {
|
||||||
|
text: qsTr("Appears on")
|
||||||
|
visible: appearsOnRepeater.count > 0
|
||||||
|
onHeaderClicked: pageStack.push(Qt.resolvedUrl("CollectionPage.qml"), {
|
||||||
|
"loader": fullAppearsOnModelComponent.createObject(albumPage),
|
||||||
|
"allowSort": false,
|
||||||
|
//: Page title for the page with an overview of all albums a specific artist appears on
|
||||||
|
"pageTitle": qsTr("%1 appears on").arg(itemData.name)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
GridLayout {
|
||||||
|
width: parent.width
|
||||||
|
columns: 3
|
||||||
|
columnSpacing: 0
|
||||||
|
rowSpacing: 0
|
||||||
|
anchors {
|
||||||
|
left: parent.left
|
||||||
|
right: parent.right
|
||||||
|
}
|
||||||
|
Repeater {
|
||||||
|
id: appearsOnRepeater
|
||||||
|
model: appearsOnModel
|
||||||
|
|
||||||
|
LibraryItemDelegate {
|
||||||
|
readonly property int _multiplier: 1
|
||||||
|
poster: Utils.itemModelImageUrl(appWindow.apiClient.baseUrl, model.jellyfinId, model.imageTags["Primary"], "Primary", {"height": height})
|
||||||
|
blurhash: model.imageBlurHashes["Primary"][model.imageTags["Primary"]]
|
||||||
|
title: model.name
|
||||||
|
Layout.alignment: Qt.AlignLeft | Qt.AlignTop
|
||||||
|
Layout.preferredWidth: Constants.libraryDelegateWidth * _multiplier
|
||||||
|
Layout.preferredHeight: Constants.libraryDelegateHeight * _multiplier
|
||||||
|
Layout.fillWidth: false
|
||||||
|
Layout.fillHeight: false
|
||||||
|
onClicked: appWindow.navigateToItem(model.jellyfinId, model.mediaType, model.type, model.isFolder)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
138
sailfish/qml/pages/itemdetails/MusicLibraryPage.qml
Normal file
138
sailfish/qml/pages/itemdetails/MusicLibraryPage.qml
Normal file
|
@ -0,0 +1,138 @@
|
||||||
|
import QtQuick 2.6
|
||||||
|
import Sailfish.Silica 1.0
|
||||||
|
|
||||||
|
import nl.netsoj.chris.Jellyfin 1.0 as J
|
||||||
|
|
||||||
|
import "../../components"
|
||||||
|
import "../.."
|
||||||
|
|
||||||
|
BaseDetailPage {
|
||||||
|
id: musicLibraryPage
|
||||||
|
property bool _firstTimeLoaded: false
|
||||||
|
|
||||||
|
onStatusChanged: {
|
||||||
|
if (status == PageStatus.Active) {
|
||||||
|
_firstTimeLoaded = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SilicaFlickable {
|
||||||
|
anchors.fill: parent
|
||||||
|
contentHeight: content.height
|
||||||
|
|
||||||
|
Component {
|
||||||
|
id: albumArtistLoaderComponent
|
||||||
|
J.AlbumArtistLoader {
|
||||||
|
apiClient: appWindow.apiClient
|
||||||
|
parentId: itemData.jellyfinId
|
||||||
|
autoReload: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Component {
|
||||||
|
id: albumLoaderComponent
|
||||||
|
J.UserItemsLoader {
|
||||||
|
apiClient: appWindow.apiClient
|
||||||
|
parentId: itemData.jellyfinId
|
||||||
|
includeItemTypes: "MusicAlbum"
|
||||||
|
recursive: true
|
||||||
|
sortBy: "SortName"
|
||||||
|
autoReload: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Component {
|
||||||
|
id: playlistLoaderComponent
|
||||||
|
J.UserItemsLoader {
|
||||||
|
apiClient: appWindow.apiClient
|
||||||
|
parentId: itemData.jellyfinId
|
||||||
|
includeItemTypes: "Playlist"
|
||||||
|
recursive: true
|
||||||
|
sortBy: "SortName"
|
||||||
|
autoReload: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Column {
|
||||||
|
id: content
|
||||||
|
width: parent.width
|
||||||
|
|
||||||
|
PageHeader {
|
||||||
|
title: itemData.name
|
||||||
|
}
|
||||||
|
|
||||||
|
ItemChildrenShowcase {
|
||||||
|
//: Header on music library: Recently added music albums
|
||||||
|
text: qsTr("Recently added")
|
||||||
|
//collapseWhenEmpty: false
|
||||||
|
extraBusy: !_firstTimeLoaded
|
||||||
|
clickable: false
|
||||||
|
loader: J.LatestMediaLoader {
|
||||||
|
apiClient: appWindow.apiClient
|
||||||
|
parentId: itemData.jellyfinId
|
||||||
|
autoReload: _firstTimeLoaded && itemData.jellyfinId.length > 0
|
||||||
|
includeItemTypes: "Audio"
|
||||||
|
limit: 12
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
ItemChildrenShowcase {
|
||||||
|
text: qsTr("Albums")
|
||||||
|
//collapseWhenEmpty: false
|
||||||
|
extraBusy: !_firstTimeLoaded
|
||||||
|
loader: J.UserItemsLoader {
|
||||||
|
apiClient: appWindow.apiClient
|
||||||
|
parentId: itemData.jellyfinId
|
||||||
|
includeItemTypes: "MusicAlbum"
|
||||||
|
autoReload: _firstTimeLoaded && itemData.jellyfinId.length > 0
|
||||||
|
sortBy: "Random"
|
||||||
|
recursive: true
|
||||||
|
limit: 12
|
||||||
|
}
|
||||||
|
onHeaderClicked: pageStack.push(Qt.resolvedUrl("CollectionPage.qml"), {
|
||||||
|
"loader": albumLoaderComponent.createObject(musicLibraryPage),
|
||||||
|
//: Page title for the list of all albums within the music library
|
||||||
|
"pageTitle": qsTr("Albums")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
ItemChildrenShowcase {
|
||||||
|
text: qsTr("Playlists")
|
||||||
|
//collapseWhenEmpty: false
|
||||||
|
extraBusy: !_firstTimeLoaded
|
||||||
|
loader: J.UserItemsLoader {
|
||||||
|
apiClient: appWindow.apiClient
|
||||||
|
parentId: itemData.jellyfinId
|
||||||
|
includeItemTypes: "Playlist"
|
||||||
|
autoReload: _firstTimeLoaded && itemData.jellyfinId.length > 0
|
||||||
|
sortBy: "Random"
|
||||||
|
recursive: true
|
||||||
|
limit: 12
|
||||||
|
}
|
||||||
|
onHeaderClicked: pageStack.push(Qt.resolvedUrl("CollectionPage.qml"), {
|
||||||
|
"loader": playlistLoaderComponent.createObject(musicLibraryPage),
|
||||||
|
//: Page title for the list of all playlists within the music library
|
||||||
|
"pageTitle": qsTr("Playlists")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
ItemChildrenShowcase {
|
||||||
|
//: Header for music artists
|
||||||
|
text: qsTr("Artists")
|
||||||
|
//collapseWhenEmpty: false
|
||||||
|
extraBusy: !_firstTimeLoaded
|
||||||
|
loader: J.AlbumArtistLoader {
|
||||||
|
apiClient: appWindow.apiClient
|
||||||
|
parentId: itemData.jellyfinId
|
||||||
|
autoReload: _firstTimeLoaded && itemData.jellyfinId.length > 0
|
||||||
|
limit: 12
|
||||||
|
}
|
||||||
|
onHeaderClicked: pageStack.push(Qt.resolvedUrl("CollectionPage.qml"), {
|
||||||
|
"loader": albumArtistLoaderComponent.createObject(musicLibraryPage),
|
||||||
|
"allowSort": false,
|
||||||
|
//: Page title for the list of all artists within the music library
|
||||||
|
"pageTitle": qsTr("Artists")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue