| 
									
										
										
										
											2017-07-19 21:37:49 +03:00
										 |  |  | /*
 | 
					
						
							|  |  |  | This file is part of Telegram Desktop, | 
					
						
							| 
									
										
										
										
											2018-01-03 13:23:14 +03:00
										 |  |  | the official desktop application for the Telegram messaging service. | 
					
						
							| 
									
										
										
										
											2017-07-19 21:37:49 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-03 13:23:14 +03:00
										 |  |  | For license and copyright information please follow this link: | 
					
						
							|  |  |  | https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | 
					
						
							| 
									
										
										
										
											2017-07-19 21:37:49 +03:00
										 |  |  | */ | 
					
						
							|  |  |  | #include "chat_helpers/emoji_suggestions_widget.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-28 12:48:32 +04:00
										 |  |  | #include "chat_helpers/emoji_keywords.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-18 22:04:16 +04:00
										 |  |  | #include "core/core_settings.h"
 | 
					
						
							|  |  |  | #include "core/application.h"
 | 
					
						
							| 
									
										
										
										
											2019-09-17 19:13:12 +03:00
										 |  |  | #include "emoji_suggestions_helper.h"
 | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | #include "ui/effects/ripple_animation.h"
 | 
					
						
							|  |  |  | #include "ui/widgets/shadow.h"
 | 
					
						
							| 
									
										
										
										
											2018-10-13 20:35:30 +03:00
										 |  |  | #include "ui/widgets/inner_dropdown.h"
 | 
					
						
							| 
									
										
										
										
											2023-08-31 14:21:24 +03:00
										 |  |  | #include "ui/widgets/fields/input_field.h"
 | 
					
						
							| 
									
										
										
										
											2018-10-13 20:35:30 +03:00
										 |  |  | #include "ui/emoji_config.h"
 | 
					
						
							| 
									
										
										
										
											2019-09-13 15:22:54 +03:00
										 |  |  | #include "ui/ui_utility.h"
 | 
					
						
							| 
									
										
										
										
											2020-10-13 18:11:53 +03:00
										 |  |  | #include "ui/cached_round_corners.h"
 | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | #include "ui/round_rect.h"
 | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | #include "platform/platform_specific.h"
 | 
					
						
							| 
									
										
										
										
											2019-03-28 12:48:32 +04:00
										 |  |  | #include "core/application.h"
 | 
					
						
							| 
									
										
										
										
											2019-09-19 12:28:36 +03:00
										 |  |  | #include "base/event_filter.h"
 | 
					
						
							| 
									
										
										
										
											2019-07-24 13:45:24 +02:00
										 |  |  | #include "main/main_session.h"
 | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | #include "data/data_session.h"
 | 
					
						
							|  |  |  | #include "data/data_document.h"
 | 
					
						
							|  |  |  | #include "data/stickers/data_custom_emoji.h"
 | 
					
						
							|  |  |  | #include "data/stickers/data_stickers.h"
 | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | #include "styles/style_chat_helpers.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-04 10:19:15 +03:00
										 |  |  | #include <QtWidgets/QApplication>
 | 
					
						
							|  |  |  | #include <QtGui/QTextBlock>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | namespace Ui { | 
					
						
							|  |  |  | namespace Emoji { | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 12:06:15 +04:00
										 |  |  | constexpr auto kShowExactDelay = crl::time(300); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | constexpr auto kMaxNonScrolledEmoji = 7; | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | constexpr auto kAnimationDuration = crl::time(120); | 
					
						
							| 
									
										
										
										
											2017-07-24 19:18:39 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | } // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | class SuggestionsWidget final : public Ui::RpWidget { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 	SuggestionsWidget( | 
					
						
							|  |  |  | 		QWidget *parent, | 
					
						
							|  |  |  | 		const style::EmojiSuggestions &st, | 
					
						
							|  |  |  | 		not_null<Main::Session*> session, | 
					
						
							|  |  |  | 		bool suggestCustomEmoji, | 
					
						
							|  |  |  | 		Fn<bool(not_null<DocumentData*>)> allowCustomWithoutPremium); | 
					
						
							|  |  |  | 	~SuggestionsWidget(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void showWithQuery(SuggestionsQuery query, bool force = false); | 
					
						
							|  |  |  | 	void selectFirstResult(); | 
					
						
							|  |  |  | 	bool handleKeyEvent(int key); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	[[nodiscard]] rpl::producer<bool> toggleAnimated() const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct Chosen { | 
					
						
							|  |  |  | 		QString emoji; | 
					
						
							|  |  |  | 		QString customData; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	[[nodiscard]] rpl::producer<Chosen> triggered() const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  | 	struct Row { | 
					
						
							|  |  |  | 		Row(not_null<EmojiPtr> emoji, const QString &replacement); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Ui::Text::CustomEmoji *custom = nullptr; | 
					
						
							|  |  |  | 		DocumentData *document = nullptr; | 
					
						
							|  |  |  | 		not_null<EmojiPtr> emoji; | 
					
						
							|  |  |  | 		QString replacement; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	struct Custom { | 
					
						
							|  |  |  | 		not_null<DocumentData*> document; | 
					
						
							|  |  |  | 		not_null<EmojiPtr> emoji; | 
					
						
							|  |  |  | 		QString replacement; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	bool eventHook(QEvent *e) override; | 
					
						
							|  |  |  | 	void paintEvent(QPaintEvent *e) override; | 
					
						
							|  |  |  | 	void keyPressEvent(QKeyEvent *e) override; | 
					
						
							|  |  |  | 	void mouseMoveEvent(QMouseEvent *e) override; | 
					
						
							|  |  |  | 	void mousePressEvent(QMouseEvent *e) override; | 
					
						
							|  |  |  | 	void mouseReleaseEvent(QMouseEvent *e) override; | 
					
						
							|  |  |  | 	void enterEventHook(QEnterEvent *e) override; | 
					
						
							|  |  |  | 	void leaveEventHook(QEvent *e) override; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void scrollByWheelEvent(not_null<QWheelEvent*> e); | 
					
						
							|  |  |  | 	void paintFadings(QPainter &p) const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	[[nodiscard]] std::vector<Row> getRowsByQuery(const QString &text) const; | 
					
						
							|  |  |  | 	[[nodiscard]] base::flat_multi_map<int, Custom> lookupCustom( | 
					
						
							|  |  |  | 		const std::vector<Row> &rows) const; | 
					
						
							|  |  |  | 	[[nodiscard]] std::vector<Row> appendCustom( | 
					
						
							|  |  |  | 		std::vector<Row> rows); | 
					
						
							|  |  |  | 	[[nodiscard]] std::vector<Row> appendCustom( | 
					
						
							|  |  |  | 		std::vector<Row> rows, | 
					
						
							|  |  |  | 		const base::flat_multi_map<int, Custom> &custom); | 
					
						
							|  |  |  | 	void resizeToRows(); | 
					
						
							|  |  |  | 	void setSelected( | 
					
						
							|  |  |  | 		int selected, | 
					
						
							|  |  |  | 		anim::type animated = anim::type::instant); | 
					
						
							|  |  |  | 	void setPressed(int pressed); | 
					
						
							|  |  |  | 	void clearMouseSelection(); | 
					
						
							|  |  |  | 	void clearSelection(); | 
					
						
							|  |  |  | 	void updateSelectedItem(); | 
					
						
							|  |  |  | 	void updateItem(int index); | 
					
						
							|  |  |  | 	[[nodiscard]] QRect inner() const; | 
					
						
							|  |  |  | 	[[nodiscard]] QPoint innerShift() const; | 
					
						
							|  |  |  | 	[[nodiscard]] QPoint mapToInner(QPoint globalPosition) const; | 
					
						
							|  |  |  | 	void selectByMouse(QPoint globalPosition); | 
					
						
							|  |  |  | 	bool triggerSelectedRow() const; | 
					
						
							|  |  |  | 	void triggerRow(const Row &row) const; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	[[nodiscard]] int scrollCurrent() const; | 
					
						
							|  |  |  | 	void scrollTo(int value, anim::type animated = anim::type::instant); | 
					
						
							|  |  |  | 	void stopAnimations(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	[[nodiscard]] not_null<Ui::Text::CustomEmoji*> resolveCustomEmoji( | 
					
						
							|  |  |  | 		not_null<DocumentData*> document); | 
					
						
							|  |  |  | 	void customEmojiRepaint(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const style::EmojiSuggestions &_st; | 
					
						
							|  |  |  | 	const not_null<Main::Session*> _session; | 
					
						
							|  |  |  | 	SuggestionsQuery _query; | 
					
						
							|  |  |  | 	std::vector<Row> _rows; | 
					
						
							|  |  |  | 	bool _suggestCustomEmoji = false; | 
					
						
							|  |  |  | 	Fn<bool(not_null<DocumentData*>)> _allowCustomWithoutPremium; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	Ui::RoundRect _overRect; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	base::flat_map< | 
					
						
							|  |  |  | 		not_null<DocumentData*>, | 
					
						
							|  |  |  | 		std::unique_ptr<Ui::Text::CustomEmoji>> _customEmoji; | 
					
						
							|  |  |  | 	bool _repaintScheduled = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	std::optional<QPoint> _lastMousePosition; | 
					
						
							|  |  |  | 	bool _mouseSelection = false; | 
					
						
							|  |  |  | 	int _selected = -1; | 
					
						
							|  |  |  | 	int _pressed = -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int _scrollValue = 0; | 
					
						
							|  |  |  | 	Ui::Animations::Simple _scrollAnimation; | 
					
						
							|  |  |  | 	Ui::Animations::Simple _selectedAnimation; | 
					
						
							|  |  |  | 	int _scrollMax = 0; | 
					
						
							|  |  |  | 	int _oneWidth = 0; | 
					
						
							|  |  |  | 	QMargins _padding; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	QPoint _mousePressPosition; | 
					
						
							|  |  |  | 	int _dragScrollStart = -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rpl::event_stream<bool> _toggleAnimated; | 
					
						
							|  |  |  | 	rpl::event_stream<Chosen> _triggered; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | SuggestionsWidget::SuggestionsWidget( | 
					
						
							|  |  |  | 	QWidget *parent, | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | 	const style::EmojiSuggestions &st, | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | 	not_null<Main::Session*> session, | 
					
						
							|  |  |  | 	bool suggestCustomEmoji, | 
					
						
							|  |  |  | 	Fn<bool(not_null<DocumentData*>)> allowCustomWithoutPremium) | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | : RpWidget(parent) | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | , _st(st) | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | , _session(session) | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | , _suggestCustomEmoji(suggestCustomEmoji) | 
					
						
							|  |  |  | , _allowCustomWithoutPremium(std::move(allowCustomWithoutPremium)) | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | , _overRect(st::roundRadiusSmall, _st.overBg) | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | , _oneWidth(st::emojiSuggestionSize) | 
					
						
							|  |  |  | , _padding(st::emojiSuggestionsPadding) { | 
					
						
							|  |  |  | 	resize( | 
					
						
							|  |  |  | 		_oneWidth + _padding.left() + _padding.right(), | 
					
						
							|  |  |  | 		_oneWidth + _padding.top() + _padding.bottom()); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	setMouseTracking(true); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | SuggestionsWidget::~SuggestionsWidget() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | rpl::producer<bool> SuggestionsWidget::toggleAnimated() const { | 
					
						
							|  |  |  | 	return _toggleAnimated.events(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | auto SuggestionsWidget::triggered() const -> rpl::producer<Chosen> { | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 	return _triggered.events(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | void SuggestionsWidget::showWithQuery(SuggestionsQuery query, bool force) { | 
					
						
							| 
									
										
										
										
											2019-03-29 20:04:05 +04:00
										 |  |  | 	if (!force && (_query == query)) { | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	_query = query; | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 	auto rows = [&] { | 
					
						
							|  |  |  | 		if (const auto emoji = std::get_if<EmojiPtr>(&query)) { | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | 			return appendCustom( | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 				{}, | 
					
						
							|  |  |  | 				lookupCustom({ Row(*emoji, (*emoji)->text()) })); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | 		return appendCustom(getRowsByQuery(v::get<QString>(query))); | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 	}(); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	if (rows.empty()) { | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 		_toggleAnimated.fire(false); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	clearSelection(); | 
					
						
							| 
									
										
										
										
											2019-04-01 14:35:32 +04:00
										 |  |  | 	setPressed(-1); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	_rows = std::move(rows); | 
					
						
							|  |  |  | 	resizeToRows(); | 
					
						
							|  |  |  | 	update(); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	Ui::PostponeCall(this, [=] { | 
					
						
							|  |  |  | 		if (!_rows.empty()) { | 
					
						
							|  |  |  | 			_toggleAnimated.fire(true); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsWidget::selectFirstResult() { | 
					
						
							|  |  |  | 	if (!_rows.empty() && _selected < 0) { | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 		setSelected(0); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | auto SuggestionsWidget::appendCustom(std::vector<Row> rows) | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | -> std::vector<Row> { | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 	const auto custom = lookupCustom(rows); | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | 	return appendCustom(std::move(rows), custom); | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | auto SuggestionsWidget::lookupCustom(const std::vector<Row> &rows) const | 
					
						
							|  |  |  | -> base::flat_multi_map<int, Custom> { | 
					
						
							| 
									
										
										
										
											2022-08-02 20:59:11 +03:00
										 |  |  | 	if (rows.empty() | 
					
						
							|  |  |  | 		|| !_suggestCustomEmoji | 
					
						
							|  |  |  | 		|| !Core::App().settings().suggestAnimatedEmoji()) { | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 		return {}; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	auto custom = base::flat_multi_map<int, Custom>(); | 
					
						
							|  |  |  | 	const auto premium = _session->premium(); | 
					
						
							|  |  |  | 	const auto stickers = &_session->data().stickers(); | 
					
						
							|  |  |  | 	for (const auto setId : stickers->emojiSetsOrder()) { | 
					
						
							|  |  |  | 		const auto i = stickers->sets().find(setId); | 
					
						
							|  |  |  | 		if (i == end(stickers->sets())) { | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		for (const auto &document : i->second->stickers) { | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | 			if (!premium | 
					
						
							|  |  |  | 				&& document->isPremiumEmoji() | 
					
						
							|  |  |  | 				&& (!_allowCustomWithoutPremium | 
					
						
							|  |  |  | 					|| !_allowCustomWithoutPremium(document))) { | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 				// Skip the whole premium emoji set.
 | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if (const auto sticker = document->sticker()) { | 
					
						
							|  |  |  | 				if (const auto emoji = Ui::Emoji::Find(sticker->alt)) { | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | 					const auto original = emoji->original(); | 
					
						
							|  |  |  | 					const auto j = ranges::find_if( | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 						rows, | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | 						[&](const Row &row) { | 
					
						
							|  |  |  | 							return row.emoji->original() == original; | 
					
						
							|  |  |  | 						}); | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 					if (j != end(rows)) { | 
					
						
							|  |  |  | 						custom.emplace(int(j - begin(rows)), Custom{ | 
					
						
							|  |  |  | 							.document = document, | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | 							.emoji = emoji, | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 							.replacement = j->replacement, | 
					
						
							|  |  |  | 						}); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 	return custom; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | auto SuggestionsWidget::appendCustom( | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 	std::vector<Row> rows, | 
					
						
							|  |  |  | 	const base::flat_multi_map<int, Custom> &custom) | 
					
						
							|  |  |  | -> std::vector<Row> { | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | 	rows.reserve(rows.size() + custom.size()); | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 	for (const auto &[position, one] : custom) { | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | 		rows.push_back(Row(one.emoji, one.replacement)); | 
					
						
							|  |  |  | 		rows.back().document = one.document; | 
					
						
							|  |  |  | 		rows.back().custom = resolveCustomEmoji(one.document); | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | 	return rows; | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-25 11:30:38 +03:00
										 |  |  | not_null<Ui::Text::CustomEmoji*> SuggestionsWidget::resolveCustomEmoji( | 
					
						
							|  |  |  | 		not_null<DocumentData*> document) { | 
					
						
							|  |  |  | 	const auto i = _customEmoji.find(document); | 
					
						
							|  |  |  | 	if (i != end(_customEmoji)) { | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 		return i->second.get(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-07-25 11:30:38 +03:00
										 |  |  | 	auto emoji = document->session().data().customEmojiManager().create( | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 		document, | 
					
						
							| 
									
										
										
										
											2022-07-25 11:30:38 +03:00
										 |  |  | 		[=] { customEmojiRepaint(); }, | 
					
						
							|  |  |  | 		Data::CustomEmojiManager::SizeTag::Large); | 
					
						
							|  |  |  | 	return _customEmoji.emplace( | 
					
						
							|  |  |  | 		document, | 
					
						
							|  |  |  | 		std::move(emoji) | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 	).first->second.get(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-22 14:39:28 +03:00
										 |  |  | void SuggestionsWidget::customEmojiRepaint() { | 
					
						
							|  |  |  | 	if (_repaintScheduled) { | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-07-22 14:39:28 +03:00
										 |  |  | 	_repaintScheduled = true; | 
					
						
							|  |  |  | 	update(); | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | SuggestionsWidget::Row::Row( | 
					
						
							|  |  |  | 	not_null<EmojiPtr> emoji, | 
					
						
							|  |  |  | 	const QString &replacement) | 
					
						
							|  |  |  | : emoji(emoji) | 
					
						
							|  |  |  | , replacement(replacement) { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | auto SuggestionsWidget::getRowsByQuery(const QString &text) const | 
					
						
							|  |  |  | -> std::vector<Row> { | 
					
						
							|  |  |  | 	if (text.isEmpty()) { | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		return {}; | 
					
						
							| 
									
										
										
										
											2017-07-25 21:16:37 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 	const auto middle = (text[0] == ':'); | 
					
						
							|  |  |  | 	const auto real = middle ? text.mid(1) : text; | 
					
						
							| 
									
										
										
										
											2019-03-29 17:51:18 +04:00
										 |  |  | 	const auto simple = [&] { | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 		if (!middle || text.size() > 2) { | 
					
						
							| 
									
										
										
										
											2019-03-29 17:51:18 +04:00
										 |  |  | 			return false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		// Suggest :D and :-P only as exact matches.
 | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 		return ranges::none_of(text, [](QChar ch) { return ch.isLower(); }); | 
					
						
							| 
									
										
										
										
											2019-03-29 17:51:18 +04:00
										 |  |  | 	}(); | 
					
						
							|  |  |  | 	const auto exact = !middle || simple; | 
					
						
							| 
									
										
										
										
											2023-01-20 20:44:08 +04:00
										 |  |  | 	const auto list = Core::App().emojiKeywords().queryMine(real, exact); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	using Entry = ChatHelpers::EmojiKeywords::Result; | 
					
						
							| 
									
										
										
										
											2023-01-20 20:44:08 +04:00
										 |  |  | 	return ranges::views::all( | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		list | 
					
						
							| 
									
										
										
										
											2021-03-13 16:12:08 +04:00
										 |  |  | 	) | ranges::views::transform([](const Entry &result) { | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		return Row(result.emoji, result.replacement); | 
					
						
							|  |  |  | 	}) | ranges::to_vector; | 
					
						
							| 
									
										
										
										
											2017-07-25 21:16:37 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | void SuggestionsWidget::resizeToRows() { | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	const auto count = int(_rows.size()); | 
					
						
							|  |  |  | 	const auto scrolled = (count > kMaxNonScrolledEmoji); | 
					
						
							|  |  |  | 	const auto fullWidth = count * _oneWidth; | 
					
						
							|  |  |  | 	const auto newWidth = scrolled | 
					
						
							|  |  |  | 		? st::emojiSuggestionsScrolledWidth | 
					
						
							|  |  |  | 		: fullWidth; | 
					
						
							|  |  |  | 	_scrollMax = std::max(0, fullWidth - newWidth); | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 	if (_scrollValue > _scrollMax || scrollCurrent() > _scrollMax) { | 
					
						
							|  |  |  | 		scrollTo(std::min(_scrollValue, _scrollMax)); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	resize(_padding.left() + newWidth + _padding.right(), height()); | 
					
						
							|  |  |  | 	update(); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | bool SuggestionsWidget::eventHook(QEvent *e) { | 
					
						
							|  |  |  | 	if (e->type() == QEvent::Wheel) { | 
					
						
							|  |  |  | 		selectByMouse(QCursor::pos()); | 
					
						
							|  |  |  | 		if (_selected >= 0 && _pressed < 0) { | 
					
						
							|  |  |  | 			scrollByWheelEvent(static_cast<QWheelEvent*>(e)); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return RpWidget::eventHook(e); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsWidget::scrollByWheelEvent(not_null<QWheelEvent*> e) { | 
					
						
							| 
									
										
										
										
											2020-11-04 18:50:17 +03:00
										 |  |  | 	const auto horizontal = (e->angleDelta().x() != 0); | 
					
						
							|  |  |  | 	const auto vertical = (e->angleDelta().y() != 0); | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 	const auto current = scrollCurrent(); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	const auto scroll = [&] { | 
					
						
							|  |  |  | 		if (horizontal) { | 
					
						
							|  |  |  | 			const auto delta = e->pixelDelta().x() | 
					
						
							|  |  |  | 				? e->pixelDelta().x() | 
					
						
							|  |  |  | 				: e->angleDelta().x(); | 
					
						
							| 
									
										
										
										
											2021-01-23 06:29:50 +03:00
										 |  |  | 			return std::clamp( | 
					
						
							|  |  |  | 				current - ((rtl() ? -1 : 1) * delta), | 
					
						
							|  |  |  | 				0, | 
					
						
							|  |  |  | 				_scrollMax); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		} else if (vertical) { | 
					
						
							|  |  |  | 			const auto delta = e->pixelDelta().y() | 
					
						
							|  |  |  | 				? e->pixelDelta().y() | 
					
						
							|  |  |  | 				: e->angleDelta().y(); | 
					
						
							| 
									
										
										
										
											2021-01-23 06:29:50 +03:00
										 |  |  | 			return std::clamp(current - delta, 0, _scrollMax); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 		return current; | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	}(); | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 	if (current != scroll) { | 
					
						
							|  |  |  | 		scrollTo(scroll); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		if (!_lastMousePosition) { | 
					
						
							|  |  |  | 			_lastMousePosition = QCursor::pos(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		selectByMouse(*_lastMousePosition); | 
					
						
							|  |  |  | 		update(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsWidget::paintEvent(QPaintEvent *e) { | 
					
						
							| 
									
										
										
										
											2022-09-17 00:23:27 +04:00
										 |  |  | 	auto p = QPainter(this); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-22 14:39:28 +03:00
										 |  |  | 	_repaintScheduled = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	const auto clip = e->rect(); | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | 	p.fillRect(clip, _st.bg); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	const auto shift = innerShift(); | 
					
						
							|  |  |  | 	p.translate(-shift); | 
					
						
							|  |  |  | 	const auto paint = clip.translated(shift); | 
					
						
							|  |  |  | 	const auto from = std::max(paint.x(), 0) / _oneWidth; | 
					
						
							|  |  |  | 	const auto till = std::min( | 
					
						
							|  |  |  | 		(paint.x() + paint.width() + _oneWidth - 1) / _oneWidth, | 
					
						
							|  |  |  | 		int(_rows.size())); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 	const auto selected = (_pressed >= 0) | 
					
						
							|  |  |  | 		? _pressed | 
					
						
							|  |  |  | 		: _selectedAnimation.value(_selected); | 
					
						
							|  |  |  | 	if (selected > -1.) { | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | 		_overRect.paint( | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 			p, | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | 			QRect(selected * _oneWidth, 0, _oneWidth, _oneWidth)); | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-15 19:10:51 +04:00
										 |  |  | 	auto context = Ui::CustomEmoji::Context{ | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | 		.textColor = _st.textFg->c, | 
					
						
							| 
									
										
										
										
											2022-12-15 19:10:51 +04:00
										 |  |  | 		.now = crl::now(), | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	for (auto i = from; i != till; ++i) { | 
					
						
							|  |  |  | 		const auto &row = _rows[i]; | 
					
						
							|  |  |  | 		const auto emoji = row.emoji; | 
					
						
							|  |  |  | 		const auto esize = Ui::Emoji::GetSizeLarge(); | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 		const auto size = esize / style::DevicePixelRatio(); | 
					
						
							|  |  |  | 		const auto x = i * _oneWidth + (_oneWidth - size) / 2; | 
					
						
							|  |  |  | 		const auto y = (_oneWidth - size) / 2; | 
					
						
							| 
									
										
										
										
											2022-07-25 11:30:38 +03:00
										 |  |  | 		if (row.custom) { | 
					
						
							| 
									
										
										
										
											2022-12-15 19:10:51 +04:00
										 |  |  | 			context.position = { x, y }; | 
					
						
							|  |  |  | 			row.custom->paint(p, context); | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			Ui::Emoji::Draw(p, emoji, esize, x, y); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	paintFadings(p); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-17 00:23:27 +04:00
										 |  |  | void SuggestionsWidget::paintFadings(QPainter &p) const { | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 	const auto scroll = scrollCurrent(); | 
					
						
							| 
									
										
										
										
											2021-01-23 06:29:50 +03:00
										 |  |  | 	const auto o_left = std::clamp( | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 		scroll / float64(st::emojiSuggestionsFadeAfter), | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		0., | 
					
						
							|  |  |  | 		1.); | 
					
						
							|  |  |  | 	const auto shift = innerShift(); | 
					
						
							|  |  |  | 	if (o_left > 0.) { | 
					
						
							|  |  |  | 		p.setOpacity(o_left); | 
					
						
							|  |  |  | 		const auto rect = myrtlrect( | 
					
						
							|  |  |  | 			shift.x(), | 
					
						
							|  |  |  | 			0, | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | 			_st.fadeLeft.width(), | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 			height()); | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | 		_st.fadeLeft.fill(p, rect); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		p.setOpacity(1.); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-01-23 06:29:50 +03:00
										 |  |  | 	const auto o_right = std::clamp( | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 		(_scrollMax - scroll) / float64(st::emojiSuggestionsFadeAfter), | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		0., | 
					
						
							|  |  |  | 		1.); | 
					
						
							|  |  |  | 	if (o_right > 0.) { | 
					
						
							|  |  |  | 		p.setOpacity(o_right); | 
					
						
							|  |  |  | 		const auto rect = myrtlrect( | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | 			shift.x() + width() - _st.fadeRight.width(), | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 			0, | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | 			_st.fadeRight.width(), | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 			height()); | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | 		_st.fadeRight.fill(p, rect); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		p.setOpacity(1.); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsWidget::keyPressEvent(QKeyEvent *e) { | 
					
						
							|  |  |  | 	handleKeyEvent(e->key()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | bool SuggestionsWidget::handleKeyEvent(int key) { | 
					
						
							|  |  |  | 	if (key == Qt::Key_Enter || key == Qt::Key_Return) { | 
					
						
							|  |  |  | 		return triggerSelectedRow(); | 
					
						
							| 
									
										
										
										
											2019-07-04 13:26:39 +02:00
										 |  |  | 	} else if (key == Qt::Key_Tab) { | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		if (_selected < 0 || _selected >= _rows.size()) { | 
					
						
							|  |  |  | 			setSelected(0); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return triggerSelectedRow(); | 
					
						
							|  |  |  | 	} else if (_rows.empty() | 
					
						
							|  |  |  | 		|| (key != Qt::Key_Up | 
					
						
							|  |  |  | 			&& key != Qt::Key_Down | 
					
						
							|  |  |  | 			&& key != Qt::Key_Left | 
					
						
							|  |  |  | 			&& key != Qt::Key_Right)) { | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	const auto delta = (key == Qt::Key_Down || key == Qt::Key_Right) | 
					
						
							|  |  |  | 		? 1 | 
					
						
							|  |  |  | 		: -1; | 
					
						
							|  |  |  | 	if (delta < 0 && _selected < 0) { | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	auto start = _selected; | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	if (start < 0 || start >= _rows.size()) { | 
					
						
							|  |  |  | 		start = (delta > 0) ? (_rows.size() - 1) : 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	auto newSelected = start + delta; | 
					
						
							|  |  |  | 	if (newSelected < 0) { | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		newSelected = -1; | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	} else if (newSelected >= _rows.size()) { | 
					
						
							|  |  |  | 		newSelected -= _rows.size(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_mouseSelection = false; | 
					
						
							| 
									
										
										
										
											2018-12-25 16:41:40 +04:00
										 |  |  | 	_lastMousePosition = std::nullopt; | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 	setSelected(newSelected, anim::type::normal); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	return true; | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | void SuggestionsWidget::setSelected(int selected, anim::type animated) { | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	if (selected >= _rows.size()) { | 
					
						
							|  |  |  | 		selected = -1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 	if (animated == anim::type::normal) { | 
					
						
							|  |  |  | 		_selectedAnimation.start( | 
					
						
							|  |  |  | 			[=] { update(); }, | 
					
						
							|  |  |  | 			_selected, | 
					
						
							|  |  |  | 			selected, | 
					
						
							|  |  |  | 			kAnimationDuration, | 
					
						
							|  |  |  | 			anim::sineInOut); | 
					
						
							|  |  |  | 		if (_scrollMax > 0) { | 
					
						
							|  |  |  | 			const auto selectedMax = int(_rows.size()) - 3; | 
					
						
							|  |  |  | 			const auto selectedForScroll = std::min( | 
					
						
							|  |  |  | 				std::max(selected, 1) - 1, | 
					
						
							|  |  |  | 				selectedMax); | 
					
						
							|  |  |  | 			scrollTo((_scrollMax * selectedForScroll) / selectedMax, animated); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else if (_selectedAnimation.animating()) { | 
					
						
							|  |  |  | 		_selectedAnimation.stop(); | 
					
						
							|  |  |  | 		update(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	if (_selected != selected) { | 
					
						
							|  |  |  | 		updateSelectedItem(); | 
					
						
							|  |  |  | 		_selected = selected; | 
					
						
							|  |  |  | 		updateSelectedItem(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | int SuggestionsWidget::scrollCurrent() const { | 
					
						
							|  |  |  | 	return _scrollAnimation.value(_scrollValue); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsWidget::scrollTo(int value, anim::type animated) { | 
					
						
							|  |  |  | 	if (animated == anim::type::instant) { | 
					
						
							|  |  |  | 		_scrollAnimation.stop(); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		_scrollAnimation.start( | 
					
						
							|  |  |  | 			[=] { update(); }, | 
					
						
							|  |  |  | 			_scrollValue, | 
					
						
							|  |  |  | 			value, | 
					
						
							|  |  |  | 			kAnimationDuration, | 
					
						
							|  |  |  | 			anim::sineInOut); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	_scrollValue = value; | 
					
						
							|  |  |  | 	update(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsWidget::stopAnimations() { | 
					
						
							|  |  |  | 	_scrollValue = _scrollAnimation.value(_scrollValue); | 
					
						
							|  |  |  | 	_scrollAnimation.stop(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | void SuggestionsWidget::setPressed(int pressed) { | 
					
						
							|  |  |  | 	if (pressed >= _rows.size()) { | 
					
						
							|  |  |  | 		pressed = -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (_pressed != pressed) { | 
					
						
							|  |  |  | 		_pressed = pressed; | 
					
						
							| 
									
										
										
										
											2019-04-01 14:35:32 +04:00
										 |  |  | 		if (_pressed >= 0) { | 
					
						
							|  |  |  | 			_mousePressPosition = QCursor::pos(); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsWidget::clearMouseSelection() { | 
					
						
							|  |  |  | 	if (_mouseSelection) { | 
					
						
							|  |  |  | 		clearSelection(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsWidget::clearSelection() { | 
					
						
							|  |  |  | 	_mouseSelection = false; | 
					
						
							| 
									
										
										
										
											2018-12-25 16:41:40 +04:00
										 |  |  | 	_lastMousePosition = std::nullopt; | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	setSelected(-1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsWidget::updateItem(int index) { | 
					
						
							|  |  |  | 	if (index >= 0 && index < _rows.size()) { | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		update( | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 			_padding.left() + index * _oneWidth - scrollCurrent(), | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 			_padding.top(), | 
					
						
							|  |  |  | 			_oneWidth, | 
					
						
							|  |  |  | 			_oneWidth); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsWidget::updateSelectedItem() { | 
					
						
							|  |  |  | 	updateItem(_selected); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | QRect SuggestionsWidget::inner() const { | 
					
						
							|  |  |  | 	return QRect(0, 0, _rows.size() * _oneWidth, _oneWidth); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QPoint SuggestionsWidget::innerShift() const { | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 	return QPoint(scrollCurrent() - _padding.left(), -_padding.top()); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QPoint SuggestionsWidget::mapToInner(QPoint globalPosition) const { | 
					
						
							|  |  |  | 	return mapFromGlobal(globalPosition) + innerShift(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | void SuggestionsWidget::mouseMoveEvent(QMouseEvent *e) { | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	const auto globalPosition = e->globalPos(); | 
					
						
							| 
									
										
										
										
											2019-04-01 14:35:32 +04:00
										 |  |  | 	if (_dragScrollStart >= 0) { | 
					
						
							|  |  |  | 		const auto delta = (_mousePressPosition.x() - globalPosition.x()); | 
					
						
							| 
									
										
										
										
											2021-01-23 06:29:50 +03:00
										 |  |  | 		const auto scroll = std::clamp( | 
					
						
							| 
									
										
										
										
											2019-04-01 14:35:32 +04:00
										 |  |  | 			_dragScrollStart + (rtl() ? -1 : 1) * delta, | 
					
						
							|  |  |  | 			0, | 
					
						
							|  |  |  | 			_scrollMax); | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 		if (scrollCurrent() != scroll) { | 
					
						
							|  |  |  | 			scrollTo(scroll); | 
					
						
							| 
									
										
										
										
											2019-04-01 14:35:32 +04:00
										 |  |  | 			update(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} else if ((_pressed >= 0) | 
					
						
							|  |  |  | 		&& (_scrollMax > 0) | 
					
						
							|  |  |  | 		&& ((_mousePressPosition - globalPosition).manhattanLength() | 
					
						
							|  |  |  | 			>= QApplication::startDragDistance())) { | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 		_dragScrollStart = scrollCurrent(); | 
					
						
							| 
									
										
										
										
											2019-04-01 14:35:32 +04:00
										 |  |  | 		_mousePressPosition = globalPosition; | 
					
						
							| 
									
										
										
										
											2019-04-01 15:16:16 +04:00
										 |  |  | 		scrollTo(_dragScrollStart); | 
					
						
							| 
									
										
										
										
											2019-04-01 14:35:32 +04:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	if (inner().contains(mapToInner(globalPosition))) { | 
					
						
							| 
									
										
										
										
											2018-12-25 16:41:40 +04:00
										 |  |  | 		if (!_lastMousePosition) { | 
					
						
							|  |  |  | 			_lastMousePosition = globalPosition; | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} else if (!_mouseSelection | 
					
						
							|  |  |  | 			&& *_lastMousePosition == globalPosition) { | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		selectByMouse(globalPosition); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		clearMouseSelection(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-25 16:41:40 +04:00
										 |  |  | void SuggestionsWidget::selectByMouse(QPoint globalPosition) { | 
					
						
							|  |  |  | 	_mouseSelection = true; | 
					
						
							|  |  |  | 	_lastMousePosition = globalPosition; | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	const auto p = mapToInner(globalPosition); | 
					
						
							|  |  |  | 	const auto selected = (p.x() >= 0) ? (p.x() / _oneWidth) : -1; | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	setSelected((selected >= 0 && selected < _rows.size()) ? selected : -1); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-12-25 16:41:40 +04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | void SuggestionsWidget::mousePressEvent(QMouseEvent *e) { | 
					
						
							| 
									
										
										
										
											2018-12-25 16:41:40 +04:00
										 |  |  | 	selectByMouse(e->globalPos()); | 
					
						
							| 
									
										
										
										
											2019-04-01 14:35:32 +04:00
										 |  |  | 	if (_selected >= 0) { | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 		setPressed(_selected); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsWidget::mouseReleaseEvent(QMouseEvent *e) { | 
					
						
							| 
									
										
										
										
											2019-04-01 14:35:32 +04:00
										 |  |  | 	if (_pressed >= 0) { | 
					
						
							|  |  |  | 		const auto pressed = _pressed; | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 		setPressed(-1); | 
					
						
							| 
									
										
										
										
											2019-04-01 14:35:32 +04:00
										 |  |  | 		if (_dragScrollStart >= 0) { | 
					
						
							|  |  |  | 			_dragScrollStart = -1; | 
					
						
							|  |  |  | 		} else if (pressed == _selected) { | 
					
						
							| 
									
										
										
										
											2017-07-27 13:22:51 +03:00
										 |  |  | 			triggerRow(_rows[_selected]); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | bool SuggestionsWidget::triggerSelectedRow() const { | 
					
						
							| 
									
										
										
										
											2019-04-01 14:35:32 +04:00
										 |  |  | 	if (_selected >= 0) { | 
					
						
							| 
									
										
										
										
											2017-07-27 13:22:51 +03:00
										 |  |  | 		triggerRow(_rows[_selected]); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		return true; | 
					
						
							| 
									
										
										
										
											2017-07-27 13:22:51 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	return false; | 
					
						
							| 
									
										
										
										
											2017-07-27 13:22:51 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | void SuggestionsWidget::triggerRow(const Row &row) const { | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 	_triggered.fire({ | 
					
						
							|  |  |  | 		row.emoji->text(), | 
					
						
							| 
									
										
										
										
											2022-08-08 14:57:33 +03:00
										 |  |  | 		(row.document | 
					
						
							|  |  |  | 			? Data::SerializeCustomEmojiId(row.document) | 
					
						
							|  |  |  | 			: QString()), | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 	}); | 
					
						
							| 
									
										
										
										
											2017-07-27 13:22:51 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-19 17:00:21 +04:00
										 |  |  | void SuggestionsWidget::enterEventHook(QEnterEvent *e) { | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	if (!inner().contains(mapToInner(QCursor::pos()))) { | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 		clearMouseSelection(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return TWidget::enterEventHook(e); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsWidget::leaveEventHook(QEvent *e) { | 
					
						
							|  |  |  | 	clearMouseSelection(); | 
					
						
							|  |  |  | 	return TWidget::leaveEventHook(e); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | SuggestionsController::SuggestionsController( | 
					
						
							|  |  |  | 	not_null<QWidget*> outer, | 
					
						
							| 
									
										
										
										
											2019-05-27 12:16:14 +02:00
										 |  |  | 	not_null<QTextEdit*> field, | 
					
						
							| 
									
										
										
										
											2019-08-02 11:40:35 +01:00
										 |  |  | 	not_null<Main::Session*> session, | 
					
						
							| 
									
										
										
										
											2019-05-27 12:16:14 +02:00
										 |  |  | 	const Options &options) | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | : _st(options.st ? *options.st : st::defaultEmojiSuggestions) | 
					
						
							|  |  |  | , _field(field) | 
					
						
							| 
									
										
										
										
											2019-08-02 11:40:35 +01:00
										 |  |  | , _session(session) | 
					
						
							| 
									
										
										
										
											2019-05-27 12:16:14 +02:00
										 |  |  | , _showExactTimer([=] { showWithQuery(getEmojiQuery()); }) | 
					
						
							|  |  |  | , _options(options) { | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | 	_container = base::make_unique_q<InnerDropdown>(outer, _st.dropdown); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	_container->setAutoHiding(false); | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 	_suggestions = _container->setOwnedWidget( | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 		object_ptr<Ui::Emoji::SuggestionsWidget>( | 
					
						
							|  |  |  | 			_container, | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | 			_st, | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | 			session, | 
					
						
							|  |  |  | 			_options.suggestCustomEmoji, | 
					
						
							|  |  |  | 			_options.allowCustomWithoutPremium)); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-13 18:56:08 +03:00
										 |  |  | 	setReplaceCallback(nullptr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-05 13:51:36 +03:00
										 |  |  | 	const auto fieldCallback = [=](not_null<QEvent*> event) { | 
					
						
							| 
									
										
										
										
											2021-06-11 19:31:16 +04:00
										 |  |  | 		return (_container && fieldFilter(event)) | 
					
						
							| 
									
										
										
										
											2019-09-19 12:28:36 +03:00
										 |  |  | 			? base::EventFilterResult::Cancel | 
					
						
							|  |  |  | 			: base::EventFilterResult::Continue; | 
					
						
							| 
									
										
										
										
											2019-09-05 13:51:36 +03:00
										 |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2019-09-19 12:28:36 +03:00
										 |  |  | 	_fieldFilter.reset(base::install_event_filter(_field, fieldCallback)); | 
					
						
							| 
									
										
										
										
											2019-09-05 13:51:36 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	const auto outerCallback = [=](not_null<QEvent*> event) { | 
					
						
							| 
									
										
										
										
											2021-06-11 19:31:16 +04:00
										 |  |  | 		return (_container && outerFilter(event)) | 
					
						
							| 
									
										
										
										
											2019-09-19 12:28:36 +03:00
										 |  |  | 			? base::EventFilterResult::Cancel | 
					
						
							|  |  |  | 			: base::EventFilterResult::Continue; | 
					
						
							| 
									
										
										
										
											2019-09-05 13:51:36 +03:00
										 |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2019-09-19 12:28:36 +03:00
										 |  |  | 	_outerFilter.reset(base::install_event_filter(outer, outerCallback)); | 
					
						
							| 
									
										
										
										
											2019-09-05 13:51:36 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 	QObject::connect( | 
					
						
							|  |  |  | 		_field, | 
					
						
							|  |  |  | 		&QTextEdit::textChanged, | 
					
						
							|  |  |  | 		_container, | 
					
						
							|  |  |  | 		[=] { handleTextChange(); }); | 
					
						
							|  |  |  | 	QObject::connect( | 
					
						
							|  |  |  | 		_field, | 
					
						
							|  |  |  | 		&QTextEdit::cursorPositionChanged, | 
					
						
							|  |  |  | 		_container, | 
					
						
							|  |  |  | 		[=] { handleCursorPositionChange(); }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	_suggestions->toggleAnimated( | 
					
						
							|  |  |  | 	) | rpl::start_with_next([=](bool visible) { | 
					
						
							|  |  |  | 		suggestionsUpdated(visible); | 
					
						
							|  |  |  | 	}, _lifetime); | 
					
						
							|  |  |  | 	_suggestions->triggered( | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 	) | rpl::start_with_next([=](const SuggestionsWidget::Chosen &chosen) { | 
					
						
							|  |  |  | 		replaceCurrent(chosen.emoji, chosen.customData); | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 	}, _lifetime); | 
					
						
							| 
									
										
										
										
											2019-03-29 20:04:05 +04:00
										 |  |  | 	Core::App().emojiKeywords().refreshed( | 
					
						
							|  |  |  | 	) | rpl::start_with_next([=] { | 
					
						
							| 
									
										
										
										
											2019-03-30 12:06:15 +04:00
										 |  |  | 		_keywordsRefreshed = true; | 
					
						
							|  |  |  | 		if (!_showExactTimer.isActive()) { | 
					
						
							|  |  |  | 			showWithQuery(_lastShownQuery); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-03-29 20:04:05 +04:00
										 |  |  | 	}, _lifetime); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	updateForceHidden(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-14 14:58:06 +04:00
										 |  |  | 	_container->shownValue( | 
					
						
							|  |  |  | 	) | rpl::filter([=](bool shown) { | 
					
						
							|  |  |  | 		return shown && !_shown; | 
					
						
							|  |  |  | 	}) | rpl::start_with_next([=] { | 
					
						
							|  |  |  | 		_container->hide(); | 
					
						
							|  |  |  | 	}, _container->lifetime()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	handleTextChange(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-02 19:23:06 +03:00
										 |  |  | not_null<SuggestionsController*> SuggestionsController::Init( | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 		not_null<QWidget*> outer, | 
					
						
							| 
									
										
										
										
											2019-05-27 12:16:14 +02:00
										 |  |  | 		not_null<Ui::InputField*> field, | 
					
						
							| 
									
										
										
										
											2019-08-02 11:40:35 +01:00
										 |  |  | 		not_null<Main::Session*> session, | 
					
						
							| 
									
										
										
										
											2019-05-27 12:16:14 +02:00
										 |  |  | 		const Options &options) { | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 	const auto result = Ui::CreateChild<SuggestionsController>( | 
					
						
							|  |  |  | 		field.get(), | 
					
						
							|  |  |  | 		outer, | 
					
						
							| 
									
										
										
										
											2019-05-27 12:16:14 +02:00
										 |  |  | 		field->rawTextEdit(), | 
					
						
							| 
									
										
										
										
											2019-08-02 11:40:35 +01:00
										 |  |  | 		session, | 
					
						
							| 
									
										
										
										
											2019-05-27 12:16:14 +02:00
										 |  |  | 		options); | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 	result->setReplaceCallback([=]( | 
					
						
							|  |  |  | 			int from, | 
					
						
							|  |  |  | 			int till, | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 			const QString &replacement, | 
					
						
							|  |  |  | 			const QString &customEmojiData) { | 
					
						
							|  |  |  | 		field->commitInstantReplacement( | 
					
						
							|  |  |  | 			from, | 
					
						
							|  |  |  | 			till, | 
					
						
							|  |  |  | 			replacement, | 
					
						
							|  |  |  | 			customEmojiData); | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 	}); | 
					
						
							|  |  |  | 	return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-13 18:56:08 +03:00
										 |  |  | void SuggestionsController::setReplaceCallback( | 
					
						
							| 
									
										
										
										
											2018-06-04 18:35:11 +03:00
										 |  |  | 	Fn<void( | 
					
						
							| 
									
										
										
										
											2018-05-13 18:56:08 +03:00
										 |  |  | 		int from, | 
					
						
							|  |  |  | 		int till, | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 		const QString &replacement, | 
					
						
							|  |  |  | 		const QString &customEmojiData)> callback) { | 
					
						
							| 
									
										
										
										
											2018-05-13 18:56:08 +03:00
										 |  |  | 	if (callback) { | 
					
						
							|  |  |  | 		_replaceCallback = std::move(callback); | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 		_replaceCallback = [=]( | 
					
						
							|  |  |  | 				int from, | 
					
						
							|  |  |  | 				int till, | 
					
						
							|  |  |  | 				const QString &replacement, | 
					
						
							|  |  |  | 				const QString &customEmojiData) { | 
					
						
							| 
									
										
										
										
											2018-05-13 18:56:08 +03:00
										 |  |  | 			auto cursor = _field->textCursor(); | 
					
						
							|  |  |  | 			cursor.setPosition(from); | 
					
						
							|  |  |  | 			cursor.setPosition(till, QTextCursor::KeepAnchor); | 
					
						
							|  |  |  | 			cursor.insertText(replacement); | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | void SuggestionsController::handleTextChange() { | 
					
						
							| 
									
										
										
										
											2020-06-18 22:04:16 +04:00
										 |  |  | 	if (Core::App().settings().suggestEmoji() | 
					
						
							| 
									
										
										
										
											2019-08-02 11:40:35 +01:00
										 |  |  | 		&& _field->textCursor().position() > 0) { | 
					
						
							| 
									
										
										
										
											2019-03-28 12:48:32 +04:00
										 |  |  | 		Core::App().emojiKeywords().refresh(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	_ignoreCursorPositionChange = true; | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 	InvokeQueued(_container, [=] { _ignoreCursorPositionChange = false; }); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 	const auto query = getEmojiQuery(); | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 	if (v::is<EmojiPtr>(query)) { | 
					
						
							|  |  |  | 		showWithQuery(query); | 
					
						
							| 
									
										
										
										
											2022-08-08 15:24:11 +03:00
										 |  |  | 		InvokeQueued(_container, [=] { | 
					
						
							|  |  |  | 			if (_shown) { | 
					
						
							|  |  |  | 				updateGeometry(); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const auto text = v::get<QString>(query); | 
					
						
							|  |  |  | 	if (text.isEmpty() || _textChangeAfterKeyPress) { | 
					
						
							|  |  |  | 		const auto exact = !text.isEmpty() && (text[0] != ':'); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		if (exact) { | 
					
						
							|  |  |  | 			const auto hidden = _container->isHidden() | 
					
						
							|  |  |  | 				|| _container->isHiding(); | 
					
						
							|  |  |  | 			_showExactTimer.callOnce(hidden ? kShowExactDelay : 0); | 
					
						
							| 
									
										
										
										
											2019-03-30 12:06:15 +04:00
										 |  |  | 		} else { | 
					
						
							|  |  |  | 			showWithQuery(query); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 			_suggestions->selectFirstResult(); | 
					
						
							| 
									
										
										
										
											2019-03-30 12:06:15 +04:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | void SuggestionsController::showWithQuery(SuggestionsQuery query) { | 
					
						
							| 
									
										
										
										
											2019-03-30 12:06:15 +04:00
										 |  |  | 	_showExactTimer.cancel(); | 
					
						
							|  |  |  | 	const auto force = base::take(_keywordsRefreshed); | 
					
						
							|  |  |  | 	_lastShownQuery = query; | 
					
						
							|  |  |  | 	_suggestions->showWithQuery(_lastShownQuery, force); | 
					
						
							| 
									
										
										
										
											2023-05-18 14:04:14 +04:00
										 |  |  | 	_container->resizeToContent(); | 
					
						
							| 
									
										
										
										
											2019-03-29 20:04:05 +04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | SuggestionsQuery SuggestionsController::getEmojiQuery() { | 
					
						
							| 
									
										
										
										
											2020-06-18 22:04:16 +04:00
										 |  |  | 	if (!Core::App().settings().suggestEmoji()) { | 
					
						
							| 
									
										
										
										
											2017-07-25 21:16:37 +03:00
										 |  |  | 		return QString(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:21:11 +04:00
										 |  |  | 	const auto cursor = _field->textCursor(); | 
					
						
							| 
									
										
										
										
											2018-05-26 17:58:21 +03:00
										 |  |  | 	if (cursor.hasSelection()) { | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 		return QString(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 17:51:18 +04:00
										 |  |  | 	const auto modernLimit = Core::App().emojiKeywords().maxQueryLength(); | 
					
						
							|  |  |  | 	const auto legacyLimit = GetSuggestionMaxLength(); | 
					
						
							| 
									
										
										
										
											2019-03-29 16:21:11 +04:00
										 |  |  | 	const auto position = cursor.position(); | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 	const auto findTextPart = [&]() -> SuggestionsQuery { | 
					
						
							| 
									
										
										
										
											2022-08-08 15:34:04 +03:00
										 |  |  | 		auto previousFragmentStart = 0; | 
					
						
							|  |  |  | 		auto previousFragmentName = QString(); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 		auto document = _field->document(); | 
					
						
							|  |  |  | 		auto block = document->findBlock(position); | 
					
						
							|  |  |  | 		for (auto i = block.begin(); !i.atEnd(); ++i) { | 
					
						
							|  |  |  | 			auto fragment = i.fragment(); | 
					
						
							| 
									
										
										
										
											2022-08-08 15:34:04 +03:00
										 |  |  | 			if (!fragment.isValid()) { | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			auto from = fragment.position(); | 
					
						
							|  |  |  | 			auto till = from + fragment.length(); | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 			const auto format = fragment.charFormat(); | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 			if (format.objectType() == InputField::kCustomEmojiFormat) { | 
					
						
							| 
									
										
										
										
											2022-08-08 15:34:04 +03:00
										 |  |  | 				previousFragmentName = QString(); | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 				continue; | 
					
						
							|  |  |  | 			} else if (format.isImageFormat()) { | 
					
						
							|  |  |  | 				const auto imageName = format.toImageFormat().name(); | 
					
						
							| 
									
										
										
										
											2022-08-08 15:34:04 +03:00
										 |  |  | 				if (from >= position || till < position) { | 
					
						
							|  |  |  | 					previousFragmentStart = from; | 
					
						
							|  |  |  | 					previousFragmentName = imageName; | 
					
						
							|  |  |  | 					continue; | 
					
						
							|  |  |  | 				} else if (const auto emoji = Emoji::FromUrl(imageName)) { | 
					
						
							| 
									
										
										
										
											2022-08-04 16:34:50 +03:00
										 |  |  | 					_queryStartPosition = position - 1; | 
					
						
							| 
									
										
										
										
											2022-08-08 15:34:04 +03:00
										 |  |  | 					const auto start = (previousFragmentName == imageName) | 
					
						
							|  |  |  | 						? previousFragmentStart | 
					
						
							|  |  |  | 						: from; | 
					
						
							|  |  |  | 					_emojiQueryLength = (position - start); | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 					return emoji; | 
					
						
							| 
									
										
										
										
											2022-08-08 15:34:04 +03:00
										 |  |  | 				} else { | 
					
						
							|  |  |  | 					continue; | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2022-08-08 15:34:04 +03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			if (from >= position || till < position) { | 
					
						
							|  |  |  | 				previousFragmentName = QString(); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 				continue; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			_queryStartPosition = from; | 
					
						
							| 
									
										
										
										
											2022-08-08 15:24:11 +03:00
										 |  |  | 			_emojiQueryLength = 0; | 
					
						
							| 
									
										
										
										
											2018-03-15 02:10:03 +03:00
										 |  |  | 			return fragment.text(); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		return QString(); | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-02 17:57:59 +03:00
										 |  |  | 	const auto part = findTextPart(); | 
					
						
							|  |  |  | 	if (const auto emoji = std::get_if<EmojiPtr>(&part)) { | 
					
						
							|  |  |  | 		return *emoji; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	const auto text = v::get<QString>(part); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	if (text.isEmpty()) { | 
					
						
							|  |  |  | 		return QString(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-03-29 17:51:18 +04:00
										 |  |  | 	const auto length = position - _queryStartPosition; | 
					
						
							|  |  |  | 	for (auto i = length; i != 0;) { | 
					
						
							| 
									
										
										
										
											2019-03-29 16:21:11 +04:00
										 |  |  | 		if (text[--i] == ':') { | 
					
						
							| 
									
										
										
										
											2019-03-29 18:23:21 +04:00
										 |  |  | 			const auto previous = (i > 0) ? text[i - 1] : QChar(0); | 
					
						
							|  |  |  | 			if (i > 0 && (previous.isLetter() || previous.isDigit())) { | 
					
						
							|  |  |  | 				return QString(); | 
					
						
							|  |  |  | 			} else if (i + 1 == length || text[i + 1].isSpace()) { | 
					
						
							| 
									
										
										
										
											2019-03-29 17:51:18 +04:00
										 |  |  | 				return QString(); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-03-29 17:51:18 +04:00
										 |  |  | 			_queryStartPosition += i + 2; | 
					
						
							|  |  |  | 			return text.mid(i, length - i); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-03-29 17:51:18 +04:00
										 |  |  | 		if (length - i > legacyLimit && length - i > modernLimit) { | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 			return QString(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-03-29 17:51:18 +04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Exact query should be full input field value.
 | 
					
						
							|  |  |  | 	const auto end = [&] { | 
					
						
							|  |  |  | 		auto cursor = _field->textCursor(); | 
					
						
							|  |  |  | 		cursor.movePosition(QTextCursor::End); | 
					
						
							|  |  |  | 		return cursor.position(); | 
					
						
							|  |  |  | 	}(); | 
					
						
							| 
									
										
										
										
											2019-05-27 12:16:14 +02:00
										 |  |  | 	if (!_options.suggestExactFirstWord | 
					
						
							|  |  |  | 		|| !length | 
					
						
							| 
									
										
										
										
											2019-03-29 18:23:21 +04:00
										 |  |  | 		|| text[0].isSpace() | 
					
						
							|  |  |  | 		|| (length > modernLimit) | 
					
						
							| 
									
										
										
										
											2019-03-29 17:51:18 +04:00
										 |  |  | 		|| (_queryStartPosition != 0) | 
					
						
							|  |  |  | 		|| (position != end)) { | 
					
						
							|  |  |  | 		return QString(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	return text; | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | void SuggestionsController::replaceCurrent( | 
					
						
							|  |  |  | 		const QString &replacement, | 
					
						
							|  |  |  | 		const QString &customEmojiData) { | 
					
						
							| 
									
										
										
										
											2022-08-08 15:24:11 +03:00
										 |  |  | 	const auto cursor = _field->textCursor(); | 
					
						
							|  |  |  | 	const auto position = cursor.position(); | 
					
						
							| 
									
										
										
										
											2019-03-29 16:21:11 +04:00
										 |  |  | 	const auto suggestion = getEmojiQuery(); | 
					
						
							| 
									
										
										
										
											2022-08-08 15:24:11 +03:00
										 |  |  | 	if (v::is<EmojiPtr>(suggestion)) { | 
					
						
							|  |  |  | 		const auto weak = Ui::MakeWeak(_container.get()); | 
					
						
							|  |  |  | 		const auto count = std::max(_emojiQueryLength, 1); | 
					
						
							|  |  |  | 		for (auto i = 0; i != count; ++i) { | 
					
						
							|  |  |  | 			const auto start = position - count + i; | 
					
						
							|  |  |  | 			_replaceCallback(start, start + 1, replacement, customEmojiData); | 
					
						
							|  |  |  | 			if (!weak) { | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else if (v::get<QString>(suggestion).isEmpty()) { | 
					
						
							| 
									
										
										
										
											2019-03-30 12:06:15 +04:00
										 |  |  | 		showWithQuery(QString()); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2022-08-08 15:24:11 +03:00
										 |  |  | 		const auto from = position - v::get<QString>(suggestion).size(); | 
					
						
							| 
									
										
										
										
											2022-07-18 20:30:28 +03:00
										 |  |  | 		_replaceCallback(from, position, replacement, customEmojiData); | 
					
						
							| 
									
										
										
										
											2017-07-25 21:16:37 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsController::handleCursorPositionChange() { | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 	InvokeQueued(_container, [=] { | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 		if (_ignoreCursorPositionChange) { | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-03-30 12:06:15 +04:00
										 |  |  | 		showWithQuery(QString()); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	}); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsController::suggestionsUpdated(bool visible) { | 
					
						
							|  |  |  | 	_shown = visible; | 
					
						
							|  |  |  | 	if (_shown) { | 
					
						
							|  |  |  | 		_container->resizeToContent(); | 
					
						
							|  |  |  | 		updateGeometry(); | 
					
						
							|  |  |  | 		if (!_forceHidden) { | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 			if (_container->isHidden() || _container->isHiding()) { | 
					
						
							|  |  |  | 				raise(); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			_container->showAnimated( | 
					
						
							|  |  |  | 				Ui::PanelAnimation::Origin::BottomLeft); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} else if (!_forceHidden) { | 
					
						
							|  |  |  | 		_container->hideAnimated(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsController::updateGeometry() { | 
					
						
							|  |  |  | 	auto cursor = _field->textCursor(); | 
					
						
							|  |  |  | 	cursor.setPosition(_queryStartPosition); | 
					
						
							|  |  |  | 	auto aroundRect = _field->cursorRect(cursor); | 
					
						
							|  |  |  | 	aroundRect.setTopLeft(_field->viewport()->mapToGlobal(aroundRect.topLeft())); | 
					
						
							|  |  |  | 	aroundRect.setTopLeft(_container->parentWidget()->mapFromGlobal(aroundRect.topLeft())); | 
					
						
							|  |  |  | 	auto boundingRect = _container->parentWidget()->rect(); | 
					
						
							|  |  |  | 	auto origin = rtl() ? PanelAnimation::Origin::BottomRight : PanelAnimation::Origin::BottomLeft; | 
					
						
							|  |  |  | 	auto point = rtl() ? (aroundRect.topLeft() + QPoint(aroundRect.width(), 0)) : aroundRect.topLeft(); | 
					
						
							| 
									
										
										
										
											2023-05-17 15:51:04 +04:00
										 |  |  | 	const auto padding = _st.dropdown.padding; | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 	const auto shift = std::min(_container->width() - padding.left() - padding.right(), st::emojiSuggestionSize) / 2; | 
					
						
							|  |  |  | 	point -= rtl() ? QPoint(_container->width() - padding.right() - shift, _container->height()) : QPoint(padding.left() + shift, _container->height()); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	if (rtl()) { | 
					
						
							|  |  |  | 		if (point.x() < boundingRect.x()) { | 
					
						
							|  |  |  | 			point.setX(boundingRect.x()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (point.x() + _container->width() > boundingRect.x() + boundingRect.width()) { | 
					
						
							|  |  |  | 			point.setX(boundingRect.x() + boundingRect.width() - _container->width()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		if (point.x() + _container->width() > boundingRect.x() + boundingRect.width()) { | 
					
						
							|  |  |  | 			point.setX(boundingRect.x() + boundingRect.width() - _container->width()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (point.x() < boundingRect.x()) { | 
					
						
							|  |  |  | 			point.setX(boundingRect.x()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (point.y() < boundingRect.y()) { | 
					
						
							|  |  |  | 		point.setY(aroundRect.y() + aroundRect.height()); | 
					
						
							|  |  |  | 		origin = (origin == PanelAnimation::Origin::BottomRight) ? PanelAnimation::Origin::TopRight : PanelAnimation::Origin::TopLeft; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	_container->move(point); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsController::updateForceHidden() { | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 	_forceHidden = !_field->isVisible() || !_field->hasFocus(); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	if (_forceHidden) { | 
					
						
							|  |  |  | 		_container->hideFast(); | 
					
						
							|  |  |  | 	} else if (_shown) { | 
					
						
							|  |  |  | 		_container->showFast(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | bool SuggestionsController::fieldFilter(not_null<QEvent*> event) { | 
					
						
							|  |  |  | 	auto type = event->type(); | 
					
						
							|  |  |  | 	switch (type) { | 
					
						
							|  |  |  | 	case QEvent::Move: | 
					
						
							|  |  |  | 	case QEvent::Resize: { | 
					
						
							|  |  |  | 		if (_shown) { | 
					
						
							|  |  |  | 			updateGeometry(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case QEvent::Show: | 
					
						
							|  |  |  | 	case QEvent::ShowToParent: | 
					
						
							|  |  |  | 	case QEvent::Hide: | 
					
						
							|  |  |  | 	case QEvent::HideToParent: | 
					
						
							|  |  |  | 	case QEvent::FocusIn: | 
					
						
							|  |  |  | 	case QEvent::FocusOut: { | 
					
						
							|  |  |  | 		updateForceHidden(); | 
					
						
							|  |  |  | 	} break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case QEvent::KeyPress: { | 
					
						
							|  |  |  | 		const auto key = static_cast<QKeyEvent*>(event.get())->key(); | 
					
						
							|  |  |  | 		switch (key) { | 
					
						
							|  |  |  | 		case Qt::Key_Enter: | 
					
						
							|  |  |  | 		case Qt::Key_Return: | 
					
						
							|  |  |  | 		case Qt::Key_Tab: | 
					
						
							|  |  |  | 		case Qt::Key_Up: | 
					
						
							|  |  |  | 		case Qt::Key_Down: | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 		case Qt::Key_Left: | 
					
						
							|  |  |  | 		case Qt::Key_Right: | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 			if (_shown && !_forceHidden) { | 
					
						
							| 
									
										
										
										
											2019-03-30 14:00:31 +04:00
										 |  |  | 				return _suggestions->handleKeyEvent(key); | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 		case Qt::Key_Escape: | 
					
						
							|  |  |  | 			if (_shown && !_forceHidden) { | 
					
						
							| 
									
										
										
										
											2019-03-30 12:06:15 +04:00
										 |  |  | 				showWithQuery(QString()); | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 				return true; | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		_textChangeAfterKeyPress = true; | 
					
						
							|  |  |  | 		InvokeQueued(_container, [=] { _textChangeAfterKeyPress = false; }); | 
					
						
							|  |  |  | 	} break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool SuggestionsController::outerFilter(not_null<QEvent*> event) { | 
					
						
							|  |  |  | 	auto type = event->type(); | 
					
						
							|  |  |  | 	switch (type) { | 
					
						
							|  |  |  | 	case QEvent::Move: | 
					
						
							|  |  |  | 	case QEvent::Resize: { | 
					
						
							| 
									
										
										
										
											2018-11-22 16:48:50 +04:00
										 |  |  | 		// updateGeometry uses not only container geometry, but also
 | 
					
						
							|  |  |  | 		// container children geometries that will be updated later.
 | 
					
						
							|  |  |  | 		InvokeQueued(_container, [=] { | 
					
						
							|  |  |  | 			if (_shown) { | 
					
						
							|  |  |  | 				updateGeometry(); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}); | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 	} break; | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-11-21 14:09:46 +04:00
										 |  |  | 	return false; | 
					
						
							| 
									
										
										
										
											2017-07-22 11:35:18 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SuggestionsController::raise() { | 
					
						
							|  |  |  | 	_container->raise(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Emoji
 | 
					
						
							|  |  |  | } // namespace Ui
 |