/* * Strawberry Music Player * This file was part of Clementine. * Copyright 2011, David Sansome * Copyright 2019-2023, Jonas Kvinge * * Strawberry is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * Strawberry 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with Strawberry. If not, see . * */ #ifndef TAGREADERCLIENT_H #define TAGREADERCLIENT_H #include "config.h" #include #include #include #include #include "core/messagehandler.h" #include "core/workerpool.h" #include "song.h" #include "tagreadermessages.pb.h" class QThread; class Song; template class WorkerPool; class TagReaderClient : public QObject { Q_OBJECT public: explicit TagReaderClient(QObject *parent = nullptr); using HandlerType = AbstractMessageHandler; using ReplyType = HandlerType::ReplyType; void Start(); void ExitAsync(); enum class SaveType { NoType = 0, Tags = 1, PlayCount = 2, Rating = 4, Cover = 8 }; Q_DECLARE_FLAGS(SaveTypes, SaveType) class SaveCoverOptions { public: explicit SaveCoverOptions(const QString &_cover_filename = QString(), const QByteArray &_cover_data = QByteArray(), const QString &_mime_type = QString()) : cover_filename(_cover_filename), cover_data(_cover_data), mime_type(_mime_type) {} explicit SaveCoverOptions(const QString &_cover_filename, const QString &_mime_type = QString()) : cover_filename(_cover_filename), mime_type(_mime_type) {} explicit SaveCoverOptions(const QByteArray &_cover_data, const QString &_mime_type = QString()) : cover_data(_cover_data), mime_type(_mime_type) {} QString cover_filename; QByteArray cover_data; QString mime_type; }; ReplyType *IsMediaFile(const QString &filename); ReplyType *ReadFile(const QString &filename); ReplyType *SaveFile(const QString &filename, const Song &metadata, const SaveTypes types = SaveType::Tags, const SaveCoverOptions &save_cover_options = SaveCoverOptions()); ReplyType *LoadEmbeddedArt(const QString &filename); ReplyType *SaveEmbeddedArt(const QString &filename, const SaveCoverOptions &save_cover_options); ReplyType *UpdateSongPlaycount(const Song &metadata); ReplyType *UpdateSongRating(const Song &metadata); // Convenience functions that call the above functions and wait for a response. // These block the calling thread with a semaphore, and must NOT be called from the TagReaderClient's thread. void ReadFileBlocking(const QString &filename, Song *song); bool SaveFileBlocking(const QString &filename, const Song &metadata, const SaveTypes types = SaveType::Tags, const SaveCoverOptions &save_cover_options = SaveCoverOptions()); bool IsMediaFileBlocking(const QString &filename); QByteArray LoadEmbeddedArtBlocking(const QString &filename); QImage LoadEmbeddedArtAsImageBlocking(const QString &filename); bool SaveEmbeddedArtBlocking(const QString &filename, const SaveCoverOptions &save_cover_options); bool UpdateSongPlaycountBlocking(const Song &metadata); bool UpdateSongRatingBlocking(const Song &metadata); // TODO: Make this not a singleton static TagReaderClient *Instance() { return sInstance; } signals: void ExitFinished(); private slots: void Exit(); void WorkerFailedToStart(); public slots: void UpdateSongsPlaycount(const SongList &songs); void UpdateSongsRating(const SongList &songs); private: static TagReaderClient *sInstance; WorkerPool *worker_pool_; QList message_queue_; QThread *original_thread_; }; using TagReaderReply = TagReaderClient::ReplyType; #endif // TAGREADERCLIENT_H