mirror of https://github.com/procxx/kepka.git
				
				
				
			
		
			
				
	
	
		
			1227 lines
		
	
	
		
			34 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			1227 lines
		
	
	
		
			34 KiB
		
	
	
	
		
			C++
		
	
	
	
| /*
 | |
| 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-2017 John Preston, https://desktop.telegram.org
 | |
| */
 | |
| #include "history/history_item.h"
 | |
| 
 | |
| #include "lang/lang_keys.h"
 | |
| #include "mainwidget.h"
 | |
| #include "history/history_service_layout.h"
 | |
| #include "history/history_media_types.h"
 | |
| #include "history/history_message.h"
 | |
| #include "media/media_clip_reader.h"
 | |
| #include "styles/style_dialogs.h"
 | |
| #include "styles/style_history.h"
 | |
| #include "ui/effects/ripple_animation.h"
 | |
| #include "storage/file_upload.h"
 | |
| #include "auth_session.h"
 | |
| #include "media/media_audio.h"
 | |
| #include "messenger.h"
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| // a new message from the same sender is attached to previous within 15 minutes
 | |
| constexpr int kAttachMessageToPreviousSecondsDelta = 900;
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| ReplyMarkupClickHandler::ReplyMarkupClickHandler(const HistoryItem *item, int row, int col)
 | |
| : _itemId(item->fullId())
 | |
| , _row(row)
 | |
| , _col(col) {
 | |
| }
 | |
| 
 | |
| // Copy to clipboard support.
 | |
