strawberry-audio-player-win.../ext/libstrawberry-common/core/messagereply.h

100 lines
2.9 KiB
C
Raw Normal View History

2018-02-27 18:06:05 +01:00
/* This file is part of Strawberry.
Copyright 2011, David Sansome <me@davidsansome.com>
2021-04-10 07:32:38 +02:00
Copyright 2018-2021, Jonas Kvinge <jonas@jkvinge.net>
2018-02-27 18:06:05 +01:00
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 <http://www.gnu.org/licenses/>.
*/
#ifndef MESSAGEREPLY_H
#define MESSAGEREPLY_H
#include <QtGlobal>
2018-02-27 18:06:05 +01:00
#include <QObject>
#include <QSemaphore>
#include <QString>
#include <QTimer>
2018-02-27 18:06:05 +01:00
#include "core/logging.h"
// Base QObject for a reply future class that is returned immediately for requests that will occur in the background.
// Similar to QNetworkReply. Use MessageReply instead.
2018-02-27 18:06:05 +01:00
class _MessageReplyBase : public QObject {
Q_OBJECT
public:
2020-06-26 22:41:38 +02:00
explicit _MessageReplyBase(QObject *parent = nullptr);
2018-02-27 18:06:05 +01:00
virtual int id() const = 0;
2018-02-27 18:06:05 +01:00
bool is_finished() const { return finished_; }
bool is_successful() const { return success_; }
// Waits for the reply to finish by waiting on a semaphore. Never call this from the MessageHandler's thread or it will block forever.
2018-02-27 18:06:05 +01:00
// Returns true if the call was successful.
bool WaitForFinished();
void Abort();
2019-02-20 21:29:14 +01:00
signals:
void Finished();
2018-02-27 18:06:05 +01:00
2019-02-20 21:29:14 +01:00
protected:
2018-02-27 18:06:05 +01:00
bool finished_;
bool success_;
QSemaphore semaphore_;
};
2019-02-20 21:29:14 +01:00
// A reply future class that is returned immediately for requests that will occur in the background. Similar to QNetworkReply.
2022-03-22 21:09:05 +01:00
template<typename MessageType>
2018-02-27 18:06:05 +01:00
class MessageReply : public _MessageReplyBase {
public:
2020-11-19 18:22:57 +01:00
explicit MessageReply(const MessageType &request_message, QObject *parent = nullptr);
2018-02-27 18:06:05 +01:00
int id() const override { return request_message_.id(); }
2020-11-19 18:22:57 +01:00
const MessageType &request_message() const { return request_message_; }
const MessageType &message() const { return reply_message_; }
2018-02-27 18:06:05 +01:00
2020-11-19 18:22:57 +01:00
void SetReply(const MessageType &message);
2018-02-27 18:06:05 +01:00
2020-11-19 18:22:57 +01:00
private:
2018-02-27 18:06:05 +01:00
MessageType request_message_;
MessageType reply_message_;
};
template<typename MessageType>
2020-11-19 18:22:57 +01:00
MessageReply<MessageType>::MessageReply(const MessageType &request_message, QObject *parent) : _MessageReplyBase(parent) {
2018-02-27 18:06:05 +01:00
request_message_.MergeFrom(request_message);
}
template<typename MessageType>
2020-11-19 18:22:57 +01:00
void MessageReply<MessageType>::SetReply(const MessageType &message) {
2019-07-10 22:40:30 +02:00
2018-02-27 18:06:05 +01:00
Q_ASSERT(!finished_);
reply_message_.MergeFrom(message);
finished_ = true;
success_ = true;
qLog(Debug) << "Releasing ID" << id() << "(finished)";
2019-07-10 22:40:30 +02:00
// Delay the signal as workaround to fix the signal periodically not emitted.
QTimer::singleShot(1, this, &_MessageReplyBase::Finished);
2019-07-10 22:40:30 +02:00
2021-02-16 18:53:51 +01:00
semaphore_.release();
2018-02-27 18:06:05 +01:00
}
#endif // MESSAGEREPLY_H