mirror of https://github.com/procxx/kepka.git
Some parts from history module moved to history_[item,media,message].
This commit is contained in:
parent
90a4b66366
commit
d277b0d4bb
|
@ -21,6 +21,11 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "core/basic_types.h"
|
#include "core/basic_types.h"
|
||||||
|
#include "history.h"
|
||||||
|
#include "history/history_item.h"
|
||||||
|
#include "history/history_media.h"
|
||||||
|
#include "history/history_message.h"
|
||||||
|
#include "layout.h"
|
||||||
|
|
||||||
class AppClass;
|
class AppClass;
|
||||||
class MainWindow;
|
class MainWindow;
|
||||||
|
@ -28,9 +33,6 @@ class MainWidget;
|
||||||
class ApiWrap;
|
class ApiWrap;
|
||||||
class FileUploader;
|
class FileUploader;
|
||||||
|
|
||||||
#include "history.h"
|
|
||||||
#include "layout.h"
|
|
||||||
|
|
||||||
using HistoryItemsMap = OrderedSet<HistoryItem*>;
|
using HistoryItemsMap = OrderedSet<HistoryItem*>;
|
||||||
using PhotoItems = QHash<PhotoData*, HistoryItemsMap>;
|
using PhotoItems = QHash<PhotoData*, HistoryItemsMap>;
|
||||||
using DocumentItems = QHash<DocumentData*, HistoryItemsMap>;
|
using DocumentItems = QHash<DocumentData*, HistoryItemsMap>;
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,889 @@
|
||||||
|
/*
|
||||||
|
This file is part of Telegram Desktop,
|
||||||
|
the official desktop version of Telegram messaging app, see https://telegram.org
|
||||||
|
|
||||||
|
Telegram Desktop 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.
|
||||||
|
|
||||||
|
It 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.
|
||||||
|
|
||||||
|
In addition, as a special exception, the copyright holders give permission
|
||||||
|
to link the code of portions of this program with the OpenSSL library.
|
||||||
|
|
||||||
|
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
|
||||||
|
Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
|
||||||
|
*/
|
||||||
|
#include "stdafx.h"
|
||||||
|
#include "history/history_item.h"
|
||||||
|
|
||||||
|
#include "lang.h"
|
||||||
|
#include "mainwidget.h"
|
||||||
|
#include "history/history_service_layout.h"
|
||||||
|
#include "media/media_clip_reader.h"
|
||||||
|
#include "styles/style_dialogs.h"
|
||||||
|
#include "fileuploader.h"
|
||||||
|
|
||||||
|
class ReplyMarkupClickHandler : public LeftButtonClickHandler {
|
||||||
|
public:
|
||||||
|
ReplyMarkupClickHandler(const HistoryItem *item, int row, int col) : _itemId(item->fullId()), _row(row), _col(col) {
|
||||||
|
}
|
||||||
|
|
||||||
|
QString tooltip() const override {
|
||||||
|
return _fullDisplayed ? QString() : buttonText();
|
||||||
|
}
|
||||||
|
|
||||||
|
void setFullDisplayed(bool full) {
|
||||||
|
_fullDisplayed = full;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Copy to clipboard support.
|
||||||
|
void copyToClipboard() const override {
|
||||||
|
if (auto button = getButton()) {
|
||||||
|
if (button->type == HistoryMessageReplyMarkup::Button::Type::Url) {
|
||||||
|
auto url = QString::fromUtf8(button->data);
|
||||||
|
if (!url.isEmpty()) {
|
||||||
|
QApplication::clipboard()->setText(url);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
QString copyToClipboardContextItemText() const override {
|
||||||
|
if (auto button = getButton()) {
|
||||||
|
if (button->type == HistoryMessageReplyMarkup::Button::Type::Url) {
|
||||||
|
return lang(lng_context_copy_link);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return QString();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Finds the corresponding button in the items markup struct.
|
||||||
|
// If the button is not found it returns nullptr.
|
||||||
|
// Note: it is possible that we will point to the different button
|
||||||
|
// than the one was used when constructing the handler, but not a big deal.
|
||||||
|
const HistoryMessageReplyMarkup::Button *getButton() const {
|
||||||
|
if (auto item = App::histItemById(_itemId)) {
|
||||||
|
if (auto markup = item->Get<HistoryMessageReplyMarkup>()) {
|
||||||
|
if (_row < markup->rows.size()) {
|
||||||
|
auto &row = markup->rows.at(_row);
|
||||||
|
if (_col < row.size()) {
|
||||||
|
return &row.at(_col);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
// We hold only FullMsgId, not HistoryItem*, because all click handlers
|
||||||
|
// are activated async and the item may be already destroyed.
|
||||||
|
void setMessageId(const FullMsgId &msgId) {
|
||||||
|
_itemId = msgId;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void onClickImpl() const override {
|
||||||
|
if (auto item = App::histItemById(_itemId)) {
|
||||||
|
App::activateBotCommand(item, _row, _col);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
FullMsgId _itemId;
|
||||||
|
int _row, _col;
|
||||||
|
bool _fullDisplayed = true;
|
||||||
|
|
||||||
|
// Returns the full text of the corresponding button.
|
||||||
|
QString buttonText() const {
|
||||||
|
if (auto button = getButton()) {
|
||||||
|
return button->text;
|
||||||
|
}
|
||||||
|
return QString();
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
ReplyKeyboard::ReplyKeyboard(const HistoryItem *item, StylePtr &&s)
|
||||||
|
: _item(item)
|
||||||
|
, _a_selected(animation(this, &ReplyKeyboard::step_selected))
|
||||||
|
, _st(std_::forward<StylePtr>(s)) {
|
||||||
|
if (auto markup = item->Get<HistoryMessageReplyMarkup>()) {
|
||||||
|
_rows.reserve(markup->rows.size());
|
||||||
|
for (int i = 0, l = markup->rows.size(); i != l; ++i) {
|
||||||
|
auto &row = markup->rows.at(i);
|
||||||
|
int s = row.size();
|
||||||
|
ButtonRow newRow(s, Button());
|
||||||
|
for (int j = 0; j != s; ++j) {
|
||||||
|
auto &button = newRow[j];
|
||||||
|
auto str = row.at(j).text;
|
||||||
|
button.type = row.at(j).type;
|
||||||
|
button.link = MakeShared<ReplyMarkupClickHandler>(item, i, j);
|
||||||
|
button.text.setText(_st->textFont(), textOneLine(str), _textPlainOptions);
|
||||||
|
button.characters = str.isEmpty() ? 1 : str.size();
|
||||||
|
}
|
||||||
|
_rows.push_back(newRow);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplyKeyboard::updateMessageId() {
|
||||||
|
auto msgId = _item->fullId();
|
||||||
|
for_const (auto &row, _rows) {
|
||||||
|
for_const (auto &button, row) {
|
||||||
|
button.link->setMessageId(msgId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplyKeyboard::resize(int width, int height) {
|
||||||
|
_width = width;
|
||||||
|
|
||||||
|
auto markup = _item->Get<HistoryMessageReplyMarkup>();
|
||||||
|
float64 y = 0, buttonHeight = _rows.isEmpty() ? _st->buttonHeight() : (float64(height + _st->buttonSkip()) / _rows.size());
|
||||||
|
for (auto &row : _rows) {
|
||||||
|
int s = row.size();
|
||||||
|
|
||||||
|
int widthForButtons = _width - ((s - 1) * _st->buttonSkip());
|
||||||
|
int widthForText = widthForButtons;
|
||||||
|
int widthOfText = 0;
|
||||||
|
int maxMinButtonWidth = 0;
|
||||||
|
for_const (auto &button, row) {
|
||||||
|
widthOfText += qMax(button.text.maxWidth(), 1);
|
||||||
|
int minButtonWidth = _st->minButtonWidth(button.type);
|
||||||
|
widthForText -= minButtonWidth;
|
||||||
|
accumulate_max(maxMinButtonWidth, minButtonWidth);
|
||||||
|
}
|
||||||
|
bool exact = (widthForText == widthOfText);
|
||||||
|
bool enough = (widthForButtons - s * maxMinButtonWidth) >= widthOfText;
|
||||||
|
|
||||||
|
float64 x = 0;
|
||||||
|
for (Button &button : row) {
|
||||||
|
int buttonw = qMax(button.text.maxWidth(), 1);
|
||||||
|
float64 textw = buttonw, minw = _st->minButtonWidth(button.type);
|
||||||
|
float64 w = textw;
|
||||||
|
if (exact) {
|
||||||
|
w += minw;
|
||||||
|
} else if (enough) {
|
||||||
|
w = (widthForButtons / float64(s));
|
||||||
|
textw = w - minw;
|
||||||
|
} else {
|
||||||
|
textw = (widthForText / float64(s));
|
||||||
|
w = minw + textw;
|
||||||
|
accumulate_max(w, 2 * float64(_st->buttonPadding()));
|
||||||
|
}
|
||||||
|
|
||||||
|
int rectx = static_cast<int>(std::floor(x));
|
||||||
|
int rectw = static_cast<int>(std::floor(x + w)) - rectx;
|
||||||
|
button.rect = QRect(rectx, qRound(y), rectw, qRound(buttonHeight - _st->buttonSkip()));
|
||||||
|
if (rtl()) button.rect.setX(_width - button.rect.x() - button.rect.width());
|
||||||
|
x += w + _st->buttonSkip();
|
||||||
|
|
||||||
|
button.link->setFullDisplayed(textw >= buttonw);
|
||||||
|
}
|
||||||
|
y += buttonHeight;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ReplyKeyboard::isEnoughSpace(int width, const style::botKeyboardButton &st) const {
|
||||||
|
for_const (auto &row, _rows) {
|
||||||
|
int s = row.size();
|
||||||
|
int widthLeft = width - ((s - 1) * st.margin + s * 2 * st.padding);
|
||||||
|
for_const (auto &button, row) {
|
||||||
|
widthLeft -= qMax(button.text.maxWidth(), 1);
|
||||||
|
if (widthLeft < 0) {
|
||||||
|
if (row.size() > 3) {
|
||||||
|
return false;
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplyKeyboard::setStyle(StylePtr &&st) {
|
||||||
|
_st = std_::move(st);
|
||||||
|
}
|
||||||
|
|
||||||
|
int ReplyKeyboard::naturalWidth() const {
|
||||||
|
auto result = 0;
|
||||||
|
for_const (auto &row, _rows) {
|
||||||
|
auto maxMinButtonWidth = 0;
|
||||||
|
for_const (auto &button, row) {
|
||||||
|
accumulate_max(maxMinButtonWidth, _st->minButtonWidth(button.type));
|
||||||
|
}
|
||||||
|
auto rowMaxButtonWidth = 0;
|
||||||
|
for_const (auto &button, row) {
|
||||||
|
accumulate_max(rowMaxButtonWidth, qMax(button.text.maxWidth(), 1) + maxMinButtonWidth);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto rowSize = row.size();
|
||||||
|
accumulate_max(result, rowSize * rowMaxButtonWidth + (rowSize - 1) * _st->buttonSkip());
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
int ReplyKeyboard::naturalHeight() const {
|
||||||
|
return (_rows.size() - 1) * _st->buttonSkip() + _rows.size() * _st->buttonHeight();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplyKeyboard::paint(Painter &p, const QRect &clip) const {
|
||||||
|
t_assert(_st != nullptr);
|
||||||
|
t_assert(_width > 0);
|
||||||
|
|
||||||
|
_st->startPaint(p);
|
||||||
|
for_const (auto &row, _rows) {
|
||||||
|
for_const (auto &button, row) {
|
||||||
|
QRect rect(button.rect);
|
||||||
|
if (rect.y() >= clip.y() + clip.height()) return;
|
||||||
|
if (rect.y() + rect.height() < clip.y()) continue;
|
||||||
|
|
||||||
|
// just ignore the buttons that didn't layout well
|
||||||
|
if (rect.x() + rect.width() > _width) break;
|
||||||
|
|
||||||
|
_st->paintButton(p, button);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ClickHandlerPtr ReplyKeyboard::getState(int x, int y) const {
|
||||||
|
t_assert(_width > 0);
|
||||||
|
|
||||||
|
for_const (auto &row, _rows) {
|
||||||
|
for_const (auto &button, row) {
|
||||||
|
QRect rect(button.rect);
|
||||||
|
|
||||||
|
// just ignore the buttons that didn't layout well
|
||||||
|
if (rect.x() + rect.width() > _width) break;
|
||||||
|
|
||||||
|
if (rect.contains(x, y)) {
|
||||||
|
return button.link;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ClickHandlerPtr();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplyKeyboard::clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) {
|
||||||
|
if (!p) return;
|
||||||
|
|
||||||
|
bool startAnimation = false;
|
||||||
|
for (int i = 0, rows = _rows.size(); i != rows; ++i) {
|
||||||
|
auto &row = _rows.at(i);
|
||||||
|
for (int j = 0, cols = row.size(); j != cols; ++j) {
|
||||||
|
if (row.at(j).link == p) {
|
||||||
|
bool startAnimation = _animations.isEmpty();
|
||||||
|
|
||||||
|
int indexForAnimation = i * MatrixRowShift + j + 1;
|
||||||
|
if (!active) {
|
||||||
|
indexForAnimation = -indexForAnimation;
|
||||||
|
}
|
||||||
|
|
||||||
|
_animations.remove(-indexForAnimation);
|
||||||
|
if (!_animations.contains(indexForAnimation)) {
|
||||||
|
_animations.insert(indexForAnimation, getms());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (startAnimation && !_a_selected.animating()) {
|
||||||
|
_a_selected.start();
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplyKeyboard::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) {
|
||||||
|
_st->repaint(_item);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplyKeyboard::step_selected(uint64 ms, bool timer) {
|
||||||
|
for (Animations::iterator i = _animations.begin(); i != _animations.end();) {
|
||||||
|
int index = qAbs(i.key()) - 1, row = (index / MatrixRowShift), col = index % MatrixRowShift;
|
||||||
|
float64 dt = float64(ms - i.value()) / st::botKbDuration;
|
||||||
|
if (dt >= 1) {
|
||||||
|
_rows[row][col].howMuchOver = (i.key() > 0) ? 1 : 0;
|
||||||
|
i = _animations.erase(i);
|
||||||
|
} else {
|
||||||
|
_rows[row][col].howMuchOver = (i.key() > 0) ? dt : (1 - dt);
|
||||||
|
++i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (timer) _st->repaint(_item);
|
||||||
|
if (_animations.isEmpty()) {
|
||||||
|
_a_selected.stop();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplyKeyboard::clearSelection() {
|
||||||
|
for (auto i = _animations.cbegin(), e = _animations.cend(); i != e; ++i) {
|
||||||
|
int index = qAbs(i.key()) - 1, row = (index / MatrixRowShift), col = index % MatrixRowShift;
|
||||||
|
_rows[row][col].howMuchOver = 0;
|
||||||
|
}
|
||||||
|
_animations.clear();
|
||||||
|
_a_selected.stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ReplyKeyboard::Style::paintButton(Painter &p, const ReplyKeyboard::Button &button) const {
|
||||||
|
const QRect &rect = button.rect;
|
||||||
|
bool pressed = ClickHandler::showAsPressed(button.link);
|
||||||
|
|
||||||
|
paintButtonBg(p, rect, pressed, button.howMuchOver);
|
||||||
|
paintButtonIcon(p, rect, button.type);
|
||||||
|
if (button.type == HistoryMessageReplyMarkup::Button::Type::Callback
|
||||||
|
|| button.type == HistoryMessageReplyMarkup::Button::Type::Game) {
|
||||||
|
if (auto data = button.link->getButton()) {
|
||||||
|
if (data->requestId) {
|
||||||
|
paintButtonLoading(p, rect);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int tx = rect.x(), tw = rect.width();
|
||||||
|
if (tw >= st::botKbFont->elidew + _st->padding * 2) {
|
||||||
|
tx += _st->padding;
|
||||||
|
tw -= _st->padding * 2;
|
||||||
|
} else if (tw > st::botKbFont->elidew) {
|
||||||
|
tx += (tw - st::botKbFont->elidew) / 2;
|
||||||
|
tw = st::botKbFont->elidew;
|
||||||
|
}
|
||||||
|
int textTop = rect.y() + (pressed ? _st->downTextTop : _st->textTop);
|
||||||
|
button.text.drawElided(p, tx, textTop + ((rect.height() - _st->height) / 2), tw, 1, style::al_top);
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryMessageReplyMarkup::createFromButtonRows(const QVector<MTPKeyboardButtonRow> &v) {
|
||||||
|
if (v.isEmpty()) {
|
||||||
|
rows.clear();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
rows.reserve(v.size());
|
||||||
|
for_const (auto &row, v) {
|
||||||
|
switch (row.type()) {
|
||||||
|
case mtpc_keyboardButtonRow: {
|
||||||
|
auto &r = row.c_keyboardButtonRow();
|
||||||
|
auto &b = r.vbuttons.c_vector().v;
|
||||||
|
if (!b.isEmpty()) {
|
||||||
|
ButtonRow buttonRow;
|
||||||
|
buttonRow.reserve(b.size());
|
||||||
|
for_const (const auto &button, b) {
|
||||||
|
switch (button.type()) {
|
||||||
|
case mtpc_keyboardButton: {
|
||||||
|
buttonRow.push_back({ Button::Type::Default, qs(button.c_keyboardButton().vtext), QByteArray(), 0 });
|
||||||
|
} break;
|
||||||
|
case mtpc_keyboardButtonCallback: {
|
||||||
|
auto &buttonData = button.c_keyboardButtonCallback();
|
||||||
|
buttonRow.push_back({ Button::Type::Callback, qs(buttonData.vtext), qba(buttonData.vdata), 0 });
|
||||||
|
} break;
|
||||||
|
case mtpc_keyboardButtonRequestGeoLocation: {
|
||||||
|
buttonRow.push_back({ Button::Type::RequestLocation, qs(button.c_keyboardButtonRequestGeoLocation().vtext), QByteArray(), 0 });
|
||||||
|
} break;
|
||||||
|
case mtpc_keyboardButtonRequestPhone: {
|
||||||
|
buttonRow.push_back({ Button::Type::RequestPhone, qs(button.c_keyboardButtonRequestPhone().vtext), QByteArray(), 0 });
|
||||||
|
} break;
|
||||||
|
case mtpc_keyboardButtonUrl: {
|
||||||
|
auto &buttonData = button.c_keyboardButtonUrl();
|
||||||
|
buttonRow.push_back({ Button::Type::Url, qs(buttonData.vtext), qba(buttonData.vurl), 0 });
|
||||||
|
} break;
|
||||||
|
case mtpc_keyboardButtonSwitchInline: {
|
||||||
|
auto &buttonData = button.c_keyboardButtonSwitchInline();
|
||||||
|
auto buttonType = buttonData.is_same_peer() ? Button::Type::SwitchInlineSame : Button::Type::SwitchInline;
|
||||||
|
buttonRow.push_back({ buttonType, qs(buttonData.vtext), qba(buttonData.vquery), 0 });
|
||||||
|
if (buttonType == Button::Type::SwitchInline) {
|
||||||
|
// Optimization flag.
|
||||||
|
// Fast check on all new messages if there is a switch button to auto-click it.
|
||||||
|
flags |= MTPDreplyKeyboardMarkup_ClientFlag::f_has_switch_inline_button;
|
||||||
|
}
|
||||||
|
} break;
|
||||||
|
case mtpc_keyboardButtonGame: {
|
||||||
|
auto &buttonData = button.c_keyboardButtonGame();
|
||||||
|
buttonRow.push_back({ Button::Type::Game, qs(buttonData.vtext), QByteArray(), 0 });
|
||||||
|
} break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!buttonRow.isEmpty()) rows.push_back(buttonRow);
|
||||||
|
}
|
||||||
|
} break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryMessageReplyMarkup::create(const MTPReplyMarkup &markup) {
|
||||||
|
flags = 0;
|
||||||
|
rows.clear();
|
||||||
|
inlineKeyboard = nullptr;
|
||||||
|
|
||||||
|
switch (markup.type()) {
|
||||||
|
case mtpc_replyKeyboardMarkup: {
|
||||||
|
const auto &d(markup.c_replyKeyboardMarkup());
|
||||||
|
flags = d.vflags.v;
|
||||||
|
|
||||||
|
createFromButtonRows(d.vrows.c_vector().v);
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case mtpc_replyInlineMarkup: {
|
||||||
|
const auto &d(markup.c_replyInlineMarkup());
|
||||||
|
flags = MTPDreplyKeyboardMarkup::Flags(0) | MTPDreplyKeyboardMarkup_ClientFlag::f_inline;
|
||||||
|
|
||||||
|
createFromButtonRows(d.vrows.c_vector().v);
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case mtpc_replyKeyboardHide: {
|
||||||
|
const auto &d(markup.c_replyKeyboardHide());
|
||||||
|
flags = mtpCastFlags(d.vflags) | MTPDreplyKeyboardMarkup_ClientFlag::f_zero;
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case mtpc_replyKeyboardForceReply: {
|
||||||
|
const auto &d(markup.c_replyKeyboardForceReply());
|
||||||
|
flags = mtpCastFlags(d.vflags) | MTPDreplyKeyboardMarkup_ClientFlag::f_force_reply;
|
||||||
|
} break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryMessageUnreadBar::init(int count) {
|
||||||
|
if (_freezed) return;
|
||||||
|
_text = lng_unread_bar(lt_count, count);
|
||||||
|
_width = st::semiboldFont->width(_text);
|
||||||
|
}
|
||||||
|
|
||||||
|
int HistoryMessageUnreadBar::height() {
|
||||||
|
return st::unreadBarHeight + st::unreadBarMargin;
|
||||||
|
}
|
||||||
|
|
||||||
|
int HistoryMessageUnreadBar::marginTop() {
|
||||||
|
return st::lineWidth + st::unreadBarMargin;
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryMessageUnreadBar::paint(Painter &p, int y, int w) const {
|
||||||
|
p.fillRect(0, y + marginTop(), w, height() - marginTop() - st::lineWidth, st::unreadBarBG);
|
||||||
|
p.fillRect(0, y + height() - st::lineWidth, w, st::lineWidth, st::unreadBarBorder);
|
||||||
|
p.setFont(st::unreadBarFont);
|
||||||
|
p.setPen(st::unreadBarColor);
|
||||||
|
|
||||||
|
int left = st::msgServiceMargin.left();
|
||||||
|
int maxwidth = w;
|
||||||
|
if (Adaptive::Wide()) {
|
||||||
|
maxwidth = qMin(maxwidth, int32(st::msgMaxWidth + 2 * st::msgPhotoSkip + 2 * st::msgMargin.left()));
|
||||||
|
}
|
||||||
|
w = maxwidth;
|
||||||
|
|
||||||
|
p.drawText((w - _width) / 2, y + marginTop() + (st::unreadBarHeight - 2 * st::lineWidth - st::unreadBarFont->height) / 2 + st::unreadBarFont->ascent, _text);
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryMessageDate::init(const QDateTime &date) {
|
||||||
|
_text = langDayOfMonthFull(date.date());
|
||||||
|
_width = st::msgServiceFont->width(_text);
|
||||||
|
}
|
||||||
|
|
||||||
|
int HistoryMessageDate::height() const {
|
||||||
|
return st::msgServiceMargin.top() + st::msgServicePadding.top() + st::msgServiceFont->height + st::msgServicePadding.bottom() + st::msgServiceMargin.bottom();
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryMessageDate::paint(Painter &p, int y, int w) const {
|
||||||
|
HistoryLayout::ServiceMessagePainter::paintDate(p, _text, _width, y, w);
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryMediaPtr::reset(HistoryMedia *p) {
|
||||||
|
if (_p) {
|
||||||
|
_p->detachFromParent();
|
||||||
|
delete _p;
|
||||||
|
}
|
||||||
|
_p = p;
|
||||||
|
if (_p) {
|
||||||
|
_p->attachToParent();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace internal {
|
||||||
|
|
||||||
|
TextSelection unshiftSelection(TextSelection selection, const Text &byText) {
|
||||||
|
if (selection == FullSelection) {
|
||||||
|
return selection;
|
||||||
|
}
|
||||||
|
return ::unshiftSelection(selection, byText);
|
||||||
|
}
|
||||||
|
|
||||||
|
TextSelection shiftSelection(TextSelection selection, const Text &byText) {
|
||||||
|
if (selection == FullSelection) {
|
||||||
|
return selection;
|
||||||
|
}
|
||||||
|
return ::shiftSelection(selection, byText);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace internal
|
||||||
|
|
||||||
|
HistoryItem::HistoryItem(History *history, MsgId msgId, MTPDmessage::Flags flags, QDateTime msgDate, int32 from) : HistoryElement()
|
||||||
|
, y(0)
|
||||||
|
, id(msgId)
|
||||||
|
, date(msgDate)
|
||||||
|
, _from(from ? App::user(from) : history->peer)
|
||||||
|
, _history(history)
|
||||||
|
, _flags(flags | MTPDmessage_ClientFlag::f_pending_init_dimensions | MTPDmessage_ClientFlag::f_pending_resize)
|
||||||
|
, _authorNameVersion(author()->nameVersion) {
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::finishCreate() {
|
||||||
|
App::historyRegItem(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::finishEdition(int oldKeyboardTop) {
|
||||||
|
setPendingInitDimensions();
|
||||||
|
if (App::main()) {
|
||||||
|
App::main()->dlgUpdated(history(), id);
|
||||||
|
}
|
||||||
|
|
||||||
|
// invalidate cache for drawInDialog
|
||||||
|
if (history()->textCachedFor == this) {
|
||||||
|
history()->textCachedFor = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (oldKeyboardTop >= 0) {
|
||||||
|
if (auto keyboard = Get<HistoryMessageReplyMarkup>()) {
|
||||||
|
keyboard->oldTop = oldKeyboardTop;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
App::historyUpdateDependent(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::finishEditionToEmpty() {
|
||||||
|
recountDisplayDate();
|
||||||
|
finishEdition(-1);
|
||||||
|
|
||||||
|
_history->removeNotification(this);
|
||||||
|
if (history()->isChannel()) {
|
||||||
|
if (history()->peer->isMegagroup() && history()->peer->asChannel()->mgInfo->pinnedMsgId == id) {
|
||||||
|
history()->peer->asChannel()->mgInfo->pinnedMsgId = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (history()->lastKeyboardId == id) {
|
||||||
|
history()->clearLastKeyboard();
|
||||||
|
}
|
||||||
|
if ((!out() || isPost()) && unread() && history()->unreadCount() > 0) {
|
||||||
|
history()->setUnreadCount(history()->unreadCount() - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auto next = nextItem()) {
|
||||||
|
next->previousItemChanged();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) {
|
||||||
|
if (auto markup = Get<HistoryMessageReplyMarkup>()) {
|
||||||
|
if (markup->inlineKeyboard) {
|
||||||
|
markup->inlineKeyboard->clickHandlerActiveChanged(p, active);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
App::hoveredLinkItem(active ? this : nullptr);
|
||||||
|
Ui::repaintHistoryItem(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) {
|
||||||
|
if (auto markup = Get<HistoryMessageReplyMarkup>()) {
|
||||||
|
if (markup->inlineKeyboard) {
|
||||||
|
markup->inlineKeyboard->clickHandlerPressedChanged(p, pressed);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
App::pressedLinkItem(pressed ? this : nullptr);
|
||||||
|
Ui::repaintHistoryItem(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::destroy() {
|
||||||
|
// All this must be done for all items manually in History::clear(false)!
|
||||||
|
eraseFromOverview();
|
||||||
|
|
||||||
|
bool wasAtBottom = history()->loadedAtBottom();
|
||||||
|
_history->removeNotification(this);
|
||||||
|
detach();
|
||||||
|
if (history()->isChannel()) {
|
||||||
|
if (history()->peer->isMegagroup() && history()->peer->asChannel()->mgInfo->pinnedMsgId == id) {
|
||||||
|
history()->peer->asChannel()->mgInfo->pinnedMsgId = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (history()->lastMsg == this) {
|
||||||
|
history()->fixLastMessage(wasAtBottom);
|
||||||
|
}
|
||||||
|
if (history()->lastKeyboardId == id) {
|
||||||
|
history()->clearLastKeyboard();
|
||||||
|
}
|
||||||
|
if ((!out() || isPost()) && unread() && history()->unreadCount() > 0) {
|
||||||
|
history()->setUnreadCount(history()->unreadCount() - 1);
|
||||||
|
}
|
||||||
|
Global::RefPendingRepaintItems().remove(this);
|
||||||
|
delete this;
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::detach() {
|
||||||
|
if (detached()) return;
|
||||||
|
|
||||||
|
if (_history->isChannel()) {
|
||||||
|
_history->asChannelHistory()->messageDetached(this);
|
||||||
|
}
|
||||||
|
_block->removeItem(this);
|
||||||
|
App::historyItemDetached(this);
|
||||||
|
|
||||||
|
_history->setPendingResize();
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::detachFast() {
|
||||||
|
_block = nullptr;
|
||||||
|
_indexInBlock = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::previousItemChanged() {
|
||||||
|
recountDisplayDate();
|
||||||
|
recountAttachToPrevious();
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::recountAttachToPrevious() {
|
||||||
|
bool attach = false;
|
||||||
|
if (!isPost() && !Has<HistoryMessageDate>() && !Has<HistoryMessageUnreadBar>()) {
|
||||||
|
if (auto previos = previousItem()) {
|
||||||
|
attach = !previos->isPost()
|
||||||
|
&& !previos->serviceMsg()
|
||||||
|
&& !previos->isEmpty()
|
||||||
|
&& previos->from() == from()
|
||||||
|
&& (qAbs(previos->date.secsTo(date)) < AttachMessageToPreviousSecondsDelta);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (attach && !(_flags & MTPDmessage_ClientFlag::f_attach_to_previous)) {
|
||||||
|
_flags |= MTPDmessage_ClientFlag::f_attach_to_previous;
|
||||||
|
setPendingInitDimensions();
|
||||||
|
} else if (!attach && (_flags & MTPDmessage_ClientFlag::f_attach_to_previous)) {
|
||||||
|
_flags &= ~MTPDmessage_ClientFlag::f_attach_to_previous;
|
||||||
|
setPendingInitDimensions();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::setId(MsgId newId) {
|
||||||
|
history()->changeMsgId(id, newId);
|
||||||
|
id = newId;
|
||||||
|
|
||||||
|
// We don't need to call Notify::replyMarkupUpdated(this) and update keyboard
|
||||||
|
// in history widget, because it can't exist for an outgoing message.
|
||||||
|
// Only inline keyboards can be in outgoing messages.
|
||||||
|
if (auto markup = inlineReplyMarkup()) {
|
||||||
|
if (markup->inlineKeyboard) {
|
||||||
|
markup->inlineKeyboard->updateMessageId();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool HistoryItem::canEdit(const QDateTime &cur) const {
|
||||||
|
auto messageToMyself = (peerToUser(_history->peer->id) == MTP::authedId());
|
||||||
|
auto messageTooOld = messageToMyself ? false : (date.secsTo(cur) >= Global::EditTimeLimit());
|
||||||
|
if (id < 0 || messageTooOld) return false;
|
||||||
|
|
||||||
|
if (auto msg = toHistoryMessage()) {
|
||||||
|
if (msg->Has<HistoryMessageVia>() || msg->Has<HistoryMessageForwarded>()) return false;
|
||||||
|
|
||||||
|
if (auto media = msg->getMedia()) {
|
||||||
|
auto type = media->type();
|
||||||
|
if (type != MediaTypePhoto &&
|
||||||
|
type != MediaTypeVideo &&
|
||||||
|
type != MediaTypeFile &&
|
||||||
|
type != MediaTypeGif &&
|
||||||
|
type != MediaTypeMusicFile &&
|
||||||
|
type != MediaTypeVoiceFile &&
|
||||||
|
type != MediaTypeWebPage) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (isPost()) {
|
||||||
|
auto channel = _history->peer->asChannel();
|
||||||
|
return (channel->amCreator() || (channel->amEditor() && out()));
|
||||||
|
}
|
||||||
|
return out() || messageToMyself;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool HistoryItem::unread() const {
|
||||||
|
// Messages from myself are always read.
|
||||||
|
if (history()->peer->isSelf()) return false;
|
||||||
|
|
||||||
|
if (out()) {
|
||||||
|
// Outgoing messages in converted chats are always read.
|
||||||
|
if (history()->peer->migrateTo()) return false;
|
||||||
|
|
||||||
|
if (id > 0) {
|
||||||
|
if (id < history()->outboxReadBefore) return false;
|
||||||
|
if (auto user = history()->peer->asUser()) {
|
||||||
|
if (user->botInfo) return false;
|
||||||
|
} else if (auto channel = history()->peer->asChannel()) {
|
||||||
|
if (!channel->isMegagroup()) return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (id > 0) {
|
||||||
|
if (id < history()->inboxReadBefore) return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return (_flags & MTPDmessage_ClientFlag::f_clientside_unread);
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::destroyUnreadBar() {
|
||||||
|
if (Has<HistoryMessageUnreadBar>()) {
|
||||||
|
RemoveComponents(HistoryMessageUnreadBar::Bit());
|
||||||
|
setPendingInitDimensions();
|
||||||
|
if (_history->unreadBar == this) {
|
||||||
|
_history->unreadBar = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
recountAttachToPrevious();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::setUnreadBarCount(int count) {
|
||||||
|
if (count > 0) {
|
||||||
|
HistoryMessageUnreadBar *bar;
|
||||||
|
if (!Has<HistoryMessageUnreadBar>()) {
|
||||||
|
AddComponents(HistoryMessageUnreadBar::Bit());
|
||||||
|
setPendingInitDimensions();
|
||||||
|
|
||||||
|
recountAttachToPrevious();
|
||||||
|
|
||||||
|
bar = Get<HistoryMessageUnreadBar>();
|
||||||
|
} else {
|
||||||
|
bar = Get<HistoryMessageUnreadBar>();
|
||||||
|
if (bar->_freezed) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
Global::RefPendingRepaintItems().insert(this);
|
||||||
|
}
|
||||||
|
bar->init(count);
|
||||||
|
} else {
|
||||||
|
destroyUnreadBar();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::setUnreadBarFreezed() {
|
||||||
|
if (auto bar = Get<HistoryMessageUnreadBar>()) {
|
||||||
|
bar->_freezed = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::clipCallback(Media::Clip::Notification notification) {
|
||||||
|
using namespace Media::Clip;
|
||||||
|
|
||||||
|
HistoryMedia *media = getMedia();
|
||||||
|
if (!media) return;
|
||||||
|
|
||||||
|
Reader *reader = media ? media->getClipReader() : 0;
|
||||||
|
if (!reader) return;
|
||||||
|
|
||||||
|
switch (notification) {
|
||||||
|
case NotificationReinit: {
|
||||||
|
bool stopped = false;
|
||||||
|
if (reader->autoPausedGif()) {
|
||||||
|
if (MainWidget *m = App::main()) {
|
||||||
|
if (!m->isItemVisible(this)) { // stop animation if it is not visible
|
||||||
|
media->stopInline();
|
||||||
|
if (DocumentData *document = media->getDocument()) { // forget data from memory
|
||||||
|
document->forget();
|
||||||
|
}
|
||||||
|
stopped = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!stopped) {
|
||||||
|
setPendingInitDimensions();
|
||||||
|
Notify::historyItemLayoutChanged(this);
|
||||||
|
}
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case NotificationRepaint: {
|
||||||
|
if (!reader->currentDisplayed()) {
|
||||||
|
Ui::repaintHistoryItem(this);
|
||||||
|
}
|
||||||
|
} break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::recountDisplayDate() {
|
||||||
|
bool displayingDate = ([this]() {
|
||||||
|
if (isEmpty()) return false;
|
||||||
|
|
||||||
|
if (auto previous = previousItem()) {
|
||||||
|
return previous->isEmpty() || (previous->date.date() != date.date());
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
})();
|
||||||
|
|
||||||
|
if (displayingDate && !Has<HistoryMessageDate>()) {
|
||||||
|
AddComponents(HistoryMessageDate::Bit());
|
||||||
|
Get<HistoryMessageDate>()->init(date);
|
||||||
|
setPendingInitDimensions();
|
||||||
|
} else if (!displayingDate && Has<HistoryMessageDate>()) {
|
||||||
|
RemoveComponents(HistoryMessageDate::Bit());
|
||||||
|
setPendingInitDimensions();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QString HistoryItem::notificationText() const {
|
||||||
|
auto getText = [this]() {
|
||||||
|
if (emptyText()) {
|
||||||
|
return _media ? _media->notificationText() : QString();
|
||||||
|
}
|
||||||
|
return _text.originalText();
|
||||||
|
};
|
||||||
|
|
||||||
|
auto result = getText();
|
||||||
|
if (result.size() > 0xFF) result = result.mid(0, 0xFF) + qsl("...");
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
QString HistoryItem::inDialogsText() const {
|
||||||
|
auto getText = [this]() {
|
||||||
|
if (emptyText()) {
|
||||||
|
return _media ? _media->inDialogsText() : QString();
|
||||||
|
}
|
||||||
|
return textClean(_text.originalText());
|
||||||
|
};
|
||||||
|
auto plainText = getText();
|
||||||
|
if ((!_history->peer->isUser() || out()) && !isPost() && !isEmpty()) {
|
||||||
|
auto fromText = author()->isSelf() ? lang(lng_from_you) : author()->shortName();
|
||||||
|
auto fromWrapped = textcmdLink(1, lng_dialogs_text_from_wrapped(lt_from, textClean(fromText)));
|
||||||
|
return lng_dialogs_text_with_from(lt_from_part, fromWrapped, lt_message, plainText);
|
||||||
|
}
|
||||||
|
return plainText;
|
||||||
|
}
|
||||||
|
|
||||||
|
void HistoryItem::drawInDialog(Painter &p, const QRect &r, bool act, const HistoryItem *&cacheFor, Text &cache) const {
|
||||||
|
if (cacheFor != this) {
|
||||||
|
cacheFor = this;
|
||||||
|
cache.setText(st::dialogsTextFont, inDialogsText(), _textDlgOptions);
|
||||||
|
}
|
||||||
|
if (r.width()) {
|
||||||
|
textstyleSet(&(act ? st::dialogsTextStyleActive : st::dialogsTextStyle));
|
||||||
|
p.setFont(st::dialogsTextFont);
|
||||||
|
p.setPen(act ? st::dialogsTextFgActive : st::dialogsTextFg);
|
||||||
|
cache.drawElided(p, r.left(), r.top(), r.width(), r.height() / st::dialogsTextFont->height);
|
||||||
|
textstyleRestore();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
HistoryItem::~HistoryItem() {
|
||||||
|
App::historyUnregItem(this);
|
||||||
|
if (id < 0 && App::uploader()) {
|
||||||
|
App::uploader()->cancel(fullId());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void GoToMessageClickHandler::onClickImpl() const {
|
||||||
|
if (App::main()) {
|
||||||
|
HistoryItem *current = App::mousedItem();
|
||||||
|
if (current && current->history()->peer->id == peer()) {
|
||||||
|
App::main()->pushReplyReturn(current);
|
||||||
|
}
|
||||||
|
Ui::showPeerHistory(peer(), msgid(), Ui::ShowWay::Forward);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,935 @@
|
||||||
|
/*
|
||||||
|
This file is part of Telegram Desktop,
|
||||||
|
the official desktop version of Telegram messaging app, see https://telegram.org
|
||||||
|
|
||||||
|
Telegram Desktop 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.
|
||||||
|
|
||||||
|
It 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.
|
||||||
|
|
||||||
|
In addition, as a special exception, the copyright holders give permission
|
||||||
|
to link the code of portions of this program with the OpenSSL library.
|
||||||
|
|
||||||
|
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
|
||||||
|
Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
class HistoryElement {
|
||||||
|
public:
|
||||||
|
HistoryElement() = default;
|
||||||
|
HistoryElement(const HistoryElement &other) = delete;
|
||||||
|
HistoryElement &operator=(const HistoryElement &other) = delete;
|
||||||
|
|
||||||
|
int maxWidth() const {
|
||||||
|
return _maxw;
|
||||||
|
}
|
||||||
|
int minHeight() const {
|
||||||
|
return _minh;
|
||||||
|
}
|
||||||
|
int height() const {
|
||||||
|
return _height;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual ~HistoryElement() = default;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
mutable int _maxw = 0;
|
||||||
|
mutable int _minh = 0;
|
||||||
|
mutable int _height = 0;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
class HistoryMessage;
|
||||||
|
|
||||||
|
enum HistoryCursorState {
|
||||||
|
HistoryDefaultCursorState,
|
||||||
|
HistoryInTextCursorState,
|
||||||
|
HistoryInDateCursorState,
|
||||||
|
HistoryInForwardedCursorState,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct HistoryTextState {
|
||||||
|
HistoryTextState() = default;
|
||||||
|
HistoryTextState(const Text::StateResult &state)
|
||||||
|
: cursor(state.uponSymbol ? HistoryInTextCursorState : HistoryDefaultCursorState)
|
||||||
|
, link(state.link)
|
||||||
|
, afterSymbol(state.afterSymbol)
|
||||||
|
, symbol(state.symbol) {
|
||||||
|
}
|
||||||
|
HistoryTextState &operator=(const Text::StateResult &state) {
|
||||||
|
cursor = state.uponSymbol ? HistoryInTextCursorState : HistoryDefaultCursorState;
|
||||||
|
link = state.link;
|
||||||
|
afterSymbol = state.afterSymbol;
|
||||||
|
symbol = state.symbol;
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
HistoryCursorState cursor = HistoryDefaultCursorState;
|
||||||
|
ClickHandlerPtr link;
|
||||||
|
bool afterSymbol = false;
|
||||||
|
uint16 symbol = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct HistoryStateRequest {
|
||||||
|
Text::StateRequest::Flags flags = Text::StateRequest::Flag::LookupLink;
|
||||||
|
Text::StateRequest forText() const {
|
||||||
|
Text::StateRequest result;
|
||||||
|
result.flags = flags;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
enum InfoDisplayType {
|
||||||
|
InfoDisplayDefault,
|
||||||
|
InfoDisplayOverImage,
|
||||||
|
InfoDisplayOverBackground,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum HistoryItemType {
|
||||||
|
HistoryItemMsg = 0,
|
||||||
|
HistoryItemJoined
|
||||||
|
};
|
||||||
|
|
||||||
|
struct HistoryMessageVia : public BaseComponent<HistoryMessageVia> {
|
||||||
|
void create(int32 userId);
|
||||||
|
void resize(int32 availw) const;
|
||||||
|
|
||||||
|
UserData *_bot = nullptr;
|
||||||
|
mutable QString _text;
|
||||||
|
mutable int _width = 0;
|
||||||
|
mutable int _maxWidth = 0;
|
||||||
|
ClickHandlerPtr _lnk;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct HistoryMessageViews : public BaseComponent<HistoryMessageViews> {
|
||||||
|
QString _viewsText;
|
||||||
|
int _views = 0;
|
||||||
|
int _viewsWidth = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct HistoryMessageSigned : public BaseComponent<HistoryMessageSigned> {
|
||||||
|
void create(UserData *from, const QDateTime &date);
|
||||||
|
int maxWidth() const;
|
||||||
|
|
||||||
|
Text _signature;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct HistoryMessageEdited : public BaseComponent<HistoryMessageEdited> {
|
||||||
|
void create(const QDateTime &editDate, const QDateTime &date);
|
||||||
|
int maxWidth() const;
|
||||||
|
|
||||||
|
QDateTime _editDate;
|
||||||
|
Text _edited;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct HistoryMessageForwarded : public BaseComponent<HistoryMessageForwarded> {
|
||||||
|
void create(const HistoryMessageVia *via) const;
|
||||||
|
|
||||||
|
PeerData *_authorOriginal = nullptr;
|
||||||
|
PeerData *_fromOriginal = nullptr;
|
||||||
|
MsgId _originalId = 0;
|
||||||
|
mutable Text _text = { 1 };
|
||||||
|
};
|
||||||
|
|
||||||
|
struct HistoryMessageReply : public BaseComponent<HistoryMessageReply> {
|
||||||
|
HistoryMessageReply &operator=(HistoryMessageReply &&other) {
|
||||||
|
replyToMsgId = other.replyToMsgId;
|
||||||
|
std::swap(replyToMsg, other.replyToMsg);
|
||||||
|
replyToLnk = std_::move(other.replyToLnk);
|
||||||
|
replyToName = std_::move(other.replyToName);
|
||||||
|
replyToText = std_::move(other.replyToText);
|
||||||
|
replyToVersion = other.replyToVersion;
|
||||||
|
_maxReplyWidth = other._maxReplyWidth;
|
||||||
|
_replyToVia = std_::move(other._replyToVia);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
~HistoryMessageReply() {
|
||||||
|
// clearData() should be called by holder
|
||||||
|
t_assert(replyToMsg == nullptr);
|
||||||
|
t_assert(_replyToVia == nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool updateData(HistoryMessage *holder, bool force = false);
|
||||||
|
void clearData(HistoryMessage *holder); // must be called before destructor
|
||||||
|
|
||||||
|
bool isNameUpdated() const;
|
||||||
|
void updateName() const;
|
||||||
|
void resize(int width) const;
|
||||||
|
void itemRemoved(HistoryMessage *holder, HistoryItem *removed);
|
||||||
|
|
||||||
|
enum PaintFlag {
|
||||||
|
PaintInBubble = 0x01,
|
||||||
|
PaintSelected = 0x02,
|
||||||
|
};
|
||||||
|
Q_DECLARE_FLAGS(PaintFlags, PaintFlag);
|
||||||
|
void paint(Painter &p, const HistoryItem *holder, int x, int y, int w, PaintFlags flags) const;
|
||||||
|
|
||||||
|
MsgId replyToId() const {
|
||||||
|
return replyToMsgId;
|
||||||
|
}
|
||||||
|
int replyToWidth() const {
|
||||||
|
return _maxReplyWidth;
|
||||||
|
}
|
||||||
|
ClickHandlerPtr replyToLink() const {
|
||||||
|
return replyToLnk;
|
||||||
|
}
|
||||||
|
|
||||||
|
MsgId replyToMsgId = 0;
|
||||||
|
HistoryItem *replyToMsg = nullptr;
|
||||||
|
ClickHandlerPtr replyToLnk;
|
||||||
|
mutable Text replyToName, replyToText;
|
||||||
|
mutable int replyToVersion = 0;
|
||||||
|
mutable int _maxReplyWidth = 0;
|
||||||
|
std_::unique_ptr<HistoryMessageVia> _replyToVia;
|
||||||
|
int toWidth = 0;
|
||||||
|
};
|
||||||
|
Q_DECLARE_OPERATORS_FOR_FLAGS(HistoryMessageReply::PaintFlags);
|
||||||
|
|
||||||
|
class ReplyKeyboard;
|
||||||
|
struct HistoryMessageReplyMarkup : public BaseComponent<HistoryMessageReplyMarkup> {
|
||||||
|
HistoryMessageReplyMarkup() = default;
|
||||||
|
HistoryMessageReplyMarkup(MTPDreplyKeyboardMarkup::Flags f) : flags(f) {
|
||||||
|
}
|
||||||
|
|
||||||
|
void create(const MTPReplyMarkup &markup);
|
||||||
|
|
||||||
|
struct Button {
|
||||||
|
enum class Type {
|
||||||
|
Default,
|
||||||
|
Url,
|
||||||
|
Callback,
|
||||||
|
RequestPhone,
|
||||||
|
RequestLocation,
|
||||||
|
SwitchInline,
|
||||||
|
SwitchInlineSame,
|
||||||
|
Game,
|
||||||
|
};
|
||||||
|
Type type;
|
||||||
|
QString text;
|
||||||
|
QByteArray data;
|
||||||
|
mutable mtpRequestId requestId;
|
||||||
|
};
|
||||||
|
using ButtonRow = QVector<Button>;
|
||||||
|
using ButtonRows = QVector<ButtonRow>;
|
||||||
|
|
||||||
|
ButtonRows rows;
|
||||||
|
MTPDreplyKeyboardMarkup::Flags flags = 0;
|
||||||
|
|
||||||
|
std_::unique_ptr<ReplyKeyboard> inlineKeyboard;
|
||||||
|
|
||||||
|
// If >= 0 it holds the y coord of the inlineKeyboard before the last edition.
|
||||||
|
int oldTop = -1;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void createFromButtonRows(const QVector<MTPKeyboardButtonRow> &v);
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
class ReplyMarkupClickHandler;
|
||||||
|
class ReplyKeyboard {
|
||||||
|
private:
|
||||||
|
struct Button;
|
||||||
|
|
||||||
|
public:
|
||||||
|
class Style {
|
||||||
|
public:
|
||||||
|
Style(const style::botKeyboardButton &st) : _st(&st) {
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void startPaint(Painter &p) const = 0;
|
||||||
|
virtual style::font textFont() const = 0;
|
||||||
|
|
||||||
|
int buttonSkip() const {
|
||||||
|
return _st->margin;
|
||||||
|
}
|
||||||
|
int buttonPadding() const {
|
||||||
|
return _st->padding;
|
||||||
|
}
|
||||||
|
int buttonHeight() const {
|
||||||
|
return _st->height;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void repaint(const HistoryItem *item) const = 0;
|
||||||
|
virtual ~Style() {
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
virtual void paintButtonBg(Painter &p, const QRect &rect, bool pressed, float64 howMuchOver) const = 0;
|
||||||
|
virtual void paintButtonIcon(Painter &p, const QRect &rect, HistoryMessageReplyMarkup::Button::Type type) const = 0;
|
||||||
|
virtual void paintButtonLoading(Painter &p, const QRect &rect) const = 0;
|
||||||
|
virtual int minButtonWidth(HistoryMessageReplyMarkup::Button::Type type) const = 0;
|
||||||
|
|
||||||
|
private:
|
||||||
|
const style::botKeyboardButton *_st;
|
||||||
|
|
||||||
|
void paintButton(Painter &p, const ReplyKeyboard::Button &button) const;
|
||||||
|
friend class ReplyKeyboard;
|
||||||
|
|
||||||
|
};
|
||||||
|
typedef std_::unique_ptr<Style> StylePtr;
|
||||||
|
|
||||||
|
ReplyKeyboard(const HistoryItem *item, StylePtr &&s);
|
||||||
|
ReplyKeyboard(const ReplyKeyboard &other) = delete;
|
||||||
|
ReplyKeyboard &operator=(const ReplyKeyboard &other) = delete;
|
||||||
|
|
||||||
|
bool isEnoughSpace(int width, const style::botKeyboardButton &st) const;
|
||||||
|
void setStyle(StylePtr &&s);
|
||||||
|
void resize(int width, int height);
|
||||||
|
|
||||||
|
// what width and height will best fit this keyboard
|
||||||
|
int naturalWidth() const;
|
||||||
|
int naturalHeight() const;
|
||||||
|
|
||||||
|
void paint(Painter &p, const QRect &clip) const;
|
||||||
|
ClickHandlerPtr getState(int x, int y) const;
|
||||||
|
|
||||||
|
void clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active);
|
||||||
|
void clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed);
|
||||||
|
|
||||||
|
void clearSelection();
|
||||||
|
void updateMessageId();
|
||||||
|
|
||||||
|
private:
|
||||||
|
const HistoryItem *_item;
|
||||||
|
int _width = 0;
|
||||||
|
|
||||||
|
friend class Style;
|
||||||
|
using ReplyMarkupClickHandlerPtr = QSharedPointer<ReplyMarkupClickHandler>;
|
||||||
|
struct Button {
|
||||||
|
Text text = { 1 };
|
||||||
|
QRect rect;
|
||||||
|
int characters = 0;
|
||||||
|
float64 howMuchOver = 0.;
|
||||||
|
HistoryMessageReplyMarkup::Button::Type type;
|
||||||
|
ReplyMarkupClickHandlerPtr link;
|
||||||
|
};
|
||||||
|
using ButtonRow = QVector<Button>;
|
||||||
|
using ButtonRows = QVector<ButtonRow>;
|
||||||
|
ButtonRows _rows;
|
||||||
|
|
||||||
|
using Animations = QMap<int, uint64>;
|
||||||
|
Animations _animations;
|
||||||
|
Animation _a_selected;
|
||||||
|
void step_selected(uint64 ms, bool timer);
|
||||||
|
|
||||||
|
StylePtr _st;
|
||||||
|
};
|
||||||
|
|
||||||
|
// any HistoryItem can have this Interface for
|
||||||
|
// displaying the day mark above the message
|
||||||
|
struct HistoryMessageDate : public BaseComponent<HistoryMessageDate> {
|
||||||
|
void init(const QDateTime &date);
|
||||||
|
|
||||||
|
int height() const;
|
||||||
|
void paint(Painter &p, int y, int w) const;
|
||||||
|
|
||||||
|
QString _text;
|
||||||
|
int _width = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
// any HistoryItem can have this Interface for
|
||||||
|
// displaying the unread messages bar above the message
|
||||||
|
struct HistoryMessageUnreadBar : public BaseComponent<HistoryMessageUnreadBar> {
|
||||||
|
void init(int count);
|
||||||
|
|
||||||
|
static int height();
|
||||||
|
static int marginTop();
|
||||||
|
|
||||||
|
void paint(Painter &p, int y, int w) const;
|
||||||
|
|
||||||
|
QString _text;
|
||||||
|
int _width = 0;
|
||||||
|
|
||||||
|
// if unread bar is freezed the new messages do not
|
||||||
|
// increment the counter displayed by this bar
|
||||||
|
//
|
||||||
|
// it happens when we've opened the conversation and
|
||||||
|
// we've seen the bar and new messages are marked as read
|
||||||
|
// as soon as they are added to the chat history
|
||||||
|
bool _freezed = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
// HistoryMedia has a special owning smart pointer
|
||||||
|
// which regs/unregs this media to the holding HistoryItem
|
||||||
|
class HistoryMedia;
|
||||||
|
class HistoryMediaPtr {
|
||||||
|
public:
|
||||||
|
HistoryMediaPtr() = default;
|
||||||
|
HistoryMediaPtr(const HistoryMediaPtr &other) = delete;
|
||||||
|
HistoryMediaPtr &operator=(const HistoryMediaPtr &other) = delete;
|
||||||
|
HistoryMedia *data() const {
|
||||||
|
return _p;
|
||||||
|
}
|
||||||
|
void reset(HistoryMedia *p = nullptr);
|
||||||
|
void clear() {
|
||||||
|
reset();
|
||||||
|
}
|
||||||
|
bool isNull() const {
|
||||||
|
return data() == nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
HistoryMedia *operator->() const {
|
||||||
|
return data();
|
||||||
|
}
|
||||||
|
HistoryMedia &operator*() const {
|
||||||
|
t_assert(!isNull());
|
||||||
|
return *data();
|
||||||
|
}
|
||||||
|
explicit operator bool() const {
|
||||||
|
return !isNull();
|
||||||
|
}
|
||||||
|
~HistoryMediaPtr() {
|
||||||
|
clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
HistoryMedia *_p = nullptr;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
namespace internal {
|
||||||
|
|
||||||
|
TextSelection unshiftSelection(TextSelection selection, const Text &byText);
|
||||||
|
TextSelection shiftSelection(TextSelection selection, const Text &byText);
|
||||||
|
|
||||||
|
} // namespace internal
|
||||||
|
|
||||||
|
class HistoryItem : public HistoryElement, public Composer, public ClickHandlerHost {
|
||||||
|
public:
|
||||||
|
int resizeGetHeight(int width) {
|
||||||
|
if (_flags & MTPDmessage_ClientFlag::f_pending_init_dimensions) {
|
||||||
|
_flags &= ~MTPDmessage_ClientFlag::f_pending_init_dimensions;
|
||||||
|
initDimensions();
|
||||||
|
}
|
||||||
|
if (_flags & MTPDmessage_ClientFlag::f_pending_resize) {
|
||||||
|
_flags &= ~MTPDmessage_ClientFlag::f_pending_resize;
|
||||||
|
}
|
||||||
|
return resizeGetHeight_(width);
|
||||||
|
}
|
||||||
|
virtual void draw(Painter &p, const QRect &r, TextSelection selection, uint64 ms) const = 0;
|
||||||
|
|
||||||
|
virtual void dependencyItemRemoved(HistoryItem *dependency) {
|
||||||
|
}
|
||||||
|
virtual bool updateDependencyItem() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
virtual MsgId dependencyMsgId() const {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
virtual bool notificationReady() const {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
UserData *viaBot() const {
|
||||||
|
if (const HistoryMessageVia *via = Get<HistoryMessageVia>()) {
|
||||||
|
return via->_bot;
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
UserData *getMessageBot() const {
|
||||||
|
if (auto bot = viaBot()) {
|
||||||
|
return bot;
|
||||||
|
}
|
||||||
|
auto bot = from()->asUser();
|
||||||
|
if (!bot) {
|
||||||
|
bot = history()->peer->asUser();
|
||||||
|
}
|
||||||
|
return (bot && bot->botInfo) ? bot : nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
History *history() const {
|
||||||
|
return _history;
|
||||||
|
}
|
||||||
|
PeerData *from() const {
|
||||||
|
return _from;
|
||||||
|
}
|
||||||
|
HistoryBlock *block() {
|
||||||
|
return _block;
|
||||||
|
}
|
||||||
|
const HistoryBlock *block() const {
|
||||||
|
return _block;
|
||||||
|
}
|
||||||
|
void destroy();
|
||||||
|
void detach();
|
||||||
|
void detachFast();
|
||||||
|
bool detached() const {
|
||||||
|
return !_block;
|
||||||
|
}
|
||||||
|
void attachToBlock(HistoryBlock *block, int index) {
|
||||||
|
t_assert(_block == nullptr);
|
||||||
|
t_assert(_indexInBlock < 0);
|
||||||
|
t_assert(block != nullptr);
|
||||||
|
t_assert(index >= 0);
|
||||||
|
|
||||||
|
_block = block;
|
||||||
|
_indexInBlock = index;
|
||||||
|
if (pendingResize()) {
|
||||||
|
_history->setHasPendingResizedItems();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void setIndexInBlock(int index) {
|
||||||
|
t_assert(_block != nullptr);
|
||||||
|
t_assert(index >= 0);
|
||||||
|
|
||||||
|
_indexInBlock = index;
|
||||||
|
}
|
||||||
|
int indexInBlock() const {
|
||||||
|
if (_indexInBlock >= 0) {
|
||||||
|
t_assert(_block != nullptr);
|
||||||
|
t_assert(_block->items.at(_indexInBlock) == this);
|
||||||
|
} else if (_block != nullptr) {
|
||||||
|
t_assert(_indexInBlock >= 0);
|
||||||
|
t_assert(_block->items.at(_indexInBlock) == this);
|
||||||
|
}
|
||||||
|
return _indexInBlock;
|
||||||
|
}
|
||||||
|
bool out() const {
|
||||||
|
return _flags & MTPDmessage::Flag::f_out;
|
||||||
|
}
|
||||||
|
bool unread() const;
|
||||||
|
bool mentionsMe() const {
|
||||||
|
return _flags & MTPDmessage::Flag::f_mentioned;
|
||||||
|
}
|
||||||
|
bool isMediaUnread() const {
|
||||||
|
return (_flags & MTPDmessage::Flag::f_media_unread) && (channelId() == NoChannel);
|
||||||
|
}
|
||||||
|
void markMediaRead() {
|
||||||
|
_flags &= ~MTPDmessage::Flag::f_media_unread;
|
||||||
|
}
|
||||||
|
bool definesReplyKeyboard() const {
|
||||||
|
if (auto markup = Get<HistoryMessageReplyMarkup>()) {
|
||||||
|
if (markup->flags & MTPDreplyKeyboardMarkup_ClientFlag::f_inline) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// optimization: don't create markup component for the case
|
||||||
|
// MTPDreplyKeyboardHide with flags = 0, assume it has f_zero flag
|
||||||
|
return (_flags & MTPDmessage::Flag::f_reply_markup);
|
||||||
|
}
|
||||||
|
MTPDreplyKeyboardMarkup::Flags replyKeyboardFlags() const {
|
||||||
|
t_assert(definesReplyKeyboard());
|
||||||
|
if (auto markup = Get<HistoryMessageReplyMarkup>()) {
|
||||||
|
return markup->flags;
|
||||||
|
}
|
||||||
|
|
||||||
|
// optimization: don't create markup component for the case
|
||||||
|
// MTPDreplyKeyboardHide with flags = 0, assume it has f_zero flag
|
||||||
|
return qFlags(MTPDreplyKeyboardMarkup_ClientFlag::f_zero);
|
||||||
|
}
|
||||||
|
bool hasSwitchInlineButton() const {
|
||||||
|
return _flags & MTPDmessage_ClientFlag::f_has_switch_inline_button;
|
||||||
|
}
|
||||||
|
bool hasTextLinks() const {
|
||||||
|
return _flags & MTPDmessage_ClientFlag::f_has_text_links;
|
||||||
|
}
|
||||||
|
bool isGroupMigrate() const {
|
||||||
|
return _flags & MTPDmessage_ClientFlag::f_is_group_migrate;
|
||||||
|
}
|
||||||
|
bool hasViews() const {
|
||||||
|
return _flags & MTPDmessage::Flag::f_views;
|
||||||
|
}
|
||||||
|
bool isPost() const {
|
||||||
|
return _flags & MTPDmessage::Flag::f_post;
|
||||||
|
}
|
||||||
|
bool indexInOverview() const {
|
||||||
|
return (id > 0) && (!history()->isChannel() || history()->isMegagroup() || isPost());
|
||||||
|
}
|
||||||
|
bool isSilent() const {
|
||||||
|
return _flags & MTPDmessage::Flag::f_silent;
|
||||||
|
}
|
||||||
|
bool hasOutLayout() const {
|
||||||
|
return out() && !isPost();
|
||||||
|
}
|
||||||
|
virtual int32 viewsCount() const {
|
||||||
|
return hasViews() ? 1 : -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual bool needCheck() const {
|
||||||
|
return out() || (id < 0 && history()->peer->isSelf());
|
||||||
|
}
|
||||||
|
virtual bool hasPoint(int x, int y) const {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual HistoryTextState getState(int x, int y, HistoryStateRequest request) const = 0;
|
||||||
|
|
||||||
|
virtual TextSelection adjustSelection(TextSelection selection, TextSelectType type) const {
|
||||||
|
return selection;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ClickHandlerHost interface
|
||||||
|
void clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) override;
|
||||||
|
void clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) override;
|
||||||
|
|
||||||
|
virtual HistoryItemType type() const {
|
||||||
|
return HistoryItemMsg;
|
||||||
|
}
|
||||||
|
virtual bool serviceMsg() const {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
virtual void applyEdition(const MTPDmessage &message) {
|
||||||
|
}
|
||||||
|
virtual void applyEdition(const MTPDmessageService &message) {
|
||||||
|
}
|
||||||
|
virtual void updateMedia(const MTPMessageMedia *media) {
|
||||||
|
}
|
||||||
|
virtual int32 addToOverview(AddToOverviewMethod method) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
virtual void eraseFromOverview() {
|
||||||
|
}
|
||||||
|
virtual bool hasBubble() const {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
virtual void previousItemChanged();
|
||||||
|
|
||||||
|
virtual TextWithEntities selectedText(TextSelection selection) const {
|
||||||
|
return { qsl("[-]"), EntitiesInText() };
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual QString notificationHeader() const {
|
||||||
|
return QString();
|
||||||
|
}
|
||||||
|
virtual QString notificationText() const;
|
||||||
|
|
||||||
|
// Returns text with link-start and link-end commands for service-color highlighting.
|
||||||
|
// Example: "[link1-start]You:[link1-end] [link1-start]Photo,[link1-end] caption text"
|
||||||
|
virtual QString inDialogsText() const;
|
||||||
|
virtual QString inReplyText() const {
|
||||||
|
return notificationText();
|
||||||
|
}
|
||||||
|
virtual TextWithEntities originalText() const {
|
||||||
|
return { QString(), EntitiesInText() };
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void drawInfo(Painter &p, int32 right, int32 bottom, int32 width, bool selected, InfoDisplayType type) const {
|
||||||
|
}
|
||||||
|
virtual void setViewsCount(int32 count) {
|
||||||
|
}
|
||||||
|
virtual void setId(MsgId newId);
|
||||||
|
void drawInDialog(Painter &p, const QRect &r, bool act, const HistoryItem *&cacheFor, Text &cache) const;
|
||||||
|
|
||||||
|
bool emptyText() const {
|
||||||
|
return _text.isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool canDelete() const {
|
||||||
|
ChannelData *channel = _history->peer->asChannel();
|
||||||
|
if (!channel) return !(_flags & MTPDmessage_ClientFlag::f_is_group_migrate);
|
||||||
|
|
||||||
|
if (id == 1) return false;
|
||||||
|
if (channel->amCreator()) return true;
|
||||||
|
if (isPost()) {
|
||||||
|
if (channel->amEditor() && out()) return true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return (channel->amEditor() || channel->amModerator() || out());
|
||||||
|
}
|
||||||
|
|
||||||
|
bool canPin() const {
|
||||||
|
return id > 0 && _history->peer->isMegagroup() && (_history->peer->asChannel()->amEditor() || _history->peer->asChannel()->amCreator()) && toHistoryMessage();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool canEdit(const QDateTime &cur) const;
|
||||||
|
|
||||||
|
bool suggestBanReportDeleteAll() const {
|
||||||
|
ChannelData *channel = history()->peer->asChannel();
|
||||||
|
if (!channel || (!channel->amEditor() && !channel->amCreator())) return false;
|
||||||
|
return !isPost() && !out() && from()->isUser() && toHistoryMessage();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool hasDirectLink() const {
|
||||||
|
return id > 0 && _history->peer->isChannel() && _history->peer->asChannel()->isPublic() && !_history->peer->isMegagroup();
|
||||||
|
}
|
||||||
|
QString directLink() const {
|
||||||
|
return hasDirectLink() ? qsl("https://telegram.me/") + _history->peer->asChannel()->username + '/' + QString::number(id) : QString();
|
||||||
|
}
|
||||||
|
|
||||||
|
int32 y;
|
||||||
|
MsgId id;
|
||||||
|
QDateTime date;
|
||||||
|
|
||||||
|
ChannelId channelId() const {
|
||||||
|
return _history->channelId();
|
||||||
|
}
|
||||||
|
FullMsgId fullId() const {
|
||||||
|
return FullMsgId(channelId(), id);
|
||||||
|
}
|
||||||
|
|
||||||
|
HistoryMedia *getMedia() const {
|
||||||
|
return _media.data();
|
||||||
|
}
|
||||||
|
virtual void setText(const TextWithEntities &textWithEntities) {
|
||||||
|
}
|
||||||
|
virtual bool textHasLinks() const {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual int infoWidth() const {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
virtual int timeLeft() const {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
virtual int timeWidth() const {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
virtual bool pointInTime(int32 right, int32 bottom, int x, int y, InfoDisplayType type) const {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32 skipBlockWidth() const {
|
||||||
|
return st::msgDateSpace + infoWidth() - st::msgDateDelta.x();
|
||||||
|
}
|
||||||
|
int32 skipBlockHeight() const {
|
||||||
|
return st::msgDateFont->height - st::msgDateDelta.y();
|
||||||
|
}
|
||||||
|
QString skipBlock() const {
|
||||||
|
return textcmdSkipBlock(skipBlockWidth(), skipBlockHeight());
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual HistoryMessage *toHistoryMessage() { // dynamic_cast optimize
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
virtual const HistoryMessage *toHistoryMessage() const { // dynamic_cast optimize
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
MsgId replyToId() const {
|
||||||
|
if (auto reply = Get<HistoryMessageReply>()) {
|
||||||
|
return reply->replyToId();
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool hasFromName() const {
|
||||||
|
return (!out() || isPost()) && !history()->peer->isUser();
|
||||||
|
}
|
||||||
|
PeerData *author() const {
|
||||||
|
return isPost() ? history()->peer : _from;
|
||||||
|
}
|
||||||
|
|
||||||
|
PeerData *fromOriginal() const {
|
||||||
|
if (const HistoryMessageForwarded *fwd = Get<HistoryMessageForwarded>()) {
|
||||||
|
return fwd->_fromOriginal;
|
||||||
|
}
|
||||||
|
return from();
|
||||||
|
}
|
||||||
|
PeerData *authorOriginal() const {
|
||||||
|
if (const HistoryMessageForwarded *fwd = Get<HistoryMessageForwarded>()) {
|
||||||
|
return fwd->_authorOriginal;
|
||||||
|
}
|
||||||
|
return author();
|
||||||
|
}
|
||||||
|
|
||||||
|
// count > 0 - creates the unread bar if necessary and
|
||||||
|
// sets unread messages count if bar is not freezed yet
|
||||||
|
// count <= 0 - destroys the unread bar
|
||||||
|
void setUnreadBarCount(int count);
|
||||||
|
void destroyUnreadBar();
|
||||||
|
|
||||||
|
// marks the unread bar as freezed so that unread
|
||||||
|
// messages count will not change for this bar
|
||||||
|
// when the new messages arrive in this chat history
|
||||||
|
void setUnreadBarFreezed();
|
||||||
|
|
||||||
|
bool pendingResize() const {
|
||||||
|
return _flags & MTPDmessage_ClientFlag::f_pending_resize;
|
||||||
|
}
|
||||||
|
void setPendingResize() {
|
||||||
|
_flags |= MTPDmessage_ClientFlag::f_pending_resize;
|
||||||
|
if (!detached()) {
|
||||||
|
_history->setHasPendingResizedItems();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
bool pendingInitDimensions() const {
|
||||||
|
return _flags & MTPDmessage_ClientFlag::f_pending_init_dimensions;
|
||||||
|
}
|
||||||
|
void setPendingInitDimensions() {
|
||||||
|
_flags |= MTPDmessage_ClientFlag::f_pending_init_dimensions;
|
||||||
|
setPendingResize();
|
||||||
|
}
|
||||||
|
|
||||||
|
int displayedDateHeight() const {
|
||||||
|
if (auto date = Get<HistoryMessageDate>()) {
|
||||||
|
return date->height();
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
int marginTop() const {
|
||||||
|
int result = 0;
|
||||||
|
if (isAttachedToPrevious()) {
|
||||||
|
result += st::msgMarginTopAttached;
|
||||||
|
} else {
|
||||||
|
result += st::msgMargin.top();
|
||||||
|
}
|
||||||
|
result += displayedDateHeight();
|
||||||
|
if (auto unreadbar = Get<HistoryMessageUnreadBar>()) {
|
||||||
|
result += unreadbar->height();
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
int marginBottom() const {
|
||||||
|
return st::msgMargin.bottom();
|
||||||
|
}
|
||||||
|
bool isAttachedToPrevious() const {
|
||||||
|
return _flags & MTPDmessage_ClientFlag::f_attach_to_previous;
|
||||||
|
}
|
||||||
|
bool displayDate() const {
|
||||||
|
return Has<HistoryMessageDate>();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool isInOneDayWithPrevious() const {
|
||||||
|
return !isEmpty() && !displayDate();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool isEmpty() const {
|
||||||
|
return _text.isEmpty() && !_media;
|
||||||
|
}
|
||||||
|
|
||||||
|
void clipCallback(Media::Clip::Notification notification);
|
||||||
|
|
||||||
|
virtual ~HistoryItem();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
HistoryItem(History *history, MsgId msgId, MTPDmessage::Flags flags, QDateTime msgDate, int32 from);
|
||||||
|
|
||||||
|
// to completely create history item we need to call
|
||||||
|
// a virtual method, it can not be done from constructor
|
||||||
|
virtual void finishCreate();
|
||||||
|
|
||||||
|
// called from resizeGetHeight() when MTPDmessage_ClientFlag::f_pending_init_dimensions is set
|
||||||
|
virtual void initDimensions() = 0;
|
||||||
|
|
||||||
|
virtual int resizeGetHeight_(int width) = 0;
|
||||||
|
|
||||||
|
void finishEdition(int oldKeyboardTop);
|
||||||
|
void finishEditionToEmpty();
|
||||||
|
|
||||||
|
PeerData *_from;
|
||||||
|
History *_history;
|
||||||
|
HistoryBlock *_block = nullptr;
|
||||||
|
int _indexInBlock = -1;
|
||||||
|
MTPDmessage::Flags _flags;
|
||||||
|
|
||||||
|
mutable int32 _authorNameVersion;
|
||||||
|
|
||||||
|
HistoryItem *previousItem() const {
|
||||||
|
if (_block && _indexInBlock >= 0) {
|
||||||
|
if (_indexInBlock > 0) {
|
||||||
|
return _block->items.at(_indexInBlock - 1);
|
||||||
|
}
|
||||||
|
if (auto previous = _block->previousBlock()) {
|
||||||
|
t_assert(!previous->items.isEmpty());
|
||||||
|
return previous->items.back();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
HistoryItem *nextItem() const {
|
||||||
|
if (_block && _indexInBlock >= 0) {
|
||||||
|
if (_indexInBlock + 1 < _block->items.size()) {
|
||||||
|
return _block->items.at(_indexInBlock + 1);
|
||||||
|
}
|
||||||
|
if (auto next = _block->nextBlock()) {
|
||||||
|
t_assert(!next->items.isEmpty());
|
||||||
|
return next->items.front();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
// this should be used only in previousItemChanged()
|
||||||
|
// to add required bits to the Composer mask
|
||||||
|
// after that always use Has<HistoryMessageDate>()
|
||||||
|
void recountDisplayDate();
|
||||||
|
|
||||||
|
// this should be used only in previousItemChanged() or when
|
||||||
|
// HistoryMessageDate or HistoryMessageUnreadBar bit is changed in the Composer mask
|
||||||
|
// then the result should be cached in a client side flag MTPDmessage_ClientFlag::f_attach_to_previous
|
||||||
|
void recountAttachToPrevious();
|
||||||
|
|
||||||
|
const HistoryMessageReplyMarkup *inlineReplyMarkup() const {
|
||||||
|
if (auto markup = Get<HistoryMessageReplyMarkup>()) {
|
||||||
|
if (markup->flags & MTPDreplyKeyboardMarkup_ClientFlag::f_inline) {
|
||||||
|
return markup;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
const ReplyKeyboard *inlineReplyKeyboard() const {
|
||||||
|
if (auto markup = inlineReplyMarkup()) {
|
||||||
|
return markup->inlineKeyboard.get();
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
HistoryMessageReplyMarkup *inlineReplyMarkup() {
|
||||||
|
return const_cast<HistoryMessageReplyMarkup*>(static_cast<const HistoryItem*>(this)->inlineReplyMarkup());
|
||||||
|
}
|
||||||
|
ReplyKeyboard *inlineReplyKeyboard() {
|
||||||
|
return const_cast<ReplyKeyboard*>(static_cast<const HistoryItem*>(this)->inlineReplyKeyboard());
|
||||||
|
}
|
||||||
|
|
||||||
|
TextSelection toMediaSelection(TextSelection selection) const {
|
||||||
|
return internal::unshiftSelection(selection, _text);
|
||||||
|
}
|
||||||
|
TextSelection fromMediaSelection(TextSelection selection) const {
|
||||||
|
return internal::shiftSelection(selection, _text);
|
||||||
|
}
|
||||||
|
|
||||||
|
Text _text = { int(st::msgMinWidth) };
|
||||||
|
int _textWidth = -1;
|
||||||
|
int _textHeight = 0;
|
||||||
|
|
||||||
|
HistoryMediaPtr _media;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
// make all the constructors in HistoryItem children protected
|
||||||
|
// and wrapped with a static create() call with the same args
|
||||||
|
// so that history item can not be created directly, without
|
||||||
|
// calling a virtual finishCreate() method
|
||||||
|
template <typename T>
|
||||||
|
class HistoryItemInstantiated {
|
||||||
|
public:
|
||||||
|
template <typename ... Args>
|
||||||
|
static T *_create(Args ... args) {
|
||||||
|
T *result = new T(args ...);
|
||||||
|
result->finishCreate();
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
class MessageClickHandler : public LeftButtonClickHandler {
|
||||||
|
public:
|
||||||
|
MessageClickHandler(PeerId peer, MsgId msgid) : _peer(peer), _msgid(msgid) {
|
||||||
|
}
|
||||||
|
MessageClickHandler(HistoryItem *item) : _peer(item->history()->peer->id), _msgid(item->id) {
|
||||||
|
}
|
||||||
|
PeerId peer() const {
|
||||||
|
return _peer;
|
||||||
|
}
|
||||||
|
MsgId msgid() const {
|
||||||
|
return _msgid;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
PeerId _peer;
|
||||||
|
MsgId _msgid;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
class GoToMessageClickHandler : public MessageClickHandler {
|
||||||
|
public:
|
||||||
|
using MessageClickHandler::MessageClickHandler;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void onClickImpl() const override;
|
||||||
|
|
||||||
|
};
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,382 @@
|
||||||
|
/*
|
||||||
|
This file is part of Telegram Desktop,
|
||||||
|
the official desktop version of Telegram messaging app, see https://telegram.org
|
||||||
|
|
||||||
|
Telegram Desktop 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.
|
||||||
|
|
||||||
|
It 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.
|
||||||
|
|
||||||
|
In addition, as a special exception, the copyright holders give permission
|
||||||
|
to link the code of portions of this program with the OpenSSL library.
|
||||||
|
|
||||||
|
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
|
||||||
|
Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
|
||||||
|
*/
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
void historyInitMessages();
|
||||||
|
|
||||||
|
class HistoryMessage : public HistoryItem, private HistoryItemInstantiated<HistoryMessage> {
|
||||||
|
public:
|
||||||
|
static HistoryMessage *create(History *history, const MTPDmessage &msg) {
|
||||||
|
return _create(history, msg);
|
||||||
|
}
|
||||||
|
static HistoryMessage *create(History *history, MsgId msgId, MTPDmessage::Flags flags, QDateTime date, int32 from, HistoryMessage *fwd) {
|
||||||
|
return _create(history, msgId, flags, date, from, fwd);
|
||||||
|
}
|
||||||
|
static HistoryMessage *create(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, const TextWithEntities &textWithEntities) {
|
||||||
|
return _create(history, msgId, flags, replyTo, viaBotId, date, from, textWithEntities);
|
||||||
|
}
|
||||||
|
static HistoryMessage *create(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, DocumentData *doc, const QString &caption, const MTPReplyMarkup &markup) {
|
||||||
|
return _create(history, msgId, flags, replyTo, viaBotId, date, from, doc, caption, markup);
|
||||||
|
}
|
||||||
|
static HistoryMessage *create(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, PhotoData *photo, const QString &caption, const MTPReplyMarkup &markup) {
|
||||||
|
return _create(history, msgId, flags, replyTo, viaBotId, date, from, photo, caption, markup);
|
||||||
|
}
|
||||||
|
|
||||||
|
void initTime();
|
||||||
|
void initMedia(const MTPMessageMedia *media, QString ¤tText);
|
||||||
|
void initMediaFromDocument(DocumentData *doc, const QString &caption);
|
||||||
|
void fromNameUpdated(int32 width) const;
|
||||||
|
|
||||||
|
int32 plainMaxWidth() const;
|
||||||
|
void countPositionAndSize(int32 &left, int32 &width) const;
|
||||||
|
|
||||||
|
bool drawBubble() const {
|
||||||
|
return _media ? (!emptyText() || _media->needsBubble()) : !isEmpty();
|
||||||
|
}
|
||||||
|
bool hasBubble() const override {
|
||||||
|
return drawBubble();
|
||||||
|
}
|
||||||
|
bool displayFromName() const {
|
||||||
|
if (!hasFromName()) return false;
|
||||||
|
if (isAttachedToPrevious()) return false;
|
||||||
|
|
||||||
|
return (!emptyText() || !_media || !_media->isDisplayed() || Has<HistoryMessageReply>() || Has<HistoryMessageForwarded>() || viaBot() || !_media->hideFromName());
|
||||||
|
}
|
||||||
|
bool displayEditedBadge(bool hasViaBot) const;
|
||||||
|
bool uploading() const {
|
||||||
|
return _media && _media->uploading();
|
||||||
|
}
|
||||||
|
|
||||||
|
void drawInfo(Painter &p, int32 right, int32 bottom, int32 width, bool selected, InfoDisplayType type) const override;
|
||||||
|
void setViewsCount(int32 count) override;
|
||||||
|
void setId(MsgId newId) override;
|
||||||
|
void draw(Painter &p, const QRect &r, TextSelection selection, uint64 ms) const override;
|
||||||
|
|
||||||
|
void dependencyItemRemoved(HistoryItem *dependency) override;
|
||||||
|
|
||||||
|
bool hasPoint(int x, int y) const override;
|
||||||
|
bool pointInTime(int32 right, int32 bottom, int x, int y, InfoDisplayType type) const override;
|
||||||
|
|
||||||
|
HistoryTextState getState(int x, int y, HistoryStateRequest request) const override;
|
||||||
|
|
||||||
|
TextSelection adjustSelection(TextSelection selection, TextSelectType type) const override;
|
||||||
|
|
||||||
|
// ClickHandlerHost interface
|
||||||
|
void clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) override {
|
||||||
|
if (_media) _media->clickHandlerActiveChanged(p, active);
|
||||||
|
HistoryItem::clickHandlerActiveChanged(p, active);
|
||||||
|
}
|
||||||
|
void clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) override {
|
||||||
|
if (_media) _media->clickHandlerPressedChanged(p, pressed);
|
||||||
|
HistoryItem::clickHandlerPressedChanged(p, pressed);
|
||||||
|
}
|
||||||
|
|
||||||
|
QString notificationHeader() const override;
|
||||||
|
|
||||||
|
void applyEdition(const MTPDmessage &message) override;
|
||||||
|
void applyEdition(const MTPDmessageService &message) override;
|
||||||
|
void updateMedia(const MTPMessageMedia *media) override;
|
||||||
|
int32 addToOverview(AddToOverviewMethod method) override;
|
||||||
|
void eraseFromOverview() override;
|
||||||
|
|
||||||
|
TextWithEntities selectedText(TextSelection selection) const override;
|
||||||
|
void setText(const TextWithEntities &textWithEntities) override;
|
||||||
|
TextWithEntities originalText() const override;
|
||||||
|
bool textHasLinks() const override;
|
||||||
|
|
||||||
|
int32 infoWidth() const override {
|
||||||
|
int32 result = _timeWidth;
|
||||||
|
if (const HistoryMessageViews *views = Get<HistoryMessageViews>()) {
|
||||||
|
result += st::msgDateViewsSpace + views->_viewsWidth + st::msgDateCheckSpace + st::msgViewsImg.pxWidth();
|
||||||
|
} else if (id < 0 && history()->peer->isSelf()) {
|
||||||
|
result += st::msgDateCheckSpace + st::msgCheckImg.pxWidth();
|
||||||
|
}
|
||||||
|
if (out() && !isPost()) {
|
||||||
|
result += st::msgDateCheckSpace + st::msgCheckImg.pxWidth();
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
int32 timeLeft() const override {
|
||||||
|
int32 result = 0;
|
||||||
|
if (const HistoryMessageViews *views = Get<HistoryMessageViews>()) {
|
||||||
|
result += st::msgDateViewsSpace + views->_viewsWidth + st::msgDateCheckSpace + st::msgViewsImg.pxWidth();
|
||||||
|
} else if (id < 0 && history()->peer->isSelf()) {
|
||||||
|
result += st::msgDateCheckSpace + st::msgCheckImg.pxWidth();
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
int32 timeWidth() const override {
|
||||||
|
return _timeWidth;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32 viewsCount() const override {
|
||||||
|
if (const HistoryMessageViews *views = Get<HistoryMessageViews>()) {
|
||||||
|
return views->_views;
|
||||||
|
}
|
||||||
|
return HistoryItem::viewsCount();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool updateDependencyItem() override {
|
||||||
|
if (auto reply = Get<HistoryMessageReply>()) {
|
||||||
|
return reply->updateData(this, true);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
MsgId dependencyMsgId() const override {
|
||||||
|
return replyToId();
|
||||||
|
}
|
||||||
|
|
||||||
|
HistoryMessage *toHistoryMessage() override { // dynamic_cast optimize
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
const HistoryMessage *toHistoryMessage() const override { // dynamic_cast optimize
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
// hasFromPhoto() returns true even if we don't display the photo
|
||||||
|
// but we need to skip a place at the left side for this photo
|
||||||
|
bool displayFromPhoto() const;
|
||||||
|
bool hasFromPhoto() const;
|
||||||
|
|
||||||
|
~HistoryMessage();
|
||||||
|
|
||||||
|
private:
|
||||||
|
HistoryMessage(History *history, const MTPDmessage &msg);
|
||||||
|
HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags flags, QDateTime date, int32 from, HistoryMessage *fwd); // local forwarded
|
||||||
|
HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, const TextWithEntities &textWithEntities); // local message
|
||||||
|
HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, DocumentData *doc, const QString &caption, const MTPReplyMarkup &markup); // local document
|
||||||
|
HistoryMessage(History *history, MsgId msgId, MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, QDateTime date, int32 from, PhotoData *photo, const QString &caption, const MTPReplyMarkup &markup); // local photo
|
||||||
|
friend class HistoryItemInstantiated<HistoryMessage>;
|
||||||
|
|
||||||
|
void setEmptyText();
|
||||||
|
|
||||||
|
void initDimensions() override;
|
||||||
|
int resizeGetHeight_(int width) override;
|
||||||
|
int performResizeGetHeight(int width);
|
||||||
|
void applyEditionToEmpty();
|
||||||
|
|
||||||
|
bool displayForwardedFrom() const {
|
||||||
|
if (const HistoryMessageForwarded *fwd = Get<HistoryMessageForwarded>()) {
|
||||||
|
return Has<HistoryMessageVia>() || !_media || !_media->isDisplayed() || fwd->_authorOriginal->isChannel() || !_media->hideForwardedFrom();
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
void paintFromName(Painter &p, QRect &trect, bool selected) const;
|
||||||
|
void paintForwardedInfo(Painter &p, QRect &trect, bool selected) const;
|
||||||
|
void paintReplyInfo(Painter &p, QRect &trect, bool selected) const;
|
||||||
|
|
||||||
|
// this method draws "via @bot" if it is not painted in forwarded info or in from name
|
||||||
|
void paintViaBotIdInfo(Painter &p, QRect &trect, bool selected) const;
|
||||||
|
|
||||||
|
void paintText(Painter &p, QRect &trect, TextSelection selection) const;
|
||||||
|
|
||||||
|
void setMedia(const MTPMessageMedia *media);
|
||||||
|
void setReplyMarkup(const MTPReplyMarkup *markup);
|
||||||
|
|
||||||
|
QString _timeText;
|
||||||
|
int _timeWidth = 0;
|
||||||
|
|
||||||
|
struct CreateConfig {
|
||||||
|
MsgId replyTo = 0;
|
||||||
|
UserId viaBotId = 0;
|
||||||
|
int viewsCount = -1;
|
||||||
|
PeerId authorIdOriginal = 0;
|
||||||
|
PeerId fromIdOriginal = 0;
|
||||||
|
MsgId originalId = 0;
|
||||||
|
QDateTime editDate;
|
||||||
|
const MTPReplyMarkup *markup = nullptr;
|
||||||
|
};
|
||||||
|
void createComponentsHelper(MTPDmessage::Flags flags, MsgId replyTo, int32 viaBotId, const MTPReplyMarkup &markup);
|
||||||
|
void createComponents(const CreateConfig &config);
|
||||||
|
|
||||||
|
class KeyboardStyle : public ReplyKeyboard::Style {
|
||||||
|
public:
|
||||||
|
using ReplyKeyboard::Style::Style;
|
||||||
|
|
||||||
|
void startPaint(Painter &p) const override;
|
||||||
|
style::font textFont() const override;
|
||||||
|
void repaint(const HistoryItem *item) const override;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void paintButtonBg(Painter &p, const QRect &rect, bool down, float64 howMuchOver) const override;
|
||||||
|
void paintButtonIcon(Painter &p, const QRect &rect, HistoryMessageReplyMarkup::Button::Type type) const override;
|
||||||
|
void paintButtonLoading(Painter &p, const QRect &rect) const override;
|
||||||
|
int minButtonWidth(HistoryMessageReplyMarkup::Button::Type type) const override;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
inline MTPDmessage::Flags newMessageFlags(PeerData *p) {
|
||||||
|
MTPDmessage::Flags result = 0;
|
||||||
|
if (!p->isSelf()) {
|
||||||
|
result |= MTPDmessage::Flag::f_out;
|
||||||
|
//if (p->isChat() || (p->isUser() && !p->asUser()->botInfo)) {
|
||||||
|
// result |= MTPDmessage::Flag::f_unread;
|
||||||
|
//}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct HistoryServiceDependentData {
|
||||||
|
MsgId msgId = 0;
|
||||||
|
HistoryItem *msg = nullptr;
|
||||||
|
ClickHandlerPtr lnk;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct HistoryServicePinned : public BaseComponent<HistoryServicePinned>, public HistoryServiceDependentData {
|
||||||
|
};
|
||||||
|
|
||||||
|
struct HistoryServiceGameScore : public BaseComponent<HistoryServiceGameScore>, public HistoryServiceDependentData {
|
||||||
|
int score = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
namespace HistoryLayout {
|
||||||
|
class ServiceMessagePainter;
|
||||||
|
} // namespace HistoryLayout
|
||||||
|
|
||||||
|
class HistoryService : public HistoryItem, private HistoryItemInstantiated<HistoryService> {
|
||||||
|
public:
|
||||||
|
static HistoryService *create(History *history, const MTPDmessageService &msg) {
|
||||||
|
return _create(history, msg);
|
||||||
|
}
|
||||||
|
static HistoryService *create(History *history, MsgId msgId, QDateTime date, const QString &msg, MTPDmessage::Flags flags = 0, int32 from = 0) {
|
||||||
|
return _create(history, msgId, date, msg, flags, from);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool updateDependencyItem() override;
|
||||||
|
MsgId dependencyMsgId() const override {
|
||||||
|
if (auto dependent = GetDependentData()) {
|
||||||
|
return dependent->msgId;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
bool notificationReady() const override {
|
||||||
|
if (auto dependent = GetDependentData()) {
|
||||||
|
return (dependent->msg || !dependent->msgId);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void countPositionAndSize(int32 &left, int32 &width) const;
|
||||||
|
|
||||||
|
void draw(Painter &p, const QRect &r, TextSelection selection, uint64 ms) const override;
|
||||||
|
bool hasPoint(int x, int y) const override;
|
||||||
|
HistoryTextState getState(int x, int y, HistoryStateRequest request) const override;
|
||||||
|
|
||||||
|
TextSelection adjustSelection(TextSelection selection, TextSelectType type) const override {
|
||||||
|
return _text.adjustSelection(selection, type);
|
||||||
|
}
|
||||||
|
|
||||||
|
void clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) override {
|
||||||
|
if (_media) _media->clickHandlerActiveChanged(p, active);
|
||||||
|
HistoryItem::clickHandlerActiveChanged(p, active);
|
||||||
|
}
|
||||||
|
void clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) override {
|
||||||
|
if (_media) _media->clickHandlerPressedChanged(p, pressed);
|
||||||
|
HistoryItem::clickHandlerPressedChanged(p, pressed);
|
||||||
|
}
|
||||||
|
|
||||||
|
void applyEdition(const MTPDmessageService &message) override;
|
||||||
|
|
||||||
|
int32 addToOverview(AddToOverviewMethod method) override;
|
||||||
|
void eraseFromOverview() override;
|
||||||
|
|
||||||
|
bool needCheck() const override {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
bool serviceMsg() const override {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
TextWithEntities selectedText(TextSelection selection) const override;
|
||||||
|
QString inDialogsText() const override;
|
||||||
|
QString inReplyText() const override;
|
||||||
|
|
||||||
|
~HistoryService();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
friend class HistoryLayout::ServiceMessagePainter;
|
||||||
|
|
||||||
|
HistoryService(History *history, const MTPDmessageService &msg);
|
||||||
|
HistoryService(History *history, MsgId msgId, QDateTime date, const QString &msg, MTPDmessage::Flags flags = 0, int32 from = 0);
|
||||||
|
friend class HistoryItemInstantiated<HistoryService>;
|
||||||
|
|
||||||
|
void initDimensions() override;
|
||||||
|
int resizeGetHeight_(int width) override;
|
||||||
|
|
||||||
|
using Links = QList<ClickHandlerPtr>;
|
||||||
|
void setServiceText(const QString &text, const Links &links);
|
||||||
|
|
||||||
|
void removeMedia();
|
||||||
|
|
||||||
|
private:
|
||||||
|
HistoryServiceDependentData *GetDependentData() {
|
||||||
|
if (auto pinned = Get<HistoryServicePinned>()) {
|
||||||
|
return pinned;
|
||||||
|
} else if (auto gamescore = Get<HistoryServiceGameScore>()) {
|
||||||
|
return gamescore;
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
const HistoryServiceDependentData *GetDependentData() const {
|
||||||
|
return const_cast<HistoryService*>(this)->GetDependentData();
|
||||||
|
}
|
||||||
|
bool updateDependent(bool force = false);
|
||||||
|
bool updateDependentText();
|
||||||
|
void clearDependency();
|
||||||
|
|
||||||
|
void createFromMtp(const MTPDmessageService &message);
|
||||||
|
void setMessageByAction(const MTPmessageAction &action);
|
||||||
|
|
||||||
|
bool preparePinnedText(const QString &from, QString *outText, Links *outLinks);
|
||||||
|
bool prepareGameScoreText(const QString &from, QString *outText, Links *outLinks);
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
class HistoryJoined : public HistoryService, private HistoryItemInstantiated<HistoryJoined> {
|
||||||
|
public:
|
||||||
|
static HistoryJoined *create(History *history, const QDateTime &date, UserData *from, MTPDmessage::Flags flags) {
|
||||||
|
return _create(history, date, from, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
HistoryItemType type() const {
|
||||||
|
return HistoryItemJoined;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
HistoryJoined(History *history, const QDateTime &date, UserData *from, MTPDmessage::Flags flags);
|
||||||
|
using HistoryItemInstantiated<HistoryJoined>::_create;
|
||||||
|
friend class HistoryItemInstantiated<HistoryJoined>;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
class ViaInlineBotClickHandler : public LeftButtonClickHandler {
|
||||||
|
public:
|
||||||
|
ViaInlineBotClickHandler(UserData *bot) : _bot(bot) {
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void onClickImpl() const override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
UserData *_bot;
|
||||||
|
|
||||||
|
};
|
|
@ -77,6 +77,10 @@ const TextParseOptions &itemTextOptions(History *h, PeerData *f) {
|
||||||
return _historyTextOptions;
|
return _historyTextOptions;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const TextParseOptions &itemTextOptions(const HistoryItem *item) {
|
||||||
|
return itemTextOptions(item->history(), item->author());
|
||||||
|
}
|
||||||
|
|
||||||
const TextParseOptions &itemTextNoMonoOptions(History *h, PeerData *f) {
|
const TextParseOptions &itemTextNoMonoOptions(History *h, PeerData *f) {
|
||||||
if ((h->peer->isUser() && h->peer->asUser()->botInfo) || (f->isUser() && f->asUser()->botInfo) || (h->peer->isChat() && h->peer->asChat()->botStatus >= 0) || (h->peer->isMegagroup() && h->peer->asChannel()->mgInfo->botStatus >= 0)) {
|
if ((h->peer->isUser() && h->peer->asUser()->botInfo) || (f->isUser() && f->asUser()->botInfo) || (h->peer->isChat() && h->peer->asChat()->botStatus >= 0) || (h->peer->isMegagroup() && h->peer->asChannel()->mgInfo->botStatus >= 0)) {
|
||||||
return _historyBotNoMonoOptions;
|
return _historyBotNoMonoOptions;
|
||||||
|
@ -84,6 +88,10 @@ const TextParseOptions &itemTextNoMonoOptions(History *h, PeerData *f) {
|
||||||
return _historyTextNoMonoOptions;
|
return _historyTextNoMonoOptions;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const TextParseOptions &itemTextNoMonoOptions(const HistoryItem *item) {
|
||||||
|
return itemTextNoMonoOptions(item->history(), item->author());
|
||||||
|
}
|
||||||
|
|
||||||
QString formatSizeText(qint64 size) {
|
QString formatSizeText(qint64 size) {
|
||||||
if (size >= 1024 * 1024) { // more than 1 mb
|
if (size >= 1024 * 1024) { // more than 1 mb
|
||||||
qint64 sizeTenthMb = (size * 10 / (1024 * 1024));
|
qint64 sizeTenthMb = (size * 10 / (1024 * 1024));
|
||||||
|
|
|
@ -26,7 +26,9 @@ extern TextParseOptions _textNameOptions, _textDlgOptions;
|
||||||
extern TextParseOptions _historyTextOptions, _historyBotOptions, _historyTextNoMonoOptions, _historyBotNoMonoOptions;
|
extern TextParseOptions _historyTextOptions, _historyBotOptions, _historyTextNoMonoOptions, _historyBotNoMonoOptions;
|
||||||
|
|
||||||
const TextParseOptions &itemTextOptions(History *h, PeerData *f);
|
const TextParseOptions &itemTextOptions(History *h, PeerData *f);
|
||||||
|
const TextParseOptions &itemTextOptions(const HistoryItem *item);
|
||||||
const TextParseOptions &itemTextNoMonoOptions(History *h, PeerData *f);
|
const TextParseOptions &itemTextNoMonoOptions(History *h, PeerData *f);
|
||||||
|
const TextParseOptions &itemTextNoMonoOptions(const HistoryItem *item);
|
||||||
|
|
||||||
enum RoundCorners {
|
enum RoundCorners {
|
||||||
SmallMaskCorners = 0x00, // for images
|
SmallMaskCorners = 0x00, // for images
|
||||||
|
|
|
@ -24,7 +24,6 @@ Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
|
||||||
#include "lang.h"
|
#include "lang.h"
|
||||||
#include "inline_bots/inline_bot_layout_item.h"
|
#include "inline_bots/inline_bot_layout_item.h"
|
||||||
#include "observer_peer.h"
|
#include "observer_peer.h"
|
||||||
#include "history.h"
|
|
||||||
#include "mainwidget.h"
|
#include "mainwidget.h"
|
||||||
#include "application.h"
|
#include "application.h"
|
||||||
#include "fileuploader.h"
|
#include "fileuploader.h"
|
||||||
|
|
|
@ -218,8 +218,14 @@
|
||||||
'<(src_loc)/dialogs/dialogs_row.h',
|
'<(src_loc)/dialogs/dialogs_row.h',
|
||||||
'<(src_loc)/history/field_autocomplete.cpp',
|
'<(src_loc)/history/field_autocomplete.cpp',
|
||||||
'<(src_loc)/history/field_autocomplete.h',
|
'<(src_loc)/history/field_autocomplete.h',
|
||||||
|
'<(src_loc)/history/history_item.cpp',
|
||||||
|
'<(src_loc)/history/history_item.h',
|
||||||
'<(src_loc)/history/history_location_manager.cpp',
|
'<(src_loc)/history/history_location_manager.cpp',
|
||||||
'<(src_loc)/history/history_location_manager.h',
|
'<(src_loc)/history/history_location_manager.h',
|
||||||
|
'<(src_loc)/history/history_media.cpp',
|
||||||
|
'<(src_loc)/history/history_media.h',
|
||||||
|
'<(src_loc)/history/history_message.cpp',
|
||||||
|
'<(src_loc)/history/history_message.h',
|
||||||
'<(src_loc)/history/history_service_layout.cpp',
|
'<(src_loc)/history/history_service_layout.cpp',
|
||||||
'<(src_loc)/history/history_service_layout.h',
|
'<(src_loc)/history/history_service_layout.h',
|
||||||
'<(src_loc)/inline_bots/inline_bot_layout_internal.cpp',
|
'<(src_loc)/inline_bots/inline_bot_layout_internal.cpp',
|
||||||
|
|
Loading…
Reference in New Issue