2014-05-30 08:53:19 +00:00
|
|
|
/*
|
|
|
|
This file is part of Telegram Desktop,
|
2018-01-03 10:23:14 +00:00
|
|
|
the official desktop application for the Telegram messaging service.
|
2014-05-30 08:53:19 +00:00
|
|
|
|
2018-01-03 10:23:14 +00:00
|
|
|
For license and copyright information please follow this link:
|
|
|
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
2014-05-30 08:53:19 +00:00
|
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
|
2017-04-06 14:38:10 +00:00
|
|
|
#include "base/variant.h"
|
2017-03-10 14:14:10 +00:00
|
|
|
|
2016-11-28 15:45:07 +00:00
|
|
|
enum class CompressConfirm {
|
|
|
|
Auto,
|
|
|
|
Yes,
|
|
|
|
No,
|
|
|
|
None,
|
2014-05-30 08:53:19 +00:00
|
|
|
};
|
|
|
|
|
2016-11-28 15:45:07 +00:00
|
|
|
enum class SendMediaType {
|
|
|
|
Photo,
|
|
|
|
Audio,
|
|
|
|
File,
|
2018-03-25 11:37:57 +00:00
|
|
|
Secure,
|
2016-11-28 15:45:07 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct SendMediaPrepare {
|
|
|
|
SendMediaPrepare(const QString &file, const PeerId &peer, SendMediaType type, MsgId replyTo) : id(rand_value<PhotoId>()), file(file), peer(peer), type(type), replyTo(replyTo) {
|
2014-05-30 08:53:19 +00:00
|
|
|
}
|
2016-11-28 15:45:07 +00:00
|
|
|
SendMediaPrepare(const QImage &img, const PeerId &peer, SendMediaType type, MsgId replyTo) : id(rand_value<PhotoId>()), img(img), peer(peer), type(type), replyTo(replyTo) {
|
2014-05-30 08:53:19 +00:00
|
|
|
}
|
2016-11-28 15:45:07 +00:00
|
|
|
SendMediaPrepare(const QByteArray &data, const PeerId &peer, SendMediaType type, MsgId replyTo) : id(rand_value<PhotoId>()), data(data), peer(peer), type(type), replyTo(replyTo) {
|
2015-05-29 18:52:43 +00:00
|
|
|
}
|
2016-11-28 15:45:07 +00:00
|
|
|
SendMediaPrepare(const QByteArray &data, int duration, const PeerId &peer, SendMediaType type, MsgId replyTo) : id(rand_value<PhotoId>()), data(data), peer(peer), type(type), duration(duration), replyTo(replyTo) {
|
2014-05-30 08:53:19 +00:00
|
|
|
}
|
|
|
|
PhotoId id;
|
|
|
|
QString file;
|
|
|
|
QImage img;
|
|
|
|
QByteArray data;
|
|
|
|
PeerId peer;
|
2016-11-28 15:45:07 +00:00
|
|
|
SendMediaType type;
|
|
|
|
int duration = 0;
|
2015-03-19 09:18:19 +00:00
|
|
|
MsgId replyTo;
|
2014-05-30 08:53:19 +00:00
|
|
|
|
2016-11-28 15:45:07 +00:00
|
|
|
};
|
|
|
|
using SendMediaPrepareList = QList<SendMediaPrepare>;
|
|
|
|
|
|
|
|
using UploadFileParts = QMap<int, QByteArray>;
|
|
|
|
struct SendMediaReady {
|
|
|
|
SendMediaReady() = default; // temp
|
2019-01-18 11:26:43 +00:00
|
|
|
SendMediaReady(
|
|
|
|
SendMediaType type,
|
|
|
|
const QString &file,
|
|
|
|
const QString &filename,
|
|
|
|
int32 filesize,
|
|
|
|
const QByteArray &data,
|
|
|
|
const uint64 &id,
|
|
|
|
const uint64 &thumbId,
|
|
|
|
const QString &thumbExt,
|
|
|
|
const PeerId &peer,
|
|
|
|
const MTPPhoto &photo,
|
|
|
|
const PreparedPhotoThumbs &photoThumbs,
|
|
|
|
const MTPDocument &document,
|
|
|
|
const QByteArray &jpeg,
|
|
|
|
MsgId replyTo);
|
|
|
|
|
2015-03-19 09:18:19 +00:00
|
|
|
MsgId replyTo;
|
2016-11-28 15:45:07 +00:00
|
|
|
SendMediaType type;
|
2014-05-30 08:53:19 +00:00
|
|
|
QString file, filename;
|
|
|
|
int32 filesize;
|
|
|
|
QByteArray data;
|
2015-01-05 20:17:33 +00:00
|
|
|
QString thumbExt;
|
|
|
|
uint64 id, thumbId; // id always file-id of media, thumbId is file-id of thumb ( == id for photos)
|
2014-05-30 08:53:19 +00:00
|
|
|
PeerId peer;
|
|
|
|
|
|
|
|
MTPPhoto photo;
|
|
|
|
MTPDocument document;
|
|
|
|
PreparedPhotoThumbs photoThumbs;
|
2015-10-27 02:39:02 +00:00
|
|
|
UploadFileParts parts;
|
2014-05-30 08:53:19 +00:00
|
|
|
QByteArray jpeg_md5;
|
2014-10-17 19:14:42 +00:00
|
|
|
|
2015-10-11 08:37:24 +00:00
|
|
|
QString caption;
|
2015-10-27 02:39:02 +00:00
|
|
|
|
2014-05-30 08:53:19 +00:00
|
|
|
};
|
2015-09-29 18:44:31 +00:00
|
|
|
|
2018-09-06 11:13:54 +00:00
|
|
|
SendMediaReady PreparePeerPhoto(PeerId peerId, QImage &&image);
|
|
|
|
|
2017-03-04 19:36:59 +00:00
|
|
|
using TaskId = void*; // no interface, just id
|
|
|
|
|
2015-09-29 18:44:31 +00:00
|
|
|
class Task {
|
|
|
|
public:
|
|
|
|
virtual void process() = 0; // is executed in a separate thread
|
|
|
|
virtual void finish() = 0; // is executed in the same as TaskQueue thread
|
2016-11-28 15:45:07 +00:00
|
|
|
virtual ~Task() = default;
|
2015-09-29 18:59:26 +00:00
|
|
|
|
2015-09-29 18:44:31 +00:00
|
|
|
TaskId id() const {
|
2016-11-28 15:45:07 +00:00
|
|
|
return static_cast<TaskId>(const_cast<Task*>(this));
|
2015-09-29 18:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
2015-09-29 18:59:26 +00:00
|
|
|
class TaskQueueWorker;
|
2015-09-29 18:44:31 +00:00
|
|
|
class TaskQueue : public QObject {
|
|
|
|
Q_OBJECT
|
|
|
|
|
|
|
|
public:
|
2017-12-19 16:57:42 +00:00
|
|
|
explicit TaskQueue(TimeMs stopTimeoutMs = 0); // <= 0 - never stop worker
|
2015-09-29 18:44:31 +00:00
|
|
|
|
2017-12-18 10:38:14 +00:00
|
|
|
TaskId addTask(std::unique_ptr<Task> &&task);
|
|
|
|
void addTasks(std::vector<std::unique_ptr<Task>> &&tasks);
|
2015-09-29 18:44:31 +00:00
|
|
|
void cancelTask(TaskId id); // this task finish() won't be called
|
2016-02-12 16:35:06 +00:00
|
|
|
|
2015-09-29 18:44:31 +00:00
|
|
|
~TaskQueue();
|
|
|
|
|
|
|
|
signals:
|
|
|
|
void taskAdded();
|
|
|
|
|
|
|
|
public slots:
|
|
|
|
void onTaskProcessed();
|
|
|
|
void stop();
|
|
|
|
|
|
|
|
private:
|
|
|
|
friend class TaskQueueWorker;
|
|
|
|
|
2015-10-28 02:41:13 +00:00
|
|
|
void wakeThread();
|
|
|
|
|
2017-12-18 10:38:14 +00:00
|
|
|
std::deque<std::unique_ptr<Task>> _tasksToProcess;
|
|
|
|
std::deque<std::unique_ptr<Task>> _tasksToFinish;
|
|
|
|
TaskId _taskInProcessId = TaskId();
|
2015-09-29 18:44:31 +00:00
|
|
|
QMutex _tasksToProcessMutex, _tasksToFinishMutex;
|
2017-12-19 16:57:42 +00:00
|
|
|
QThread *_thread = nullptr;
|
|
|
|
TaskQueueWorker *_worker = nullptr;
|
|
|
|
QTimer *_stopTimer = nullptr;
|
2015-09-29 18:44:31 +00:00
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
class TaskQueueWorker : public QObject {
|
|
|
|
Q_OBJECT
|
|
|
|
|
|
|
|
public:
|
2016-11-28 15:45:07 +00:00
|
|
|
TaskQueueWorker(TaskQueue *queue) : _queue(queue) {
|
2015-09-29 18:44:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
signals:
|
|
|
|
void taskProcessed();
|
|
|
|
|
|
|
|
public slots:
|
|
|
|
void onTaskAdded();
|
|
|
|
|
|
|
|
private:
|
|
|
|
TaskQueue *_queue;
|
2016-11-28 15:45:07 +00:00
|
|
|
bool _inTaskAdded = false;
|
2015-09-29 18:44:31 +00:00
|
|
|
|
|
|
|
};
|
2015-10-27 02:39:02 +00:00
|
|
|
|
2017-12-19 16:57:42 +00:00
|
|
|
struct SendingAlbum {
|
|
|
|
struct Item {
|
|
|
|
explicit Item(TaskId taskId) : taskId(taskId) {
|
|
|
|
}
|
2018-12-26 07:22:36 +00:00
|
|
|
|
2017-12-19 16:57:42 +00:00
|
|
|
TaskId taskId;
|
|
|
|
FullMsgId msgId;
|
2018-09-21 16:28:46 +00:00
|
|
|
std::optional<MTPInputSingleMedia> media;
|
2017-12-19 16:57:42 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
SendingAlbum();
|
|
|
|
|
2018-12-26 07:22:36 +00:00
|
|
|
void fillMedia(
|
|
|
|
not_null<HistoryItem*> item,
|
|
|
|
const MTPInputMedia &media,
|
|
|
|
uint64 randomId);
|
|
|
|
void refreshMediaCaption(not_null<HistoryItem*> item);
|
|
|
|
void removeItem(not_null<HistoryItem*> item);
|
|
|
|
|
2017-12-25 14:17:00 +00:00
|
|
|
uint64 groupId = 0;
|
2017-12-19 16:57:42 +00:00
|
|
|
std::vector<Item> items;
|
2017-12-25 14:17:00 +00:00
|
|
|
bool silent = false;
|
2017-12-19 16:57:42 +00:00
|
|
|
|
|
|
|
};
|
|
|
|
|
2015-10-27 02:39:02 +00:00
|
|
|
struct FileLoadTo {
|
2016-05-27 16:47:46 +00:00
|
|
|
FileLoadTo(const PeerId &peer, bool silent, MsgId replyTo)
|
2016-02-25 16:19:54 +00:00
|
|
|
: peer(peer)
|
|
|
|
, silent(silent)
|
|
|
|
, replyTo(replyTo) {
|
2015-10-27 02:39:02 +00:00
|
|
|
}
|
|
|
|
PeerId peer;
|
2016-05-27 16:47:46 +00:00
|
|
|
bool silent;
|
2015-10-27 02:39:02 +00:00
|
|
|
MsgId replyTo;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct FileLoadResult {
|
2017-12-19 16:57:42 +00:00
|
|
|
FileLoadResult(
|
|
|
|
TaskId taskId,
|
|
|
|
uint64 id,
|
|
|
|
const FileLoadTo &to,
|
2018-05-24 13:03:21 +00:00
|
|
|
const TextWithTags &caption,
|
2017-12-19 16:57:42 +00:00
|
|
|
std::shared_ptr<SendingAlbum> album);
|
|
|
|
|
|
|
|
TaskId taskId;
|
2015-10-27 02:39:02 +00:00
|
|
|
uint64 id;
|
|
|
|
FileLoadTo to;
|
2017-12-19 16:57:42 +00:00
|
|
|
std::shared_ptr<SendingAlbum> album;
|
2017-09-17 11:53:38 +00:00
|
|
|
SendMediaType type = SendMediaType::File;
|
2015-10-27 02:39:02 +00:00
|
|
|
QString filepath;
|
|
|
|
QByteArray content;
|
|
|
|
|
|
|
|
QString filename;
|
2015-12-19 14:37:28 +00:00
|
|
|
QString filemime;
|
2016-11-28 15:45:07 +00:00
|
|
|
int32 filesize = 0;
|
2015-10-27 02:39:02 +00:00
|
|
|
UploadFileParts fileparts;
|
|
|
|
QByteArray filemd5;
|
2015-12-24 19:26:28 +00:00
|
|
|
int32 partssize;
|
2015-10-27 02:39:02 +00:00
|
|
|
|
2016-11-28 15:45:07 +00:00
|
|
|
uint64 thumbId = 0; // id is always file-id of media, thumbId is file-id of thumb ( == id for photos)
|
2015-10-27 02:39:02 +00:00
|
|
|
QString thumbname;
|
|
|
|
UploadFileParts thumbparts;
|
|
|
|
QByteArray thumbmd5;
|
2018-10-12 16:41:51 +00:00
|
|
|
QImage thumb;
|
2015-10-27 02:39:02 +00:00
|
|
|
|
2018-10-25 08:47:06 +00:00
|
|
|
QImage goodThumbnail;
|
|
|
|
QByteArray goodThumbnailBytes;
|
|
|
|
|
2015-10-27 02:39:02 +00:00
|
|
|
MTPPhoto photo;
|
|
|
|
MTPDocument document;
|
|
|
|
|
|
|
|
PreparedPhotoThumbs photoThumbs;
|
2018-05-24 13:03:21 +00:00
|
|
|
TextWithTags caption;
|
2015-10-27 02:39:02 +00:00
|
|
|
|
2019-01-18 11:26:43 +00:00
|
|
|
void setFileData(const QByteArray &filedata);
|
|
|
|
void setThumbData(const QByteArray &thumbdata);
|
|
|
|
|
2015-10-27 02:39:02 +00:00
|
|
|
};
|
|
|
|
|
2017-12-19 16:57:42 +00:00
|
|
|
struct FileMediaInformation {
|
2017-03-10 14:14:10 +00:00
|
|
|
struct Image {
|
|
|
|
QImage data;
|
|
|
|
bool animated = false;
|
|
|
|
};
|
|
|
|
struct Song {
|
2017-03-27 12:50:40 +00:00
|
|
|
int duration = -1;
|
2017-03-10 14:14:10 +00:00
|
|
|
QString title;
|
|
|
|
QString performer;
|
|
|
|
QImage cover;
|
|
|
|
};
|
|
|
|
struct Video {
|
|
|
|
bool isGifv = false;
|
2018-11-07 14:07:20 +00:00
|
|
|
bool supportsStreaming = false;
|
2017-03-27 12:50:40 +00:00
|
|
|
int duration = -1;
|
2017-03-10 14:14:10 +00:00
|
|
|
QImage thumbnail;
|
|
|
|
};
|
|
|
|
|
2017-12-19 16:57:42 +00:00
|
|
|
QString filemime;
|
|
|
|
base::variant<Image, Song, Video> media;
|
|
|
|
};
|
|
|
|
|
|
|
|
class FileLoadTask final : public Task {
|
|
|
|
public:
|
|
|
|
static std::unique_ptr<FileMediaInformation> ReadMediaInformation(
|
|
|
|
const QString &filepath,
|
|
|
|
const QByteArray &content,
|
|
|
|
const QString &filemime);
|
2017-12-21 16:15:37 +00:00
|
|
|
static bool FillImageInformation(
|
|
|
|
QImage &&image,
|
|
|
|
bool animated,
|
|
|
|
std::unique_ptr<FileMediaInformation> &result);
|
2017-12-19 16:57:42 +00:00
|
|
|
|
|
|
|
FileLoadTask(
|
|
|
|
const QString &filepath,
|
|
|
|
const QByteArray &content,
|
2017-12-21 16:15:37 +00:00
|
|
|
std::unique_ptr<FileMediaInformation> information,
|
2017-12-19 16:57:42 +00:00
|
|
|
SendMediaType type,
|
|
|
|
const FileLoadTo &to,
|
2018-05-24 13:03:21 +00:00
|
|
|
const TextWithTags &caption,
|
2017-12-19 16:57:42 +00:00
|
|
|
std::shared_ptr<SendingAlbum> album = nullptr);
|
|
|
|
FileLoadTask(
|
|
|
|
const QByteArray &voice,
|
|
|
|
int32 duration,
|
|
|
|
const VoiceWaveform &waveform,
|
|
|
|
const FileLoadTo &to,
|
2018-05-24 13:03:21 +00:00
|
|
|
const TextWithTags &caption);
|
2015-10-27 02:39:02 +00:00
|
|
|
|
|
|
|
uint64 fileid() const {
|
|
|
|
return _id;
|
|
|
|
}
|
|
|
|
|
|
|
|
void process();
|
|
|
|
void finish();
|
|
|
|
|
2017-03-10 14:14:10 +00:00
|
|
|
private:
|
2017-12-21 16:15:37 +00:00
|
|
|
static bool CheckForSong(
|
|
|
|
const QString &filepath,
|
|
|
|
const QByteArray &content,
|
|
|
|
std::unique_ptr<FileMediaInformation> &result);
|
|
|
|
static bool CheckForVideo(
|
|
|
|
const QString &filepath,
|
|
|
|
const QByteArray &content,
|
|
|
|
std::unique_ptr<FileMediaInformation> &result);
|
|
|
|
static bool CheckForImage(
|
|
|
|
const QString &filepath,
|
|
|
|
const QByteArray &content,
|
|
|
|
std::unique_ptr<FileMediaInformation> &result);
|
2017-03-10 14:14:10 +00:00
|
|
|
|
|
|
|
template <typename Mimes, typename Extensions>
|
|
|
|
static bool CheckMimeOrExtensions(const QString &filepath, const QString &filemime, Mimes &mimes, Extensions &extensions);
|
|
|
|
|
2017-12-19 16:57:42 +00:00
|
|
|
std::unique_ptr<FileMediaInformation> readMediaInformation(const QString &filemime) const {
|
2017-03-10 14:14:10 +00:00
|
|
|
return ReadMediaInformation(_filepath, _content, filemime);
|
|
|
|
}
|
2017-12-19 16:57:42 +00:00
|
|
|
void removeFromAlbum();
|
2017-03-10 14:14:10 +00:00
|
|
|
|
2015-10-27 02:39:02 +00:00
|
|
|
uint64 _id;
|
|
|
|
FileLoadTo _to;
|
2017-12-19 16:57:42 +00:00
|
|
|
const std::shared_ptr<SendingAlbum> _album;
|
2015-10-27 02:39:02 +00:00
|
|
|
QString _filepath;
|
|
|
|
QByteArray _content;
|
2017-12-19 16:57:42 +00:00
|
|
|
std::unique_ptr<FileMediaInformation> _information;
|
2016-10-12 19:34:25 +00:00
|
|
|
int32 _duration = 0;
|
2016-02-12 16:35:06 +00:00
|
|
|
VoiceWaveform _waveform;
|
2016-11-28 15:45:07 +00:00
|
|
|
SendMediaType _type;
|
2018-05-24 13:03:21 +00:00
|
|
|
TextWithTags _caption;
|
2015-10-27 02:39:02 +00:00
|
|
|
|
2017-12-19 16:57:42 +00:00
|
|
|
std::shared_ptr<FileLoadResult> _result;
|
2015-10-27 02:39:02 +00:00
|
|
|
|
|
|
|
};
|