2019-11-09 03:06:10 +01:00
|
|
|
#include "TimelineViewManager.h"
|
|
|
|
|
|
|
|
#include <QMetaType>
|
|
|
|
#include <QPalette>
|
|
|
|
#include <QQmlContext>
|
|
|
|
|
|
|
|
#include "ChatPage.h"
|
|
|
|
#include "ColorImageProvider.h"
|
|
|
|
#include "DelegateChooser.h"
|
2018-07-17 15:37:25 +02:00
|
|
|
#include "Logging.h"
|
2020-01-17 01:25:14 +01:00
|
|
|
#include "MatrixClient.h"
|
2019-11-09 03:06:10 +01:00
|
|
|
#include "MxcImageProvider.h"
|
|
|
|
#include "UserSettingsPage.h"
|
|
|
|
#include "dialogs/ImageOverlay.h"
|
2017-04-06 01:06:42 +02:00
|
|
|
|
2020-01-11 18:53:32 +01:00
|
|
|
Q_DECLARE_METATYPE(mtx::events::collections::TimelineEvents)
|
|
|
|
|
2018-07-17 22:50:18 +02:00
|
|
|
void
|
2019-11-09 03:06:10 +01:00
|
|
|
TimelineViewManager::updateColorPalette()
|
2018-07-17 22:50:18 +02:00
|
|
|
{
|
2020-01-27 15:59:25 +01:00
|
|
|
if (settings->theme() == "light") {
|
2019-11-09 03:06:10 +01:00
|
|
|
QPalette lightActive(/*windowText*/ QColor("#333"),
|
|
|
|
/*button*/ QColor("#333"),
|
|
|
|
/*light*/ QColor(),
|
|
|
|
/*dark*/ QColor(220, 220, 220, 120),
|
|
|
|
/*mid*/ QColor(),
|
|
|
|
/*text*/ QColor("#333"),
|
|
|
|
/*bright_text*/ QColor(),
|
|
|
|
/*base*/ QColor("white"),
|
|
|
|
/*window*/ QColor("white"));
|
|
|
|
view->rootContext()->setContextProperty("currentActivePalette", lightActive);
|
|
|
|
view->rootContext()->setContextProperty("currentInactivePalette", lightActive);
|
2020-01-27 15:59:25 +01:00
|
|
|
} else if (settings->theme() == "dark") {
|
2019-11-09 03:06:10 +01:00
|
|
|
QPalette darkActive(/*windowText*/ QColor("#caccd1"),
|
|
|
|
/*button*/ QColor("#caccd1"),
|
|
|
|
/*light*/ QColor(),
|
2020-01-23 21:28:17 +01:00
|
|
|
/*dark*/ QColor("#2d3139"),
|
2019-11-09 03:06:10 +01:00
|
|
|
/*mid*/ QColor(),
|
|
|
|
/*text*/ QColor("#caccd1"),
|
|
|
|
/*bright_text*/ QColor(),
|
2020-01-23 21:28:17 +01:00
|
|
|
/*base*/ QColor("#202228"),
|
|
|
|
/*window*/ QColor("#202228"));
|
2019-11-09 03:06:10 +01:00
|
|
|
darkActive.setColor(QPalette::Highlight, QColor("#e7e7e9"));
|
|
|
|
view->rootContext()->setContextProperty("currentActivePalette", darkActive);
|
|
|
|
view->rootContext()->setContextProperty("currentInactivePalette", darkActive);
|
|
|
|
} else {
|
|
|
|
view->rootContext()->setContextProperty("currentActivePalette", QPalette());
|
|
|
|
view->rootContext()->setContextProperty("currentInactivePalette", nullptr);
|
2018-07-17 22:50:18 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-27 15:59:25 +01:00
|
|
|
TimelineViewManager::TimelineViewManager(QSharedPointer<UserSettings> userSettings, QWidget *parent)
|
2019-11-09 03:06:10 +01:00
|
|
|
: imgProvider(new MxcImageProvider())
|
|
|
|
, colorImgProvider(new ColorImageProvider())
|
2020-01-27 15:59:25 +01:00
|
|
|
, settings(userSettings)
|
2017-04-13 03:11:22 +02:00
|
|
|
{
|
2019-11-09 03:06:10 +01:00
|
|
|
qmlRegisterUncreatableMetaObject(qml_mtx_events::staticMetaObject,
|
2019-11-30 01:43:39 +01:00
|
|
|
"im.nheko",
|
2019-11-09 03:06:10 +01:00
|
|
|
1,
|
|
|
|
0,
|
|
|
|
"MtxEvent",
|
|
|
|
"Can't instantiate enum!");
|
2019-11-30 01:43:39 +01:00
|
|
|
qmlRegisterType<DelegateChoice>("im.nheko", 1, 0, "DelegateChoice");
|
|
|
|
qmlRegisterType<DelegateChooser>("im.nheko", 1, 0, "DelegateChooser");
|
2020-01-11 18:53:32 +01:00
|
|
|
qRegisterMetaType<mtx::events::collections::TimelineEvents>();
|
2019-11-09 03:06:10 +01:00
|
|
|
|
|
|
|
#ifdef USE_QUICK_VIEW
|
|
|
|
view = new QQuickView();
|
|
|
|
container = QWidget::createWindowContainer(view, parent);
|
|
|
|
#else
|
|
|
|
view = new QQuickWidget(parent);
|
|
|
|
container = view;
|
|
|
|
view->setResizeMode(QQuickWidget::SizeRootObjectToView);
|
|
|
|
container->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
|
|
|
|
|
|
|
|
connect(view, &QQuickWidget::statusChanged, this, [](QQuickWidget::Status status) {
|
|
|
|
nhlog::ui()->debug("Status changed to {}", status);
|
|
|
|
});
|
|
|
|
#endif
|
|
|
|
container->setMinimumSize(200, 200);
|
|
|
|
view->rootContext()->setContextProperty("timelineManager", this);
|
|
|
|
updateColorPalette();
|
|
|
|
view->engine()->addImageProvider("MxcImage", imgProvider);
|
|
|
|
view->engine()->addImageProvider("colorimage", colorImgProvider);
|
|
|
|
view->setSource(QUrl("qrc:///qml/TimelineView.qml"));
|
|
|
|
|
|
|
|
connect(dynamic_cast<ChatPage *>(parent),
|
|
|
|
&ChatPage::themeChanged,
|
|
|
|
this,
|
|
|
|
&TimelineViewManager::updateColorPalette);
|
2017-11-15 17:38:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2019-11-09 03:06:10 +01:00
|
|
|
TimelineViewManager::sync(const mtx::responses::Rooms &rooms)
|
2017-04-13 03:11:22 +02:00
|
|
|
{
|
2020-01-17 01:25:14 +01:00
|
|
|
for (const auto &[room_id, room] : rooms.join) {
|
2019-11-09 03:06:10 +01:00
|
|
|
// addRoom will only add the room, if it doesn't exist
|
2020-01-17 01:25:14 +01:00
|
|
|
addRoom(QString::fromStdString(room_id));
|
|
|
|
const auto &room_model = models.value(QString::fromStdString(room_id));
|
|
|
|
room_model->addEvents(room.timeline);
|
|
|
|
|
|
|
|
if (ChatPage::instance()->userSettings()->isTypingNotificationsEnabled()) {
|
|
|
|
std::vector<QString> typing;
|
|
|
|
typing.reserve(room.ephemeral.typing.size());
|
|
|
|
for (const auto &user : room.ephemeral.typing) {
|
|
|
|
if (user != http::client()->user_id().to_string())
|
|
|
|
typing.push_back(QString::fromStdString(user));
|
|
|
|
}
|
|
|
|
room_model->updateTypingUsers(typing);
|
|
|
|
}
|
2019-11-09 03:06:10 +01:00
|
|
|
}
|
2019-11-10 00:30:02 +01:00
|
|
|
|
|
|
|
this->isInitialSync_ = false;
|
|
|
|
emit initialSyncChanged(false);
|
2017-09-03 10:43:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2019-11-09 03:06:10 +01:00
|
|
|
TimelineViewManager::addRoom(const QString &room_id)
|
2017-09-03 10:43:45 +02:00
|
|
|
{
|
2019-12-03 23:34:16 +01:00
|
|
|
if (!models.contains(room_id)) {
|
|
|
|
QSharedPointer<TimelineModel> newRoom(new TimelineModel(this, room_id));
|
|
|
|
connect(newRoom.data(),
|
|
|
|
&TimelineModel::newEncryptedImage,
|
|
|
|
imgProvider,
|
|
|
|
&MxcImageProvider::addEncryptionInfo);
|
|
|
|
models.insert(room_id, std::move(newRoom));
|
|
|
|
}
|
2017-04-13 03:11:22 +02:00
|
|
|
}
|
|
|
|
|
2019-06-10 01:03:18 +02:00
|
|
|
void
|
2019-11-09 03:06:10 +01:00
|
|
|
TimelineViewManager::setHistoryView(const QString &room_id)
|
2019-06-10 01:03:18 +02:00
|
|
|
{
|
2019-11-09 03:06:10 +01:00
|
|
|
nhlog::ui()->info("Trying to activate room {}", room_id.toStdString());
|
2019-06-10 01:03:18 +02:00
|
|
|
|
2019-11-09 03:06:10 +01:00
|
|
|
auto room = models.find(room_id);
|
|
|
|
if (room != models.end()) {
|
|
|
|
timeline_ = room.value().data();
|
|
|
|
emit activeTimelineChanged(timeline_);
|
|
|
|
nhlog::ui()->info("Activated room {}", room_id.toStdString());
|
|
|
|
}
|
2019-06-10 01:03:18 +02:00
|
|
|
}
|
|
|
|
|
2017-09-10 11:58:00 +02:00
|
|
|
void
|
2019-12-03 02:26:41 +01:00
|
|
|
TimelineViewManager::openImageOverlay(QString mxcUrl, QString eventId) const
|
2017-09-10 11:58:00 +02:00
|
|
|
{
|
2019-11-09 03:06:10 +01:00
|
|
|
QQuickImageResponse *imgResponse =
|
|
|
|
imgProvider->requestImageResponse(mxcUrl.remove("mxc://"), QSize());
|
2019-12-03 02:26:41 +01:00
|
|
|
connect(imgResponse, &QQuickImageResponse::finished, this, [this, eventId, imgResponse]() {
|
|
|
|
if (!imgResponse->errorString().isEmpty()) {
|
|
|
|
nhlog::ui()->error("Error when retrieving image for overlay: {}",
|
|
|
|
imgResponse->errorString().toStdString());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto pixmap = QPixmap::fromImage(imgResponse->textureFactory()->image());
|
2019-11-09 03:06:10 +01:00
|
|
|
|
2019-12-03 02:26:41 +01:00
|
|
|
auto imgDialog = new dialogs::ImageOverlay(pixmap);
|
|
|
|
imgDialog->show();
|
|
|
|
connect(imgDialog, &dialogs::ImageOverlay::saving, timeline_, [this, eventId]() {
|
|
|
|
timeline_->saveMedia(eventId);
|
|
|
|
});
|
|
|
|
});
|
2017-12-01 16:33:49 +01:00
|
|
|
}
|
|
|
|
|
2017-08-20 12:47:22 +02:00
|
|
|
void
|
2019-11-09 03:06:10 +01:00
|
|
|
TimelineViewManager::updateReadReceipts(const QString &room_id,
|
|
|
|
const std::vector<QString> &event_ids)
|
2017-04-06 01:06:42 +02:00
|
|
|
{
|
2019-11-09 03:06:10 +01:00
|
|
|
auto room = models.find(room_id);
|
|
|
|
if (room != models.end()) {
|
|
|
|
room.value()->markEventsAsRead(event_ids);
|
2017-08-26 10:33:26 +02:00
|
|
|
}
|
2017-04-06 01:06:42 +02:00
|
|
|
}
|
|
|
|
|
2018-06-28 15:16:43 +02:00
|
|
|
void
|
|
|
|
TimelineViewManager::initWithMessages(const std::map<QString, mtx::responses::Timeline> &msgs)
|
|
|
|
{
|
2019-11-09 03:06:10 +01:00
|
|
|
for (const auto &e : msgs) {
|
|
|
|
addRoom(e.first);
|
2018-06-28 15:16:43 +02:00
|
|
|
|
2019-11-09 03:06:10 +01:00
|
|
|
models.value(e.first)->addEvents(e.second);
|
2018-06-28 15:16:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-20 12:47:22 +02:00
|
|
|
void
|
2020-01-12 16:39:01 +01:00
|
|
|
TimelineViewManager::queueTextMessage(const QString &msg, const std::optional<RelatedInfo> &related)
|
2017-07-29 10:49:00 +02:00
|
|
|
{
|
2019-11-09 03:06:10 +01:00
|
|
|
mtx::events::msg::Text text = {};
|
|
|
|
text.body = msg.trimmed().toStdString();
|
|
|
|
text.format = "org.matrix.custom.html";
|
|
|
|
text.formatted_body = utils::markdownToHtml(msg).toStdString();
|
|
|
|
|
2020-01-12 16:39:01 +01:00
|
|
|
if (related) {
|
|
|
|
QString body;
|
|
|
|
bool firstLine = true;
|
|
|
|
for (const auto &line : related->quoted_body.split("\n")) {
|
|
|
|
if (firstLine) {
|
|
|
|
firstLine = false;
|
|
|
|
body = QString("> <%1> %2\n").arg(related->quoted_user).arg(line);
|
|
|
|
} else {
|
|
|
|
body = QString("%1\n> %2\n").arg(body).arg(line);
|
|
|
|
}
|
2019-11-09 03:06:10 +01:00
|
|
|
}
|
2017-08-26 10:33:26 +02:00
|
|
|
|
2020-01-12 16:39:01 +01:00
|
|
|
text.body = QString("%1\n%2").arg(body).arg(msg).toStdString();
|
|
|
|
text.formatted_body =
|
|
|
|
utils::getFormattedQuoteBody(*related, utils::markdownToHtml(msg)).toStdString();
|
|
|
|
text.relates_to.in_reply_to.event_id = related->related_event;
|
|
|
|
}
|
2017-08-26 10:33:26 +02:00
|
|
|
|
2020-01-27 15:59:25 +01:00
|
|
|
if (!settings->isMarkdownEnabled()) {
|
|
|
|
text.format = "";
|
|
|
|
text.formatted_body = "";
|
|
|
|
}
|
|
|
|
|
2019-11-09 03:06:10 +01:00
|
|
|
if (timeline_)
|
|
|
|
timeline_->sendMessage(text);
|
2017-10-01 18:49:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2019-11-09 03:06:10 +01:00
|
|
|
TimelineViewManager::queueEmoteMessage(const QString &msg)
|
2017-10-01 18:49:36 +02:00
|
|
|
{
|
2019-11-09 03:06:10 +01:00
|
|
|
auto html = utils::markdownToHtml(msg);
|
2018-04-21 15:34:50 +02:00
|
|
|
|
2019-11-09 03:06:10 +01:00
|
|
|
mtx::events::msg::Emote emote;
|
|
|
|
emote.body = msg.trimmed().toStdString();
|
2017-10-01 18:49:36 +02:00
|
|
|
|
2020-01-27 15:59:25 +01:00
|
|
|
if (html != msg.trimmed().toHtmlEscaped() && settings->isMarkdownEnabled()) {
|
2019-11-09 03:06:10 +01:00
|
|
|
emote.formatted_body = html.toStdString();
|
2020-01-27 15:59:25 +01:00
|
|
|
emote.format = "org.matrix.custom.html";
|
|
|
|
}
|
2017-10-01 18:49:36 +02:00
|
|
|
|
2019-11-09 03:06:10 +01:00
|
|
|
if (timeline_)
|
|
|
|
timeline_->sendMessage(emote);
|
2017-07-29 10:49:00 +02:00
|
|
|
}
|
|
|
|
|
2017-08-20 12:47:22 +02:00
|
|
|
void
|
2019-11-09 03:06:10 +01:00
|
|
|
TimelineViewManager::queueImageMessage(const QString &roomid,
|
|
|
|
const QString &filename,
|
2019-12-14 17:08:36 +01:00
|
|
|
const std::optional<mtx::crypto::EncryptedFile> &file,
|
2019-11-09 03:06:10 +01:00
|
|
|
const QString &url,
|
|
|
|
const QString &mime,
|
|
|
|
uint64_t dsize,
|
2020-01-12 16:39:01 +01:00
|
|
|
const QSize &dimensions,
|
|
|
|
const std::optional<RelatedInfo> &related)
|
2017-04-06 01:06:42 +02:00
|
|
|
{
|
2019-11-09 03:06:10 +01:00
|
|
|
mtx::events::msg::Image image;
|
|
|
|
image.info.mimetype = mime.toStdString();
|
|
|
|
image.info.size = dsize;
|
|
|
|
image.body = filename.toStdString();
|
|
|
|
image.url = url.toStdString();
|
|
|
|
image.info.h = dimensions.height();
|
|
|
|
image.info.w = dimensions.width();
|
2019-12-05 15:31:53 +01:00
|
|
|
image.file = file;
|
2020-01-12 16:39:01 +01:00
|
|
|
|
|
|
|
if (related)
|
|
|
|
image.relates_to.in_reply_to.event_id = related->related_event;
|
|
|
|
|
2019-11-09 03:06:10 +01:00
|
|
|
models.value(roomid)->sendMessage(image);
|
2017-04-06 01:06:42 +02:00
|
|
|
}
|
|
|
|
|
2017-08-20 12:47:22 +02:00
|
|
|
void
|
2019-12-05 15:31:53 +01:00
|
|
|
TimelineViewManager::queueFileMessage(
|
|
|
|
const QString &roomid,
|
|
|
|
const QString &filename,
|
2019-12-14 17:08:36 +01:00
|
|
|
const std::optional<mtx::crypto::EncryptedFile> &encryptedFile,
|
2019-12-05 15:31:53 +01:00
|
|
|
const QString &url,
|
|
|
|
const QString &mime,
|
2020-01-12 16:39:01 +01:00
|
|
|
uint64_t dsize,
|
|
|
|
const std::optional<RelatedInfo> &related)
|
2017-04-06 01:06:42 +02:00
|
|
|
{
|
2019-11-09 03:06:10 +01:00
|
|
|
mtx::events::msg::File file;
|
|
|
|
file.info.mimetype = mime.toStdString();
|
|
|
|
file.info.size = dsize;
|
|
|
|
file.body = filename.toStdString();
|
|
|
|
file.url = url.toStdString();
|
2019-12-05 15:31:53 +01:00
|
|
|
file.file = encryptedFile;
|
2020-01-12 16:39:01 +01:00
|
|
|
|
|
|
|
if (related)
|
|
|
|
file.relates_to.in_reply_to.event_id = related->related_event;
|
|
|
|
|
2019-11-09 03:06:10 +01:00
|
|
|
models.value(roomid)->sendMessage(file);
|
2017-04-06 01:06:42 +02:00
|
|
|
}
|
2017-04-11 21:48:02 +02:00
|
|
|
|
2019-11-09 03:06:10 +01:00
|
|
|
void
|
|
|
|
TimelineViewManager::queueAudioMessage(const QString &roomid,
|
|
|
|
const QString &filename,
|
2019-12-14 17:08:36 +01:00
|
|
|
const std::optional<mtx::crypto::EncryptedFile> &file,
|
2019-11-09 03:06:10 +01:00
|
|
|
const QString &url,
|
|
|
|
const QString &mime,
|
2020-01-12 16:39:01 +01:00
|
|
|
uint64_t dsize,
|
|
|
|
const std::optional<RelatedInfo> &related)
|
2017-04-11 21:48:02 +02:00
|
|
|
{
|
2019-11-09 03:06:10 +01:00
|
|
|
mtx::events::msg::Audio audio;
|
|
|
|
audio.info.mimetype = mime.toStdString();
|
|
|
|
audio.info.size = dsize;
|
|
|
|
audio.body = filename.toStdString();
|
|
|
|
audio.url = url.toStdString();
|
2019-12-05 15:31:53 +01:00
|
|
|
audio.file = file;
|
2020-01-12 16:39:01 +01:00
|
|
|
|
|
|
|
if (related)
|
|
|
|
audio.relates_to.in_reply_to.event_id = related->related_event;
|
|
|
|
|
2019-11-09 03:06:10 +01:00
|
|
|
models.value(roomid)->sendMessage(audio);
|
2017-04-13 03:11:22 +02:00
|
|
|
}
|
2017-05-08 18:44:01 +02:00
|
|
|
|
2019-11-09 03:06:10 +01:00
|
|
|
void
|
|
|
|
TimelineViewManager::queueVideoMessage(const QString &roomid,
|
|
|
|
const QString &filename,
|
2019-12-14 17:08:36 +01:00
|
|
|
const std::optional<mtx::crypto::EncryptedFile> &file,
|
2019-11-09 03:06:10 +01:00
|
|
|
const QString &url,
|
|
|
|
const QString &mime,
|
2020-01-12 16:39:01 +01:00
|
|
|
uint64_t dsize,
|
|
|
|
const std::optional<RelatedInfo> &related)
|
2017-10-07 19:50:32 +02:00
|
|
|
{
|
2019-11-09 03:06:10 +01:00
|
|
|
mtx::events::msg::Video video;
|
|
|
|
video.info.mimetype = mime.toStdString();
|
|
|
|
video.info.size = dsize;
|
|
|
|
video.body = filename.toStdString();
|
|
|
|
video.url = url.toStdString();
|
2019-12-05 15:31:53 +01:00
|
|
|
video.file = file;
|
2020-01-12 16:39:01 +01:00
|
|
|
|
|
|
|
if (related)
|
|
|
|
video.relates_to.in_reply_to.event_id = related->related_event;
|
|
|
|
|
2019-11-09 03:06:10 +01:00
|
|
|
models.value(roomid)->sendMessage(video);
|
2017-10-07 19:50:32 +02:00
|
|
|
}
|