2014-05-30 08:53:19 +00:00
|
|
|
/*
|
|
|
|
This file is part of Telegram Desktop,
|
2014-12-01 10:47:38 +00:00
|
|
|
the official desktop version of Telegram messaging app, see https://telegram.org
|
2014-05-30 08:53:19 +00:00
|
|
|
|
|
|
|
Telegram Desktop 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.
|
|
|
|
|
|
|
|
It 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.
|
|
|
|
|
2015-10-03 13:16:42 +00:00
|
|
|
In addition, as a special exception, the copyright holders give permission
|
|
|
|
to link the code of portions of this program with the OpenSSL library.
|
|
|
|
|
2014-05-30 08:53:19 +00:00
|
|
|
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
|
2015-10-03 13:16:42 +00:00
|
|
|
Copyright (c) 2014-2015 John Preston, https://desktop.telegram.org
|
2014-05-30 08:53:19 +00:00
|
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
|
2015-10-27 02:39:02 +00:00
|
|
|
enum PrepareMediaType {
|
|
|
|
PrepareAuto,
|
|
|
|
PreparePhoto,
|
|
|
|
PrepareAudio,
|
|
|
|
PrepareVideo,
|
|
|
|
PrepareDocument,
|
2014-05-30 08:53:19 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ToPrepareMedia {
|
2015-10-27 02:39:02 +00:00
|
|
|
ToPrepareMedia(const QString &file, const PeerId &peer, PrepareMediaType t, bool broadcast, bool ctrlShiftEnter, MsgId replyTo) : id(MTP::nonce<PhotoId>()), file(file), peer(peer), type(t), duration(0), ctrlShiftEnter(ctrlShiftEnter), replyTo(replyTo) {
|
2014-05-30 08:53:19 +00:00
|
|
|
}
|
2015-10-27 02:39:02 +00:00
|
|
|
ToPrepareMedia(const QImage &img, const PeerId &peer, PrepareMediaType t, bool broadcast, bool ctrlShiftEnter, MsgId replyTo) : id(MTP::nonce<PhotoId>()), img(img), peer(peer), type(t), duration(0), ctrlShiftEnter(ctrlShiftEnter), replyTo(replyTo) {
|
2014-05-30 08:53:19 +00:00
|
|
|
}
|
2015-10-27 02:39:02 +00:00
|
|
|
ToPrepareMedia(const QByteArray &data, const PeerId &peer, PrepareMediaType t, bool broadcast, bool ctrlShiftEnter, MsgId replyTo) : id(MTP::nonce<PhotoId>()), data(data), peer(peer), type(t), duration(0), ctrlShiftEnter(ctrlShiftEnter), replyTo(replyTo) {
|
2015-05-29 18:52:43 +00:00
|
|
|
}
|
2015-10-27 02:39:02 +00:00
|
|
|
ToPrepareMedia(const QByteArray &data, int32 duration, const PeerId &peer, PrepareMediaType t, bool broadcast, bool ctrlShiftEnter, MsgId replyTo) : id(MTP::nonce<PhotoId>()), data(data), peer(peer), type(t), duration(duration), ctrlShiftEnter(ctrlShiftEnter), replyTo(replyTo) {
|
2014-05-30 08:53:19 +00:00
|
|
|
}
|
|
|
|
PhotoId id;
|
|
|
|
QString file;
|
|
|
|
QImage img;
|
|
|
|
QByteArray data;
|
|
|
|
PeerId peer;
|
2015-10-27 02:39:02 +00:00
|
|
|
PrepareMediaType type;
|
2015-05-29 18:52:43 +00:00
|
|
|
int32 duration;
|
2015-09-21 20:57:42 +00:00
|
|
|
bool broadcast;
|
2014-10-17 19:14:42 +00:00
|
|
|
bool ctrlShiftEnter;
|
2015-03-19 09:18:19 +00:00
|
|
|
MsgId replyTo;
|
2014-05-30 08:53:19 +00:00
|
|
|
};
|
|
|
|
typedef QList<ToPrepareMedia> ToPrepareMedias;
|
|
|
|
|
2015-10-27 02:39:02 +00:00
|
|
|
typedef QMap<int32, QByteArray> UploadFileParts;
|
2014-05-30 08:53:19 +00:00
|
|
|
struct ReadyLocalMedia {
|
2015-10-27 02:39:02 +00:00
|
|
|
ReadyLocalMedia(PrepareMediaType 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 MTPAudio &audio, const PreparedPhotoThumbs &photoThumbs, const MTPDocument &document, const QByteArray &jpeg, bool broadcast, bool ctrlShiftEnter, MsgId replyTo) :
|
2015-09-21 20:57:42 +00:00
|
|
|
replyTo(replyTo), type(type), file(file), filename(filename), filesize(filesize), data(data), thumbExt(thumbExt), id(id), thumbId(thumbId), peer(peer), photo(photo), document(document), audio(audio), photoThumbs(photoThumbs), broadcast(broadcast), ctrlShiftEnter(ctrlShiftEnter) {
|
2014-05-30 08:53:19 +00:00
|
|
|
if (!jpeg.isEmpty()) {
|
|
|
|
int32 size = jpeg.size();
|
|
|
|
for (int32 i = 0, part = 0; i < size; i += UploadPartSize, ++part) {
|
|
|
|
parts.insert(part, jpeg.mid(i, UploadPartSize));
|
|
|
|
}
|
|
|
|
jpeg_md5.resize(32);
|
|
|
|
hashMd5Hex(jpeg.constData(), jpeg.size(), jpeg_md5.data());
|
|
|
|
}
|
|
|
|
}
|
2015-03-19 09:18:19 +00:00
|
|
|
MsgId replyTo;
|
2015-10-27 02:39:02 +00:00
|
|
|
PrepareMediaType 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;
|
2015-05-29 18:52:43 +00:00
|
|
|
MTPAudio audio;
|
2014-05-30 08:53:19 +00:00
|
|
|
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-09-21 20:57:42 +00:00
|
|
|
bool broadcast;
|
2014-10-17 19:14:42 +00:00
|
|
|
bool ctrlShiftEnter;
|
2015-10-11 08:37:24 +00:00
|
|
|
QString caption;
|
2015-10-27 02:39:02 +00:00
|
|
|
|
|
|
|
ReadyLocalMedia() : type(PrepareAuto) { // temp
|
|
|
|
}
|
2014-05-30 08:53:19 +00:00
|
|
|
};
|
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
|
2015-09-29 18:59:26 +00:00
|
|
|
virtual ~Task() {
|
|
|
|
}
|
|
|
|
|
2015-09-29 18:44:31 +00:00
|
|
|
TaskId id() const {
|
|
|
|
return TaskId(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
typedef QSharedPointer<Task> TaskPtr;
|
2015-10-28 02:41:13 +00:00
|
|
|
typedef QList<TaskPtr> TasksList;
|
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:
|
|
|
|
|
|
|
|
TaskQueue(QObject *parent, int32 stopTimeoutMs = 0); // <= 0 - never stop worker
|
|
|
|
|
|
|
|
TaskId addTask(TaskPtr task);
|
2015-10-28 02:41:13 +00:00
|
|
|
void addTasks(const TasksList &tasks);
|
2015-09-29 18:44:31 +00:00
|
|
|
void cancelTask(TaskId id); // this task finish() won't be called
|
|
|
|
|
2015-10-27 02:39:02 +00:00
|
|
|
TaskId addTask(Task *task) {
|
2015-09-29 18:44:31 +00:00
|
|
|
return addTask(TaskPtr(task));
|
|
|
|
}
|
|
|
|
|
|
|
|
~TaskQueue();
|
|
|
|
|
|
|
|
signals:
|
|
|
|
|
|
|
|
void taskAdded();
|
|
|
|
|
|
|
|
public slots:
|
|
|
|
|
|
|
|
void onTaskProcessed();
|
|
|
|
void stop();
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
friend class TaskQueueWorker;
|
|
|
|
|
2015-10-28 02:41:13 +00:00
|
|
|
void wakeThread();
|
|
|
|
|
|
|
|
TasksList _tasksToProcess, _tasksToFinish;
|
2015-09-29 18:44:31 +00:00
|
|
|
QMutex _tasksToProcessMutex, _tasksToFinishMutex;
|
|
|
|
QThread *_thread;
|
|
|
|
TaskQueueWorker *_worker;
|
|
|
|
QTimer *_stopTimer;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
class TaskQueueWorker : public QObject {
|
|
|
|
Q_OBJECT
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
TaskQueueWorker(TaskQueue *queue) : _queue(queue), _inTaskAdded(false) {
|
|
|
|
}
|
|
|
|
|
|
|
|
signals:
|
|
|
|
|
|
|
|
void taskProcessed();
|
|
|
|
|
|
|
|
public slots:
|
|
|
|
|
|
|
|
void onTaskAdded();
|
|
|
|
|
|
|
|
private:
|
|
|
|
TaskQueue *_queue;
|
|
|
|
bool _inTaskAdded;
|
|
|
|
|
|
|
|
};
|
2015-10-27 02:39:02 +00:00
|
|
|
|
|
|
|
struct FileLoadTo {
|
|
|
|
FileLoadTo(const PeerId &peer, bool broadcast, MsgId replyTo) : peer(peer), broadcast(broadcast), replyTo(replyTo) {
|
|
|
|
}
|
|
|
|
PeerId peer;
|
|
|
|
bool broadcast;
|
|
|
|
MsgId replyTo;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct FileLoadResult {
|
2015-10-28 04:12:01 +00:00
|
|
|
FileLoadResult(const uint64 &id, const FileLoadTo &to) : id(id), to(to), type(PrepareAuto), filesize(0), thumbId(0) {
|
2015-10-27 02:39:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
uint64 id;
|
|
|
|
FileLoadTo to;
|
|
|
|
PrepareMediaType type;
|
|
|
|
QString filepath;
|
|
|
|
QByteArray content;
|
|
|
|
|
|
|
|
QString filename;
|
|
|
|
int32 filesize;
|
|
|
|
UploadFileParts fileparts;
|
|
|
|
QByteArray filemd5;
|
|
|
|
|
|
|
|
uint64 thumbId; // id is always file-id of media, thumbId is file-id of thumb ( == id for photos)
|
|
|
|
QString thumbname;
|
|
|
|
UploadFileParts thumbparts;
|
|
|
|
QByteArray thumbmd5;
|
|
|
|
QPixmap thumb;
|
|
|
|
|
|
|
|
MTPPhoto photo;
|
|
|
|
MTPAudio audio;
|
|
|
|
MTPDocument document;
|
|
|
|
|
|
|
|
PreparedPhotoThumbs photoThumbs;
|
|
|
|
QString photoCaption;
|
|
|
|
|
|
|
|
QString originalText; // when pasted had an image mime save text mime here to insert if image send was cancelled
|
|
|
|
|
|
|
|
void setFileData(const QByteArray &filedata) {
|
|
|
|
if (!filedata.isEmpty()) {
|
|
|
|
int32 size = filedata.size();
|
|
|
|
for (int32 i = 0, part = 0; i < size; i += UploadPartSize, ++part) {
|
|
|
|
fileparts.insert(part, filedata.mid(i, UploadPartSize));
|
|
|
|
}
|
|
|
|
filemd5.resize(32);
|
|
|
|
hashMd5Hex(filedata.constData(), filedata.size(), filemd5.data());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void setThumbData(const QByteArray &thumbdata) {
|
|
|
|
if (!thumbdata.isEmpty()) {
|
|
|
|
int32 size = thumbdata.size();
|
|
|
|
for (int32 i = 0, part = 0; i < size; i += UploadPartSize, ++part) {
|
|
|
|
thumbparts.insert(part, thumbdata.mid(i, UploadPartSize));
|
|
|
|
}
|
|
|
|
thumbmd5.resize(32);
|
|
|
|
hashMd5Hex(thumbdata.constData(), thumbdata.size(), thumbmd5.data());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
typedef QSharedPointer<FileLoadResult> FileLoadResultPtr;
|
|
|
|
|
|
|
|
enum FileLoadForceConfirmType {
|
|
|
|
FileLoadNoForceConfirm,
|
|
|
|
FileLoadNeverConfirm,
|
|
|
|
FileLoadAlwaysConfirm,
|
|
|
|
};
|
|
|
|
|
|
|
|
class FileLoadTask : public Task {
|
|
|
|
public:
|
|
|
|
|
|
|
|
FileLoadTask(const QString &filepath, PrepareMediaType type, const FileLoadTo &to, FileLoadForceConfirmType confirm = FileLoadNoForceConfirm);
|
|
|
|
FileLoadTask(const QByteArray &content, PrepareMediaType type, const FileLoadTo &to);
|
2015-10-28 02:41:13 +00:00
|
|
|
FileLoadTask(const QImage &image, PrepareMediaType type, const FileLoadTo &to, FileLoadForceConfirmType confirm = FileLoadNoForceConfirm, const QString &originalText = QString());
|
2015-10-27 02:39:02 +00:00
|
|
|
FileLoadTask(const QByteArray &audio, int32 duration, const FileLoadTo &to);
|
|
|
|
|
|
|
|
uint64 fileid() const {
|
|
|
|
return _id;
|
|
|
|
}
|
|
|
|
|
|
|
|
void process();
|
|
|
|
void finish();
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
|
|
|
uint64 _id;
|
|
|
|
FileLoadTo _to;
|
|
|
|
QString _filepath;
|
|
|
|
QImage _image;
|
|
|
|
QByteArray _content;
|
|
|
|
int32 _duration;
|
|
|
|
PrepareMediaType _type;
|
|
|
|
FileLoadForceConfirmType _confirm;
|
|
|
|
QString _originalText;
|
|
|
|
|
|
|
|
FileLoadResultPtr _result;
|
|
|
|
|
|
|
|
};
|