#ifndef JELLYFINITEM_H #define JELLYFINITEM_H #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "jellyfinapiclient.h" namespace Jellyfin { /** * @brief Base class for a serializable object. * * This class will be (de)serialized based on its properties. * Note: it must have a constructor without arguments marked with Q_INVOKABLE */ class JsonSerializable : public QObject { Q_OBJECT public: Q_INVOKABLE JsonSerializable(QObject *parent); /** * @brief Sets this objects properties based on obj. * @param obj The data to load into this object. */ void deserialize(const QJsonObject &obj); QJsonObject serialize() const; private: QVariant jsonToVariant(QMetaProperty prop, const QJsonValue &val, const QJsonObject &root); QJsonValue variantToJson(const QVariant var) const; /** * @brief Sets the first letter of the string to lower case (to make it camelCase). * @param str The string to modify * @return THe modified string */ static QString fromPascalCase(QString str); /** * @brief Sets the first letter of the string to uper case (to make it PascalCase). * @param str The string to modify * @return THe modified string */ static QString toPascalCase(QString str); static const QRegularExpression m_listExpression; /** * @brief Qt is doing weird. I'll keep track of the metatypes myself. */ QHash m_nameMetatypeMap; }; /** * @brief An "interface" for a remote data source * * This class is basically a base class for JSON data that can be fetched from over the network. * Subclasses should reimplement reload and call setStatus to update the QML part of the code * appropiatly. */ class RemoteData : public JsonSerializable { Q_OBJECT public: enum Status { /// The data is unitialized and not loading either. Uninitialised, /// The data is being loaded over the network Loading, /// The data is ready, the properties in this object are up to date. Ready, /// An error has occurred while loading the data. See error() for more details. Error }; Q_ENUM(Status) explicit RemoteData(QObject *parent = nullptr); Q_PROPERTY(ApiClient *apiClient MEMBER m_apiClient WRITE setApiClient NOTIFY apiClientChanged STORED false) Q_PROPERTY(Status status READ status NOTIFY statusChanged STORED false) Q_PROPERTY(QNetworkReply::NetworkError error READ error NOTIFY errorChanged STORED false) Q_PROPERTY(QString errorString READ errorString NOTIFY errorStringChanged STORED false) Status status() const { return m_status; } QNetworkReply::NetworkError error() const { return m_error; } QString errorString() const { return m_errorString; } void setApiClient(ApiClient *newApiClient); signals: void statusChanged(Status newStatus); void apiClientChanged(ApiClient *newApiClient); void errorChanged(QNetworkReply::NetworkError newError); void errorStringChanged(QString newErrorString); public slots: virtual void reload() = 0; protected: void setStatus(Status newStatus); void setError(QNetworkReply::NetworkError error); void setErrorString(const QString &newErrorString); ApiClient *m_apiClient = nullptr; private: Status m_status = Uninitialised; QNetworkReply::NetworkError m_error = QNetworkReply::NoError; QString m_errorString; }; class MediaStream : public JsonSerializable { Q_OBJECT public: Q_INVOKABLE explicit MediaStream(QObject *parent = nullptr); MediaStream(const MediaStream &other); bool operator==(const MediaStream &other); virtual ~MediaStream() { qDebug() << "MediaStream destroyed"; } enum MediaStreamType { Undefined, Audio, Video, Subtitle, EmbeddedImage }; Q_ENUM(MediaStreamType) Q_PROPERTY(QString codec MEMBER m_codec NOTIFY codecChanged) Q_PROPERTY(QString codecTag MEMBER m_codecTag NOTIFY codecTagChanged) Q_PROPERTY(QString language MEMBER m_language NOTIFY languageChanged) Q_PROPERTY(QString displayTitle MEMBER m_displayTitle NOTIFY displayTitleChanged) Q_PROPERTY(MediaStreamType type MEMBER m_type NOTIFY typeChanged) Q_PROPERTY(int index MEMBER m_index NOTIFY indexChanged) signals: void codecChanged(const QString &newCodec); void codecTagChanged(const QString &newCodecTag); void languageChanged(const QString &newLanguage); void displayTitleChanged(const QString &newDisplayTitle); void typeChanged(MediaStreamType newType); void indexChanged(int newIndex); private: QString m_codec; QString m_codecTag; QString m_language; QString m_displayTitle; MediaStreamType m_type = Undefined; int m_index = -1; }; class Item : public RemoteData { Q_OBJECT public: Q_INVOKABLE explicit Item(QObject *parent = nullptr); Q_PROPERTY(QString jellyfinId READ jellyfinId WRITE setJellyfinId NOTIFY jellyfinIdChanged) // Based on https://github.com/jellyfin/jellyfin/blob/907695dec7fda152d0e17c1197637bc0e17c9928/MediaBrowser.Model/Dto/BaseItemDto.cs // I copy, pasted and replaced. I feel like a Go programmer implementing generic containers. // If this were D, I would've writed a compile-time C# parser to parse that source code at compile time, extract // the properties and generate a class based on that. // Doing that in C++ would be more difficult and I dislike qmake. Does it even support running programs at compile time? // But here I am, using ctrl-C++ Q_PROPERTY(QString name MEMBER m_name NOTIFY nameChanged) Q_PROPERTY(QString originalTitle MEMBER m_originalTitle NOTIFY originalTitleChanged) Q_PROPERTY(QString serverId MEMBER m_serverId NOTIFY serverIdChanged) Q_PROPERTY(QString etag MEMBER m_etag NOTIFY etagChanged) Q_PROPERTY(QString sourceType MEMBER m_sourceType NOTIFY sourceTypeChanged) Q_PROPERTY(QString playlistItemId MEMBER m_playlistItemId NOTIFY playlistItemIdChanged) Q_PROPERTY(QDateTime dateCreated MEMBER m_dateCreated NOTIFY dateCreatedChanged) Q_PROPERTY(QDateTime dateLastMediaAdded MEMBER m_dateLastMediaAdded NOTIFY dateLastMediaAddedChanged) Q_PROPERTY(QString extraType MEMBER m_extraType NOTIFY extraTypeChanged) Q_PROPERTY(int airsBeforeSeasonNumber READ airsBeforeSeasonNumber WRITE setAirsBeforeSeasonNumber NOTIFY airsBeforeSeasonNumberChanged) Q_PROPERTY(int airsAfterSeasonNumber READ airsAfterSeasonNumber WRITE setAirsAfterSeasonNumber NOTIFY airsAfterSeasonNumberChanged) Q_PROPERTY(int airsBeforeEpisodeNumber READ airsBeforeEpisodeNumber WRITE setAirsBeforeEpisodeNumber NOTIFY airsBeforeEpisodeNumberChanged) Q_PROPERTY(bool canDelete READ canDelete WRITE setCanDelete NOTIFY canDeleteChanged) Q_PROPERTY(bool canDownload READ canDownload WRITE setCanDownload NOTIFY canDownloadChanged) Q_PROPERTY(bool hasSubtitles READ hasSubtitles WRITE setHasSubtitles NOTIFY hasSubtitlesChanged) Q_PROPERTY(QString preferredMetadataLanguage MEMBER m_preferredMetadataLanguage NOTIFY preferredMetadataLanguageChanged) Q_PROPERTY(QString preferredMetadataCountryCode MEMBER m_preferredMetadataCountryCode NOTIFY preferredMetadataCountryCodeChanged) Q_PROPERTY(bool supportsSync READ supportsSync WRITE setSupportsSync NOTIFY supportsSyncChanged) Q_PROPERTY(QString container MEMBER m_container NOTIFY containerChanged) Q_PROPERTY(QString sortName MEMBER m_sortName NOTIFY sortNameChanged) Q_PROPERTY(QString forcedSortName MEMBER m_forcedSortName NOTIFY forcedSortNameChanged) //SKIP: Video3DFormat Q_PROPERTY(QDateTime premiereData MEMBER m_premiereDate NOTIFY premiereDateChanged) //SKIP: ExternalUrls //SKIP: MediaSources Q_PROPERTY(float criticRating READ criticRating WRITE setCriticRating NOTIFY criticRatingChanged) Q_PROPERTY(QStringList productionLocations MEMBER m_productionLocations NOTIFY productionLocationsChanged) // Handpicked, important ones Q_PROPERTY(qint64 runTimeTicks READ runTimeTicks WRITE setRunTimeTicks NOTIFY runTimeTicksChanged) Q_PROPERTY(QString overview MEMBER m_overview NOTIFY overviewChanged) Q_PROPERTY(int productionYear READ productionYear WRITE setProductionYear NOTIFY productionYearChanged) Q_PROPERTY(int indexNumber READ indexNumber WRITE setIndexNumber NOTIFY indexNumberChanged) Q_PROPERTY(int indexNumberEnd READ indexNumberEnd WRITE setIndexNumberEnd NOTIFY indexNumberEndChanged) Q_PROPERTY(bool isFolder READ isFolder WRITE setIsFolder NOTIFY isFolderChanged) Q_PROPERTY(QString type MEMBER m_type NOTIFY typeChanged) Q_PROPERTY(QString seriesName MEMBER m_seriesName NOTIFY seriesNameChanged) Q_PROPERTY(QString seasonName MEMBER m_seasonName NOTIFY seasonNameChanged) Q_PROPERTY(QList __list__mediaStreams MEMBER __list__m_mediaStreams NOTIFY mediaStreamsChanged) Q_PROPERTY(QVariantList mediaStreams MEMBER m_mediaStreams NOTIFY mediaStreamsChanged STORED false) QString jellyfinId() const { return m_id; } void setJellyfinId(QString newId); int airsBeforeSeasonNumber() const { return m_airsBeforeSeasonNumber.value_or(-1); } void setAirsBeforeSeasonNumber(int newAirsBeforeSeasonNumber) { m_airsBeforeSeasonNumber = newAirsBeforeSeasonNumber; emit airsBeforeSeasonNumberChanged(newAirsBeforeSeasonNumber); } int airsAfterSeasonNumber() const { return m_airsAfterSeasonNumber.value_or(-1); } void setAirsAfterSeasonNumber(int newAirsAfterSeasonNumber) { m_airsAfterSeasonNumber = newAirsAfterSeasonNumber; emit airsAfterSeasonNumberChanged(newAirsAfterSeasonNumber); } int airsBeforeEpisodeNumber() const { return m_airsBeforeEpisodeNumber.value_or(-1); } void setAirsBeforeEpisodeNumber(int newAirsBeforeEpisodeNumber) { m_airsBeforeEpisodeNumber = newAirsBeforeEpisodeNumber; emit airsBeforeEpisodeNumberChanged(newAirsBeforeEpisodeNumber); } bool canDelete() const { return m_canDelete.value_or(false); } void setCanDelete(bool newCanDelete) { m_canDelete = newCanDelete; emit canDeleteChanged(newCanDelete); } bool canDownload() const { return m_canDownload.value_or(false); } void setCanDownload(bool newCanDownload) { m_canDownload = newCanDownload; emit canDownloadChanged(newCanDownload); } bool hasSubtitles() const { return m_hasSubtitles.value_or(false); } void setHasSubtitles(bool newHasSubtitles) { m_hasSubtitles = newHasSubtitles; emit hasSubtitlesChanged(newHasSubtitles); } bool supportsSync() const { return m_supportsSync.value_or(false); } void setSupportsSync(bool newSupportsSync) { m_supportsSync = newSupportsSync; emit supportsSyncChanged(newSupportsSync); } float criticRating() const { return m_criticRating.value_or(std::nanf("")); } void setCriticRating(float newCriticRating) { m_criticRating = newCriticRating; emit criticRatingChanged(newCriticRating); } // Handpicked, important ones qint64 runTimeTicks() const { return m_runTimeTicks.value_or(-1); } void setRunTimeTicks(qint64 newRunTimeTicks) { m_runTimeTicks = newRunTimeTicks; emit runTimeTicksChanged(newRunTimeTicks); } int productionYear() const { return m_productionYear.value_or(-1); } void setProductionYear(int newProductionYear) { m_productionYear = std::optional(newProductionYear); emit productionYearChanged(newProductionYear); } int indexNumber() const { return m_indexNumber.value_or(-1); } void setIndexNumber(int newIndexNumber) { m_indexNumber = std::optional(newIndexNumber); emit indexNumberChanged(newIndexNumber); } int indexNumberEnd() const { return m_indexNumberEnd.value_or(-1); } void setIndexNumberEnd(int newIndexNumberEnd) { m_indexNumberEnd = std::optional(newIndexNumberEnd); emit indexNumberEndChanged(newIndexNumberEnd); } bool isFolder() const { return m_isFolder.value_or(false); } void setIsFolder(bool newIsFolder) { m_isFolder = newIsFolder; emit isFolderChanged(newIsFolder); } //QQmlListProperty mediaStreams() { return toReadOnlyQmlListProperty(m_mediaStreams); } //QList mediaStreams() { return *reinterpret_cast *>(&m_mediaStreams); } QVariantList mediaStreams() { QVariantList l; for (auto e: m_mediaStreams) l.append(QVariant::fromValue(e)); return l;} signals: void jellyfinIdChanged(const QString &newId); void nameChanged(const QString &newName); void originalTitleChanged(const QString &newOriginalTitle); void serverIdChanged(const QString &newServerId); void etagChanged(const QString &newEtag); void sourceTypeChanged(const QString &sourceType); void playlistItemIdChanged(const QString &playlistItemIdChanged); void dateCreatedChanged(QDateTime newDateCreatedChanged); void dateLastMediaAddedChanged(QDateTime newDateLastMediaAdded); void extraTypeChanged(const QString &newExtraType); void airsBeforeSeasonNumberChanged(int newAirsBeforeSeasonNumber); void airsAfterSeasonNumberChanged(int newAirsAfterSeasonNumber); void airsBeforeEpisodeNumberChanged(int newAirsAfterEpisodeNumber); bool canDeleteChanged(bool newCanDelete); void canDownloadChanged(bool newCanDownload); void hasSubtitlesChanged(bool newHasSubtitles); void preferredMetadataLanguageChanged(const QString &newPreferredMetadataLanguage); void preferredMetadataCountryCodeChanged(const QString &newPreferredMetadataCountryCode); void supportsSyncChanged(bool newSupportsSync); void containerChanged(const QString &newContainer); void sortNameChanged(const QString &newSortName); void forcedSortNameChanged(const QString &newForcedSortName); void premiereDateChanged(QDateTime newPremiereDate); void criticRatingChanged(float newCriticRating); void productionLocationsChanged(QStringList newProductionLocations); // Handpicked, important ones void runTimeTicksChanged(qint64 newRunTimeTicks); void overviewChanged(const QString &newOverview); void productionYearChanged(int newProductionYear); void indexNumberChanged(int newIndexNumber); void indexNumberEndChanged(int newIndexNumberEnd); void isFolderChanged(bool newIsFolder); void typeChanged(const QString &newType); void seriesNameChanged(const QString &newSeriesName); void seasonNameChanged(const QString &newSeasonName); void mediaStreamsChanged(/*const QList &newMediaStreams*/); public slots: /** * @brief (Re)loads the item from the Jellyfin server. */ void reload() override; protected: QString m_id; QString m_name; QString m_originalTitle; QString m_serverId; QString m_etag; QString m_sourceType; QString m_playlistItemId; QDateTime m_dateCreated; QDateTime m_dateLastMediaAdded; QString m_extraType; std::optional m_airsBeforeSeasonNumber = std::nullopt; std::optional m_airsAfterSeasonNumber = std::nullopt; std::optional m_airsBeforeEpisodeNumber = std::nullopt; std::optional m_canDelete = std::nullopt; std::optional m_canDownload = std::nullopt; std::optional m_hasSubtitles = std::nullopt; QString m_preferredMetadataLanguage; QString m_preferredMetadataCountryCode; std::optional m_supportsSync = std::nullopt; QString m_container; QString m_sortName; QString m_forcedSortName; QDateTime m_premiereDate; std::optional m_criticRating = std::nullopt; QStringList m_productionLocations; // Handpicked, important ones std::optional m_runTimeTicks = std::nullopt; QString m_overview; std::optional m_productionYear = std::nullopt; std::optional m_indexNumber = std::nullopt; std::optional m_indexNumberEnd = std::nullopt; std::optional m_isFolder = std::nullopt; QString m_type; QString m_seriesName; QString m_seasonName; QList __list__m_mediaStreams; QVariantList m_mediaStreams; template QQmlListProperty toReadOnlyQmlListProperty(QList &list) { return QQmlListProperty(this, std::addressof(list), &qlist_count, &qlist_at); } template static int qlist_count(QQmlListProperty *p) { return reinterpret_cast *>(p->data)->count(); } template static T *qlist_at(QQmlListProperty *p, int idx) { return reinterpret_cast *>(p->data)->at(idx); } }; void registerSerializableJsonTypes(const char* URI); } #endif // JELLYFINITEM_H