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
|
|
|
|
2018-05-23 16:33:30 +03:00
|
|
|
#include <QAbstractTextDocumentLayout>
|
2018-05-26 22:44:49 +03:00
|
|
|
#include <QApplication>
|
2017-11-30 00:39:35 +03:00
|
|
|
#include <QDateTime>
|
2017-04-06 02:06:42 +03:00
|
|
|
#include <QHBoxLayout>
|
|
|
|
#include <QLabel>
|
2017-11-16 17:33:52 +03:00
|
|
|
#include <QPainter>
|
2017-12-01 16:39:50 +03:00
|
|
|
#include <QSettings>
|
2017-11-16 17:33:52 +03:00
|
|
|
#include <QStyle>
|
|
|
|
#include <QStyleOption>
|
2018-05-23 16:33:30 +03:00
|
|
|
#include <QTextBrowser>
|
|
|
|
#include <QTimer>
|
2017-04-06 02:06:42 +03:00
|
|
|
|
2017-12-01 16:39:50 +03:00
|
|
|
#include "AvatarProvider.h"
|
|
|
|
#include "RoomInfoListItem.h"
|
2018-01-05 01:27:32 +03:00
|
|
|
#include "Utils.h"
|
2017-05-07 17:15:38 +03:00
|
|
|
|
2018-04-29 15:42:40 +03:00
|
|
|
#include "Cache.h"
|
2018-05-08 18:43:56 +03:00
|
|
|
#include "MatrixClient.h"
|
2018-04-29 15:42:40 +03:00
|
|
|
|
2017-10-28 15:46:39 +03:00
|
|
|
class ImageItem;
|
2018-04-27 22:15:44 +03:00
|
|
|
class StickerItem;
|
2017-12-01 16:39:50 +03:00
|
|
|
class AudioItem;
|
2017-12-01 19:28:26 +03:00
|
|
|
class VideoItem;
|
2017-11-28 03:01:37 +03:00
|
|
|
class FileItem;
|
2017-10-28 15:46:39 +03:00
|
|
|
class Avatar;
|
|
|
|
|
2018-05-23 16:33:30 +03:00
|
|
|
class TextLabel : public QTextBrowser
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
|
|
|
|
public:
|
|
|
|
TextLabel(const QString &text, QWidget *parent = 0)
|
|
|
|
: QTextBrowser(parent)
|
|
|
|
{
|
|
|
|
setText(text);
|
|
|
|
setOpenExternalLinks(true);
|
|
|
|
|
|
|
|
// Make it look and feel like an ordinary label.
|
|
|
|
setReadOnly(true);
|
|
|
|
setFrameStyle(QFrame::NoFrame);
|
|
|
|
QPalette pal = palette();
|
|
|
|
pal.setColor(QPalette::Base, Qt::transparent);
|
|
|
|
setPalette(pal);
|
|
|
|
|
|
|
|
// Wrap anywhere but prefer words, adjust minimum height on the fly.
|
|
|
|
setLineWrapMode(QTextEdit::WidgetWidth);
|
|
|
|
setWordWrapMode(QTextOption::WrapAtWordBoundaryOrAnywhere);
|
|
|
|
connect(document()->documentLayout(),
|
|
|
|
&QAbstractTextDocumentLayout::documentSizeChanged,
|
|
|
|
this,
|
|
|
|
&TextLabel::adjustHeight);
|
|
|
|
document()->setDocumentMargin(0);
|
2018-05-26 10:34:34 +03:00
|
|
|
|
|
|
|
setFixedHeight(20);
|
2018-05-23 16:33:30 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void wheelEvent(QWheelEvent *event) override { event->ignore(); }
|
|
|
|
|
|
|
|
private slots:
|
2018-05-26 10:34:34 +03:00
|
|
|
void adjustHeight(const QSizeF &size) { setFixedHeight(size.height()); }
|
2018-05-23 16:33:30 +03:00
|
|
|
};
|
|
|
|
|
2018-05-26 22:44:49 +03:00
|
|
|
class UserProfileFilter : public QObject
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
|
|
|
|
public:
|
|
|
|
explicit UserProfileFilter(const QString &user_id, QLabel *parent)
|
|
|
|
: QObject(parent)
|
|
|
|
, user_id_{user_id}
|
|
|
|
{}
|
|
|
|
|
|
|
|
signals:
|
|
|
|
void hoverOff();
|
|
|
|
void hoverOn();
|
|
|
|
|
|
|
|
protected:
|
|
|
|
bool eventFilter(QObject *obj, QEvent *event)
|
|
|
|
{
|
|
|
|
if (event->type() == QEvent::MouseButtonRelease) {
|
|
|
|
// QMouseEvent *mouseEvent = static_cast<QMouseEvent *>(event);
|
|
|
|
// TODO: Open user profile
|
|
|
|
return true;
|
|
|
|
} else if (event->type() == QEvent::HoverLeave) {
|
|
|
|
emit hoverOff();
|
|
|
|
return true;
|
|
|
|
} else if (event->type() == QEvent::HoverEnter) {
|
|
|
|
emit hoverOn();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return QObject::eventFilter(obj, event);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
QString user_id_;
|
|
|
|
};
|
|
|
|
|
2017-04-25 14:37:54 +03:00
|
|
|
class TimelineItem : 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
|
|
|
TimelineItem(const mtx::events::RoomEvent<mtx::events::msg::Notice> &e,
|
2017-08-26 11:33:26 +03:00
|
|
|
bool with_sender,
|
2018-04-21 16:34:50 +03:00
|
|
|
const QString &room_id,
|
2017-08-26 11:33:26 +03:00
|
|
|
QWidget *parent = 0);
|
2017-12-04 19:41:19 +03:00
|
|
|
TimelineItem(const mtx::events::RoomEvent<mtx::events::msg::Text> &e,
|
2017-08-26 11:33:26 +03:00
|
|
|
bool with_sender,
|
2018-04-21 16:34:50 +03:00
|
|
|
const QString &room_id,
|
2017-08-26 11:33:26 +03:00
|
|
|
QWidget *parent = 0);
|
2017-12-04 19:41:19 +03:00
|
|
|
TimelineItem(const mtx::events::RoomEvent<mtx::events::msg::Emote> &e,
|
2017-09-02 16:47:59 +03:00
|
|
|
bool with_sender,
|
2018-04-21 16:34:50 +03:00
|
|
|
const QString &room_id,
|
2017-09-02 16:47:59 +03:00
|
|
|
QWidget *parent = 0);
|
2017-04-13 04:11:22 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
// For local messages.
|
2017-09-10 12:58:00 +03:00
|
|
|
// m.text & m.emote
|
2017-12-04 19:41:19 +03:00
|
|
|
TimelineItem(mtx::events::MessageType ty,
|
2017-09-03 11:43:45 +03:00
|
|
|
const QString &userid,
|
|
|
|
QString body,
|
|
|
|
bool withSender,
|
2018-04-21 16:34:50 +03:00
|
|
|
const QString &room_id,
|
2017-09-03 11:43:45 +03:00
|
|
|
QWidget *parent = 0);
|
2017-09-10 12:58:00 +03:00
|
|
|
// m.image
|
2018-04-21 16:34:50 +03:00
|
|
|
TimelineItem(ImageItem *item,
|
|
|
|
const QString &userid,
|
|
|
|
bool withSender,
|
|
|
|
const QString &room_id,
|
|
|
|
QWidget *parent = 0);
|
|
|
|
TimelineItem(FileItem *item,
|
|
|
|
const QString &userid,
|
|
|
|
bool withSender,
|
|
|
|
const QString &room_id,
|
|
|
|
QWidget *parent = 0);
|
|
|
|
TimelineItem(AudioItem *item,
|
|
|
|
const QString &userid,
|
|
|
|
bool withSender,
|
|
|
|
const QString &room_id,
|
|
|
|
QWidget *parent = 0);
|
|
|
|
TimelineItem(VideoItem *item,
|
|
|
|
const QString &userid,
|
|
|
|
bool withSender,
|
|
|
|
const QString &room_id,
|
|
|
|
QWidget *parent = 0);
|
2017-04-13 04:11:22 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
TimelineItem(ImageItem *img,
|
2017-12-04 19:41:19 +03:00
|
|
|
const mtx::events::RoomEvent<mtx::events::msg::Image> &e,
|
2017-08-26 11:33:26 +03:00
|
|
|
bool with_sender,
|
2018-04-21 16:34:50 +03:00
|
|
|
const QString &room_id,
|
2017-08-26 11:33:26 +03:00
|
|
|
QWidget *parent);
|
2018-04-27 22:15:44 +03:00
|
|
|
TimelineItem(StickerItem *img,
|
|
|
|
const mtx::events::Sticker &e,
|
|
|
|
bool with_sender,
|
|
|
|
const QString &room_id,
|
|
|
|
QWidget *parent);
|
2017-11-28 03:01:37 +03:00
|
|
|
TimelineItem(FileItem *file,
|
2017-12-04 19:41:19 +03:00
|
|
|
const mtx::events::RoomEvent<mtx::events::msg::File> &e,
|
2017-11-28 03:01:37 +03:00
|
|
|
bool with_sender,
|
2018-04-21 16:34:50 +03:00
|
|
|
const QString &room_id,
|
2017-11-28 03:01:37 +03:00
|
|
|
QWidget *parent);
|
2017-12-01 16:39:50 +03:00
|
|
|
TimelineItem(AudioItem *audio,
|
2017-12-04 19:41:19 +03:00
|
|
|
const mtx::events::RoomEvent<mtx::events::msg::Audio> &e,
|
2017-12-01 16:39:50 +03:00
|
|
|
bool with_sender,
|
2018-04-21 16:34:50 +03:00
|
|
|
const QString &room_id,
|
2017-12-01 19:28:26 +03:00
|
|
|
QWidget *parent);
|
|
|
|
TimelineItem(VideoItem *video,
|
2017-12-04 19:41:19 +03:00
|
|
|
const mtx::events::RoomEvent<mtx::events::msg::Video> &e,
|
2017-12-01 19:28:26 +03:00
|
|
|
bool with_sender,
|
2018-04-21 16:34:50 +03:00
|
|
|
const QString &room_id,
|
2017-12-01 16:39:50 +03:00
|
|
|
QWidget *parent);
|
2017-04-28 14:56:45 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
void setUserAvatar(const QImage &pixmap);
|
2017-11-24 01:10:58 +03:00
|
|
|
DescInfo descriptionMessage() const { return descriptionMsg_; }
|
|
|
|
QString eventId() const { return event_id_; }
|
2018-01-05 16:28:38 +03:00
|
|
|
void setEventId(const QString &event_id) { event_id_ = event_id; }
|
2018-01-14 13:54:17 +03:00
|
|
|
void markReceived();
|
2018-06-17 02:29:19 +03:00
|
|
|
bool isReceived() { return isReceived_; };
|
2018-04-21 16:34:50 +03:00
|
|
|
void setRoomId(QString room_id) { room_id_ = room_id; }
|
2018-06-14 02:28:35 +03:00
|
|
|
void sendReadReceipt() const;
|
2017-06-05 02:14:05 +03:00
|
|
|
|
2018-03-17 22:23:46 +03:00
|
|
|
//! Add a user avatar for this event.
|
|
|
|
void addAvatar();
|
|
|
|
|
2018-06-09 16:03:14 +03:00
|
|
|
signals:
|
|
|
|
void eventRedacted(const QString &event_id);
|
|
|
|
void redactionFailed(const QString &msg);
|
|
|
|
|
2017-11-16 17:33:52 +03:00
|
|
|
protected:
|
|
|
|
void paintEvent(QPaintEvent *event) override;
|
2018-01-03 19:05:49 +03:00
|
|
|
void contextMenuEvent(QContextMenuEvent *event) override;
|
2017-11-16 17:33:52 +03:00
|
|
|
|
2017-04-06 02:06:42 +03:00
|
|
|
private:
|
2017-08-26 11:33:26 +03:00
|
|
|
void init();
|
2018-03-14 22:31:09 +03:00
|
|
|
//! Add a context menu option to save the image of the timeline item.
|
|
|
|
void addSaveImageAction(ImageItem *image);
|
2018-06-30 15:13:15 +03:00
|
|
|
//! Add the reply action in the context menu for widgets that support it.
|
|
|
|
void addReplyAction();
|
2017-06-05 02:14:05 +03:00
|
|
|
|
2017-11-30 00:39:35 +03:00
|
|
|
template<class Widget>
|
2018-04-29 15:42:40 +03:00
|
|
|
void setupLocalWidgetLayout(Widget *widget, const QString &userid, bool withSender);
|
2017-11-30 00:39:35 +03:00
|
|
|
|
2017-12-01 16:39:50 +03:00
|
|
|
template<class Event, class Widget>
|
2018-04-29 15:42:40 +03:00
|
|
|
void setupWidgetLayout(Widget *widget, const Event &event, bool withSender);
|
2017-12-01 16:39:50 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
void generateBody(const QString &body);
|
2018-05-26 22:44:49 +03:00
|
|
|
void generateBody(const QString &user_id, const QString &displayname, const QString &body);
|
2017-08-26 11:33:26 +03:00
|
|
|
void generateTimestamp(const QDateTime &time);
|
2017-04-13 04:11:22 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
void setupAvatarLayout(const QString &userName);
|
|
|
|
void setupSimpleLayout();
|
2017-06-05 02:14:05 +03:00
|
|
|
|
2018-06-17 02:29:19 +03:00
|
|
|
//! Whether or not the event associated with the widget
|
|
|
|
//! has been acknowledged by the server.
|
|
|
|
bool isReceived_ = false;
|
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
QString replaceEmoji(const QString &body);
|
2017-11-24 01:10:58 +03:00
|
|
|
QString event_id_;
|
2018-03-11 18:56:40 +03:00
|
|
|
QString room_id_;
|
2017-04-19 19:38:39 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
DescInfo descriptionMsg_;
|
2017-08-06 18:53:31 +03:00
|
|
|
|
2018-03-11 18:56:40 +03:00
|
|
|
QMenu *contextMenu_;
|
2018-01-03 19:05:49 +03:00
|
|
|
QAction *showReadReceipts_;
|
2018-03-11 18:56:40 +03:00
|
|
|
QAction *markAsRead_;
|
2018-03-17 22:23:46 +03:00
|
|
|
QAction *redactMsg_;
|
2018-06-30 15:13:15 +03:00
|
|
|
QAction *replyMsg_;
|
2018-01-03 19:05:49 +03:00
|
|
|
|
2018-03-17 22:23:46 +03:00
|
|
|
QHBoxLayout *topLayout_ = nullptr;
|
|
|
|
QHBoxLayout *messageLayout_ = nullptr;
|
|
|
|
QVBoxLayout *mainLayout_ = nullptr;
|
|
|
|
QVBoxLayout *headerLayout_ = nullptr;
|
|
|
|
QHBoxLayout *widgetLayout_ = nullptr;
|
2017-06-05 02:14:05 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
Avatar *userAvatar_;
|
2017-04-13 04:11:22 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
QFont font_;
|
2018-03-17 22:23:46 +03:00
|
|
|
QFont usernameFont_;
|
2017-07-15 17:11:46 +03:00
|
|
|
|
2017-08-26 11:33:26 +03:00
|
|
|
QLabel *timestamp_;
|
2018-02-05 04:22:55 +03:00
|
|
|
QLabel *checkmark_;
|
2017-08-26 11:33:26 +03:00
|
|
|
QLabel *userName_;
|
2018-05-23 16:33:30 +03:00
|
|
|
TextLabel *body_;
|
2017-04-06 02:06:42 +03:00
|
|
|
};
|
2017-11-30 00:39:35 +03:00
|
|
|
|
|
|
|
template<class Widget>
|
|
|
|
void
|
2018-04-29 15:42:40 +03:00
|
|
|
TimelineItem::setupLocalWidgetLayout(Widget *widget, const QString &userid, bool withSender)
|
2017-11-30 00:39:35 +03:00
|
|
|
{
|
2018-04-21 16:34:50 +03:00
|
|
|
auto displayName = Cache::displayName(room_id_, userid);
|
2017-11-30 00:39:35 +03:00
|
|
|
auto timestamp = QDateTime::currentDateTime();
|
|
|
|
|
2018-01-05 01:27:32 +03:00
|
|
|
descriptionMsg_ = {"You",
|
|
|
|
userid,
|
2018-04-29 15:42:40 +03:00
|
|
|
QString(" %1").arg(utils::messageDescription<Widget>()),
|
2018-01-05 01:27:32 +03:00
|
|
|
utils::descriptiveTime(timestamp),
|
|
|
|
timestamp};
|
2017-11-30 00:39:35 +03:00
|
|
|
|
|
|
|
generateTimestamp(timestamp);
|
|
|
|
|
2018-03-17 22:23:46 +03:00
|
|
|
widgetLayout_ = new QHBoxLayout;
|
2018-05-26 17:05:57 +03:00
|
|
|
widgetLayout_->setContentsMargins(0, 2, 0, 2);
|
2018-03-17 22:23:46 +03:00
|
|
|
widgetLayout_->addWidget(widget);
|
|
|
|
widgetLayout_->addStretch(1);
|
2018-01-14 13:29:54 +03:00
|
|
|
|
2017-11-30 00:39:35 +03:00
|
|
|
if (withSender) {
|
2018-05-26 22:44:49 +03:00
|
|
|
generateBody(userid, displayName, "");
|
2017-11-30 00:39:35 +03:00
|
|
|
setupAvatarLayout(displayName);
|
2018-01-14 13:29:54 +03:00
|
|
|
|
2018-03-17 22:23:46 +03:00
|
|
|
headerLayout_->addLayout(widgetLayout_);
|
2018-01-14 13:29:54 +03:00
|
|
|
messageLayout_->addLayout(headerLayout_, 1);
|
2017-11-30 00:39:35 +03:00
|
|
|
|
2018-03-25 15:59:47 +03:00
|
|
|
AvatarProvider::resolve(
|
2018-04-21 16:34:50 +03:00
|
|
|
room_id_, userid, this, [this](const QImage &img) { setUserAvatar(img); });
|
2017-11-30 00:39:35 +03:00
|
|
|
} else {
|
|
|
|
setupSimpleLayout();
|
2018-01-14 13:29:54 +03:00
|
|
|
|
2018-03-17 22:23:46 +03:00
|
|
|
messageLayout_->addLayout(widgetLayout_, 1);
|
2017-11-30 00:39:35 +03:00
|
|
|
}
|
|
|
|
|
2018-02-06 07:42:55 +03:00
|
|
|
messageLayout_->addWidget(checkmark_);
|
2018-01-14 13:29:54 +03:00
|
|
|
messageLayout_->addWidget(timestamp_);
|
|
|
|
mainLayout_->addLayout(messageLayout_);
|
2017-11-30 00:39:35 +03:00
|
|
|
}
|
2017-12-01 16:39:50 +03:00
|
|
|
|
|
|
|
template<class Event, class Widget>
|
|
|
|
void
|
2018-04-29 15:42:40 +03:00
|
|
|
TimelineItem::setupWidgetLayout(Widget *widget, const Event &event, bool withSender)
|
2017-12-01 16:39:50 +03:00
|
|
|
{
|
|
|
|
init();
|
|
|
|
|
2017-12-04 19:41:19 +03:00
|
|
|
event_id_ = QString::fromStdString(event.event_id);
|
|
|
|
const auto sender = QString::fromStdString(event.sender);
|
2017-12-01 16:39:50 +03:00
|
|
|
|
2017-12-04 19:41:19 +03:00
|
|
|
auto timestamp = QDateTime::fromMSecsSinceEpoch(event.origin_server_ts);
|
2018-04-21 16:34:50 +03:00
|
|
|
auto displayName = Cache::displayName(room_id_, sender);
|
2017-12-01 16:39:50 +03:00
|
|
|
|
|
|
|
QSettings settings;
|
2017-12-04 19:41:19 +03:00
|
|
|
descriptionMsg_ = {sender == settings.value("auth/user_id") ? "You" : displayName,
|
|
|
|
sender,
|
2018-04-29 15:42:40 +03:00
|
|
|
QString(" %1").arg(utils::messageDescription<Widget>()),
|
2018-01-05 01:27:32 +03:00
|
|
|
utils::descriptiveTime(timestamp),
|
2017-12-26 00:02:33 +03:00
|
|
|
timestamp};
|
2017-12-01 16:39:50 +03:00
|
|
|
|
|
|
|
generateTimestamp(timestamp);
|
|
|
|
|
2018-03-17 22:23:46 +03:00
|
|
|
widgetLayout_ = new QHBoxLayout();
|
2018-05-26 17:05:57 +03:00
|
|
|
widgetLayout_->setContentsMargins(0, 2, 0, 2);
|
2018-03-17 22:23:46 +03:00
|
|
|
widgetLayout_->addWidget(widget);
|
|
|
|
widgetLayout_->addStretch(1);
|
2018-01-14 13:29:54 +03:00
|
|
|
|
2017-12-01 16:39:50 +03:00
|
|
|
if (withSender) {
|
2018-05-26 22:44:49 +03:00
|
|
|
generateBody(sender, displayName, "");
|
2017-12-01 16:39:50 +03:00
|
|
|
setupAvatarLayout(displayName);
|
|
|
|
|
2018-03-17 22:23:46 +03:00
|
|
|
headerLayout_->addLayout(widgetLayout_);
|
2018-01-14 13:29:54 +03:00
|
|
|
messageLayout_->addLayout(headerLayout_, 1);
|
2017-12-01 16:39:50 +03:00
|
|
|
|
2018-03-25 15:59:47 +03:00
|
|
|
AvatarProvider::resolve(
|
2018-04-21 16:34:50 +03:00
|
|
|
room_id_, sender, this, [this](const QImage &img) { setUserAvatar(img); });
|
2017-12-01 16:39:50 +03:00
|
|
|
} else {
|
|
|
|
setupSimpleLayout();
|
2018-01-14 13:29:54 +03:00
|
|
|
|
2018-03-17 22:23:46 +03:00
|
|
|
messageLayout_->addLayout(widgetLayout_, 1);
|
2017-12-01 16:39:50 +03:00
|
|
|
}
|
|
|
|
|
2018-02-06 07:42:55 +03:00
|
|
|
messageLayout_->addWidget(checkmark_);
|
2018-01-14 13:29:54 +03:00
|
|
|
messageLayout_->addWidget(timestamp_);
|
|
|
|
mainLayout_->addLayout(messageLayout_);
|
2017-12-01 16:39:50 +03:00
|
|
|
}
|