2017-04-06 02:06:42 +03:00
|
|
|
/*
|
|
|
|
* nheko Copyright (C) 2017 Konstantinos Sideris <siderisk@auth.gr>
|
|
|
|
*
|
|
|
|
* This program 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.
|
|
|
|
*
|
|
|
|
* This program 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.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2017-05-16 21:46:45 +03:00
|
|
|
#pragma once
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-11-30 00:39:35 +03:00
|
|
|
#include <QApplication>
|
2017-10-28 15:46:39 +03:00
|
|
|
#include <QLayout>
|
2017-04-06 02:06:42 +03:00
|
|
|
#include <QList>
|
2017-11-15 19:38:50 +03:00
|
|
|
#include <QQueue>
|
2017-04-06 02:06:42 +03:00
|
|
|
#include <QScrollArea>
|
2017-11-16 17:33:52 +03:00
|
|
|
#include <QStyle>
|
|
|
|
#include <QStyleOption>
|
2018-05-08 18:43:56 +03:00
|
|
|
#include <QTimer>
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-12-23 14:50:11 +03:00
|
|
|
#include <mtx/events.hpp>
|
|
|
|
#include <mtx/responses/messages.hpp>
|
2017-12-01 16:39:50 +03:00
|
|
|
|
|
|
|
#include "MatrixClient.h"
|
2018-07-17 16:37:25 +03:00
|
|
|
#include "timeline/TimelineItem.h"
|
|
|
|
#include "ui/ScrollBar.h"
|
2017-05-07 17:15:38 +03:00
|
|
|
|
2018-06-15 16:45:39 +03:00
|
|
|
class StateKeeper
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
StateKeeper(std::function<void()> &&fn)
|
|
|
|
: fn_(std::move(fn))
|
|
|
|
{}
|
|
|
|
|
|
|
|
~StateKeeper() { fn_(); }
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::function<void()> fn_;
|
|
|
|
};
|
|
|
|
|
2018-07-07 12:35:29 +03:00
|
|
|
struct DecryptionResult
|
|
|
|
{
|
|
|
|
//! The decrypted content as a normal plaintext event.
|
|
|
|
utils::TimelineEvent event;
|
|
|
|
//! Whether or not the decryption was successful.
|
|
|
|
bool isDecrypted = false;
|
|
|
|
};
|
|
|
|
|
2017-10-27 22:20:33 +03:00
|
|
|
class FloatingButton;
|
2017-10-28 15:46:39 +03:00
|
|
|
struct DescInfo;
|
2017-10-27 22:20:33 +03:00
|
|
|
|
2017-04-13 04:11:22 +03:00
|
|
|
// Contains info about a message shown in the history view
|
|
|
|
// but not yet confirmed by the homeserver through sync.
|
2017-10-15 22:08:51 +03:00
|
|
|
struct PendingMessage
|
|
|
|
{
|
2017-12-04 19:41:19 +03:00
|
|
|
mtx::events::MessageType ty;
|
2018-06-09 16:03:14 +03:00
|
|
|
std::string txn_id;
|
2017-08-26 11:33:26 +03:00
|
|
|
QString body;
|
2017-11-15 19:38:50 +03:00
|
|
|
QString filename;
|
2018-02-18 23:52:31 +03:00
|
|
|
QString mime;
|
2018-02-19 23:09:21 +03:00
|
|
|
uint64_t media_size;
|
2017-08-26 11:33:26 +03:00
|
|
|
QString event_id;
|
|
|
|
TimelineItem *widget;
|
2018-07-10 23:31:51 +03:00
|
|
|
QSize dimensions;
|
2018-06-12 22:35:10 +03:00
|
|
|
bool is_encrypted = false;
|
2017-04-13 04:11:22 +03:00
|
|
|
};
|
|
|
|
|
2018-06-12 22:35:10 +03:00
|
|
|
template<class MessageT>
|
|
|
|
MessageT
|
|
|
|
toRoomMessage(const PendingMessage &) = delete;
|
|
|
|
|
|
|
|
template<>
|
|
|
|
mtx::events::msg::Audio
|
|
|
|
toRoomMessage<mtx::events::msg::Audio>(const PendingMessage &m);
|
|
|
|
|
|
|
|
template<>
|
|
|
|
mtx::events::msg::Emote
|
|
|
|
toRoomMessage<mtx::events::msg::Emote>(const PendingMessage &m);
|
|
|
|
|
|
|
|
template<>
|
|
|
|
mtx::events::msg::File
|
|
|
|
toRoomMessage<mtx::events::msg::File>(const PendingMessage &);
|
|
|
|
|
|
|
|
template<>
|
|
|
|
mtx::events::msg::Image
|
|
|
|
toRoomMessage<mtx::events::msg::Image>(const PendingMessage &m);
|
|
|
|
|
|
|
|
template<>
|
|
|
|
mtx::events::msg::Text
|
|
|
|
toRoomMessage<mtx::events::msg::Text>(const PendingMessage &);
|
|
|
|
|
|
|
|
template<>
|
|
|
|
mtx::events::msg::Video
|
|
|
|
toRoomMessage<mtx::events::msg::Video>(const PendingMessage &m);
|
|
|
|
|
2017-05-12 15:43:35 +03:00
|
|
|
// In which place new TimelineItems should be inserted.
|
2017-10-15 22:08:51 +03:00
|
|
|
enum class TimelineDirection
|
|
|
|
{
|
2017-08-26 11:33:26 +03:00
|
|
|
Top,
|
|
|
|
Bottom,
|
2017-05-12 15:43:35 +03:00
|
|
|
};
|
|
|
|
|
2017-04-25 14:37:54 +03:00
|
|
|
class TimelineView : public QWidget
|
2017-04-06 02:06:42 +03:00
|
|
|
{
|
2017-08-26 11:33:26 +03:00
|
|
|
Q_OBJECT
|
2017-04-06 02:06:42 +03:00
|
|
|
|
|
|
|
public:
|
2017-12-04 19:41:19 +03:00
|
|
|
TimelineView(const mtx::responses::Timeline &timeline,
|
2017-08-26 11:33:26 +03:00
|
|
|
const QString &room_id,
|
|
|
|
QWidget *parent = 0);
|
2018-05-08 18:43:56 +03:00
|
|
|
TimelineView(const QString &room_id, QWidget *parent = 0);
|
2017-08-26 11:33:26 +03:00
|
|
|
|
|
|
|
// Add new events at the end of the timeline.
|
2018-02-15 22:58:57 +03:00
|
|
|
void addEvents(const mtx::responses::Timeline &timeline);
|
2017-12-04 19:41:19 +03:00
|
|
|
void addUserMessage(mtx::events::MessageType ty, const QString &msg);
|
2017-11-30 00:39:35 +03:00
|
|
|
|
2017-12-04 19:41:19 +03:00
|
|
|
template<class Widget, mtx::events::MessageType MsgType>
|
2018-02-18 23:52:31 +03:00
|
|
|
void addUserMessage(const QString &url,
|
|
|
|
const QString &filename,
|
|
|
|
const QString &mime,
|
2018-07-10 23:31:51 +03:00
|
|
|
uint64_t size,
|
|
|
|
const QSize &dimensions = QSize());
|
2018-06-09 16:03:14 +03:00
|
|
|
void updatePendingMessage(const std::string &txn_id, const QString &event_id);
|
2017-08-26 11:33:26 +03:00
|
|
|
void scrollDown();
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2018-03-17 22:23:46 +03:00
|
|
|
//! Remove an item from the timeline with the given Event ID.
|
|
|
|
void removeEvent(const QString &event_id);
|
2018-06-28 16:16:43 +03:00
|
|
|
void setPrevBatchToken(const QString &token) { prev_batch_token_ = token; }
|
2018-03-17 22:23:46 +03:00
|
|
|
|
2017-04-06 02:06:42 +03:00
|
|
|
public slots:
|
2017-08-26 11:33:26 +03:00
|
|
|
void sliderRangeChanged(int min, int max);
|
|
|
|
void sliderMoved(int position);
|
|
|
|
void fetchHistory();
|
2017-05-12 15:43:35 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
// Add old events at the top of the timeline.
|
2018-06-09 16:03:14 +03:00
|
|
|
void addBackwardsEvents(const mtx::responses::Messages &msgs);
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-10-07 20:50:32 +03:00
|
|
|
// Whether or not the initial batch has been loaded.
|
2018-07-23 21:54:13 +03:00
|
|
|
bool hasLoaded() { return scroll_layout_->count() > 0 || isTimelineFinished; }
|
2017-10-07 20:50:32 +03:00
|
|
|
|
2018-06-09 16:03:14 +03:00
|
|
|
void handleFailedMessage(const std::string &txn_id);
|
2017-11-15 19:38:50 +03:00
|
|
|
|
|
|
|
private slots:
|
|
|
|
void sendNextPendingMessage();
|
|
|
|
|
2017-08-06 18:53:31 +03:00
|
|
|
signals:
|
2017-08-26 11:33:26 +03:00
|
|
|
void updateLastTimelineMessage(const QString &user, const DescInfo &info);
|
2018-06-09 16:03:14 +03:00
|
|
|
void messagesRetrieved(const mtx::responses::Messages &res);
|
|
|
|
void messageFailed(const std::string &txn_id);
|
|
|
|
void messageSent(const std::string &txn_id, const QString &event_id);
|
2018-07-17 23:50:18 +03:00
|
|
|
void markReadEvents(const std::vector<QString> &event_ids);
|
2017-08-06 18:53:31 +03:00
|
|
|
|
2017-11-16 17:33:52 +03:00
|
|
|
protected:
|
|
|
|
void paintEvent(QPaintEvent *event) override;
|
2017-11-24 01:10:58 +03:00
|
|
|
void showEvent(QShowEvent *event) override;
|
2018-07-21 12:09:23 +03:00
|
|
|
void hideEvent(QHideEvent *event) override;
|
2017-11-24 01:10:58 +03:00
|
|
|
bool event(QEvent *event) override;
|
2017-11-16 17:33:52 +03:00
|
|
|
|
2017-04-06 02:06:42 +03:00
|
|
|
private:
|
2018-01-05 01:27:32 +03:00
|
|
|
using TimelineEvent = mtx::events::collections::TimelineEvents;
|
|
|
|
|
2018-07-17 23:50:18 +03:00
|
|
|
//! Mark our own widgets as read if they have more than one receipt.
|
|
|
|
void displayReadReceipts(std::vector<TimelineEvent> events);
|
2018-08-31 10:47:27 +03:00
|
|
|
//! Determine if the start of the timeline is reached from the response of /messages.
|
|
|
|
bool isStartOfTimeline(const mtx::responses::Messages &msgs);
|
2018-07-17 23:50:18 +03:00
|
|
|
|
2018-07-15 20:32:22 +03:00
|
|
|
QWidget *relativeWidget(QWidget *item, int dt) const;
|
2018-03-17 22:23:46 +03:00
|
|
|
|
2018-07-07 12:35:29 +03:00
|
|
|
DecryptionResult parseEncryptedEvent(
|
2018-06-10 20:03:45 +03:00
|
|
|
const mtx::events::EncryptedEvent<mtx::events::msg::Encrypted> &e);
|
|
|
|
|
2018-06-15 16:45:39 +03:00
|
|
|
void handleClaimedKeys(std::shared_ptr<StateKeeper> keeper,
|
2018-06-25 17:19:52 +03:00
|
|
|
const std::map<std::string, std::string> &room_key,
|
|
|
|
const std::map<std::string, DevicePublicKeys> &pks,
|
2018-06-15 16:45:39 +03:00
|
|
|
const std::string &user_id,
|
|
|
|
const mtx::responses::ClaimKeys &res,
|
|
|
|
mtx::http::RequestErr err);
|
|
|
|
|
2018-06-09 16:03:14 +03:00
|
|
|
//! Callback for all message sending.
|
|
|
|
void sendRoomMessageHandler(const std::string &txn_id,
|
|
|
|
const mtx::responses::EventId &res,
|
|
|
|
mtx::http::RequestErr err);
|
2018-06-13 12:28:00 +03:00
|
|
|
void prepareEncryptedMessage(const PendingMessage &msg);
|
2018-06-09 16:03:14 +03:00
|
|
|
|
|
|
|
//! Call the /messages endpoint to fill the timeline.
|
|
|
|
void getMessages();
|
2018-01-16 23:24:23 +03:00
|
|
|
//! HACK: Fixing layout flickering when adding to the bottom
|
|
|
|
//! of the timeline.
|
2018-07-07 13:39:53 +03:00
|
|
|
void pushTimelineItem(QWidget *item)
|
2018-01-16 23:24:23 +03:00
|
|
|
{
|
2018-08-28 00:19:39 +03:00
|
|
|
setUpdatesEnabled(false);
|
2018-01-16 23:24:23 +03:00
|
|
|
item->hide();
|
2018-01-29 16:29:07 +03:00
|
|
|
scroll_layout_->addWidget(item);
|
2018-08-28 00:19:39 +03:00
|
|
|
QTimer::singleShot(0, this, [item, this]() {
|
|
|
|
item->show();
|
|
|
|
item->adjustSize();
|
|
|
|
setUpdatesEnabled(true);
|
|
|
|
});
|
2018-01-16 23:24:23 +03:00
|
|
|
};
|
|
|
|
|
2018-01-23 18:34:57 +03:00
|
|
|
//! Decides whether or not to show or hide the scroll down button.
|
|
|
|
void toggleScrollDownButton();
|
2017-08-26 11:33:26 +03:00
|
|
|
void init();
|
2018-07-07 13:39:53 +03:00
|
|
|
void addTimelineItem(QWidget *item,
|
2018-02-10 02:09:30 +03:00
|
|
|
TimelineDirection direction = TimelineDirection::Bottom);
|
2017-08-26 11:33:26 +03:00
|
|
|
void updateLastSender(const QString &user_id, TimelineDirection direction);
|
|
|
|
void notifyForLastEvent();
|
2018-01-05 01:27:32 +03:00
|
|
|
void notifyForLastEvent(const TimelineEvent &event);
|
2018-05-01 23:32:11 +03:00
|
|
|
//! Keep track of the sender and the timestamp of the current message.
|
|
|
|
void saveLastMessageInfo(const QString &sender, const QDateTime &datetime)
|
|
|
|
{
|
|
|
|
lastSender_ = sender;
|
|
|
|
lastMsgTimestamp_ = datetime;
|
|
|
|
}
|
|
|
|
void saveFirstMessageInfo(const QString &sender, const QDateTime &datetime)
|
|
|
|
{
|
|
|
|
firstSender_ = sender;
|
|
|
|
firstMsgTimestamp_ = datetime;
|
|
|
|
}
|
|
|
|
//! Keep track of the sender and the timestamp of the current message.
|
|
|
|
void saveMessageInfo(const QString &sender,
|
|
|
|
uint64_t origin_server_ts,
|
|
|
|
TimelineDirection direction);
|
2018-03-18 14:29:21 +03:00
|
|
|
|
|
|
|
TimelineEvent findFirstViewableEvent(const std::vector<TimelineEvent> &events);
|
|
|
|
TimelineEvent findLastViewableEvent(const std::vector<TimelineEvent> &events);
|
|
|
|
|
2018-03-27 21:07:39 +03:00
|
|
|
//! Mark the last event as read.
|
2017-11-24 01:10:58 +03:00
|
|
|
void readLastEvent() const;
|
2018-03-27 21:07:39 +03:00
|
|
|
//! Whether or not the scrollbar is visible (non-zero height).
|
2017-12-23 14:50:11 +03:00
|
|
|
bool isScrollbarActivated() { return scroll_area_->verticalScrollBar()->value() != 0; }
|
2018-03-27 21:07:39 +03:00
|
|
|
//! Retrieve the event id of the last item.
|
2017-11-24 01:10:58 +03:00
|
|
|
QString getLastEventId() const;
|
2017-05-12 15:43:35 +03:00
|
|
|
|
2017-11-30 14:19:34 +03:00
|
|
|
template<class Event, class Widget>
|
2017-12-04 19:41:19 +03:00
|
|
|
TimelineItem *processMessageEvent(const Event &event, TimelineDirection direction);
|
2017-11-30 14:19:34 +03:00
|
|
|
|
|
|
|
// TODO: Remove this eventually.
|
|
|
|
template<class Event>
|
2017-12-04 19:41:19 +03:00
|
|
|
TimelineItem *processMessageEvent(const Event &event, TimelineDirection direction);
|
2017-11-30 14:19:34 +03:00
|
|
|
|
|
|
|
// For events with custom display widgets.
|
|
|
|
template<class Event, class Widget>
|
|
|
|
TimelineItem *createTimelineItem(const Event &event, bool withSender);
|
|
|
|
|
|
|
|
// For events without custom display widgets.
|
|
|
|
// TODO: All events should have custom widgets.
|
|
|
|
template<class Event>
|
|
|
|
TimelineItem *createTimelineItem(const Event &event, bool withSender);
|
|
|
|
|
2017-10-01 12:11:33 +03:00
|
|
|
// Used to determine whether or not we should prefix a message with the
|
|
|
|
// sender's name.
|
2018-05-01 23:32:11 +03:00
|
|
|
bool isSenderRendered(const QString &user_id,
|
|
|
|
uint64_t origin_server_ts,
|
|
|
|
TimelineDirection direction);
|
2017-09-03 11:43:45 +03:00
|
|
|
|
2018-06-09 16:03:14 +03:00
|
|
|
bool isPendingMessage(const std::string &txn_id,
|
|
|
|
const QString &sender,
|
|
|
|
const QString &userid);
|
|
|
|
void removePendingMessage(const std::string &txn_id);
|
2017-09-03 11:43:45 +03:00
|
|
|
|
2017-11-16 17:33:52 +03:00
|
|
|
bool isDuplicate(const QString &event_id) { return eventIds_.contains(event_id); }
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-11-15 19:38:50 +03:00
|
|
|
void handleNewUserMessage(PendingMessage msg);
|
2018-05-01 23:32:11 +03:00
|
|
|
bool isDateDifference(const QDateTime &first,
|
|
|
|
const QDateTime &second = QDateTime::currentDateTime()) const;
|
2017-11-15 19:38:50 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
// Return nullptr if the event couldn't be parsed.
|
2018-07-07 13:39:53 +03:00
|
|
|
QWidget *parseMessageEvent(const mtx::events::collections::TimelineEvents &event,
|
|
|
|
TimelineDirection direction);
|
2017-05-12 15:43:35 +03:00
|
|
|
|
2018-07-15 20:32:22 +03:00
|
|
|
//! Store the event id associated with the given widget.
|
|
|
|
void saveEventId(QWidget *widget);
|
2018-07-21 12:09:23 +03:00
|
|
|
//! Remove all widgets from the timeline layout.
|
|
|
|
void clearTimeline();
|
2018-07-15 20:32:22 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
QVBoxLayout *top_layout_;
|
|
|
|
QVBoxLayout *scroll_layout_;
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
QScrollArea *scroll_area_;
|
|
|
|
ScrollBar *scrollbar_;
|
|
|
|
QWidget *scroll_widget_;
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
QString firstSender_;
|
2018-05-01 23:32:11 +03:00
|
|
|
QDateTime firstMsgTimestamp_;
|
|
|
|
QString lastSender_;
|
|
|
|
QDateTime lastMsgTimestamp_;
|
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
QString room_id_;
|
|
|
|
QString prev_batch_token_;
|
|
|
|
QString local_user_;
|
2017-05-12 15:43:35 +03:00
|
|
|
|
2017-09-30 22:26:33 +03:00
|
|
|
bool isPaginationInProgress_ = false;
|
|
|
|
|
|
|
|
// Keeps track whether or not the user has visited the view.
|
2017-10-09 01:32:25 +03:00
|
|
|
bool isInitialized = false;
|
|
|
|
bool isTimelineFinished = false;
|
|
|
|
bool isInitialSync = true;
|
2017-05-12 15:43:35 +03:00
|
|
|
|
2017-10-21 18:53:15 +03:00
|
|
|
const int SCROLL_BAR_GAP = 200;
|
2017-05-12 15:43:35 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
QTimer *paginationTimer_;
|
2017-08-05 15:59:24 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
int scroll_height_ = 0;
|
|
|
|
int previous_max_height_ = 0;
|
2017-04-13 04:11:22 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
int oldPosition_;
|
|
|
|
int oldHeight_;
|
2017-06-05 19:21:19 +03:00
|
|
|
|
2017-10-27 22:20:33 +03:00
|
|
|
FloatingButton *scrollDownBtn_;
|
|
|
|
|
2017-10-27 13:36:26 +03:00
|
|
|
TimelineDirection lastMessageDirection_;
|
|
|
|
|
2018-01-05 01:27:32 +03:00
|
|
|
//! Messages received by sync not added to the timeline.
|
|
|
|
std::vector<TimelineEvent> bottomMessages_;
|
2018-01-30 22:56:01 +03:00
|
|
|
//! Messages received by /messages not added to the timeline.
|
|
|
|
std::vector<TimelineEvent> topMessages_;
|
2018-01-05 01:27:32 +03:00
|
|
|
|
|
|
|
//! Render the given timeline events to the bottom of the timeline.
|
|
|
|
void renderBottomEvents(const std::vector<TimelineEvent> &events);
|
2018-01-30 22:56:01 +03:00
|
|
|
//! Render the given timeline events to the top of the timeline.
|
|
|
|
void renderTopEvents(const std::vector<TimelineEvent> &events);
|
2018-01-05 01:27:32 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
// The events currently rendered. Used for duplicate detection.
|
2018-07-15 20:32:22 +03:00
|
|
|
QMap<QString, QWidget *> eventIds_;
|
2017-11-15 19:38:50 +03:00
|
|
|
QQueue<PendingMessage> pending_msgs_;
|
|
|
|
QList<PendingMessage> pending_sent_msgs_;
|
2017-04-06 02:06:42 +03:00
|
|
|
};
|
2017-11-30 00:39:35 +03:00
|
|
|
|
2017-12-04 19:41:19 +03:00
|
|
|
template<class Widget, mtx::events::MessageType MsgType>
|
2017-11-30 00:39:35 +03:00
|
|
|
void
|
2018-01-10 10:52:59 +03:00
|
|
|
TimelineView::addUserMessage(const QString &url,
|
2018-01-12 12:27:24 +03:00
|
|
|
const QString &filename,
|
2018-02-18 23:52:31 +03:00
|
|
|
const QString &mime,
|
2018-07-10 23:31:51 +03:00
|
|
|
uint64_t size,
|
|
|
|
const QSize &dimensions)
|
2017-11-30 00:39:35 +03:00
|
|
|
{
|
2018-05-01 23:32:11 +03:00
|
|
|
auto with_sender = (lastSender_ != local_user_) || isDateDifference(lastMsgTimestamp_);
|
2018-02-18 23:52:31 +03:00
|
|
|
auto trimmed = QFileInfo{filename}.fileName(); // Trim file path.
|
2017-11-30 00:39:35 +03:00
|
|
|
|
2018-05-08 18:43:56 +03:00
|
|
|
auto widget = new Widget(url, trimmed, size, this);
|
2017-11-30 00:39:35 +03:00
|
|
|
|
2018-01-05 01:27:32 +03:00
|
|
|
TimelineItem *view_item =
|
2018-04-21 16:34:50 +03:00
|
|
|
new TimelineItem(widget, local_user_, with_sender, room_id_, scroll_widget_);
|
2018-01-16 23:24:23 +03:00
|
|
|
|
2018-02-10 02:09:30 +03:00
|
|
|
addTimelineItem(view_item);
|
2017-11-30 00:39:35 +03:00
|
|
|
|
|
|
|
lastMessageDirection_ = TimelineDirection::Bottom;
|
|
|
|
|
2018-05-01 23:32:11 +03:00
|
|
|
// Keep track of the sender and the timestamp of the current message.
|
|
|
|
saveLastMessageInfo(local_user_, QDateTime::currentDateTime());
|
2017-11-30 00:39:35 +03:00
|
|
|
|
2018-06-09 16:03:14 +03:00
|
|
|
PendingMessage message;
|
|
|
|
message.ty = MsgType;
|
2018-07-15 19:09:08 +03:00
|
|
|
message.txn_id = http::client()->generate_txn_id();
|
2018-06-09 16:03:14 +03:00
|
|
|
message.body = url;
|
|
|
|
message.filename = trimmed;
|
|
|
|
message.mime = mime;
|
|
|
|
message.media_size = size;
|
|
|
|
message.widget = view_item;
|
2018-07-10 23:31:51 +03:00
|
|
|
message.dimensions = dimensions;
|
2017-11-30 00:39:35 +03:00
|
|
|
|
|
|
|
handleNewUserMessage(message);
|
|
|
|
}
|
2017-11-30 13:55:30 +03:00
|
|
|
|
|
|
|
template<class Event>
|
|
|
|
TimelineItem *
|
|
|
|
TimelineView::createTimelineItem(const Event &event, bool withSender)
|
|
|
|
{
|
2018-04-21 16:34:50 +03:00
|
|
|
TimelineItem *item = new TimelineItem(event, withSender, room_id_, scroll_widget_);
|
2017-11-30 13:55:30 +03:00
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class Event, class Widget>
|
|
|
|
TimelineItem *
|
|
|
|
TimelineView::createTimelineItem(const Event &event, bool withSender)
|
|
|
|
{
|
2018-05-08 18:43:56 +03:00
|
|
|
auto eventWidget = new Widget(event);
|
2018-04-21 16:34:50 +03:00
|
|
|
auto item = new TimelineItem(eventWidget, event, withSender, room_id_, scroll_widget_);
|
2017-11-30 13:55:30 +03:00
|
|
|
|
|
|
|
return item;
|
|
|
|
}
|
2017-11-30 14:19:34 +03:00
|
|
|
|
|
|
|
template<class Event>
|
|
|
|
TimelineItem *
|
2017-12-04 19:41:19 +03:00
|
|
|
TimelineView::processMessageEvent(const Event &event, TimelineDirection direction)
|
2017-11-30 14:19:34 +03:00
|
|
|
{
|
2017-12-04 19:41:19 +03:00
|
|
|
const auto event_id = QString::fromStdString(event.event_id);
|
|
|
|
const auto sender = QString::fromStdString(event.sender);
|
2017-11-30 14:19:34 +03:00
|
|
|
|
2018-06-09 16:03:14 +03:00
|
|
|
const auto txn_id = event.unsigned_data.transaction_id;
|
|
|
|
if ((!txn_id.empty() && isPendingMessage(txn_id, sender, local_user_)) ||
|
2018-03-17 22:23:46 +03:00
|
|
|
isDuplicate(event_id)) {
|
2018-06-09 16:03:14 +03:00
|
|
|
removePendingMessage(txn_id);
|
2017-11-30 14:19:34 +03:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2018-05-01 23:32:11 +03:00
|
|
|
auto with_sender = isSenderRendered(sender, event.origin_server_ts, direction);
|
2017-11-30 14:19:34 +03:00
|
|
|
|
2018-05-01 23:32:11 +03:00
|
|
|
saveMessageInfo(sender, event.origin_server_ts, direction);
|
2017-11-30 14:19:34 +03:00
|
|
|
|
2018-03-17 22:23:46 +03:00
|
|
|
auto item = createTimelineItem<Event>(event, with_sender);
|
|
|
|
|
|
|
|
eventIds_[event_id] = item;
|
|
|
|
|
|
|
|
return item;
|
2017-11-30 14:19:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
template<class Event, class Widget>
|
|
|
|
TimelineItem *
|
2017-12-04 19:41:19 +03:00
|
|
|
TimelineView::processMessageEvent(const Event &event, TimelineDirection direction)
|
2017-11-30 14:19:34 +03:00
|
|
|
{
|
2017-12-04 19:41:19 +03:00
|
|
|
const auto event_id = QString::fromStdString(event.event_id);
|
|
|
|
const auto sender = QString::fromStdString(event.sender);
|
2017-11-30 14:19:34 +03:00
|
|
|
|
2018-06-09 16:03:14 +03:00
|
|
|
const auto txn_id = event.unsigned_data.transaction_id;
|
|
|
|
if ((!txn_id.empty() && isPendingMessage(txn_id, sender, local_user_)) ||
|
2018-03-17 22:23:46 +03:00
|
|
|
isDuplicate(event_id)) {
|
2018-06-09 16:03:14 +03:00
|
|
|
removePendingMessage(txn_id);
|
2017-11-30 14:19:34 +03:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2018-05-01 23:32:11 +03:00
|
|
|
auto with_sender = isSenderRendered(sender, event.origin_server_ts, direction);
|
2017-11-30 14:19:34 +03:00
|
|
|
|
2018-05-01 23:32:11 +03:00
|
|
|
saveMessageInfo(sender, event.origin_server_ts, direction);
|
2017-11-30 14:19:34 +03:00
|
|
|
|
2018-03-17 22:23:46 +03:00
|
|
|
auto item = createTimelineItem<Event, Widget>(event, with_sender);
|
|
|
|
|
|
|
|
eventIds_[event_id] = item;
|
|
|
|
|
|
|
|
return item;
|
2017-11-30 14:19:34 +03:00
|
|
|
}
|