#pragma once #include #include #include #include #include #include #include #include #include "CacheCryptoStructs.h" #include "Logging.h" namespace mtx::http { using RequestErr = const std::optional &; } 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, Redacted, UnknownMessage, }; Q_ENUM_NS(EventType) 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) } class StateKeeper { public: StateKeeper(std::function &&fn) : fn_(std::move(fn)) {} ~StateKeeper() { fn_(); } private: std::function fn_; }; 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; }; class TimelineViewManager; class TimelineModel : public QAbstractListModel { Q_OBJECT Q_PROPERTY( int currentIndex READ currentIndex WRITE setCurrentIndex NOTIFY currentIndexChanged) public: explicit TimelineModel(TimelineViewManager *manager, QString room_id, QObject *parent = 0); enum Roles { Section, Type, Body, FormattedBody, UserId, UserName, Timestamp, Url, ThumbnailUrl, Filename, Filesize, MimeType, Height, Width, ProportionalHeight, Id, State, IsEncrypted, ReplyTo, RoomName, RoomTopic, }; QHash roleNames() const override; int rowCount(const QModelIndex &parent = QModelIndex()) const override; QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override; Q_INVOKABLE QColor userColor(QString id, QColor background); Q_INVOKABLE QString displayName(QString id) const; Q_INVOKABLE QString avatarUrl(QString id) const; Q_INVOKABLE QString formatDateSeparator(QDate date) const; Q_INVOKABLE QString escapeEmoji(QString str) const; Q_INVOKABLE void viewRawMessage(QString id) const; Q_INVOKABLE void openUserProfile(QString userid) const; Q_INVOKABLE void replyAction(QString id); Q_INVOKABLE void readReceiptsAction(QString id) const; Q_INVOKABLE void redactEvent(QString id); Q_INVOKABLE int idToIndex(QString id) const; Q_INVOKABLE QString indexToId(int index) const; Q_INVOKABLE void cacheMedia(QString eventId); Q_INVOKABLE void saveMedia(QString eventId) const; void addEvents(const mtx::responses::Timeline &events); template void sendMessage(const T &msg); public slots: void fetchHistory(); void setCurrentIndex(int index); int currentIndex() const { return idToIndex(currentId); } void markEventsAsRead(const std::vector &event_ids); 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); void messageFailed(QString txn_id); void messageSent(QString txn_id, QString event_id); void currentIndexChanged(int index); void redactionFailed(QString id); void eventRedacted(QString id); void nextPendingMessage(); void newMessageToSend(mtx::events::collections::TimelineEvents event); void mediaCached(QString mxcUrl, QString cacheUrl); void newEncryptedImage(mtx::crypto::EncryptedFile encryptionInfo); private: DecryptionResult decryptEvent( const mtx::events::EncryptedEvent &e) const; std::vector internalAddEvents( const std::vector &timeline); void sendEncryptedMessage(const std::string &txn_id, nlohmann::json content); void handleClaimedKeys(std::shared_ptr keeper, const std::map &room_key, const std::map &pks, const std::string &user_id, const mtx::responses::ClaimKeys &res, mtx::http::RequestErr err); void updateLastMessage(); void readEvent(const std::string &id); QHash events; QSet failed, read; QList pending; std::vector eventOrder; QString room_id_; QString prev_batch_token_; bool isInitialSync = true; bool paginationInProgress = false; bool isProcessingPending = false; QHash userColors; QString currentId; TimelineViewManager *manager_; friend struct SendMessageVisitor; }; template void TimelineModel::sendMessage(const T &msg) { mtx::events::RoomEvent msgCopy = {}; msgCopy.content = msg; emit newMessageToSend(msgCopy); }