summary refs log tree commit diff
path: root/src/TimelineView.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/TimelineView.cc')
-rw-r--r--src/TimelineView.cc190
1 files changed, 190 insertions, 0 deletions
diff --git a/src/TimelineView.cc b/src/TimelineView.cc
new file mode 100644

index 00000000..00a42557 --- /dev/null +++ b/src/TimelineView.cc
@@ -0,0 +1,190 @@ +/* + * 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/>. + */ + +#include <QDebug> +#include <QScrollBar> +#include <QSettings> +#include <QtWidgets/QLabel> +#include <QtWidgets/QSpacerItem> + +#include "TimelineItem.h" +#include "TimelineView.h" +#include "TimelineViewManager.h" + +TimelineView::TimelineView(const QList<Event> &events, QWidget *parent) + : QWidget(parent) +{ + init(); + addEvents(events); +} + +TimelineView::TimelineView(QWidget *parent) + : QWidget(parent) +{ + init(); +} + +void TimelineView::clear() +{ + for (const auto msg : scroll_layout_->children()) + msg->deleteLater(); +} + +void TimelineView::sliderRangeChanged(int min, int max) +{ + Q_UNUSED(min); + scroll_area_->verticalScrollBar()->setValue(max); +} + +int TimelineView::addEvents(const QList<Event> &events) +{ + QSettings settings; + auto local_user = settings.value("auth/user_id").toString(); + + int message_count = 0; + + for (const auto &event : events) { + if (event.type() == "m.room.message") { + auto msg_type = event.content().value("msgtype").toString(); + + if (isPendingMessage(event, local_user)) { + removePendingMessage(event); + continue; + } + + if (msg_type == "m.text" || msg_type == "m.notice") { + auto with_sender = last_sender_ != event.sender(); + auto color = TimelineViewManager::getUserColor(event.sender()); + + addHistoryItem(event, color, with_sender); + last_sender_ = event.sender(); + + message_count += 1; + } + } + } + + return message_count; +} + +void TimelineView::init() +{ + top_layout_ = new QVBoxLayout(this); + top_layout_->setSpacing(0); + top_layout_->setMargin(0); + + scroll_area_ = new QScrollArea(this); + scroll_area_->setWidgetResizable(true); + scroll_area_->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + + scroll_widget_ = new QWidget(); + + scroll_layout_ = new QVBoxLayout(); + scroll_layout_->addStretch(1); + scroll_layout_->setSpacing(0); + + scroll_widget_->setLayout(scroll_layout_); + + scroll_area_->setWidget(scroll_widget_); + + top_layout_->addWidget(scroll_area_); + + setLayout(top_layout_); + + connect(scroll_area_->verticalScrollBar(), + SIGNAL(rangeChanged(int, int)), + this, + SLOT(sliderRangeChanged(int, int))); +} + +void TimelineView::addHistoryItem(const Event &event, const QString &color, bool with_sender) +{ + TimelineItem *item = new TimelineItem(event, with_sender, color, scroll_widget_); + scroll_layout_->addWidget(item); +} + +void TimelineView::updatePendingMessage(int txn_id, QString event_id) +{ + for (auto &msg : pending_msgs_) { + if (msg.txn_id == txn_id) { + msg.event_id = event_id; + break; + } + } +} + +bool TimelineView::isPendingMessage(const Event &event, const QString &userid) +{ + if (event.sender() != userid || event.type() != "m.room.message") + return false; + + auto msgtype = event.content().value("msgtype").toString(); + auto body = event.content().value("body").toString(); + + // FIXME: should contain more checks later on for other types of messages. + if (msgtype != "m.text") + return false; + + for (const auto &msg : pending_msgs_) { + if (msg.event_id == event.eventId() || msg.body == body) + return true; + } + + return false; +} + +void TimelineView::removePendingMessage(const Event &event) +{ + auto body = event.content().value("body").toString(); + + for (auto it = pending_msgs_.begin(); it != pending_msgs_.end(); it++) { + int index = std::distance(pending_msgs_.begin(), it); + + if (it->event_id == event.eventId() || it->body == body) { + pending_msgs_.removeAt(index); + break; + } + } +} + +void TimelineView::addUserTextMessage(const QString &body, int txn_id) +{ + QSettings settings; + auto user_id = settings.value("auth/user_id").toString(); + + auto with_sender = last_sender_ != user_id; + auto color = TimelineViewManager::getUserColor(user_id); + + TimelineItem *view_item; + + if (with_sender) + view_item = new TimelineItem(user_id, color, body, scroll_widget_); + else + view_item = new TimelineItem(body, scroll_widget_); + + scroll_layout_->addWidget(view_item); + + last_sender_ = user_id; + + PendingMessage message(txn_id, body, "", view_item); + + pending_msgs_.push_back(message); +} + +TimelineView::~TimelineView() +{ +}