| void ReplyMarkupClickHandler::copyToClipboard() const {
 | |
| 	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 ReplyMarkupClickHandler::copyToClipboardContextItemText() const {
 | |
| 	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 *ReplyMarkupClickHandler::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;
 | |
| }
 | |
| 
 | |
| void ReplyMarkupClickHandler::onClickImpl() const {
 | |
| 	if (auto item = App::histItemById(_itemId)) {
 | |
| 		App::activateBotCommand(item, _row, _col);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Returns the full text of the corresponding button.
 | |
| QString ReplyMarkupClickHandler::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::move(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->textStyle(), TextUtilities::SingleLine(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>();
 | |
| 	double y = 0, buttonHeight = _rows.isEmpty() ? _st->buttonHeight() : (double(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 += std::max(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;
 | |
| 
 | |
| 		double x = 0;
 | |
| 		for (Button &button : row) {
 | |
| 			int buttonw = std::max(button.text.maxWidth(), 1);
 | |
| 			double textw = buttonw, minw = _st->minButtonWidth(button.type);
 | |
| 			double w = textw;
 | |
| 			if (exact) {
 | |
| 				w += minw;
 | |
| 			} else if (enough) {
 | |
| 				w = (widthForButtons / double(s));
 | |
| 				textw = w - minw;
 | |
| 			} else {
 | |
| 				textw = (widthForText / double(s));
 | |
| 				w = minw + textw;
 | |
| 				accumulate_max(w, 2 * double(_st->buttonPadding()));
 | |
| 			}
 | |
| 
 | |
| 			int rectx = static_cast<int>(std::floor(x));
 | |
| 			int rectw = static_cast<int>(std::floor(x + w)) - rectx;
 | |
| 			button.rect = QRect(rectx, std::round(y), rectw, std::round(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 -= std::max(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, std::max(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, int outerWidth, const QRect &clip, TimeMs ms) const {
 | |
| 	Assert(_st != nullptr);
 | |
| 	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, outerWidth, button, ms);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| ClickHandlerPtr ReplyKeyboard::getState(QPoint point) const {
 | |
| 	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(point)) {
 | |
| 				_savedCoords = point;
 | |
| 				return button.link;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return ClickHandlerPtr();
 | |
| }
 | |
| 
 | |
| void ReplyKeyboard::clickHandlerActiveChanged(const ClickHandlerPtr &p, bool active) {
 | |
| 	if (!p) return;
 | |
| 
 | |
| 	_savedActive = active ? p : ClickHandlerPtr();
 | |
| 	auto coords = findButtonCoordsByClickHandler(p);
 | |
| 	if (coords.i >= 0 && _savedPressed != p) {
 | |
| 		startAnimation(coords.i, coords.j, active ? 1 : -1);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| ReplyKeyboard::ButtonCoords ReplyKeyboard::findButtonCoordsByClickHandler(const ClickHandlerPtr &p) {
 | |
| 	for (int i = 0, rows = _rows.size(); i != rows; ++i) {
 | |
| 		auto &row = _rows[i];
 | |
| 		for (int j = 0, cols = row.size(); j != cols; ++j) {
 | |
| 			if (row[j].link == p) {
 | |
| 				return { i, j };
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return { -1, -1 };
 | |
| }
 | |
| 
 | |
| void ReplyKeyboard::clickHandlerPressedChanged(const ClickHandlerPtr &p, bool pressed) {
 | |
| 	if (!p) return;
 | |
| 
 | |
| 	_savedPressed = pressed ? p : ClickHandlerPtr();
 | |
| 	auto coords = findButtonCoordsByClickHandler(p);
 | |
| 	if (coords.i >= 0) {
 | |
| 		auto &button = _rows[coords.i][coords.j];
 | |
| 		if (pressed) {
 | |
| 			if (!button.ripple) {
 | |
| 				auto mask = Ui::RippleAnimation::roundRectMask(button.rect.size(), _st->buttonRadius());
 | |
| 				button.ripple = MakeShared<Ui::RippleAnimation>(_st->_st->ripple, std::move(mask), [this] { _st->repaint(_item); });
 | |
| 			}
 | |
| 			button.ripple->add(_savedCoords - button.rect.topLeft());
 | |
| 		} else {
 | |
| 			if (button.ripple) {
 | |
| 				button.ripple->lastStop();
 | |
| 			}
 | |
| 			if (_savedActive != p) {
 | |
| 				startAnimation(coords.i, coords.j, -1);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ReplyKeyboard::startAnimation(int i, int j, int direction) {
 | |
| 	auto notStarted = _animations.isEmpty();
 | |
| 
 | |
| 	int indexForAnimation = (i * MatrixRowShift + j + 1) * direction;
 | |
| 
 | |
| 	_animations.remove(-indexForAnimation);
 | |
| 	if (!_animations.contains(indexForAnimation)) {
 | |
| 		_animations.insert(indexForAnimation, getms());
 | |
| 	}
 | |
| 
 | |
| 	if (notStarted && !_a_selected.animating()) {
 | |
| 		_a_selected.start();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ReplyKeyboard::step_selected(TimeMs ms, bool timer) {
 | |
| 	for (Animations::iterator i = _animations.begin(); i != _animations.end();) {
 | |
| 		int index = qAbs(i.key()) - 1, row = (index / MatrixRowShift), col = index % MatrixRowShift;
 | |
| 		double dt = double(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();
 | |
| }
 | |
| 
 | |
| int ReplyKeyboard::Style::buttonSkip() const {
 | |
| 	return _st->margin;
 | |
| }
 | |
| 
 | |
| int ReplyKeyboard::Style::buttonPadding() const {
 | |
| 	return _st->padding;
 | |
| }
 | |
| 
 | |
| int ReplyKeyboard::Style::buttonHeight() const {
 | |
| 	return _st->height;
 | |
| }
 | |
| 
 | |
| void ReplyKeyboard::Style::paintButton(Painter &p, int outerWidth, const ReplyKeyboard::Button &button, TimeMs ms) const {
 | |
| 	const QRect &rect = button.rect;
 | |
| 	paintButtonBg(p, rect, button.howMuchOver);
 | |
| 	if (button.ripple) {
 | |
| 		button.ripple->paint(p, rect.x(), rect.y(), outerWidth, ms);
 | |
| 		if (button.ripple->empty()) {
 | |
| 			button.ripple.reset();
 | |
| 		}
 | |
| 	}
 | |
| 	paintButtonIcon(p, rect, outerWidth, 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::botKbStyle.font->elidew + _st->padding * 2) {
 | |
| 		tx += _st->padding;
 | |
| 		tw -= _st->padding * 2;
 | |
| 	} else if (tw > st::botKbStyle.font->elidew) {
 | |
| 		tx += (tw - st::botKbStyle.font->elidew) / 2;
 | |
| 		tw = st::botKbStyle.font->elidew;
 | |
| 	}
 | |
| 	button.text.drawElided(p, tx, rect.y() + _st->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.v;
 | |
| 			if (!b.isEmpty()) {
 | |
| 				ButtonRow buttonRow;
 | |
| 				buttonRow.reserve(b.size());
 | |
| 				for_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;
 | |
| 					case mtpc_keyboardButtonBuy: {
 | |
| 						auto &buttonData = button.c_keyboardButtonBuy();
 | |
| 						buttonRow.push_back({ Button::Type::Buy, qs(buttonData.vtext), QByteArray(), 0 });
 | |
| 					}
 | |
| 					}
 | |
| 				}
 | |
| 				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: {
 | |
| 		auto &d = markup.c_replyKeyboardMarkup();
 | |
| 		flags = d.vflags.v;
 | |
| 
 | |
| 		createFromButtonRows(d.vrows.v);
 | |
| 	} break;
 | |
| 
 | |
| 	case mtpc_replyInlineMarkup: {
 | |
| 		auto &d = markup.c_replyInlineMarkup();
 | |
| 		flags = MTPDreplyKeyboardMarkup::Flags(0) | MTPDreplyKeyboardMarkup_ClientFlag::f_inline;
 | |
| 
 | |
| 		createFromButtonRows(d.vrows.v);
 | |
| 	} break;
 | |
| 
 | |
| 	case mtpc_replyKeyboardHide: {
 | |
| 		auto &d = markup.c_replyKeyboardHide();
 | |
| 		flags = mtpCastFlags(d.vflags) | MTPDreplyKeyboardMarkup_ClientFlag::f_zero;
 | |
| 	} break;
 | |
| 
 | |
| 	case mtpc_replyKeyboardForceReply: {
 | |
| 		auto &d = markup.c_replyKeyboardForceReply();
 | |
| 		flags = mtpCastFlags(d.vflags) | MTPDreplyKeyboardMarkup_ClientFlag::f_force_reply;
 | |
| 	} break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryMessageReplyMarkup::create(const HistoryMessageReplyMarkup &markup) {
 | |
| 	flags = markup.flags;
 | |
| 	inlineKeyboard = nullptr;
 | |
| 
 | |
| 	rows.clear();
 | |
| 	for_const (auto &row, markup.rows) {
 | |
| 		ButtonRow buttonRow;
 | |
| 		buttonRow.reserve(row.size());
 | |
| 		for_const (auto &button, row) {
 | |
| 			buttonRow.push_back({ button.type, button.text, button.data, 0 });
 | |
| 		}
 | |
| 		if (!buttonRow.isEmpty()) rows.push_back(buttonRow);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 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::historyUnreadBarHeight + st::historyUnreadBarMargin;
 | |
| }
 | |
| 
 | |
| int HistoryMessageUnreadBar::marginTop() {
 | |
| 	return st::lineWidth + st::historyUnreadBarMargin;
 | |
| }
 | |
| 
 | |
| void HistoryMessageUnreadBar::paint(Painter &p, int y, int w) const {
 | |
| 	p.fillRect(0, y + marginTop(), w, height() - marginTop() - st::lineWidth, st::historyUnreadBarBg);
 | |
| 	p.fillRect(0, y + height() - st::lineWidth, w, st::lineWidth, st::historyUnreadBarBorder);
 | |
| 	p.setFont(st::historyUnreadBarFont);
 | |
| 	p.setPen(st::historyUnreadBarFg);
 | |
| 
 | |
| 	int left = st::msgServiceMargin.left();
 | |
| 	int maxwidth = w;
 | |
| 	if (Adaptive::ChatWide()) {
 | |
| 		maxwidth = std::min(maxwidth, qint32(st::msgMaxWidth + 2 * st::msgPhotoSkip + 2 * st::msgMargin.left()));
 | |
| 	}
 | |
| 	w = maxwidth;
 | |
| 
 | |
| 	p.drawText((w - _width) / 2, y + marginTop() + (st::historyUnreadBarHeight - 2 * st::lineWidth - st::historyUnreadBarFont->height) / 2 + st::historyUnreadBarFont->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);
 | |
| }
 | |
| 
 | |
| HistoryMessageLogEntryOriginal::HistoryMessageLogEntryOriginal() = default;
 | |
| 
 | |
| HistoryMessageLogEntryOriginal::HistoryMessageLogEntryOriginal(HistoryMessageLogEntryOriginal &&other) : _page(std::move(other._page)) {
 | |
| }
 | |
| 
 | |
| HistoryMessageLogEntryOriginal &HistoryMessageLogEntryOriginal::operator=(HistoryMessageLogEntryOriginal &&other) {
 | |
| 	_page = std::move(other._page);
 | |
| 	return *this;
 | |
| }
 | |
| 
 | |
| HistoryMessageLogEntryOriginal::~HistoryMessageLogEntryOriginal() = default;
 | |
| 
 | |
| HistoryMediaPtr::HistoryMediaPtr() = default;
 | |
| 
 | |
| HistoryMediaPtr::HistoryMediaPtr(std::unique_ptr<HistoryMedia> pointer) : _pointer(std::move(pointer)) {
 | |
| 	if (_pointer) {
 | |
| 		_pointer->attachToParent();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryMediaPtr::reset(std::unique_ptr<HistoryMedia> pointer) {
 | |
| 	*this = std::move(pointer);
 | |
| }
 | |
| 
 | |
| HistoryMediaPtr &HistoryMediaPtr::operator=(std::unique_ptr<HistoryMedia> pointer) {
 | |
| 	if (_pointer) {
 | |
| 		_pointer->detachFromParent();
 | |
| 	}
 | |
| 	_pointer = std::move(pointer);
 | |
| 	if (_pointer) {
 | |
| 		_pointer->attachToParent();
 | |
| 	}
 | |
| 	return *this;
 | |
| }
 | |
| 
 | |
| HistoryMediaPtr::~HistoryMediaPtr() {
 | |
| 	reset();
 | |
| }
 | |
| 
 | |
| namespace internal {
 | |
| 
 | |
| TextSelection unshiftSelection(TextSelection selection, quint16 byLength) {
 | |
| 	if (selection == FullSelection) {
 | |
| 		return selection;
 | |
| 	}
 | |
| 	return ::unshiftSelection(selection, byLength);
 | |
| }
 | |
| 
 | |
| TextSelection shiftSelection(TextSelection selection, quint16 byLength) {
 | |
| 	if (selection == FullSelection) {
 | |
| 		return selection;
 | |
| 	}
 | |
| 	return ::shiftSelection(selection, byLength);
 | |
| }
 | |
| 
 | |
| } // namespace internal
 | |
| 
 | |
| HistoryItem::HistoryItem(
 | |
| 	not_null<History*> history,
 | |
| 	MsgId id,
 | |
| 	MTPDmessage::Flags flags,
 | |
| 	QDateTime date,
 | |
| 	UserId from) : HistoryElement()
 | |
| , id(id)
 | |
| , date(date)
 | |
| , _history(history)
 | |
| , _from(from ? App::user(from) : history->peer)
 | |
| , _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()->peer, 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();
 | |
| 	}
 | |
| 	if (auto previous = previousItem()) {
 | |
| 		previous->nextItemChanged();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::markMediaRead() {
 | |
| 	_flags &= ~MTPDmessage::Flag::f_media_unread;
 | |
| 
 | |
| 	if (mentionsMe()) {
 | |
| 		history()->updateChatListEntry();
 | |
| 		history()->eraseFromUnreadMentions(id);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 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::addLogEntryOriginal(WebPageId localId, const QString &label, const TextWithEntities &content) {
 | |
| 	Expects(isLogEntry());
 | |
| 	AddComponents(HistoryMessageLogEntryOriginal::Bit());
 | |
| 	auto original = Get<HistoryMessageLogEntryOriginal>();
 | |
| 	auto webpage = App::feedWebPage(localId, label, content);
 | |
| 	original->_page = std::make_unique<HistoryWebPage>(this, webpage);
 | |
| }
 | |
| 
 | |
| void HistoryItem::destroy() {
 | |
| 	if (isLogEntry()) {
 | |
| 		Assert(detached());
 | |
| 	} else {
 | |
| 		// All this must be done for all items manually in History::clear(false)!
 | |
| 		eraseFromOverview();
 | |
| 
 | |
| 		auto 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() {
 | |
| 	Expects(!isLogEntry());
 | |
| 	recountDisplayDate();
 | |
| 	recountAttachToPrevious();
 | |
| }
 | |
| 
 | |
| // Called only if there is no more next item! Not always when it changes!
 | |
| void HistoryItem::nextItemChanged() {
 | |
| 	Expects(!isLogEntry());
 | |
| 	setAttachToNext(false);
 | |
| }
 | |
| 
 | |
| bool HistoryItem::computeIsAttachToPrevious(not_null<HistoryItem*> previous) {
 | |
| 	if (!Has<HistoryMessageDate>() && !Has<HistoryMessageUnreadBar>()) {
 | |
| 		return !isPost() && !previous->isPost()
 | |
| 			&& !serviceMsg() && !previous->serviceMsg()
 | |
| 			&& !isEmpty() && !previous->isEmpty()
 | |
| 			&& previous->from() == from()
 | |
| 			&& (qAbs(previous->date.secsTo(date)) < kAttachMessageToPreviousSecondsDelta);
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| void HistoryItem::recountAttachToPrevious() {
 | |
| 	Expects(!isLogEntry());
 | |
| 	auto attachToPrevious = false;
 | |
| 	if (auto previous = previousItem()) {
 | |
| 		attachToPrevious = computeIsAttachToPrevious(previous);
 | |
| 		previous->setAttachToNext(attachToPrevious);
 | |
| 	}
 | |
| 	setAttachToPrevious(attachToPrevious);
 | |
| }
 | |
| 
 | |
| void HistoryItem::setAttachToNext(bool attachToNext) {
 | |
| 	if (attachToNext && !(_flags & MTPDmessage_ClientFlag::f_attach_to_next)) {
 | |
| 		_flags |= MTPDmessage_ClientFlag::f_attach_to_next;
 | |
| 		Global::RefPendingRepaintItems().insert(this);
 | |
| 	} else if (!attachToNext && (_flags & MTPDmessage_ClientFlag::f_attach_to_next)) {
 | |
| 		_flags &= ~MTPDmessage_ClientFlag::f_attach_to_next;
 | |
| 		Global::RefPendingRepaintItems().insert(this);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::setAttachToPrevious(bool attachToPrevious) {
 | |
| 	if (attachToPrevious && !(_flags & MTPDmessage_ClientFlag::f_attach_to_previous)) {
 | |
| 		_flags |= MTPDmessage_ClientFlag::f_attach_to_previous;
 | |
| 		setPendingInitDimensions();
 | |
| 	} else if (!attachToPrevious && (_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();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (_media) {
 | |
| 		_media->updateMessageId();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool HistoryItem::canPin() const {
 | |
| 	if (id < 0 || !_history->peer->isMegagroup() || !toHistoryMessage()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	if (auto channel = _history->peer->asMegagroup()) {
 | |
| 		return channel->canPinMessages();
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::canForward() const {
 | |
| 	if (id < 0 || isLogEntry()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	if (auto message = toHistoryMessage()) {
 | |
| 		if (auto media = message->getMedia()) {
 | |
| 			if (media->type() == MediaTypeCall) {
 | |
| 				return false;
 | |
| 			}
 | |
| 		}
 | |
| 		return true;
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::canEdit(const QDateTime &cur) const {
 | |
| 	auto messageToMyself = _history->peer->isSelf();
 | |
| 	auto messageTooOld = messageToMyself ? false : (date.secsTo(cur) >= Global::EditTimeLimit());
 | |
| 	if (id < 0 || messageTooOld) {
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	if (auto message = toHistoryMessage()) {
 | |
| 		if (message->Has<HistoryMessageVia>() || message->Has<HistoryMessageForwarded>()) {
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		if (auto media = message->getMedia()) {
 | |
| 			if (!media->canEditCaption() && media->type() != MediaTypeWebPage) {
 | |
| 				return false;
 | |
| 			}
 | |
| 		}
 | |
| 		if (messageToMyself) {
 | |
| 			return true;
 | |
| 		}
 | |
| 		if (auto channel = _history->peer->asChannel()) {
 | |
| 			if (isPost() && channel->canEditMessages()) {
 | |
| 				return true;
 | |
| 			}
 | |
| 			if (out()) {
 | |
| 				return !isPost() || channel->canPublish();
 | |
| 			}
 | |
| 		} else {
 | |
| 			return out();
 | |
| 		}
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::canDelete() const {
 | |
| 	if (isLogEntry()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	auto channel = _history->peer->asChannel();
 | |
| 	if (!channel) {
 | |
| 		return !(_flags & MTPDmessage_ClientFlag::f_is_group_migrate);
 | |
| 	}
 | |
| 
 | |
| 	if (id == 1) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	if (channel->canDeleteMessages()) {
 | |
| 		return true;
 | |
| 	}
 | |
| 	if (out() && toHistoryMessage()) {
 | |
| 		return isPost() ? channel->canPublish() : true;
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::canDeleteForEveryone(const QDateTime &cur) const {
 | |
| 	auto messageToMyself = _history->peer->isSelf();
 | |
| 	auto messageTooOld = messageToMyself ? false : (date.secsTo(cur) >= Global::EditTimeLimit());
 | |
| 	if (id < 0 || messageToMyself || messageTooOld || isPost()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	if (history()->peer->isChannel()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	if (!toHistoryMessage()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	if (auto media = getMedia()) {
 | |
| 		if (media->type() == MediaTypeCall) {
 | |
| 			return false;
 | |
| 		}
 | |
| 	}
 | |
| 	if (!out()) {
 | |
| 		if (auto chat = history()->peer->asChat()) {
 | |
| 			if (!chat->amCreator() && (!chat->amAdmin() || !chat->adminsEnabled())) {
 | |
| 				return false;
 | |
| 			}
 | |
| 		} else {
 | |
| 			return false;
 | |
| 		}
 | |
| 	}
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool HistoryItem::suggestBanReport() const {
 | |
| 	auto channel = history()->peer->asChannel();
 | |
| 	auto fromUser = from()->asUser();
 | |
| 	if (!channel || !fromUser || !channel->canRestrictUser(fromUser)) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	return !isPost() && !out() && toHistoryMessage();
 | |
| }
 | |
| 
 | |
| bool HistoryItem::suggestDeleteAllReport() const {
 | |
| 	auto channel = history()->peer->asChannel();
 | |
| 	if (!channel || !channel->canDeleteMessages()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	return !isPost() && !out() && from()->isUser() && toHistoryMessage();
 | |
| }
 | |
| 
 | |
| bool HistoryItem::hasDirectLink() const {
 | |
| 	if (id <= 0) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	if (auto channel = _history->peer->asChannel()) {
 | |
| 		return channel->isPublic();
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| QString HistoryItem::directLink() const {
 | |
| 	if (hasDirectLink()) {
 | |
| 		auto channel = _history->peer->asChannel();
 | |
| 		Assert(channel != nullptr);
 | |
| 		auto query = channel->username + '/' + QString::number(id);
 | |
| 		if (!channel->isMegagroup()) {
 | |
| 			if (auto media = getMedia()) {
 | |
| 				if (auto document = media->getDocument()) {
 | |
| 					if (document->isRoundVideo()) {
 | |
| 						return qsl("https://telesco.pe/") + query;
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		return Messenger::Instance().createInternalLinkFull(query);
 | |
| 	}
 | |
| 	return QString();
 | |
| }
 | |
| 
 | |
| 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>()) {
 | |
| 		Assert(!isLogEntry());
 | |
| 
 | |
| 		RemoveComponents(HistoryMessageUnreadBar::Bit());
 | |
| 		setPendingInitDimensions();
 | |
| 		if (_history->unreadBar == this) {
 | |
| 			_history->unreadBar = nullptr;
 | |
| 		}
 | |
| 
 | |
| 		recountAttachToPrevious();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::setUnreadBarCount(int count) {
 | |
| 	Expects(!isLogEntry());
 | |
| 	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() {
 | |
| 	Expects(!isLogEntry());
 | |
| 	if (auto bar = Get<HistoryMessageUnreadBar>()) {
 | |
| 		bar->_freezed = true;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::clipCallback(Media::Clip::Notification notification) {
 | |
| 	using namespace Media::Clip;
 | |
| 
 | |
| 	auto media = getMedia();
 | |
| 	if (!media) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	auto reader = media->getClipReader();
 | |
| 	if (!reader) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	switch (notification) {
 | |
| 	case NotificationReinit: {
 | |
| 		auto stopped = false;
 | |
| 		if (reader->autoPausedGif()) {
 | |
| 			auto amVisible = false;
 | |
| 			Auth().data().queryItemVisibility().notify({ this, &amVisible }, true);
 | |
| 			if (!amVisible) { // stop animation if it is not visible
 | |
| 				media->stopInline();
 | |
| 				if (auto document = media->getDocument()) { // forget data from memory
 | |
| 					document->forget();
 | |
| 				}
 | |
| 				stopped = true;
 | |
| 			}
 | |
| 		} else if (reader->mode() == Media::Clip::Reader::Mode::Video && reader->state() == Media::Clip::State::Finished) {
 | |
| 			// Stop finished video message.
 | |
| 			media->stopInline();
 | |
| 		}
 | |
| 		if (!stopped) {
 | |
| 			setPendingInitDimensions();
 | |
| 			Notify::historyItemLayoutChanged(this);
 | |
| 			Global::RefPendingRepaintItems().insert(this);
 | |
| 		}
 | |
| 	} break;
 | |
| 
 | |
| 	case NotificationRepaint: {
 | |
| 		if (!reader->currentDisplayed()) {
 | |
| 			Ui::repaintHistoryItem(this);
 | |
| 		}
 | |
| 	} break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::audioTrackUpdated() {
 | |
| 	auto media = getMedia();
 | |
| 	if (!media) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	auto reader = media->getClipReader();
 | |
| 	if (!reader || reader->mode() != Media::Clip::Reader::Mode::Video) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	auto audio = reader->audioMsgId();
 | |
| 	auto current = Media::Player::mixer()->currentState(audio.type());
 | |
| 	if (current.id != audio || Media::Player::IsStoppedOrStopping(current.state)) {
 | |
| 		media->stopInline();
 | |
| 	} else if (Media::Player::IsPaused(current.state) || current.state == Media::Player::State::Pausing) {
 | |
| 		if (!reader->videoPaused()) {
 | |
| 			reader->pauseResumeVideo();
 | |
| 		}
 | |
| 	} else {
 | |
| 		if (reader->videoPaused()) {
 | |
| 			reader->pauseResumeVideo();
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void HistoryItem::recountDisplayDate() {
 | |
| 	Expects(!isLogEntry());
 | |
| 	setDisplayDate(([this]() {
 | |
| 		if (isEmpty()) {
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		if (auto previous = previousItem()) {
 | |
| 			return previous->isEmpty() || (previous->date.date() != date.date());
 | |
| 		}
 | |
| 		return true;
 | |
| 	})());
 | |
| }
 | |
| 
 | |
| void HistoryItem::setDisplayDate(bool displayDate) {
 | |
| 	if (displayDate && !Has<HistoryMessageDate>()) {
 | |
| 		AddComponents(HistoryMessageDate::Bit());
 | |
| 		Get<HistoryMessageDate>()->init(date);
 | |
| 		setPendingInitDimensions();
 | |
| 	} else if (!displayDate && 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(DrawInDialog way) const {
 | |
| 	auto getText = [this]() {
 | |
| 		if (emptyText()) {
 | |
| 			return _media ? _media->inDialogsText() : QString();
 | |
| 		}
 | |
| 		return TextUtilities::Clean(_text.originalText());
 | |
| 	};
 | |
| 	auto plainText = getText();
 | |
| 	if ((!_history->peer->isUser() || out())
 | |
| 		&& !isPost()
 | |
| 		&& !isEmpty()
 | |
| 		&& (way != DrawInDialog::WithoutSender)) {
 | |
| 		auto fromText = author()->isSelf() ? lang(lng_from_you) : author()->shortName();
 | |
| 		auto fromWrapped = textcmdLink(1, lng_dialogs_text_from_wrapped(lt_from, TextUtilities::Clean(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 active,
 | |
| 		bool selected,
 | |
| 		DrawInDialog way,
 | |
| 		const HistoryItem *&cacheFor,
 | |
| 		Text &cache) const {
 | |
| 	if (cacheFor != this) {
 | |
| 		cacheFor = this;
 | |
| 		cache.setText(st::dialogsTextStyle, inDialogsText(way), _textDlgOptions);
 | |
| 	}
 | |
| 	if (r.width()) {
 | |
| 		p.setTextPalette(active ? st::dialogsTextPaletteActive : (selected ? st::dialogsTextPaletteOver : st::dialogsTextPalette));
 | |
| 		p.setFont(st::dialogsTextFont);
 | |
| 		p.setPen(active ? st::dialogsTextFgActive : (selected ? st::dialogsTextFgOver : st::dialogsTextFg));
 | |
| 		cache.drawElided(p, r.left(), r.top(), r.width(), r.height() / st::dialogsTextFont->height);
 | |
| 		p.restoreTextPalette();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| HistoryItem::~HistoryItem() {
 | |
| 	App::historyUnregItem(this);
 | |
| 	if (id < 0 && !App::quitting()) {
 | |
| 		Auth().uploader().cancel(fullId());
 | |
| 	}
 | |
| }
 | |
| 
 | |
| ClickHandlerPtr goToMessageClickHandler(PeerData *peer, MsgId msgId) {
 | |
| 	return MakeShared<LambdaClickHandler>([peer, msgId] {
 | |
| 		if (App::main()) {
 | |
| 			auto current = App::mousedItem();
 | |
| 			if (current && current->history()->peer == peer) {
 | |
| 				App::main()->pushReplyReturn(current);
 | |
| 			}
 | |
| 			Ui::showPeerHistory(peer, msgId, Ui::ShowWay::Forward);
 | |
| 		}
 | |
| 	});
 | |
| }
 |