nheko/src/timeline/TimelineModel.h

262 lines
8.0 KiB
C
Raw Normal View History

2019-08-30 23:20:53 +02:00
#pragma once
#include <QAbstractListModel>
#include <QColor>
#include <QDate>
2019-08-31 22:43:31 +02:00
#include <QHash>
2019-09-18 22:58:25 +02:00
#include <QSet>
2019-08-30 23:20:53 +02:00
#include <mtx/common.hpp>
#include <mtx/responses.hpp>
#include <mtxclient/http/errors.hpp>
#include "CacheCryptoStructs.h"
#include "Logging.h"
namespace mtx::http {
using RequestErr = const std::optional<mtx::http::ClientError> &;
}
namespace qml_mtx_events {
Q_NAMESPACE
enum EventType
{
// Unsupported event
Unsupported,
/// m.room_key_request
KeyRequest,
/// m.room.aliases
Aliases,
/// m.room.avatar
Avatar,
/// m.room.canonical_alias
CanonicalAlias,
/// m.room.create
Create,
/// m.room.encrypted.
Encrypted,
/// m.room.encryption.
Encryption,
/// m.room.guest_access
GuestAccess,
/// m.room.history_visibility
HistoryVisibility,
/// m.room.join_rules
JoinRules,
/// m.room.member
Member,
/// m.room.name
Name,
/// m.room.power_levels
PowerLevels,
/// m.room.tombstone
Tombstone,
/// m.room.topic
Topic,
/// m.room.redaction
Redaction,
/// m.room.pinned_events
PinnedEvents,
// m.sticker
Sticker,
// m.tag
Tag,
/// m.room.message
AudioMessage,
EmoteMessage,
FileMessage,
ImageMessage,
LocationMessage,
NoticeMessage,
TextMessage,
VideoMessage,
2019-09-09 21:42:33 +02:00
Redacted,
UnknownMessage,
};
Q_ENUM_NS(EventType)
2019-09-18 22:58:25 +02:00
enum EventState
{
//! The plaintext message was received by the server.
Received,
//! At least one of the participants has read the message.
Read,
//! The client sent the message. Not yet received.
Sent,
//! When the message is loaded from cache or backfill.
Empty,
//! When the message failed to send
Failed,
};
Q_ENUM_NS(EventState)
}
2019-09-19 22:44:25 +02:00
class StateKeeper
{
public:
StateKeeper(std::function<void()> &&fn)
: fn_(std::move(fn))
{}
~StateKeeper() { fn_(); }
private:
std::function<void()> fn_;
};
2019-09-08 16:50:32 +02:00
struct DecryptionResult
{
//! The decrypted content as a normal plaintext event.
mtx::events::collections::TimelineEvents event;
//! Whether or not the decryption was successful.
bool isDecrypted = false;
};
2019-10-03 18:07:01 +02:00
class TimelineViewManager;
2019-08-30 23:20:53 +02:00
class TimelineModel : public QAbstractListModel
{
Q_OBJECT
2019-09-18 20:34:30 +02:00
Q_PROPERTY(
int currentIndex READ currentIndex WRITE setCurrentIndex NOTIFY currentIndexChanged)
2020-01-17 01:25:14 +01:00
Q_PROPERTY(std::vector<QString> typingUsers READ typingUsers WRITE updateTypingUsers NOTIFY
typingUsersChanged)
2019-08-30 23:20:53 +02:00
public:
2019-10-03 18:07:01 +02:00
explicit TimelineModel(TimelineViewManager *manager, QString room_id, QObject *parent = 0);
2019-08-30 23:20:53 +02:00
enum Roles
{
Section,
2019-08-30 23:20:53 +02:00
Type,
Body,
FormattedBody,
UserId,
UserName,
Timestamp,
Url,
2019-10-09 00:36:03 +02:00
ThumbnailUrl,
2019-09-29 10:45:35 +02:00
Filename,
2019-10-04 01:10:46 +02:00
Filesize,
2019-09-29 10:45:35 +02:00
MimeType,
Height,
Width,
ProportionalHeight,
2019-09-08 15:26:46 +02:00
Id,
2019-09-18 22:58:25 +02:00
State,
2019-09-19 23:02:56 +02:00
IsEncrypted,
ReplyTo,
RoomName,
RoomTopic,
2020-01-03 23:21:33 +01:00
Dump,
2019-08-30 23:20:53 +02:00
};
QHash<int, QByteArray> roleNames() const override;
int rowCount(const QModelIndex &parent = QModelIndex()) const override;
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
2020-01-11 18:53:32 +01:00
QVariant data(const QString &id, int role) const;
2019-08-30 23:20:53 +02:00
bool canFetchMore(const QModelIndex &) const override;
void fetchMore(const QModelIndex &) override;
2019-08-30 23:20:53 +02:00
Q_INVOKABLE QColor userColor(QString id, QColor background);
Q_INVOKABLE QString displayName(QString id) const;
2019-09-07 22:22:07 +02:00
Q_INVOKABLE QString avatarUrl(QString id) const;
Q_INVOKABLE QString formatDateSeparator(QDate date) const;
2020-01-17 01:25:14 +01:00
Q_INVOKABLE QString formatTypingUsers(const std::vector<QString> &users, QColor bg);
2019-10-03 18:07:01 +02:00
2019-09-07 02:01:44 +02:00
Q_INVOKABLE QString escapeEmoji(QString str) const;
2019-09-08 15:26:46 +02:00
Q_INVOKABLE void viewRawMessage(QString id) const;
Q_INVOKABLE void openUserProfile(QString userid) const;
2019-09-11 00:54:40 +02:00
Q_INVOKABLE void replyAction(QString id);
Q_INVOKABLE void readReceiptsAction(QString id) const;
2019-09-29 12:29:17 +02:00
Q_INVOKABLE void redactEvent(QString id);
2019-09-18 20:34:30 +02:00
Q_INVOKABLE int idToIndex(QString id) const;
Q_INVOKABLE QString indexToId(int index) const;
2019-12-03 02:26:41 +01:00
Q_INVOKABLE void cacheMedia(QString eventId);
Q_INVOKABLE void saveMedia(QString eventId) const;
2019-08-31 22:43:31 +02:00
void addEvents(const mtx::responses::Timeline &events);
template<class T>
void sendMessage(const T &msg);
2019-08-31 22:43:31 +02:00
public slots:
2019-10-03 22:39:56 +02:00
void setCurrentIndex(int index);
2019-09-18 20:34:30 +02:00
int currentIndex() const { return idToIndex(currentId); }
void markEventsAsRead(const std::vector<QString> &event_ids);
QVariantMap getDump(QString eventId) const;
2020-01-17 01:25:14 +01:00
void updateTypingUsers(const std::vector<QString> &users)
{
if (this->typingUsers_ != users) {
this->typingUsers_ = users;
emit typingUsersChanged(typingUsers_);
}
}
std::vector<QString> typingUsers() const { return typingUsers_; }
private slots:
// Add old events at the top of the timeline.
void addBackwardsEvents(const mtx::responses::Messages &msgs);
void processOnePendingMessage();
void addPendingMessage(mtx::events::collections::TimelineEvents event);
signals:
void oldMessagesRetrieved(const mtx::responses::Messages &res);
2019-09-18 22:58:25 +02:00
void messageFailed(QString txn_id);
void messageSent(QString txn_id, QString event_id);
2019-09-18 20:34:30 +02:00
void currentIndexChanged(int index);
2019-09-29 12:29:17 +02:00
void redactionFailed(QString id);
void eventRedacted(QString id);
void nextPendingMessage();
void newMessageToSend(mtx::events::collections::TimelineEvents event);
2019-12-03 02:26:41 +01:00
void mediaCached(QString mxcUrl, QString cacheUrl);
void newEncryptedImage(mtx::crypto::EncryptedFile encryptionInfo);
2020-01-11 18:53:32 +01:00
void replyFetched(QString requestingEvent, mtx::events::collections::TimelineEvents event);
2020-01-17 01:25:14 +01:00
void typingUsersChanged(std::vector<QString> users);
2019-08-30 23:20:53 +02:00
private:
2019-09-08 16:50:32 +02:00
DecryptionResult decryptEvent(
const mtx::events::EncryptedEvent<mtx::events::msg::Encrypted> &e) const;
2019-09-09 21:42:33 +02:00
std::vector<QString> internalAddEvents(
const std::vector<mtx::events::collections::TimelineEvents> &timeline);
2019-09-19 22:44:25 +02:00
void sendEncryptedMessage(const std::string &txn_id, nlohmann::json content);
void handleClaimedKeys(std::shared_ptr<StateKeeper> keeper,
const std::map<std::string, std::string> &room_key,
const std::map<std::string, DevicePublicKeys> &pks,
const std::string &user_id,
const mtx::responses::ClaimKeys &res,
mtx::http::RequestErr err);
void updateLastMessage();
void readEvent(const std::string &id);
2019-09-08 16:50:32 +02:00
2019-08-31 22:43:31 +02:00
QHash<QString, mtx::events::collections::TimelineEvents> events;
QSet<QString> failed, read;
QList<QString> pending;
2019-08-30 23:20:53 +02:00
std::vector<QString> eventOrder;
QString room_id_;
QString prev_batch_token_;
2019-09-03 08:23:07 +02:00
bool isInitialSync = true;
bool paginationInProgress = false;
bool isProcessingPending = false;
2019-08-31 22:43:31 +02:00
QHash<QString, QColor> userColors;
2019-09-18 20:34:30 +02:00
QString currentId;
2020-01-17 01:25:14 +01:00
std::vector<QString> typingUsers_;
2019-10-03 18:07:01 +02:00
TimelineViewManager *manager_;
friend struct SendMessageVisitor;
2019-08-30 23:20:53 +02:00
};
template<class T>
void
TimelineModel::sendMessage(const T &msg)
{
2019-09-18 22:58:25 +02:00
mtx::events::RoomEvent<T> msgCopy = {};
msgCopy.content = msg;
emit newMessageToSend(msgCopy);
}