Use a QSortFilterProxyModel instead of resetting the model

This commit is contained in:
Loren Burkholder 2021-07-28 21:31:37 -04:00
parent 1777a1b52f
commit 7e538851d6
7 changed files with 71 additions and 29 deletions

View file

@ -10,7 +10,7 @@ import im.nheko 1.0
ApplicationWindow { ApplicationWindow {
id: readReceiptsRoot id: readReceiptsRoot
property ReadReceiptsModel readReceipts property ReadReceiptsProxy readReceipts
x: MainWindow.x + (MainWindow.width / 2) - (width / 2) x: MainWindow.x + (MainWindow.width / 2) - (width / 2)
y: MainWindow.y + (MainWindow.height / 2) - (height / 2) y: MainWindow.y + (MainWindow.height / 2) - (height / 2)
@ -86,7 +86,7 @@ ApplicationWindow {
ToolTip.text: model.mxid ToolTip.text: model.mxid
TapHandler { TapHandler {
onSingleTapped: chat.model.openUserProfile(userId) onSingleTapped: Rooms.currentRoom.openUserProfile(userId)
} }
CursorShape { CursorShape {

View file

@ -46,10 +46,13 @@ ReadReceiptsModel::update()
QHash<int, QByteArray> QHash<int, QByteArray>
ReadReceiptsModel::roleNames() const ReadReceiptsModel::roleNames() const
{ {
return {{Mxid, "mxid"}, // Note: RawTimestamp is purposely not included here
return {
{Mxid, "mxid"},
{DisplayName, "displayName"}, {DisplayName, "displayName"},
{AvatarUrl, "avatarUrl"}, {AvatarUrl, "avatarUrl"},
{Timestamp, "timestamp"}}; {Timestamp, "timestamp"},
};
} }
QVariant QVariant
@ -67,6 +70,8 @@ ReadReceiptsModel::data(const QModelIndex &index, int role) const
return cache::avatarUrl(room_id_, readReceipts_[index.row()].first); return cache::avatarUrl(room_id_, readReceipts_[index.row()].first);
case Timestamp: case Timestamp:
return dateFormat(readReceipts_[index.row()].second); return dateFormat(readReceipts_[index.row()].second);
case RawTimestamp:
return readReceipts_[index.row()].second;
default: default:
return {}; return {};
} }
@ -76,21 +81,22 @@ void
ReadReceiptsModel::addUsers( ReadReceiptsModel::addUsers(
const std::multimap<uint64_t, std::string, std::greater<uint64_t>> &users) const std::multimap<uint64_t, std::string, std::greater<uint64_t>> &users)
{ {
beginResetModel(); auto newReceipts = users.size() - readReceipts_.size();
if (newReceipts > 0) {
beginInsertRows(
QModelIndex{}, readReceipts_.size(), readReceipts_.size() + newReceipts - 1);
readReceipts_.clear();
for (const auto &user : users) { for (const auto &user : users) {
readReceipts_.push_back({QString::fromStdString(user.second), QPair<QString, QDateTime> item = {
QDateTime::fromMSecsSinceEpoch(user.first)}); QString::fromStdString(user.second),
QDateTime::fromMSecsSinceEpoch(user.first)};
if (!readReceipts_.contains(item))
readReceipts_.push_back(item);
} }
std::sort(readReceipts_.begin(), endInsertRows();
readReceipts_.end(), }
[](const QPair<QString, QDateTime> &a, const QPair<QString, QDateTime> &b) {
return a.second > b.second;
});
endResetModel();
} }
QString QString
@ -112,3 +118,18 @@ ReadReceiptsModel::dateFormat(const QDateTime &then) const
return QLocale::system().toString(then.time(), QLocale::ShortFormat); return QLocale::system().toString(then.time(), QLocale::ShortFormat);
} }
ReadReceiptsProxy::ReadReceiptsProxy(QString event_id, QString room_id, QObject *parent)
: QSortFilterProxyModel{parent}
, model_{event_id, room_id, this}
{
setSourceModel(&model_);
setSortRole(ReadReceiptsModel::RawTimestamp);
}
bool
ReadReceiptsProxy::lessThan(const QModelIndex &source_left, const QModelIndex &source_right) const
{
// since we are sorting from greatest to least timestamp, return something that looks totally backwards!
return source_left.data().toULongLong() > source_right.data().toULongLong();
}

View file

@ -8,15 +8,13 @@
#include <QAbstractListModel> #include <QAbstractListModel>
#include <QDateTime> #include <QDateTime>
#include <QObject> #include <QObject>
#include <QSortFilterProxyModel>
#include <QString> #include <QString>
class ReadReceiptsModel : public QAbstractListModel class ReadReceiptsModel : public QAbstractListModel
{ {
Q_OBJECT Q_OBJECT
Q_PROPERTY(QString eventId READ eventId CONSTANT)
Q_PROPERTY(QString roomId READ roomId CONSTANT)
public: public:
enum Roles enum Roles
{ {
@ -24,6 +22,7 @@ public:
DisplayName, DisplayName,
AvatarUrl, AvatarUrl,
Timestamp, Timestamp,
RawTimestamp,
}; };
explicit ReadReceiptsModel(QString event_id, QString room_id, QObject *parent = nullptr); explicit ReadReceiptsModel(QString event_id, QString room_id, QObject *parent = nullptr);
@ -51,4 +50,26 @@ private:
QVector<QPair<QString, QDateTime>> readReceipts_; QVector<QPair<QString, QDateTime>> readReceipts_;
}; };
class ReadReceiptsProxy : public QSortFilterProxyModel
{
Q_OBJECT
Q_PROPERTY(QString eventId READ eventId CONSTANT)
Q_PROPERTY(QString roomId READ roomId CONSTANT)
public:
explicit ReadReceiptsProxy(QString event_id, QString room_id, QObject *parent = nullptr);
QString eventId() const { return event_id_; }
QString roomId() const { return room_id_; }
bool lessThan(const QModelIndex &source_left, const QModelIndex &source_right) const;
private:
QString event_id_;
QString room_id_;
ReadReceiptsModel model_;
};
#endif // READRECEIPTSMODEL_H #endif // READRECEIPTSMODEL_H

View file

@ -1092,7 +1092,7 @@ TimelineModel::relatedInfo(QString id)
void void
TimelineModel::showReadReceipts(QString id) TimelineModel::showReadReceipts(QString id)
{ {
emit openReadReceiptsDialog(new ReadReceiptsModel{id, roomId(), this}); emit openReadReceiptsDialog(new ReadReceiptsProxy{id, roomId(), this});
} }
void void

View file

@ -349,7 +349,7 @@ signals:
void typingUsersChanged(std::vector<QString> users); void typingUsersChanged(std::vector<QString> users);
void replyChanged(QString reply); void replyChanged(QString reply);
void editChanged(QString reply); void editChanged(QString reply);
void openReadReceiptsDialog(ReadReceiptsModel *rr); void openReadReceiptsDialog(ReadReceiptsProxy *rr);
void paginationInProgressChanged(const bool); void paginationInProgressChanged(const bool);
void newCallEvent(const mtx::events::collections::TimelineEvents &event); void newCallEvent(const mtx::events::collections::TimelineEvents &event);
void scrollToIndex(int index); void scrollToIndex(int index);

View file

@ -206,12 +206,12 @@ TimelineViewManager::TimelineViewManager(CallManager *callManager, ChatPage *par
0, 0,
"InviteesModel", "InviteesModel",
"InviteesModel needs to be instantiated on the C++ side"); "InviteesModel needs to be instantiated on the C++ side");
qmlRegisterUncreatableType<ReadReceiptsModel>( qmlRegisterUncreatableType<ReadReceiptsProxy>(
"im.nheko", "im.nheko",
1, 1,
0, 0,
"ReadReceiptsModel", "ReadReceiptsProxy",
"ReadReceiptsModel needs to be instantiated on the C++ side"); "ReadReceiptsProxy needs to be instantiated on the C++ side");
static auto self = this; static auto self = this;
qmlRegisterSingletonType<MainWindow>( qmlRegisterSingletonType<MainWindow>(