matrixion/src/TimelineView.cc

651 lines
21 KiB
C++
Raw Normal View History

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/>.
*/
#include <QApplication>
2017-04-06 02:06:42 +03:00
#include <QDebug>
#include <QFileInfo>
#include <QSettings>
2017-10-28 15:46:39 +03:00
#include <QTimer>
2017-10-27 22:20:33 +03:00
#include "FloatingButton.h"
2017-04-28 14:56:45 +03:00
#include "ImageItem.h"
2017-10-28 15:46:39 +03:00
#include "RoomMessages.h"
#include "ScrollBar.h"
#include "Sync.h"
#include "TimelineItem.h"
#include "TimelineView.h"
2017-04-06 02:06:42 +03:00
namespace events = matrix::events;
2017-08-26 11:33:26 +03:00
namespace msgs = matrix::events::messages;
static bool
isRedactedEvent(const QJsonObject &event)
{
if (event.contains("redacted_because"))
return true;
if (event.contains("unsigned") &&
event.value("unsigned").toObject().contains("redacted_because"))
return true;
return false;
}
TimelineView::TimelineView(const Timeline &timeline,
2017-08-26 11:33:26 +03:00
QSharedPointer<MatrixClient> client,
const QString &room_id,
QWidget *parent)
2017-08-20 13:47:22 +03:00
: QWidget(parent)
2017-11-06 00:04:55 +03:00
, room_id_{room_id}
, client_{client}
2017-04-06 02:06:42 +03:00
{
2017-08-26 11:33:26 +03:00
init();
addEvents(timeline);
2017-04-06 02:06:42 +03:00
}
2017-08-26 11:33:26 +03:00
TimelineView::TimelineView(QSharedPointer<MatrixClient> client,
const QString &room_id,
QWidget *parent)
2017-08-20 13:47:22 +03:00
: QWidget(parent)
2017-11-06 00:04:55 +03:00
, room_id_{room_id}
, client_{client}
{
2017-08-26 11:33:26 +03:00
init();
client_->messages(room_id_, "");
}
2017-08-20 13:47:22 +03:00
void
TimelineView::sliderRangeChanged(int min, int max)
2017-04-06 02:06:42 +03:00
{
2017-08-26 11:33:26 +03:00
Q_UNUSED(min);
2017-08-26 11:33:26 +03:00
if (!scroll_area_->verticalScrollBar()->isVisible()) {
scroll_area_->verticalScrollBar()->setValue(max);
return;
}
// If the scrollbar is close to the bottom and a new message
// is added we move the scrollbar.
if (max - scroll_area_->verticalScrollBar()->value() < SCROLL_BAR_GAP) {
2017-08-26 11:33:26 +03:00
scroll_area_->verticalScrollBar()->setValue(max);
return;
}
2017-10-09 01:32:25 +03:00
int currentHeight = scroll_widget_->size().height();
int diff = currentHeight - oldHeight_;
int newPosition = oldPosition_ + diff;
2017-10-09 01:32:25 +03:00
// Keep the scroll bar to the bottom if it hasn't been activated yet.
if (oldPosition_ == 0 && !scroll_area_->verticalScrollBar()->isVisible())
newPosition = max;
if (lastMessageDirection_ == TimelineDirection::Top)
scroll_area_->verticalScrollBar()->setValue(newPosition);
}
2017-08-20 13:47:22 +03:00
void
TimelineView::fetchHistory()
{
bool hasEnoughMessages = scroll_area_->verticalScrollBar()->isVisible();
2017-08-26 11:33:26 +03:00
if (!hasEnoughMessages && !isTimelineFinished) {
isPaginationInProgress_ = true;
client_->messages(room_id_, prev_batch_token_);
paginationTimer_->start(500);
return;
}
2017-08-26 11:33:26 +03:00
paginationTimer_->stop();
2017-04-06 02:06:42 +03:00
}
2017-08-20 13:47:22 +03:00
void
TimelineView::scrollDown()
2017-04-06 02:06:42 +03:00
{
2017-08-26 11:33:26 +03:00
int current = scroll_area_->verticalScrollBar()->value();
int max = scroll_area_->verticalScrollBar()->maximum();
// The first time we enter the room move the scroll bar to the bottom.
if (!isInitialized) {
scroll_area_->verticalScrollBar()->setValue(max);
isInitialized = true;
return;
}
// If the gap is small enough move the scroll bar down. e.g when a new
// message appears.
if (max - current < SCROLL_BAR_GAP)
scroll_area_->verticalScrollBar()->setValue(max);
}
2017-08-20 13:47:22 +03:00
void
TimelineView::sliderMoved(int position)
{
2017-08-26 11:33:26 +03:00
if (!scroll_area_->verticalScrollBar()->isVisible())
return;
2017-10-27 22:20:33 +03:00
const int maxScroll = scroll_area_->verticalScrollBar()->maximum();
const int currentScroll = scroll_area_->verticalScrollBar()->value();
if (maxScroll - currentScroll > SCROLL_BAR_GAP) {
scrollDownBtn_->show();
scrollDownBtn_->raise();
} else {
scrollDownBtn_->hide();
}
2017-08-26 11:33:26 +03:00
// The scrollbar is high enough so we can start retrieving old events.
if (position < SCROLL_BAR_GAP) {
if (isTimelineFinished)
return;
// Prevent user from moving up when there is pagination in
// progress.
// TODO: Keep a map of the event ids to filter out duplicates.
if (isPaginationInProgress_)
return;
isPaginationInProgress_ = true;
// FIXME: Maybe move this to TimelineViewManager to remove the
// extra calls?
client_->messages(room_id_, prev_batch_token_);
}
}
2017-04-06 02:06:42 +03:00
2017-08-20 13:47:22 +03:00
void
TimelineView::addBackwardsEvents(const QString &room_id, const RoomMessages &msgs)
{
2017-08-26 11:33:26 +03:00
if (room_id_ != room_id)
return;
2017-08-26 11:33:26 +03:00
if (msgs.chunk().count() == 0) {
isTimelineFinished = true;
return;
}
2017-08-26 11:33:26 +03:00
isTimelineFinished = false;
QList<TimelineItem *> items;
// Reset the sender of the first message in the timeline
// cause we're about to insert a new one.
firstSender_.clear();
2017-08-26 11:33:26 +03:00
// Parse in reverse order to determine where we should not show sender's
// name.
2017-09-22 21:34:31 +03:00
auto ii = msgs.chunk().size();
while (ii != 0) {
--ii;
2017-09-22 21:34:31 +03:00
TimelineItem *item =
parseMessageEvent(msgs.chunk().at(ii).toObject(), TimelineDirection::Top);
2017-08-26 11:33:26 +03:00
if (item != nullptr)
items.push_back(item);
}
2017-08-26 11:33:26 +03:00
// Reverse again to render them.
std::reverse(items.begin(), items.end());
2017-08-26 11:33:26 +03:00
oldPosition_ = scroll_area_->verticalScrollBar()->value();
oldHeight_ = scroll_widget_->size().height();
2017-08-26 11:33:26 +03:00
for (const auto &item : items)
addTimelineItem(item, TimelineDirection::Top);
2017-04-28 14:56:45 +03:00
lastMessageDirection_ = TimelineDirection::Top;
QApplication::processEvents();
2017-10-09 01:32:25 +03:00
prev_batch_token_ = msgs.end();
isPaginationInProgress_ = false;
2017-08-26 11:33:26 +03:00
// Exclude the top stretch.
if (!msgs.chunk().isEmpty() && scroll_layout_->count() > 1)
notifyForLastEvent();
2017-08-26 11:33:26 +03:00
// If this batch is the first being rendered (i.e the first and the last
// events originate from this batch), set the last sender.
if (lastSender_.isEmpty() && !items.isEmpty())
lastSender_ = items.constFirst()->descriptionMessage().userid;
}
2017-08-20 13:47:22 +03:00
TimelineItem *
TimelineView::parseMessageEvent(const QJsonObject &event, TimelineDirection direction)
{
2017-08-26 11:33:26 +03:00
events::EventType ty = events::extractEventType(event);
2017-08-26 11:33:26 +03:00
if (ty == events::EventType::RoomMessage) {
events::MessageEventType msg_type = events::extractMessageEventType(event);
2017-08-26 11:33:26 +03:00
if (msg_type == events::MessageEventType::Text) {
events::MessageEvent<msgs::Text> text;
2017-08-26 11:33:26 +03:00
try {
text.deserialize(event);
} catch (const DeserializationException &e) {
qWarning() << e.what() << event;
return nullptr;
}
2017-08-26 11:33:26 +03:00
if (isDuplicate(text.eventId()))
return nullptr;
2017-08-26 11:33:26 +03:00
eventIds_[text.eventId()] = true;
QString txnid = text.unsignedData().transactionId();
2017-11-15 19:42:21 +03:00
if (!txnid.isEmpty() &&
isPendingMessage(txnid, text.sender(), local_user_)) {
removePendingMessage(txnid);
2017-08-26 11:33:26 +03:00
return nullptr;
}
2017-08-26 11:33:26 +03:00
auto with_sender = isSenderRendered(text.sender(), direction);
2017-08-13 21:04:43 +03:00
2017-08-26 11:33:26 +03:00
updateLastSender(text.sender(), direction);
2017-04-28 14:56:45 +03:00
2017-08-26 11:33:26 +03:00
return createTimelineItem(text, with_sender);
} else if (msg_type == events::MessageEventType::Notice) {
events::MessageEvent<msgs::Notice> notice;
2017-04-28 14:56:45 +03:00
2017-08-26 11:33:26 +03:00
try {
notice.deserialize(event);
} catch (const DeserializationException &e) {
qWarning() << e.what() << event;
return nullptr;
}
2017-04-28 14:56:45 +03:00
2017-08-26 11:33:26 +03:00
if (isDuplicate(notice.eventId()))
return nullptr;
2017-08-26 11:33:26 +03:00
eventIds_[notice.eventId()] = true;
2017-08-26 11:33:26 +03:00
auto with_sender = isSenderRendered(notice.sender(), direction);
2017-08-13 21:04:43 +03:00
2017-08-26 11:33:26 +03:00
updateLastSender(notice.sender(), direction);
2017-08-26 11:33:26 +03:00
return createTimelineItem(notice, with_sender);
} else if (msg_type == events::MessageEventType::Image) {
events::MessageEvent<msgs::Image> img;
2017-08-26 11:33:26 +03:00
try {
img.deserialize(event);
} catch (const DeserializationException &e) {
qWarning() << e.what() << event;
return nullptr;
}
2017-08-26 11:33:26 +03:00
if (isDuplicate(img.eventId()))
return nullptr;
2017-08-26 11:33:26 +03:00
eventIds_[img.eventId()] = true;
QString txnid = img.unsignedData().transactionId();
2017-11-15 19:42:21 +03:00
if (!txnid.isEmpty() &&
isPendingMessage(txnid, img.sender(), local_user_)) {
removePendingMessage(txnid);
2017-09-10 12:58:00 +03:00
return nullptr;
}
2017-08-26 11:33:26 +03:00
auto with_sender = isSenderRendered(img.sender(), direction);
2017-08-13 21:04:43 +03:00
2017-08-26 11:33:26 +03:00
updateLastSender(img.sender(), direction);
2017-08-26 11:33:26 +03:00
return createTimelineItem(img, with_sender);
} else if (msg_type == events::MessageEventType::Emote) {
events::MessageEvent<msgs::Emote> emote;
try {
emote.deserialize(event);
} catch (const DeserializationException &e) {
qWarning() << e.what() << event;
return nullptr;
}
if (isDuplicate(emote.eventId()))
return nullptr;
eventIds_[emote.eventId()] = true;
QString txnid = emote.unsignedData().transactionId();
2017-11-15 19:42:21 +03:00
if (!txnid.isEmpty() &&
isPendingMessage(txnid, emote.sender(), local_user_)) {
removePendingMessage(txnid);
return nullptr;
}
auto with_sender = isSenderRendered(emote.sender(), direction);
updateLastSender(emote.sender(), direction);
return createTimelineItem(emote, with_sender);
2017-08-26 11:33:26 +03:00
} else if (msg_type == events::MessageEventType::Unknown) {
// TODO Handle redacted messages.
// Silenced for now.
if (!isRedactedEvent(event))
qWarning() << "Unknown message type" << event;
2017-08-26 11:33:26 +03:00
return nullptr;
}
}
2017-08-26 11:33:26 +03:00
return nullptr;
}
2017-08-20 13:47:22 +03:00
int
TimelineView::addEvents(const Timeline &timeline)
{
2017-08-26 11:33:26 +03:00
int message_count = 0;
2017-08-26 11:33:26 +03:00
QSettings settings;
QString localUser = settings.value("auth/user_id").toString();
2017-08-26 11:33:26 +03:00
for (const auto &event : timeline.events()) {
TimelineItem *item = parseMessageEvent(event.toObject(), TimelineDirection::Bottom);
2017-08-26 11:33:26 +03:00
if (item != nullptr) {
addTimelineItem(item, TimelineDirection::Bottom);
2017-08-26 11:33:26 +03:00
if (localUser != event.toObject().value("sender").toString())
message_count += 1;
}
}
lastMessageDirection_ = TimelineDirection::Bottom;
QApplication::processEvents();
2017-08-26 11:33:26 +03:00
if (isInitialSync) {
prev_batch_token_ = timeline.previousBatch();
isInitialSync = false;
2017-06-05 19:54:45 +03:00
2017-08-26 11:33:26 +03:00
client_->messages(room_id_, prev_batch_token_);
}
2017-06-05 19:54:45 +03:00
2017-08-26 11:33:26 +03:00
// Exclude the top stretch.
if (!timeline.events().isEmpty() && scroll_layout_->count() > 1)
notifyForLastEvent();
2017-08-26 11:33:26 +03:00
return message_count;
2017-04-06 02:06:42 +03:00
}
2017-08-20 13:47:22 +03:00
void
TimelineView::init()
2017-04-06 02:06:42 +03:00
{
QSettings settings;
local_user_ = settings.value("auth/user_id").toString();
2017-10-27 22:20:33 +03:00
QIcon icon;
icon.addFile(":/icons/icons/ui/angle-arrow-down.png");
scrollDownBtn_ = new FloatingButton(icon, this);
scrollDownBtn_->setBackgroundColor(QColor("#F5F5F5"));
scrollDownBtn_->setForegroundColor(QColor("black"));
scrollDownBtn_->hide();
connect(scrollDownBtn_, &QPushButton::clicked, this, [=]() {
const int max = scroll_area_->verticalScrollBar()->maximum();
scroll_area_->verticalScrollBar()->setValue(max);
});
2017-08-26 11:33:26 +03:00
top_layout_ = new QVBoxLayout(this);
top_layout_->setSpacing(0);
top_layout_->setMargin(0);
2017-04-06 02:06:42 +03:00
2017-08-26 11:33:26 +03:00
scroll_area_ = new QScrollArea(this);
scroll_area_->setWidgetResizable(true);
scroll_area_->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
2017-04-06 02:06:42 +03:00
2017-08-26 11:33:26 +03:00
scrollbar_ = new ScrollBar(scroll_area_);
scroll_area_->setVerticalScrollBar(scrollbar_);
2017-05-24 22:45:13 +03:00
2017-10-01 18:15:23 +03:00
scroll_widget_ = new QWidget(this);
2017-04-06 02:06:42 +03:00
2017-10-01 18:15:23 +03:00
scroll_layout_ = new QVBoxLayout(scroll_widget_);
2017-08-26 11:33:26 +03:00
scroll_layout_->addStretch(1);
scroll_layout_->setSpacing(0);
scroll_layout_->setObjectName("timelinescrollarea");
2017-04-06 02:06:42 +03:00
2017-08-26 11:33:26 +03:00
scroll_area_->setWidget(scroll_widget_);
2017-04-06 02:06:42 +03:00
2017-08-26 11:33:26 +03:00
top_layout_->addWidget(scroll_area_);
2017-04-06 02:06:42 +03:00
2017-08-26 11:33:26 +03:00
setLayout(top_layout_);
2017-04-06 02:06:42 +03:00
2017-08-26 11:33:26 +03:00
paginationTimer_ = new QTimer(this);
connect(paginationTimer_, &QTimer::timeout, this, &TimelineView::fetchHistory);
2017-08-26 11:33:26 +03:00
connect(client_.data(),
&MatrixClient::messagesRetrieved,
this,
&TimelineView::addBackwardsEvents);
2017-08-26 11:33:26 +03:00
connect(scroll_area_->verticalScrollBar(),
SIGNAL(valueChanged(int)),
this,
SLOT(sliderMoved(int)));
connect(scroll_area_->verticalScrollBar(),
SIGNAL(rangeChanged(int, int)),
this,
SLOT(sliderRangeChanged(int, int)));
2017-04-06 02:06:42 +03:00
}
2017-08-20 13:47:22 +03:00
void
TimelineView::updateLastSender(const QString &user_id, TimelineDirection direction)
{
2017-08-26 11:33:26 +03:00
if (direction == TimelineDirection::Bottom)
lastSender_ = user_id;
else
firstSender_ = user_id;
}
2017-08-20 13:47:22 +03:00
bool
TimelineView::isSenderRendered(const QString &user_id, TimelineDirection direction)
{
2017-08-26 11:33:26 +03:00
if (direction == TimelineDirection::Bottom)
return lastSender_ != user_id;
else
return firstSender_ != user_id;
}
2017-08-20 13:47:22 +03:00
TimelineItem *
2017-08-26 11:33:26 +03:00
TimelineView::createTimelineItem(const events::MessageEvent<msgs::Image> &event, bool with_sender)
2017-04-28 14:56:45 +03:00
{
2017-08-26 11:33:26 +03:00
auto image = new ImageItem(client_, event);
auto item = new TimelineItem(image, event, with_sender, scroll_widget_);
2017-08-26 11:33:26 +03:00
return item;
2017-04-28 14:56:45 +03:00
}
2017-08-20 13:47:22 +03:00
TimelineItem *
2017-08-26 11:33:26 +03:00
TimelineView::createTimelineItem(const events::MessageEvent<msgs::Notice> &event, bool with_sender)
{
2017-08-26 11:33:26 +03:00
TimelineItem *item = new TimelineItem(event, with_sender, scroll_widget_);
return item;
}
2017-08-20 13:47:22 +03:00
TimelineItem *
2017-08-26 11:33:26 +03:00
TimelineView::createTimelineItem(const events::MessageEvent<msgs::Text> &event, bool with_sender)
2017-04-06 02:06:42 +03:00
{
2017-08-26 11:33:26 +03:00
TimelineItem *item = new TimelineItem(event, with_sender, scroll_widget_);
return item;
}
TimelineItem *
TimelineView::createTimelineItem(const events::MessageEvent<msgs::Emote> &event, bool with_sender)
{
TimelineItem *item = new TimelineItem(event, with_sender, scroll_widget_);
return item;
}
2017-08-20 13:47:22 +03:00
void
TimelineView::addTimelineItem(TimelineItem *item, TimelineDirection direction)
{
2017-08-26 11:33:26 +03:00
if (direction == TimelineDirection::Bottom)
scroll_layout_->addWidget(item);
else
scroll_layout_->insertWidget(1, item);
2017-04-06 02:06:42 +03:00
}
2017-08-20 13:47:22 +03:00
void
TimelineView::updatePendingMessage(int txn_id, QString event_id)
{
if (pending_msgs_.head().txn_id == txn_id) { // We haven't received it yet
2017-11-15 19:42:21 +03:00
auto msg = pending_msgs_.dequeue();
msg.event_id = event_id;
pending_sent_msgs_.append(msg);
2017-08-26 11:33:26 +03:00
}
sendNextPendingMessage();
}
2017-08-20 13:47:22 +03:00
void
TimelineView::addUserMessage(matrix::events::MessageEventType ty, const QString &body)
{
2017-08-26 11:33:26 +03:00
QSettings settings;
auto user_id = settings.value("auth/user_id").toString();
2017-08-26 11:33:26 +03:00
auto with_sender = lastSender_ != user_id;
TimelineItem *view_item = new TimelineItem(ty, user_id, body, with_sender, scroll_widget_);
2017-08-26 11:33:26 +03:00
scroll_layout_->addWidget(view_item);
lastMessageDirection_ = TimelineDirection::Bottom;
QApplication::processEvents();
2017-10-09 01:32:25 +03:00
2017-08-26 11:33:26 +03:00
lastSender_ = user_id;
int txn_id = client_->incrementTransactionId();
PendingMessage message(ty, txn_id, body, "", "", view_item);
handleNewUserMessage(message);
}
2017-09-10 12:58:00 +03:00
void
TimelineView::addUserMessage(const QString &url, const QString &filename)
2017-09-10 12:58:00 +03:00
{
QSettings settings;
auto user_id = settings.value("auth/user_id").toString();
auto with_sender = lastSender_ != user_id;
auto image = new ImageItem(client_, url, filename, this);
TimelineItem *view_item = new TimelineItem(image, user_id, with_sender, scroll_widget_);
scroll_layout_->addWidget(view_item);
lastMessageDirection_ = TimelineDirection::Bottom;
QApplication::processEvents();
2017-10-09 01:32:25 +03:00
2017-09-10 12:58:00 +03:00
lastSender_ = user_id;
int txn_id = client_->incrementTransactionId();
2017-11-15 19:42:21 +03:00
PendingMessage message(
matrix::events::MessageEventType::Image, txn_id, url, filename, "", view_item);
handleNewUserMessage(message);
}
void
TimelineView::handleNewUserMessage(PendingMessage msg)
{
pending_msgs_.enqueue(msg);
if (pending_msgs_.size() == 1 && pending_sent_msgs_.size() == 0)
sendNextPendingMessage();
}
void
TimelineView::sendNextPendingMessage()
{
if (pending_msgs_.size() == 0)
return;
PendingMessage &m = pending_msgs_.head();
switch (m.ty) {
case matrix::events::MessageEventType::Image:
2017-11-15 19:42:21 +03:00
client_->sendRoomMessage(
m.ty, m.txn_id, room_id_, QFileInfo(m.filename).fileName(), m.body);
break;
default:
client_->sendRoomMessage(m.ty, m.txn_id, room_id_, m.body);
break;
}
2017-09-10 12:58:00 +03:00
}
2017-08-20 13:47:22 +03:00
void
TimelineView::notifyForLastEvent()
{
2017-08-26 11:33:26 +03:00
auto lastItem = scroll_layout_->itemAt(scroll_layout_->count() - 1);
auto *lastTimelineItem = qobject_cast<TimelineItem *>(lastItem->widget());
2017-08-26 11:33:26 +03:00
if (lastTimelineItem)
emit updateLastTimelineMessage(room_id_, lastTimelineItem->descriptionMessage());
else
qWarning() << "Cast to TimelineView failed" << room_id_;
}
2017-09-10 12:58:00 +03:00
bool
TimelineView::isPendingMessage(const QString &txnid,
2017-09-10 12:58:00 +03:00
const QString &sender,
const QString &local_userid)
{
if (sender != local_userid)
return false;
for (const auto &msg : pending_msgs_) {
if (QString::number(msg.txn_id) == txnid)
return true;
}
for (const auto &msg : pending_sent_msgs_) {
if (QString::number(msg.txn_id) == txnid)
2017-09-10 12:58:00 +03:00
return true;
}
return false;
}
void
TimelineView::removePendingMessage(const QString &txnid)
2017-09-10 12:58:00 +03:00
{
for (auto it = pending_sent_msgs_.begin(); it != pending_sent_msgs_.end(); ++it) {
if (QString::number(it->txn_id) == txnid) {
int index = std::distance(pending_sent_msgs_.begin(), it);
pending_sent_msgs_.removeAt(index);
return;
}
}
for (auto it = pending_msgs_.begin(); it != pending_msgs_.end(); ++it) {
if (QString::number(it->txn_id) == txnid) {
int index = std::distance(pending_msgs_.begin(), it);
2017-09-10 12:58:00 +03:00
pending_msgs_.removeAt(index);
return;
2017-09-10 12:58:00 +03:00
}
}
}
void
TimelineView::handleFailedMessage(int txnid)
{
Q_UNUSED(txnid);
// Note: We do this even if the message has already been echoed.
QTimer::singleShot(500, this, SLOT(sendNextPendingMessage()));
}
void
TimelineView::paintEvent(QPaintEvent *)
{
QStyleOption opt;
opt.init(this);
QPainter p(this);
style()->drawPrimitive(QStyle::PE_Widget, &opt, &p, this);
}