4170 lines
		
	
	
	
		
			116 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			4170 lines
		
	
	
	
		
			116 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // This file is part of Desktop App Toolkit,
 | |
| // a set of libraries for developing nice desktop applications.
 | |
| //
 | |
| // For license and copyright information please follow this link:
 | |
| // https://github.com/desktop-app/legal/blob/master/LEGAL
 | |
| //
 | |
| #include "ui/widgets/fields/input_field.h"
 | |
| 
 | |
| #include "ui/text/text.h"
 | |
| #include "ui/widgets/labels.h"
 | |
| #include "ui/widgets/popup_menu.h"
 | |
| #include "ui/emoji_config.h"
 | |
| #include "ui/ui_utility.h"
 | |
| #include "ui/painter.h"
 | |
| #include "base/invoke_queued.h"
 | |
| #include "base/random.h"
 | |
| #include "base/platform/base_platform_info.h"
 | |
| #include "base/qt_signal_producer.h"
 | |
| #include "emoji_suggestions_helper.h"
 | |
| #include "base/qthelp_regex.h"
 | |
| #include "base/qt/qt_common_adapters.h"
 | |
| #include "styles/style_widgets.h"
 | |
| #include "styles/palette.h"
 | |
| 
 | |
| #include <QtCore/QMimeData>
 | |
| #include <QtCore/QRegularExpression>
 | |
| #include <QtGui/QClipboard>
 | |
| #include <QtGui/QTextBlock>
 | |
| #include <QtGui/QTextDocumentFragment>
 | |
| #include <QtWidgets/QApplication>
 | |
| #include <QtWidgets/QCommonStyle>
 | |
| #include <QtWidgets/QScrollBar>
 | |
| #include <QtWidgets/QTextEdit>
 | |
| #include <QtCore/QMap>
 | |
| 
 | |
| #ifdef Q_OS_WIN
 | |
| #include <windows.h>
 | |
| #include <WinUser.h>
 | |
| #elif !defined DESKTOP_APP_DISABLE_X11_INTEGRATION // Q_OS_WIN
 | |
| #include "base/platform/linux/base_linux_xcb_utilities.h"
 | |
| 
 | |
| #include <xcb/xcb_keysyms.h>
 | |
| #include <xkbcommon/xkbcommon-keysyms.h>
 | |
| #endif // !Q_OS_WIN && !DESKTOP_APP_DISABLE_X11_INTEGRATION
 | |
| 
 | |
| namespace Ui {
 | |
| namespace {
 | |
| 
 | |
| constexpr auto kInstantReplaceRandomId = QTextFormat::UserProperty;
 | |
| constexpr auto kInstantReplaceWhatId = QTextFormat::UserProperty + 1;
 | |
| constexpr auto kInstantReplaceWithId = QTextFormat::UserProperty + 2;
 | |
| constexpr auto kReplaceTagId = QTextFormat::UserProperty + 3;
 | |
| constexpr auto kTagProperty = QTextFormat::UserProperty + 4;
 | |
| constexpr auto kCustomEmojiText = QTextFormat::UserProperty + 5;
 | |
| constexpr auto kCustomEmojiLink = QTextFormat::UserProperty + 6;
 | |
| const auto kObjectReplacementCh = QChar(QChar::ObjectReplacementCharacter);
 | |
| const auto kObjectReplacement = QString::fromRawData(
 | |
| 	&kObjectReplacementCh,
 | |
| 	1);
 | |
| const auto &kTagBold = InputField::kTagBold;
 | |
| const auto &kTagItalic = InputField::kTagItalic;
 | |
| const auto &kTagUnderline = InputField::kTagUnderline;
 | |
| const auto &kTagStrikeOut = InputField::kTagStrikeOut;
 | |
| const auto &kTagCode = InputField::kTagCode;
 | |
| const auto &kTagPre = InputField::kTagPre;
 | |
| const auto &kTagBlockquote = InputField::kTagBlockquote;
 | |
| const auto &kTagSpoiler = InputField::kTagSpoiler;
 | |
| const auto &kCustomEmojiFormat = InputField::kCustomEmojiFormat;
 | |
| const auto &kCustomEmojiId = InputField::kCustomEmojiId;
 | |
| const auto kTagCheckLinkMeta = u"^:/:/:^"_q;
 | |
| const auto kNewlineChars = QString("\r\n")
 | |
| 	+ QChar(0xfdd0) // QTextBeginningOfFrame
 | |
| 	+ QChar(0xfdd1) // QTextEndOfFrame
 | |
| 	+ QChar(QChar::ParagraphSeparator)
 | |
| 	+ QChar(QChar::LineSeparator);
 | |
| 
 | |
| QMap<QString, QString> customReplacesMap;
 | |
| 
 | |
| // We need unique tags otherwise same custom emoji would join in a single
 | |
| // QTextCharFormat with the same properties, including kCustomEmojiText.
 | |
| auto GlobalCustomEmojiCounter = 0;
 | |
| 
 | |
| [[nodiscard]] bool IsTagPre(QStringView tag) {
 | |
| 	return tag.startsWith(kTagPre);
 | |
| }
 | |
| 
 | |
| class InputDocument : public QTextDocument {
 | |
| public:
 | |
| 	InputDocument(QObject *parent, const style::InputField &st);
 | |
| 
 | |
| protected:
 | |
| 	QVariant loadResource(int type, const QUrl &name) override;
 | |
| 
 | |
| private:
 | |
| 	const style::InputField &_st;
 | |
| 	std::map<QUrl, QVariant> _emojiCache;
 | |
| 	rpl::lifetime _lifetime;
 | |
| 
 | |
| };
 | |
| 
 | |
| InputDocument::InputDocument(QObject *parent, const style::InputField &st)
 | |
| : QTextDocument(parent)
 | |
| , _st(st) {
 | |
| 	Emoji::Updated(
 | |
| 	) | rpl::start_with_next([=] {
 | |
| 		_emojiCache.clear();
 | |
| 	}, _lifetime);
 | |
| }
 | |
| 
 | |
| QVariant InputDocument::loadResource(int type, const QUrl &name) {
 | |
| 	if (type != QTextDocument::ImageResource
 | |
| 		|| name.scheme() != qstr("emoji")) {
 | |
| 		return QTextDocument::loadResource(type, name);
 | |
| 	}
 | |
| 	const auto i = _emojiCache.find(name);
 | |
| 	if (i != _emojiCache.end()) {
 | |
| 		return i->second;
 | |
| 	}
 | |
| 	auto result = [&] {
 | |
| 		if (const auto emoji = Emoji::FromUrl(name.toDisplayString())) {
 | |
| 			const auto height = std::max(
 | |
| 				_st.font->height * style::DevicePixelRatio(),
 | |
| 				Emoji::GetSizeNormal());
 | |
| 			return QVariant(Emoji::SinglePixmap(emoji, height));
 | |
| 		}
 | |
| 		return QVariant();
 | |
| 	}();
 | |
| 	_emojiCache.emplace(name, result);
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| bool IsNewline(QChar ch) {
 | |
| 	return (kNewlineChars.indexOf(ch) >= 0);
 | |
| }
 | |
| 
 | |
| [[nodiscard]] bool IsValidMarkdownLink(QStringView link) {
 | |
| 	return (link.indexOf('.') >= 0) || (link.indexOf(':') >= 0);
 | |
| }
 | |
| 
 | |
| [[nodiscard]] bool IsCustomEmojiLink(QStringView link) {
 | |
| 	return link.startsWith(InputField::kCustomEmojiTagStart);
 | |
| }
 | |
| 
 | |
| [[nodiscard]] QString MakeUniqueCustomEmojiLink(QStringView link) {
 | |
| 	if (!IsCustomEmojiLink(link)) {
 | |
| 		return link.toString();
 | |
| 	}
 | |
| 	const auto index = link.indexOf('?');
 | |
| 	return u"%1?%2"_q
 | |
| 		.arg((index < 0) ? link : base::StringViewMid(link, 0, index))
 | |
| 		.arg(++GlobalCustomEmojiCounter);
 | |
| }
 | |
| 
 | |
| [[nodiscard]] QString DefaultTagMimeProcessor(QStringView mimeTag) {
 | |
| 	// By default drop formatting in InputField-s.
 | |
| 	return {};
 | |
| }
 | |
| 
 | |
| [[nodiscard]] uint64 CustomEmojiIdFromLink(QStringView link) {
 | |
| 	const auto skip = InputField::kCustomEmojiTagStart.size();
 | |
| 	const auto index = link.indexOf('?', skip + 1);
 | |
| 	return base::StringViewMid(
 | |
| 		link,
 | |
| 		skip,
 | |
| 		(index <= skip) ? -1 : (index - skip)
 | |
| 	).toULongLong();
 | |
| }
 | |
| 
 | |
| [[nodiscard]] QString CheckFullTextTag(
 | |
| 		const TextWithTags &textWithTags,
 | |
| 		const QString &tag) {
 | |
| 	auto resultLink = QString();
 | |
| 	const auto checkingLink = (tag == kTagCheckLinkMeta);
 | |
| 	const auto &text = textWithTags.text;
 | |
| 	auto from = 0;
 | |
| 	auto till = int(text.size());
 | |
| 	const auto adjust = [&] {
 | |
| 		for (; from != till; ++from) {
 | |
| 			if (!IsNewline(text[from]) && !Text::IsSpace(text[from])) {
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 	};
 | |
| 	for (const auto &existing : textWithTags.tags) {
 | |
| 		adjust();
 | |
| 		if (existing.offset > from) {
 | |
| 			return QString();
 | |
| 		}
 | |
| 		auto found = false;
 | |
| 		for (const auto &single : TextUtilities::SplitTags(existing.id)) {
 | |
| 			const auto normalized = IsTagPre(single)
 | |
| 				? QStringView(kTagCode)
 | |
| 				: single;
 | |
| 			if (checkingLink && IsValidMarkdownLink(single)) {
 | |
| 				if (resultLink.isEmpty()) {
 | |
| 					resultLink = single.toString();
 | |
| 					found = true;
 | |
| 					break;
 | |
| 				} else if (QStringView(resultLink) == single) {
 | |
| 					found = true;
 | |
| 					break;
 | |
| 				}
 | |
| 				return QString();
 | |
| 			} else if (!checkingLink && QStringView(tag) == normalized) {
 | |
| 				found = true;
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 		if (!found) {
 | |
| 			return QString();
 | |
| 		}
 | |
| 		from = std::clamp(existing.offset + existing.length, from, till);
 | |
| 	}
 | |
| 	while (till != from) {
 | |
| 		if (!IsNewline(text[till - 1]) && !Text::IsSpace(text[till - 1])) {
 | |
| 			break;
 | |
| 		}
 | |
| 		--till;
 | |
| 	}
 | |
| 	return (from < till) ? QString() : checkingLink ? resultLink : tag;
 | |
| }
 | |
| 
 | |
| [[nodiscard]] bool HasFullTextTag(
 | |
| 		const TextWithTags &textWithTags,
 | |
| 		const QString &tag) {
 | |
| 	return !CheckFullTextTag(textWithTags, tag).isEmpty();
 | |
| }
 | |
| 
 | |
| [[nodiscard]] QString ReadPreLanguageName(
 | |
| 		const QString &text,
 | |
| 		int preStart,
 | |
| 		int preLength) {
 | |
| 	auto view = QStringView(text).mid(preStart, preLength);
 | |
| 	static const auto expression = QRegularExpression(
 | |
| 		"^([a-zA-Z0-9\\-]+)[\\r\\n]");
 | |
| 	const auto m = expression.match(view);
 | |
| 	return m.hasMatch() ? m.captured(1).toLower() : QString();
 | |
| }
 | |
| 
 | |
| class TagAccumulator {
 | |
| public:
 | |
| 	TagAccumulator(TextWithTags::Tags &tags) : _tags(tags) {
 | |
| 	}
 | |
| 
 | |
| 	[[nodiscard]] bool changed() const {
 | |
| 		return _changed;
 | |
| 	}
 | |
| 	[[nodiscard]] QString currentTag() const {
 | |
| 		return _currentTagId;
 | |
| 	}
 | |
| 
 | |
| 	void feed(const QString &randomTagId, int currentPosition) {
 | |
| 		if (randomTagId == _currentTagId) {
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		if (!_currentTagId.isEmpty()) {
 | |
| 			const auto tag = TextWithTags::Tag{
 | |
| 				_currentStart,
 | |
| 				currentPosition - _currentStart,
 | |
| 				_currentTagId
 | |
| 			};
 | |
| 			if (tag.length > 0) {
 | |
| 				if (_currentTag >= _tags.size()) {
 | |
| 					_changed = true;
 | |
| 					_tags.push_back(tag);
 | |
| 				} else if (_tags[_currentTag] != tag) {
 | |
| 					_changed = true;
 | |
| 					_tags[_currentTag] = tag;
 | |
| 				}
 | |
| 				++_currentTag;
 | |
| 			}
 | |
| 		}
 | |
| 		_currentTagId = randomTagId;
 | |
| 		_currentStart = currentPosition;
 | |
| 	};
 | |
| 
 | |
| 	void finish() {
 | |
| 		if (_currentTag < _tags.size()) {
 | |
| 			_tags.resize(_currentTag);
 | |
| 			_changed = true;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| private:
 | |
| 	TextWithTags::Tags &_tags;
 | |
| 	bool _changed = false;
 | |
| 
 | |
| 	int _currentTag = 0;
 | |
| 	int _currentStart = 0;
 | |
| 	QString _currentTagId;
 | |
| 
 | |
| };
 | |
| 
 | |
| struct TagStartExpression {
 | |
| 	QString tag;
 | |
| 	QString goodBefore;
 | |
| 	QString badAfter;
 | |
| 	QString badBefore;
 | |
| 	QString goodAfter;
 | |
| };
 | |
| 
 | |
| constexpr auto kTagBoldIndex = 0;
 | |
| constexpr auto kTagItalicIndex = 1;
 | |
| //constexpr auto kTagUnderlineIndex = 2;
 | |
| constexpr auto kTagStrikeOutIndex = 2;
 | |
| constexpr auto kTagCodeIndex = 3;
 | |
| constexpr auto kTagPreIndex = 4;
 | |
| constexpr auto kTagSpoilerIndex = 5;
 | |
| constexpr auto kInvalidPosition = std::numeric_limits<int>::max() / 2;
 | |
| 
 | |
| class TagSearchItem {
 | |
| public:
 | |
| 	enum class Edge {
 | |
| 		Open,
 | |
| 		Close,
 | |
| 	};
 | |
| 
 | |
| 	int matchPosition(Edge edge) const {
 | |
| 		return (_position >= 0) ? _position : kInvalidPosition;
 | |
| 	}
 | |
| 
 | |
| 	void applyOffset(int offset) {
 | |
| 		if (_position < offset) {
 | |
| 			_position = -1;
 | |
| 		}
 | |
| 		accumulate_max(_offset, offset);
 | |
| 	}
 | |
| 
 | |
| 	void fill(
 | |
| 			const QString &text,
 | |
| 			Edge edge,
 | |
| 			const TagStartExpression &expression) {
 | |
| 		const auto length = text.size();
 | |
| 		const auto &tag = expression.tag;
 | |
| 		const auto tagLength = tag.size();
 | |
| 		const auto isGoodBefore = [&](QChar ch) {
 | |
| 			return expression.goodBefore.isEmpty()
 | |
| 				|| (expression.goodBefore.indexOf(ch) >= 0);
 | |
| 		};
 | |
| 		const auto isBadAfter = [&](QChar ch) {
 | |
| 			return !expression.badAfter.isEmpty()
 | |
| 				&& (expression.badAfter.indexOf(ch) >= 0);
 | |
| 		};
 | |
| 		const auto isBadBefore = [&](QChar ch) {
 | |
| 			return !expression.badBefore.isEmpty()
 | |
| 				&& (expression.badBefore.indexOf(ch) >= 0);
 | |
| 		};
 | |
| 		const auto isGoodAfter = [&](QChar ch) {
 | |
| 			return expression.goodAfter.isEmpty()
 | |
| 				|| (expression.goodAfter.indexOf(ch) >= 0);
 | |
| 		};
 | |
| 		const auto check = [&](Edge edge) {
 | |
| 			if (_position > 0) {
 | |
| 				const auto before = text[_position - 1];
 | |
| 				if ((edge == Edge::Open && !isGoodBefore(before))
 | |
| 					|| (edge == Edge::Close && isBadBefore(before))) {
 | |
| 					return false;
 | |
| 				}
 | |
| 			}
 | |
| 			if (_position + tagLength < length) {
 | |
| 				const auto after = text[_position + tagLength];
 | |
| 				if ((edge == Edge::Open && isBadAfter(after))
 | |
| 					|| (edge == Edge::Close && !isGoodAfter(after))) {
 | |
| 					return false;
 | |
| 				}
 | |
| 			}
 | |
| 			return true;
 | |
| 		};
 | |
| 		const auto edgeIndex = static_cast<int>(edge);
 | |
| 		if (_position >= 0) {
 | |
| 			if (_checked[edgeIndex]) {
 | |
| 				return;
 | |
| 			} else if (check(edge)) {
 | |
| 				_checked[edgeIndex] = true;
 | |
| 				return;
 | |
| 			} else {
 | |
| 				_checked = { { false, false } };
 | |
| 			}
 | |
| 		}
 | |
| 		while (true) {
 | |
| 			_position = text.indexOf(tag, _offset);
 | |
| 			if (_position < 0) {
 | |
| 				_offset = _position = kInvalidPosition;
 | |
| 				break;
 | |
| 			}
 | |
| 			_offset = _position + tagLength;
 | |
| 			if (check(edge)) {
 | |
| 				break;
 | |
| 			} else {
 | |
| 				continue;
 | |
| 			}
 | |
| 		}
 | |
| 		if (_position == kInvalidPosition) {
 | |
| 			_checked = { { true, true } };
 | |
| 		} else {
 | |
| 			_checked = { { false, false } };
 | |
| 			_checked[edgeIndex] = true;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| private:
 | |
| 	int _offset = 0;
 | |
| 	int _position = -1;
 | |
| 	std::array<bool, 2> _checked = { { false, false } };
 | |
| 
 | |
| };
 | |
| 
 | |
| const std::vector<TagStartExpression> &TagStartExpressions() {
 | |
| 	static auto cached = std::vector<TagStartExpression> {
 | |
| 		{
 | |
| 			kTagBold,
 | |
| 			TextUtilities::MarkdownBoldGoodBefore(),
 | |
| 			TextUtilities::MarkdownBoldBadAfter(),
 | |
| 			TextUtilities::MarkdownBoldBadAfter(),
 | |
| 			TextUtilities::MarkdownBoldGoodBefore()
 | |
| 		},
 | |
| 		{
 | |
| 			kTagItalic,
 | |
| 			TextUtilities::MarkdownItalicGoodBefore(),
 | |
| 			TextUtilities::MarkdownItalicBadAfter(),
 | |
| 			TextUtilities::MarkdownItalicBadAfter(),
 | |
| 			TextUtilities::MarkdownItalicGoodBefore()
 | |
| 		},
 | |
| 		//{
 | |
| 		//	kTagUnderline,
 | |
| 		//	TextUtilities::MarkdownUnderlineGoodBefore(),
 | |
| 		//	TextUtilities::MarkdownUnderlineBadAfter(),
 | |
| 		//	TextUtilities::MarkdownUnderlineBadAfter(),
 | |
| 		//	TextUtilities::MarkdownUnderlineGoodBefore()
 | |
| 		//},
 | |
| 		{
 | |
| 			kTagStrikeOut,
 | |
| 			TextUtilities::MarkdownStrikeOutGoodBefore(),
 | |
| 			TextUtilities::MarkdownStrikeOutBadAfter(),
 | |
| 			TextUtilities::MarkdownStrikeOutBadAfter(),
 | |
| 			QString(),
 | |
| 		},
 | |
| 		{
 | |
| 			kTagCode,
 | |
| 			TextUtilities::MarkdownCodeGoodBefore(),
 | |
| 			TextUtilities::MarkdownCodeBadAfter(),
 | |
| 			TextUtilities::MarkdownCodeBadAfter(),
 | |
| 			TextUtilities::MarkdownCodeGoodBefore()
 | |
| 		},
 | |
| 		{
 | |
| 			kTagPre,
 | |
| 			TextUtilities::MarkdownPreGoodBefore(),
 | |
| 			TextUtilities::MarkdownPreBadAfter(),
 | |
| 			TextUtilities::MarkdownPreBadAfter(),
 | |
| 			TextUtilities::MarkdownPreGoodBefore()
 | |
| 		},
 | |
| 		{
 | |
| 			kTagSpoiler,
 | |
| 			TextUtilities::MarkdownSpoilerGoodBefore(),
 | |
| 			TextUtilities::MarkdownSpoilerBadAfter(),
 | |
| 			TextUtilities::MarkdownSpoilerBadAfter(),
 | |
| 			TextUtilities::MarkdownSpoilerGoodBefore()
 | |
| 		},
 | |
| 	};
 | |
| 	return cached;
 | |
| }
 | |
| 
 | |
| const std::map<QString, int> &TagIndices() {
 | |
| 	static auto cached = std::map<QString, int> {
 | |
| 		{ kTagBold, kTagBoldIndex },
 | |
| 		{ kTagItalic, kTagItalicIndex },
 | |
| 		//{ kTagUnderline, kTagUnderlineIndex },
 | |
| 		{ kTagStrikeOut, kTagStrikeOutIndex },
 | |
| 		{ kTagCode, kTagCodeIndex },
 | |
| 		{ kTagPre, kTagPreIndex },
 | |
| 		{ kTagSpoiler, kTagSpoilerIndex },
 | |
| 	};
 | |
| 	return cached;
 | |
| }
 | |
| 
 | |
| bool DoesTagFinishByNewline(const QString &tag) {
 | |
| 	return (tag == kTagCode);
 | |
| }
 | |
| 
 | |
| class MarkdownTagAccumulator {
 | |
| public:
 | |
| 	using Edge = TagSearchItem::Edge;
 | |
| 
 | |
| 	MarkdownTagAccumulator(std::vector<InputField::MarkdownTag> *tags)
 | |
| 	: _tags(tags)
 | |
| 	, _expressions(TagStartExpressions())
 | |
| 	, _tagIndices(TagIndices())
 | |
| 	, _items(_expressions.size()) {
 | |
| 	}
 | |
| 
 | |
| 	// Here we use the fact that text either contains only emoji
 | |
| 	// { adjustedTextLength = text.size() * (emojiLength - 1) }
 | |
| 	// or contains no emoji at all and can have tag edges in the middle
 | |
| 	// { adjustedTextLength = 0 }.
 | |
| 	//
 | |
| 	// Otherwise we would have to pass emoji positions inside text.
 | |
| 	void feed(
 | |
| 			const QString &text,
 | |
| 			int adjustedTextLength,
 | |
| 			const QString &textTag) {
 | |
| 		if (!_tags) {
 | |
| 			return;
 | |
| 		}
 | |
| 		const auto guard = gsl::finally([&] {
 | |
| 			_currentInternalLength += text.size();
 | |
| 			_currentAdjustedLength += adjustedTextLength;
 | |
| 		});
 | |
| 		if (!textTag.isEmpty()) {
 | |
| 			finishTags();
 | |
| 			return;
 | |
| 		}
 | |
| 		for (auto &item : _items) {
 | |
| 			item = TagSearchItem();
 | |
| 		}
 | |
| 		auto tryFinishTag = _currentTag;
 | |
| 		while (true) {
 | |
| 			for (; tryFinishTag != _currentFreeTag; ++tryFinishTag) {
 | |
| 				auto &tag = (*_tags)[tryFinishTag];
 | |
| 				if (tag.internalLength >= 0) {
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				const auto i = _tagIndices.find(tag.tag);
 | |
| 				Assert(i != end(_tagIndices));
 | |
| 				const auto tagIndex = i->second;
 | |
| 
 | |
| 				const auto atLeastOffset =
 | |
| 					tag.internalStart
 | |
| 					+ tag.tag.size()
 | |
| 					+ 1
 | |
| 					- _currentInternalLength;
 | |
| 				_items[tagIndex].applyOffset(atLeastOffset);
 | |
| 
 | |
| 				fillItem(
 | |
| 					tagIndex,
 | |
| 					text,
 | |
| 					Edge::Close);
 | |
| 				if (finishByNewline(tryFinishTag, text, tagIndex)) {
 | |
| 					continue;
 | |
| 				}
 | |
| 				const auto position = matchPosition(tagIndex, Edge::Close);
 | |
| 				if (position < kInvalidPosition) {
 | |
| 					const auto till = position + tag.tag.size();
 | |
| 					finishTag(tryFinishTag, till, true);
 | |
| 					_items[tagIndex].applyOffset(till);
 | |
| 				}
 | |
| 			}
 | |
| 			for (auto i = 0, count = int(_items.size()); i != count; ++i) {
 | |
| 				fillItem(i, text, Edge::Open);
 | |
| 			}
 | |
| 			const auto min = minIndex(Edge::Open);
 | |
| 			if (min < 0) {
 | |
| 				return;
 | |
| 			}
 | |
| 			startTag(matchPosition(min, Edge::Open), _expressions[min].tag);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void finish() {
 | |
| 		if (!_tags) {
 | |
| 			return;
 | |
| 		}
 | |
| 		finishTags();
 | |
| 		if (_currentTag < _tags->size()) {
 | |
| 			_tags->resize(_currentTag);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| private:
 | |
| 	void finishTag(int index, int offsetFromAccumulated, bool closed) {
 | |
| 		Expects(_tags != nullptr);
 | |
| 		Expects(index >= 0 && index < _tags->size());
 | |
| 
 | |
| 		auto &tag = (*_tags)[index];
 | |
| 		if (tag.internalLength < 0) {
 | |
| 			tag.internalLength = _currentInternalLength
 | |
| 				+ offsetFromAccumulated
 | |
| 				- tag.internalStart;
 | |
| 			tag.adjustedLength = _currentAdjustedLength
 | |
| 				+ offsetFromAccumulated
 | |
| 				- tag.adjustedStart;
 | |
| 			tag.closed = closed;
 | |
| 		}
 | |
| 		if (index == _currentTag) {
 | |
| 			++_currentTag;
 | |
| 		}
 | |
| 	}
 | |
| 	bool finishByNewline(
 | |
| 			int index,
 | |
| 			const QString &text,
 | |
| 			int tagIndex) {
 | |
| 		Expects(_tags != nullptr);
 | |
| 		Expects(index >= 0 && index < _tags->size());
 | |
| 
 | |
| 		auto &tag = (*_tags)[index];
 | |
| 
 | |
| 		if (!DoesTagFinishByNewline(tag.tag)) {
 | |
| 			return false;
 | |
| 		}
 | |
| 		const auto endPosition = newlinePosition(
 | |
| 			text,
 | |
| 			std::max(0, tag.internalStart + 1 - _currentInternalLength));
 | |
| 		if (matchPosition(tagIndex, Edge::Close) <= endPosition) {
 | |
| 			return false;
 | |
| 		}
 | |
| 		finishTag(index, endPosition, false);
 | |
| 		return true;
 | |
| 	}
 | |
| 	void finishTags() {
 | |
| 		while (_currentTag != _currentFreeTag) {
 | |
| 			finishTag(_currentTag, 0, false);
 | |
| 		}
 | |
| 	}
 | |
| 	void startTag(int offsetFromAccumulated, const QString &tag) {
 | |
| 		Expects(_tags != nullptr);
 | |
| 
 | |
| 		const auto newTag = InputField::MarkdownTag{
 | |
| 			_currentInternalLength + offsetFromAccumulated,
 | |
| 			-1,
 | |
| 			_currentAdjustedLength + offsetFromAccumulated,
 | |
| 			-1,
 | |
| 			false,
 | |
| 			tag
 | |
| 		};
 | |
| 		if (_currentFreeTag < _tags->size()) {
 | |
| 			(*_tags)[_currentFreeTag] = newTag;
 | |
| 		} else {
 | |
| 			_tags->push_back(newTag);
 | |
| 		}
 | |
| 		++_currentFreeTag;
 | |
| 	}
 | |
| 	void fillItem(int index, const QString &text, Edge edge) {
 | |
| 		Expects(index >= 0 && index < _items.size());
 | |
| 
 | |
| 		_items[index].fill(text, edge, _expressions[index]);
 | |
| 	}
 | |
| 	int matchPosition(int index, Edge edge) const {
 | |
| 		Expects(index >= 0 && index < _items.size());
 | |
| 
 | |
| 		return _items[index].matchPosition(edge);
 | |
| 	}
 | |
| 	int newlinePosition(const QString &text, int offset) const {
 | |
| 		const auto length = text.size();
 | |
| 		if (offset < length) {
 | |
| 			const auto begin = text.data();
 | |
| 			const auto end = begin + length;
 | |
| 			for (auto ch = begin + offset; ch != end; ++ch) {
 | |
| 				if (IsNewline(*ch)) {
 | |
| 					return (ch - begin);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		return kInvalidPosition;
 | |
| 	}
 | |
| 	int minIndex(Edge edge) const {
 | |
| 		auto result = -1;
 | |
| 		auto minPosition = kInvalidPosition;
 | |
| 		for (auto i = 0, count = int(_items.size()); i != count; ++i) {
 | |
| 			const auto position = matchPosition(i, edge);
 | |
| 			if (position < minPosition) {
 | |
| 				minPosition = position;
 | |
| 				result = i;
 | |
| 			}
 | |
| 		}
 | |
| 		return result;
 | |
| 	}
 | |
| 	int minIndexForFinish(const std::vector<int> &indices) const {
 | |
| 		const auto tagIndex = indices[0];
 | |
| 		auto result = -1;
 | |
| 		auto minPosition = kInvalidPosition;
 | |
| 		for (auto i : indices) {
 | |
| 			const auto edge = (i == tagIndex) ? Edge::Close : Edge::Open;
 | |
| 			const auto position = matchPosition(i, edge);
 | |
| 			if (position < minPosition) {
 | |
| 				minPosition = position;
 | |
| 				result = i;
 | |
| 			}
 | |
| 		}
 | |
| 		return result;
 | |
| 	}
 | |
| 
 | |
| 	std::vector<InputField::MarkdownTag> *_tags = nullptr;
 | |
| 	const std::vector<TagStartExpression> &_expressions;
 | |
| 	const std::map<QString, int> &_tagIndices;
 | |
| 	std::vector<TagSearchItem> _items;
 | |
| 
 | |
| 	int _currentTag = 0;
 | |
| 	int _currentFreeTag = 0;
 | |
| 	int _currentInternalLength = 0;
 | |
| 	int _currentAdjustedLength = 0;
 | |
| 
 | |
| };
 | |
| 
 | |
| template <typename Iterator>
 | |
| QString AccumulateText(Iterator begin, Iterator end) {
 | |
| 	auto result = QString();
 | |
| 	result.reserve(end - begin);
 | |
| 	for (auto i = end; i != begin;) {
 | |
| 		result.push_back(*--i);
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| QTextImageFormat PrepareEmojiFormat(EmojiPtr emoji, int lineHeight) {
 | |
| 	const auto factor = style::DevicePixelRatio();
 | |
| 	const auto size = Emoji::GetSizeNormal();
 | |
| 	const auto width = size + st::emojiPadding * factor * 2;
 | |
| 	const auto height = std::max(lineHeight * factor, size);
 | |
| 	auto result = QTextImageFormat();
 | |
| 	result.setWidth(width / factor);
 | |
| 	result.setHeight(height / factor);
 | |
| 	result.setName(emoji->toUrl());
 | |
| 	result.setVerticalAlignment(QTextCharFormat::AlignTop);
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| void RemoveDocumentTags(
 | |
| 		const style::InputField &st,
 | |
| 		not_null<QTextDocument*> document,
 | |
| 		int from,
 | |
| 		int end) {
 | |
| 	auto cursor = QTextCursor(document);
 | |
| 	cursor.setPosition(from);
 | |
| 	cursor.setPosition(end, QTextCursor::KeepAnchor);
 | |
| 
 | |
| 	auto format = QTextCharFormat();
 | |
| 	format.setProperty(kTagProperty, QString());
 | |
| 	format.setProperty(kReplaceTagId, QString());
 | |
| 	format.setForeground(st.textFg);
 | |
| 	format.setBackground(QBrush());
 | |
| 	format.setFont(st.font);
 | |
| 	cursor.mergeCharFormat(format);
 | |
| }
 | |
| 
 | |
| QTextCharFormat PrepareTagFormat(
 | |
| 		const style::InputField &st,
 | |
| 		QString tag) {
 | |
| 	auto result = QTextCharFormat();
 | |
| 	auto font = st.font;
 | |
| 	auto color = std::optional<style::color>();
 | |
| 	auto bg = std::optional<style::color>();
 | |
| 	auto replaceWhat = QString();
 | |
| 	auto replaceWith = QString();
 | |
| 	const auto applyOne = [&](QStringView tag) {
 | |
| 		if (IsCustomEmojiLink(tag)) {
 | |
| 			replaceWhat = tag.toString();
 | |
| 			replaceWith = MakeUniqueCustomEmojiLink(tag);
 | |
| 			result.setObjectType(kCustomEmojiFormat);
 | |
| 			result.setProperty(kCustomEmojiLink, replaceWith);
 | |
| 			result.setProperty(
 | |
| 				kCustomEmojiId,
 | |
| 				CustomEmojiIdFromLink(replaceWith));
 | |
| 			result.setVerticalAlignment(QTextCharFormat::AlignTop);
 | |
| 		} else if (IsValidMarkdownLink(tag)) {
 | |
| 			color = st::defaultTextPalette.linkFg;
 | |
| 		} else if (tag == kTagBold) {
 | |
| 			font = font->bold();
 | |
| 		} else if (tag == kTagItalic) {
 | |
| 			font = font->italic();
 | |
| 		} else if (tag == kTagUnderline) {
 | |
| 			font = font->underline();
 | |
| 		} else if (tag == kTagStrikeOut) {
 | |
| 			font = font->strikeout();
 | |
| 		} else if (tag == kTagBlockquote) {
 | |
| 			color = st::defaultTextPalette.monoFg;
 | |
| 			font = font->italic();
 | |
| 		} else if (tag == kTagCode || IsTagPre(tag)) {
 | |
| 			color = st::defaultTextPalette.monoFg;
 | |
| 			font = font->monospace();
 | |
| 		} else if (tag == kTagSpoiler) {
 | |
| 			bg = st::msgInDateFg;
 | |
| 		}
 | |
| 	};
 | |
| 	for (const auto &tag : TextUtilities::SplitTags(tag)) {
 | |
| 		applyOne(tag);
 | |
| 	}
 | |
| 	result.setFont(font);
 | |
| 	result.setForeground(color.value_or(st.textFg));
 | |
| 	result.setProperty(
 | |
| 		kTagProperty,
 | |
| 		(replaceWhat.isEmpty()
 | |
| 			? tag
 | |
| 			: std::move(tag).replace(replaceWhat, replaceWith)));
 | |
| 	if (bg) {
 | |
| 		result.setBackground(*bg);
 | |
| 	} else {
 | |
| 		result.setBackground(QBrush());
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| [[nodiscard]] QString TagWithoutCustomEmoji(QStringView tag) {
 | |
| 	auto tags = TextUtilities::SplitTags(tag);
 | |
| 	for (auto i = tags.begin(); i != tags.end();) {
 | |
| 		if (IsCustomEmojiLink(*i)) {
 | |
| 			i = tags.erase(i);
 | |
| 		} else {
 | |
| 			++i;
 | |
| 		}
 | |
| 	}
 | |
| 	return TextUtilities::JoinTag(tags);
 | |
| }
 | |
| 
 | |
| void RemoveCustomEmojiTag(
 | |
| 		const style::InputField &st,
 | |
| 		not_null<QTextDocument*> document,
 | |
| 		const QString &existingTags,
 | |
| 		int from,
 | |
| 		int end) {
 | |
| 	auto cursor = QTextCursor(document);
 | |
| 	cursor.setPosition(from);
 | |
| 	cursor.setPosition(end, QTextCursor::KeepAnchor);
 | |
| 
 | |
| 	auto format = PrepareTagFormat(st, TagWithoutCustomEmoji(existingTags));
 | |
| 	format.setProperty(kCustomEmojiLink, QString());
 | |
| 	format.setProperty(kCustomEmojiId, QString());
 | |
| 	cursor.mergeCharFormat(format);
 | |
| }
 | |
| 
 | |
| void ApplyTagFormat(QTextCharFormat &to, const QTextCharFormat &from) {
 | |
| 	if (from.hasProperty(kTagProperty)) {
 | |
| 		to.setProperty(
 | |
| 			kTagProperty,
 | |
| 			TagWithoutCustomEmoji(from.property(kTagProperty).toString()));
 | |
| 	}
 | |
| 	to.setProperty(kReplaceTagId, from.property(kReplaceTagId));
 | |
| 	to.setFont(from.font());
 | |
| 	if (from.hasProperty(QTextFormat::ForegroundBrush)) {
 | |
| 		to.setForeground(from.brushProperty(QTextFormat::ForegroundBrush));
 | |
| 	}
 | |
| 	if (from.hasProperty(QTextFormat::BackgroundBrush)) {
 | |
| 		to.setBackground(from.brushProperty(QTextFormat::BackgroundBrush));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Returns the position of the first inserted tag or "changedEnd" value if none found.
 | |
| int ProcessInsertedTags(
 | |
| 		const style::InputField &st,
 | |
| 		not_null<QTextDocument*> document,
 | |
| 		int changedPosition,
 | |
| 		int changedEnd,
 | |
| 		const TextWithTags::Tags &tags,
 | |
| 		Fn<QString(QStringView)> processor) {
 | |
| 	int firstTagStart = changedEnd;
 | |
| 	int applyNoTagFrom = changedEnd;
 | |
| 	for (const auto &tag : tags) {
 | |
| 		int tagFrom = changedPosition + tag.offset;
 | |
| 		int tagTo = tagFrom + tag.length;
 | |
| 		accumulate_max(tagFrom, changedPosition);
 | |
| 		accumulate_min(tagTo, changedEnd);
 | |
| 		auto tagId = processor ? processor(tag.id) : tag.id;
 | |
| 		if (tagTo > tagFrom && !tagId.isEmpty()) {
 | |
| 			accumulate_min(firstTagStart, tagFrom);
 | |
| 
 | |
| 			PrepareFormattingOptimization(document);
 | |
| 
 | |
| 			if (applyNoTagFrom < tagFrom) {
 | |
| 				RemoveDocumentTags(
 | |
| 					st,
 | |
| 					document,
 | |
| 					applyNoTagFrom,
 | |
| 					tagFrom);
 | |
| 			}
 | |
| 			QTextCursor c(document);
 | |
| 			c.setPosition(tagFrom);
 | |
| 			c.setPosition(tagTo, QTextCursor::KeepAnchor);
 | |
| 			c.mergeCharFormat(PrepareTagFormat(st, tagId));
 | |
| 
 | |
| 			applyNoTagFrom = tagTo;
 | |
| 		}
 | |
| 	}
 | |
| 	if (applyNoTagFrom < changedEnd) {
 | |
| 		RemoveDocumentTags(st, document, applyNoTagFrom, changedEnd);
 | |
| 	}
 | |
| 
 | |
| 	return firstTagStart;
 | |
| }
 | |
| 
 | |
| // When inserting a part of text inside a tag we need to have
 | |
| // a way to know if the insertion replaced the end of the tag
 | |
| // or it was strictly inside (in the middle) of the tag.
 | |
| bool WasInsertTillTheEndOfTag(
 | |
| 		QTextBlock block,
 | |
| 		QTextBlock::iterator fragmentIt,
 | |
| 		int insertionEnd) {
 | |
| 	const auto format = fragmentIt.fragment().charFormat();
 | |
| 	const auto insertTagName = format.property(kTagProperty);
 | |
| 	while (true) {
 | |
| 		for (; !fragmentIt.atEnd(); ++fragmentIt) {
 | |
| 			const auto fragment = fragmentIt.fragment();
 | |
| 			const auto position = fragment.position();
 | |
| 			const auto outsideInsertion = (position >= insertionEnd);
 | |
| 			if (outsideInsertion) {
 | |
| 				const auto format = fragment.charFormat();
 | |
| 				const auto tag = format.property(kTagProperty).toString();
 | |
| 				return TagWithoutCustomEmoji(tag)
 | |
| 					!= TagWithoutCustomEmoji(insertTagName.toString());
 | |
| 			}
 | |
| 			const auto end = position + fragment.length();
 | |
| 			const auto notFullFragmentInserted = (end > insertionEnd);
 | |
| 			if (notFullFragmentInserted) {
 | |
| 				return false;
 | |
| 			}
 | |
| 		}
 | |
| 		block = block.next();
 | |
| 		if (block.isValid()) {
 | |
| 			fragmentIt = block.begin();
 | |
| 		} else {
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 	// Insertion goes till the end of the text => not strictly inside a tag.
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| struct FormattingAction {
 | |
| 	enum class Type {
 | |
| 		Invalid,
 | |
| 		InsertEmoji,
 | |
| 		InsertCustomEmoji,
 | |
| 		RemoveCustomEmoji,
 | |
| 		TildeFont,
 | |
| 		RemoveTag,
 | |
| 		RemoveNewline,
 | |
| 		ClearInstantReplace,
 | |
| 		FixLineHeight,
 | |
| 	};
 | |
| 
 | |
| 	Type type = Type::Invalid;
 | |
| 	EmojiPtr emoji = nullptr;
 | |
| 	bool isTilde = false;
 | |
| 	QString tildeTag;
 | |
| 	QString existingTags;
 | |
| 	QString customEmojiText;
 | |
| 	QString customEmojiLink;
 | |
| 	int intervalStart = 0;
 | |
| 	int intervalEnd = 0;
 | |
| 
 | |
| };
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| void AddCustomReplacement(QString from, QString to) {
 | |
| 	customReplacesMap.insert(from, to);
 | |
| }
 | |
| 
 | |
| // kTagUnderline is not used for Markdown.
 | |
| 
 | |
| const QString InputField::kTagBold = u"**"_q;
 | |
| const QString InputField::kTagItalic = u"__"_q;
 | |
| const QString InputField::kTagUnderline = u"^^"_q;
 | |
| const QString InputField::kTagStrikeOut = u"~~"_q;
 | |
| const QString InputField::kTagCode = u"`"_q;
 | |
| const QString InputField::kTagPre = u"```"_q;
 | |
| const QString InputField::kTagSpoiler = u"||"_q;
 | |
| const QString InputField::kTagBlockquote = u">"_q;
 | |
| const QString InputField::kCustomEmojiTagStart = u"custom-emoji://"_q;
 | |
| const int InputField::kCustomEmojiFormat = QTextFormat::UserObject + 1;
 | |
| const int InputField::kCustomEmojiId = QTextFormat::UserProperty + 7;
 | |
| 
 | |
| class InputField::Inner final : public QTextEdit {
 | |
| public:
 | |
| 	Inner(not_null<InputField*> parent) : QTextEdit(parent) {
 | |
| 	}
 | |
| 
 | |
| protected:
 | |
| 	bool viewportEvent(QEvent *e) override {
 | |
| 		return outer()->viewportEventInner(e);
 | |
| 	}
 | |
| 	void focusInEvent(QFocusEvent *e) override {
 | |
| 		return outer()->focusInEventInner(e);
 | |
| 	}
 | |
| 	void focusOutEvent(QFocusEvent *e) override {
 | |
| 		return outer()->focusOutEventInner(e);
 | |
| 	}
 | |
| 	void keyPressEvent(QKeyEvent *e) override {
 | |
| 		return outer()->keyPressEventInner(e);
 | |
| 	}
 | |
| 	void contextMenuEvent(QContextMenuEvent *e) override {
 | |
| 		return outer()->contextMenuEventInner(e);
 | |
| 	}
 | |
| 	void dropEvent(QDropEvent *e) override {
 | |
| 		return outer()->dropEventInner(e);
 | |
| 	}
 | |
| 	void inputMethodEvent(QInputMethodEvent *e) override {
 | |
| 		return outer()->inputMethodEventInner(e);
 | |
| 	}
 | |
| 	void paintEvent(QPaintEvent *e) override {
 | |
| 		return outer()->paintEventInner(e);
 | |
| 	}
 | |
| 
 | |
| 	void mousePressEvent(QMouseEvent *e) override {
 | |
| 		return outer()->mousePressEventInner(e);
 | |
| 	}
 | |
| 	void mouseReleaseEvent(QMouseEvent *e) override {
 | |
| 		return outer()->mouseReleaseEventInner(e);
 | |
| 	}
 | |
| 	void mouseMoveEvent(QMouseEvent *e) override {
 | |
| 		return outer()->mouseMoveEventInner(e);
 | |
| 	}
 | |
| 
 | |
| 	bool canInsertFromMimeData(const QMimeData *source) const override {
 | |
| 		return outer()->canInsertFromMimeDataInner(source);
 | |
| 	}
 | |
| 	void insertFromMimeData(const QMimeData *source) override {
 | |
| 		return outer()->insertFromMimeDataInner(source);
 | |
| 	}
 | |
| 	QMimeData *createMimeDataFromSelection() const override {
 | |
| 		return outer()->createMimeDataFromSelectionInner();
 | |
| 	}
 | |
| 
 | |
| private:
 | |
| 	not_null<InputField*> outer() const {
 | |
| 		return static_cast<InputField*>(parentWidget());
 | |
| 	}
 | |
| 	friend class InputField;
 | |
| 
 | |
| #ifndef DESKTOP_APP_DISABLE_X11_INTEGRATION
 | |
| 	base::Platform::XCB::ObjectWithConnection<
 | |
| 		xcb_key_symbols_t,
 | |
| 		xcb_key_symbols_alloc,
 | |
| 		xcb_key_symbols_free
 | |
| 	> _xcbKeySymbols;
 | |
| #endif // !DESKTOP_APP_DISABLE_X11_INTEGRATION
 | |
| 
 | |
| };
 | |
| 
 | |
| void InsertEmojiAtCursor(QTextCursor cursor, EmojiPtr emoji) {
 | |
| 	const auto currentFormat = cursor.charFormat();
 | |
| 	const auto blockFormat = cursor.blockFormat();
 | |
| 	const auto type = blockFormat.lineHeightType();
 | |
| 	const auto height = (type == QTextBlockFormat::FixedHeight)
 | |
| 		? blockFormat.lineHeight()
 | |
| 		: QFontMetrics(cursor.charFormat().font()).height();
 | |
| 	auto format = PrepareEmojiFormat(emoji, height);
 | |
| 	ApplyTagFormat(format, currentFormat);
 | |
| 	cursor.insertText(kObjectReplacement, format);
 | |
| }
 | |
| 
 | |
| void InsertCustomEmojiAtCursor(
 | |
| 		not_null<InputField*> field,
 | |
| 		QTextCursor cursor,
 | |
| 		const QString &text,
 | |
| 		const QString &link) {
 | |
| 	const auto currentFormat = cursor.charFormat();
 | |
| 	const auto unique = MakeUniqueCustomEmojiLink(link);
 | |
| 	auto format = QTextCharFormat();
 | |
| 	format.setObjectType(kCustomEmojiFormat);
 | |
| 	format.setProperty(kCustomEmojiText, text);
 | |
| 	format.setProperty(kCustomEmojiLink, unique);
 | |
| 	format.setProperty(kCustomEmojiId, CustomEmojiIdFromLink(link));
 | |
| 	format.setVerticalAlignment(QTextCharFormat::AlignTop);
 | |
| 	format.setFont(field->st().font);
 | |
| 	format.setForeground(field->st().textFg);
 | |
| 	format.setBackground(QBrush());
 | |
| 	ApplyTagFormat(format, currentFormat);
 | |
| 	format.setProperty(kTagProperty, TextUtilities::TagWithAdded(
 | |
| 		format.property(kTagProperty).toString(),
 | |
| 		unique));
 | |
| 	cursor.insertText(kObjectReplacement, format);
 | |
| }
 | |
| 
 | |
| void InstantReplaces::add(const QString &what, const QString &with) {
 | |
| 	auto node = &reverseMap;
 | |
| 	for (auto i = what.end(), b = what.begin(); i != b;) {
 | |
| 		node = &node->tail.emplace(*--i, Node()).first->second;
 | |
| 	}
 | |
| 	node->text = with;
 | |
| 	accumulate_max(maxLength, int(what.size()));
 | |
| }
 | |
| 
 | |
| const InstantReplaces &InstantReplaces::Default() {
 | |
| 	static const auto result = [] {
 | |
| 		auto result = InstantReplaces();
 | |
| 		result.add("--", QString(1, QChar(8212)));
 | |
| 		result.add("<<", QString(1, QChar(171)));
 | |
| 		result.add(">>", QString(1, QChar(187)));
 | |
| 		result.add(
 | |
| 			":shrug:",
 | |
| 			QChar(175) + QString("\\_(") + QChar(12484) + ")_/" + QChar(175));
 | |
| 		result.add(":o ", QString(1, QChar(0xD83D)) + QChar(0xDE28));
 | |
| 		result.add("xD ", QString(1, QChar(0xD83D)) + QChar(0xDE06));
 | |
| 		const auto &replacements = Emoji::internal::GetAllReplacements();
 | |
| 		for (const auto &one : replacements) {
 | |
| 			const auto with = Emoji::QStringFromUTF16(one.emoji);
 | |
| 			const auto what = Emoji::QStringFromUTF16(one.replacement);
 | |
| 			result.add(what, with);
 | |
| 		}
 | |
| 		const auto &pairs = Emoji::internal::GetReplacementPairs();
 | |
| 		for (const auto &[what, index] : pairs) {
 | |
| 			const auto emoji = Emoji::internal::ByIndex(index);
 | |
| 			Assert(emoji != nullptr);
 | |
| 			result.add(what, emoji->text());
 | |
| 		}
 | |
| 		for (auto i = customReplacesMap.constBegin(), e = customReplacesMap.constEnd(); i != e; ++i) {
 | |
| 			result.add(i.key(), i.value());
 | |
| 		}
 | |
| 		return result;
 | |
| 	}();
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| const InstantReplaces &InstantReplaces::TextOnly() {
 | |
| 	static const auto result = [] {
 | |
| 		auto result = InstantReplaces();
 | |
| 		result.add("--", QString(1, QChar(8212)));
 | |
| 		result.add("<<", QString(1, QChar(171)));
 | |
| 		result.add(">>", QString(1, QChar(187)));
 | |
| 		result.add(
 | |
| 			":shrug:",
 | |
| 			QChar(175) + QString("\\_(") + QChar(12484) + ")_/" + QChar(175));
 | |
| 		for (auto i = customReplacesMap.constBegin(), e = customReplacesMap.constEnd(); i != e; ++i) {
 | |
| 			result.add(i.key(), i.value());
 | |
| 		}
 | |
| 		return result;
 | |
| 	}();
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| const InstantReplaces &InstantReplaces::Custom() {
 | |
| 	static const auto result = [] {
 | |
| 		auto result = InstantReplaces();
 | |
| 		for (auto i = customReplacesMap.constBegin(), e = customReplacesMap.constEnd(); i != e; ++i) {
 | |
| 			result.add(i.key(), i.value());
 | |
| 		}
 | |
| 		return result;
 | |
| 	}();
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| CustomEmojiObject::CustomEmojiObject(
 | |
| 	const style::font &font,
 | |
| 	Factory factory,
 | |
| 	Fn<bool()> paused)
 | |
| : _font(font)
 | |
| , _factory(std::move(factory))
 | |
| , _paused(std::move(paused))
 | |
| , _now(crl::now()) {
 | |
| }
 | |
| 
 | |
| CustomEmojiObject::~CustomEmojiObject() = default;
 | |
| 
 | |
| void *CustomEmojiObject::qt_metacast(const char *iid) {
 | |
| 	if (QLatin1String(iid) == qobject_interface_iid<QTextObjectInterface*>()) {
 | |
| 		return static_cast<QTextObjectInterface*>(this);
 | |
| 	}
 | |
| 	return QObject::qt_metacast(iid);
 | |
| }
 | |
| 
 | |
| QSizeF CustomEmojiObject::intrinsicSize(
 | |
| 		QTextDocument *doc,
 | |
| 		int posInDocument,
 | |
| 		const QTextFormat &format) {
 | |
| 	const auto size = st::emojiSize * 1.;
 | |
| 	const auto width = size + st::emojiPadding * 2.;
 | |
| 	const auto height = std::max(_font->height * 1., size);
 | |
| 	if (!_skip) {
 | |
| 		const auto emoji = Text::AdjustCustomEmojiSize(st::emojiSize);
 | |
| 		_skip = (st::emojiSize - emoji) / 2;
 | |
| 	}
 | |
| 	return { width, height };
 | |
| }
 | |
| 
 | |
| void CustomEmojiObject::drawObject(
 | |
| 		QPainter *painter,
 | |
| 		const QRectF &rect,
 | |
| 		QTextDocument *doc,
 | |
| 		int posInDocument,
 | |
| 		const QTextFormat &format) {
 | |
| 	const auto id = format.property(kCustomEmojiId).toULongLong();
 | |
| 	if (!id) {
 | |
| 		return;
 | |
| 	}
 | |
| 	auto i = _emoji.find(id);
 | |
| 	if (i == end(_emoji)) {
 | |
| 		const auto link = format.property(kCustomEmojiLink).toString();
 | |
| 		const auto data = InputField::CustomEmojiEntityData(link);
 | |
| 		if (auto emoji = _factory(data)) {
 | |
| 			i = _emoji.emplace(id, std::move(emoji)).first;
 | |
| 		}
 | |
| 	}
 | |
| 	if (i == end(_emoji)) {
 | |
| 		return;
 | |
| 	}
 | |
| 	i->second->paint(*painter, {
 | |
| 		.textColor = format.foreground().color(),
 | |
| 		.now = _now,
 | |
| 		.position = QPoint(
 | |
| 			int(base::SafeRound(rect.x())) + st::emojiPadding + _skip,
 | |
| 			int(base::SafeRound(rect.y())) + _skip),
 | |
| 		.paused = _paused && _paused(),
 | |
| 	});
 | |
| }
 | |
| 
 | |
| void CustomEmojiObject::clear() {
 | |
| 	_emoji.clear();
 | |
| }
 | |
| 
 | |
| void CustomEmojiObject::setNow(crl::time now) {
 | |
| 	_now = now;
 | |
| }
 | |
| 
 | |
| InputField::InputField(
 | |
| 	QWidget *parent,
 | |
| 	const style::InputField &st,
 | |
| 	rpl::producer<QString> placeholder,
 | |
| 	const QString &value)
 | |
| : InputField(
 | |
| 	parent,
 | |
| 	st,
 | |
| 	Mode::SingleLine,
 | |
| 	std::move(placeholder),
 | |
| 	{ value, {} }) {
 | |
| }
 | |
| 
 | |
| InputField::InputField(
 | |
| 	QWidget *parent,
 | |
| 	const style::InputField &st,
 | |
| 	Mode mode,
 | |
| 	rpl::producer<QString> placeholder,
 | |
| 	const QString &value)
 | |
| : InputField(
 | |
| 	parent,
 | |
| 	st,
 | |
| 	mode,
 | |
| 	std::move(placeholder),
 | |
| 	{ value, {} }) {
 | |
| }
 | |
| 
 | |
| InputField::InputField(
 | |
| 	QWidget *parent,
 | |
| 	const style::InputField &st,
 | |
| 	Mode mode,
 | |
| 	rpl::producer<QString> placeholder,
 | |
| 	const TextWithTags &value)
 | |
| : RpWidget(parent)
 | |
| , _st(st)
 | |
| , _mode(mode)
 | |
| , _minHeight(st.heightMin)
 | |
| , _maxHeight(st.heightMax)
 | |
| , _inner(std::make_unique<Inner>(this))
 | |
| , _lastTextWithTags(value)
 | |
| , _placeholderFull(std::move(placeholder)) {
 | |
| 	_inner->setDocument(CreateChild<InputDocument>(_inner.get(), _st));
 | |
| 	_inner->setAcceptRichText(false);
 | |
| 	resize(_st.width, _minHeight);
 | |
| 
 | |
| 	{ // In case of default fonts all those should be zero.
 | |
| 		const auto metrics = QFontMetricsF(_st.font->f);
 | |
| 		const auto realAscent = int(base::SafeRound(metrics.ascent()));
 | |
| 		const auto ascentAdd = _st.font->ascent - realAscent;
 | |
| 		//const auto realHeight = int(base::SafeRound(metrics.height()));
 | |
| 		//const auto heightAdd = _st.font->height - realHeight - ascentAdd;
 | |
| 		//_customFontMargins = QMargins(0, ascentAdd, 0, heightAdd);
 | |
| 		_customFontMargins = QMargins(0, ascentAdd, 0, -ascentAdd);
 | |
| 		// We move _inner down by ascentAdd for the first line to look
 | |
| 		// at the same vertical position as in the default font.
 | |
| 		//
 | |
| 		// But we don't want to get vertical scroll in case the field
 | |
| 		// fits pixel-perfect with the default font, so we allow the
 | |
| 		// bottom margin to be the same shift, but negative.
 | |
| 
 | |
| 		if (_mode != Mode::SingleLine) {
 | |
| 			const auto metrics = QFontMetricsF(_st.font->f);
 | |
| 			const auto leading = qMax(metrics.leading(), qreal(0.0));
 | |
| 			const auto adjustment = (metrics.ascent() + leading)
 | |
| 				- ((_st.font->height * 4) / 5);
 | |
| 			_placeholderCustomFontSkip = int(base::SafeRound(-adjustment));
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (_st.textBg->c.alphaF() >= 1.
 | |
| 		&& !_st.borderRadius) {
 | |
| 		setAttribute(Qt::WA_OpaquePaintEvent);
 | |
| 	}
 | |
| 
 | |
| 	_inner->setFont(_st.font->f);
 | |
| 	_inner->setAlignment(_st.textAlign);
 | |
| 	if (_mode == Mode::SingleLine) {
 | |
| 		_inner->setWordWrapMode(QTextOption::NoWrap);
 | |
| 	}
 | |
| 
 | |
| 	_placeholderFull.value(
 | |
| 	) | rpl::start_with_next([=](const QString &text) {
 | |
| 		refreshPlaceholder(text);
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	style::PaletteChanged(
 | |
| 	) | rpl::start_with_next([=] {
 | |
| 		updatePalette();
 | |
| 	}, lifetime());
 | |
| 	{
 | |
| 		auto cursor = _inner->textCursor();
 | |
| 
 | |
| 		_defaultCharFormat = cursor.charFormat();
 | |
| 		updatePalette();
 | |
| 		cursor.setCharFormat(_defaultCharFormat);
 | |
| 
 | |
| 		_defaultBlockFormat = cursor.blockFormat();
 | |
| 		_defaultBlockFormat.setLineHeight(
 | |
| 			_st.font->height,
 | |
| 			QTextBlockFormat::FixedHeight);
 | |
| 		cursor.setBlockFormat(_defaultBlockFormat);
 | |
| 
 | |
| 		_inner->setTextCursor(cursor);
 | |
| 	}
 | |
| 	_inner->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
 | |
| 	_inner->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
 | |
| 
 | |
| 	_inner->setFrameStyle(int(QFrame::NoFrame) | QFrame::Plain);
 | |
| 	_inner->viewport()->setAutoFillBackground(false);
 | |
| 
 | |
| 	_inner->setContentsMargins(0, 0, 0, 0);
 | |
| 	_inner->document()->setDocumentMargin(0);
 | |
| 
 | |
| 	setAttribute(Qt::WA_AcceptTouchEvents);
 | |
| 	_inner->viewport()->setAttribute(Qt::WA_AcceptTouchEvents);
 | |
| 
 | |
| 	_touchTimer.setCallback([=] { _touchRightButton = true; });
 | |
| 
 | |
| 	base::qt_signal_producer(
 | |
| 		_inner->document(),
 | |
| 		&QTextDocument::contentsChange
 | |
| 	) | rpl::start_with_next([=](int position, int removed, int added) {
 | |
| 		documentContentsChanged(position, removed, added);
 | |
| 	}, lifetime());
 | |
| 	base::qt_signal_producer(
 | |
| 		_inner.get(),
 | |
| 		&QTextEdit::undoAvailable
 | |
| 	) | rpl::start_with_next([=](bool undoAvailable) {
 | |
| 		_undoAvailable = undoAvailable;
 | |
| 		Integration::Instance().textActionsUpdated();
 | |
| 	}, lifetime());
 | |
| 	base::qt_signal_producer(
 | |
| 		_inner.get(),
 | |
| 		&QTextEdit::redoAvailable
 | |
| 	) | rpl::start_with_next([=](bool redoAvailable) {
 | |
| 		_redoAvailable = redoAvailable;
 | |
| 		Integration::Instance().textActionsUpdated();
 | |
| 	}, lifetime());
 | |
| 	base::qt_signal_producer(
 | |
| 		_inner.get(),
 | |
| 		&QTextEdit::cursorPositionChanged
 | |
| 	) | rpl::start_with_next([=] {
 | |
| 		auto cursor = textCursor();
 | |
| 		if (!cursor.hasSelection() && !cursor.position()) {
 | |
| 			cursor.setCharFormat(_defaultCharFormat);
 | |
| 			cursor.setBlockFormat(_defaultBlockFormat);
 | |
| 			setTextCursor(cursor);
 | |
| 		}
 | |
| 	}, lifetime());
 | |
| 	base::qt_signal_producer(
 | |
| 		_inner.get(),
 | |
| 		&Inner::selectionChanged
 | |
| 	) | rpl::start_with_next([] {
 | |
| 		Integration::Instance().textActionsUpdated();
 | |
| 	}, lifetime());
 | |
| 
 | |
| 	const auto bar = _inner->verticalScrollBar();
 | |
| 	_scrollTop = bar->value();
 | |
| 	connect(bar, &QScrollBar::valueChanged, [=] {
 | |
| 		_scrollTop = bar->value();
 | |
| 	});
 | |
| 
 | |
| 	setCursor(style::cur_text);
 | |
| 	heightAutoupdated();
 | |
| 
 | |
| 	if (!value.text.isEmpty()) {
 | |
| 		setTextWithTags(value, HistoryAction::Clear);
 | |
| 	}
 | |
| 
 | |
| 	startBorderAnimation();
 | |
| 	startPlaceholderAnimation();
 | |
| 	finishAnimating();
 | |
| }
 | |
| 
 | |
| const rpl::variable<int> &InputField::scrollTop() const {
 | |
| 	return _scrollTop;
 | |
| }
 | |
| 
 | |
| int InputField::scrollTopMax() const {
 | |
| 	return _inner->verticalScrollBar()->maximum();
 | |
| }
 | |
| 
 | |
| void InputField::scrollTo(int top) {
 | |
| 	_inner->verticalScrollBar()->setValue(top);
 | |
| }
 | |
| 
 | |
| 
 | |
| bool InputField::menuShown() const {
 | |
| 	return _contextMenu != nullptr;
 | |
| }
 | |
| 
 | |
| rpl::producer<bool> InputField::menuShownValue() const {
 | |
| 	return _menuShownChanges.events_starting_with(menuShown());
 | |
| }
 | |
| 
 | |
| bool InputField::viewportEventInner(QEvent *e) {
 | |
| 	if (e->type() == QEvent::TouchBegin
 | |
| 		|| e->type() == QEvent::TouchUpdate
 | |
| 		|| e->type() == QEvent::TouchEnd
 | |
| 		|| e->type() == QEvent::TouchCancel) {
 | |
| 		const auto ev = static_cast<QTouchEvent*>(e);
 | |
| 		if (ev->device()->type() == base::TouchDevice::TouchScreen) {
 | |
| 			handleTouchEvent(ev);
 | |
| 			return false;
 | |
| 		}
 | |
| 	} else if (e->type() == QEvent::Paint && _customEmojiObject) {
 | |
| 		_customEmojiObject->setNow(crl::now());
 | |
| 	}
 | |
| 	return _inner->QTextEdit::viewportEvent(e);
 | |
| }
 | |
| 
 | |
| void InputField::updatePalette() {
 | |
| 	auto p = _inner->palette();
 | |
| 	p.setColor(QPalette::Text, _st.textFg->c);
 | |
| 	p.setColor(QPalette::Highlight, st::msgInBgSelected->c);
 | |
| 	p.setColor(QPalette::HighlightedText, st::historyTextInFgSelected->c);
 | |
| 	_inner->setPalette(p);
 | |
| 
 | |
| 	_defaultCharFormat.merge(PrepareTagFormat(_st, QString()));
 | |
| 	auto cursor = textCursor();
 | |
| 
 | |
| 	const auto document = _inner->document();
 | |
| 	auto block = document->begin();
 | |
| 	const auto end = document->end();
 | |
| 	for (; block != end; block = block.next()) {
 | |
| 		auto till = block.position();
 | |
| 		for (auto i = block.begin(); !i.atEnd();) {
 | |
| 			for (; !i.atEnd(); ++i) {
 | |
| 				const auto fragment = i.fragment();
 | |
| 				if (!fragment.isValid() || fragment.position() < till) {
 | |
| 					continue;
 | |
| 				}
 | |
| 				till = fragment.position() + fragment.length();
 | |
| 
 | |
| 				auto format = fragment.charFormat();
 | |
| 				const auto tag = format.property(kTagProperty).toString();
 | |
| 				const auto updatedFormat = PrepareTagFormat(_st, tag);
 | |
| 				format.setForeground(updatedFormat.foreground());
 | |
| 				format.setBackground(updatedFormat.background());
 | |
| 				cursor.setPosition(fragment.position());
 | |
| 				cursor.setPosition(till, QTextCursor::KeepAnchor);
 | |
| 				cursor.mergeCharFormat(format);
 | |
| 				i = block.begin();
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	cursor = textCursor();
 | |
| 	if (!cursor.hasSelection()) {
 | |
| 		auto format = cursor.charFormat();
 | |
| 		format.merge(PrepareTagFormat(
 | |
| 			_st,
 | |
| 			TagWithoutCustomEmoji(
 | |
| 				format.property(kTagProperty).toString())));
 | |
| 		cursor.setCharFormat(format);
 | |
| 		setTextCursor(cursor);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::setExtendedContextMenu(
 | |
| 	rpl::producer<ExtendedContextMenu> value) {
 | |
| 	std::move(
 | |
| 		value
 | |
| 	) | rpl::start_with_next([=](auto pair) {
 | |
| 		auto &[menu, e] = pair;
 | |
| 		contextMenuEventInner(e.get(), std::move(menu));
 | |
| 	}, lifetime());
 | |
| }
 | |
| 
 | |
| void InputField::setInstantReplaces(const InstantReplaces &replaces) {
 | |
| 	_mutableInstantReplaces = replaces;
 | |
| }
 | |
| 
 | |
| void InputField::setInstantReplaces(rpl::producer<InstantReplaces> producer) {
 | |
| 	std::move(
 | |
| 		producer
 | |
| 	) | rpl::start_with_next([=](InstantReplaces replaces) {
 | |
| 		_mutableInstantReplaces = replaces;
 | |
| 	}, lifetime());
 | |
| }
 | |
| 
 | |
| void InputField::setInstantReplacesEnabled(rpl::producer<bool> enabled) {
 | |
| 	std::move(
 | |
| 		enabled
 | |
| 	) | rpl::start_with_next([=](bool value) {
 | |
| 		_instantReplacesEnabled = value;
 | |
| 	}, lifetime());
 | |
| }
 | |
| 
 | |
| void InputField::setMarkdownReplacesEnabled(rpl::producer<bool> enabled) {
 | |
| 	std::move(
 | |
| 		enabled
 | |
| 	) | rpl::start_with_next([=](bool value) {
 | |
| 		if (_markdownEnabled != value) {
 | |
| 			_markdownEnabled = value;
 | |
| 			if (_markdownEnabled) {
 | |
| 				handleContentsChanged();
 | |
| 			} else {
 | |
| 				_lastMarkdownTags = {};
 | |
| 			}
 | |
| 		}
 | |
| 	}, lifetime());
 | |
| }
 | |
| 
 | |
| void InputField::setTagMimeProcessor(Fn<QString(QStringView)> processor) {
 | |
| 	_tagMimeProcessor = std::move(processor);
 | |
| }
 | |
| 
 | |
| void InputField::setCustomEmojiFactory(
 | |
| 		CustomEmojiFactory factory,
 | |
| 		Fn<bool()> paused) {
 | |
| 	_customEmojiObject = std::make_unique<CustomEmojiObject>(_st.font, [=](
 | |
| 			QStringView data) {
 | |
| 		return factory(data, [=] { customEmojiRepaint(); });
 | |
| 	}, std::move(paused));
 | |
| 	_inner->document()->documentLayout()->registerHandler(
 | |
| 		kCustomEmojiFormat,
 | |
| 		_customEmojiObject.get());
 | |
| }
 | |
| 
 | |
| void InputField::customEmojiRepaint() {
 | |
| 	if (_customEmojiRepaintScheduled) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_customEmojiRepaintScheduled = true;
 | |
| 	_inner->update();
 | |
| }
 | |
| 
 | |
| void InputField::paintEventInner(QPaintEvent *e) {
 | |
| 	_customEmojiRepaintScheduled = false;
 | |
| 	_inner->QTextEdit::paintEvent(e);
 | |
| }
 | |
| 
 | |
| void InputField::setAdditionalMargin(int margin) {
 | |
| 	setAdditionalMargins({ margin, margin, margin, margin });
 | |
| }
 | |
| 
 | |
| void InputField::setAdditionalMargins(QMargins margins) {
 | |
| 	_additionalMargins = margins;
 | |
| 	QResizeEvent e(size(), size());
 | |
| 	QCoreApplication::sendEvent(this, &e);
 | |
| }
 | |
| 
 | |
| void InputField::setMaxLength(int length) {
 | |
| 	if (_maxLength != length) {
 | |
| 		_maxLength = length;
 | |
| 		if (_maxLength > 0) {
 | |
| 			const auto document = _inner->document();
 | |
| 			_correcting = true;
 | |
| 			QTextCursor(document).joinPreviousEditBlock();
 | |
| 			const auto guard = gsl::finally([&] {
 | |
| 				_correcting = false;
 | |
| 				QTextCursor(document).endEditBlock();
 | |
| 				handleContentsChanged();
 | |
| 			});
 | |
| 
 | |
| 			auto cursor = QTextCursor(document);
 | |
| 			cursor.movePosition(QTextCursor::End);
 | |
| 			chopByMaxLength(0, cursor.position());
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::setMinHeight(int height) {
 | |
| 	_minHeight = height;
 | |
| }
 | |
| 
 | |
| void InputField::setMaxHeight(int height) {
 | |
| 	_maxHeight = height;
 | |
| }
 | |
| 
 | |
| void InputField::insertTag(const QString &text, QString tagId) {
 | |
| 	auto cursor = textCursor();
 | |
| 	const auto position = cursor.position();
 | |
| 
 | |
| 	const auto document = _inner->document();
 | |
| 	auto block = document->findBlock(position);
 | |
| 	for (auto iter = block.begin(); !iter.atEnd(); ++iter) {
 | |
| 		auto fragment = iter.fragment();
 | |
| 		Assert(fragment.isValid());
 | |
| 
 | |
| 		const auto fragmentPosition = fragment.position();
 | |
| 		const auto fragmentEnd = (fragmentPosition + fragment.length());
 | |
| 		if (fragmentPosition >= position || fragmentEnd < position) {
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		const auto format = fragment.charFormat();
 | |
| 		if (format.isImageFormat()) {
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		auto mentionInCommand = false;
 | |
| 		const auto fragmentText = fragment.text();
 | |
| 		for (auto i = position - fragmentPosition; i > 0; --i) {
 | |
| 			const auto previous = fragmentText[i - 1];
 | |
| 			if (previous == '@' || previous == '#' || previous == '/') {
 | |
| 				if ((i == position - fragmentPosition
 | |
| 					|| (previous == '/'
 | |
| 						? fragmentText[i].isLetterOrNumber()
 | |
| 						: fragmentText[i].isLetter())
 | |
| 					|| previous == '#') &&
 | |
| 					(i < 2 || !(fragmentText[i - 2].isLetterOrNumber()
 | |
| 						|| fragmentText[i - 2] == '_'))) {
 | |
| 					cursor.setPosition(fragmentPosition + i - 1);
 | |
| 					auto till = fragmentPosition + i;
 | |
| 					for (; (till < fragmentEnd && till < position); ++till) {
 | |
| 						const auto ch = fragmentText[till - fragmentPosition];
 | |
| 						if (!ch.isLetterOrNumber() && ch != '_' && ch != '@') {
 | |
| 							break;
 | |
| 						}
 | |
| 					}
 | |
| 					if (till < fragmentEnd
 | |
| 						&& fragmentText[till - fragmentPosition] == ' ') {
 | |
| 						++till;
 | |
| 					}
 | |
| 					cursor.setPosition(till, QTextCursor::KeepAnchor);
 | |
| 					break;
 | |
| 				} else if ((i == position - fragmentPosition
 | |
| 					|| fragmentText[i].isLetter())
 | |
| 					&& fragmentText[i - 1] == '@'
 | |
| 					&& (i > 2)
 | |
| 					&& (fragmentText[i - 2].isLetterOrNumber()
 | |
| 						|| fragmentText[i - 2] == '_')
 | |
| 					&& !mentionInCommand) {
 | |
| 					mentionInCommand = true;
 | |
| 					--i;
 | |
| 					continue;
 | |
| 				}
 | |
| 				break;
 | |
| 			}
 | |
| 			if (position - fragmentPosition - i > 127
 | |
| 				|| (!mentionInCommand
 | |
| 					&& (position - fragmentPosition - i > 63))
 | |
| 				|| (!fragmentText[i - 1].isLetterOrNumber()
 | |
| 					&& fragmentText[i - 1] != '_')) {
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 		break;
 | |
| 	}
 | |
| 	if (tagId.isEmpty()) {
 | |
| 		cursor.insertText(text + ' ', _defaultCharFormat);
 | |
| 	} else {
 | |
| 		_insertedTags.clear();
 | |
| 		_insertedTags.push_back({ 0, int(text.size()), tagId });
 | |
| 		_insertedTagsAreFromMime = false;
 | |
| 		cursor.insertText(text + ' ');
 | |
| 		_insertedTags.clear();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool InputField::heightAutoupdated() {
 | |
| 	if (_minHeight < 0
 | |
| 		|| _maxHeight < 0
 | |
| 		|| _inHeightCheck
 | |
| 		|| _mode == Mode::SingleLine) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	_inHeightCheck = true;
 | |
| 	const auto guard = gsl::finally([&] { _inHeightCheck = false; });
 | |
| 
 | |
| 	SendPendingMoveResizeEvents(this);
 | |
| 
 | |
| 	const auto contentHeight = int(std::ceil(document()->size().height()))
 | |
| 		+ _st.textMargins.top()
 | |
| 		+ _st.textMargins.bottom()
 | |
| 		+ _additionalMargins.top()
 | |
| 		+ _additionalMargins.bottom();
 | |
| 	const auto newHeight = std::clamp(contentHeight, _minHeight, _maxHeight);
 | |
| 	if (height() != newHeight) {
 | |
| 		resize(width(), newHeight);
 | |
| 		return true;
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| void InputField::checkContentHeight() {
 | |
| 	if (heightAutoupdated()) {
 | |
| 		_heightChanges.fire({});
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::handleTouchEvent(QTouchEvent *e) {
 | |
| 	switch (e->type()) {
 | |
| 	case QEvent::TouchBegin: {
 | |
| 		if (_touchPress || e->touchPoints().isEmpty()) {
 | |
| 			return;
 | |
| 		}
 | |
| 		_touchTimer.callOnce(QApplication::startDragTime());
 | |
| 		_touchPress = true;
 | |
| 		_touchMove = _touchRightButton = false;
 | |
| 		_touchStart = e->touchPoints().cbegin()->screenPos().toPoint();
 | |
| 	} break;
 | |
| 
 | |
| 	case QEvent::TouchUpdate: {
 | |
| 		if (!e->touchPoints().isEmpty()) {
 | |
| 			touchUpdate(e->touchPoints().cbegin()->screenPos().toPoint());
 | |
| 		}
 | |
| 	} break;
 | |
| 
 | |
| 	case QEvent::TouchEnd: {
 | |
| 		touchFinish();
 | |
| 	} break;
 | |
| 
 | |
| 	case QEvent::TouchCancel: {
 | |
| 		_touchPress = false;
 | |
| 		_touchTimer.cancel();
 | |
| 	} break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::touchUpdate(QPoint globalPosition) {
 | |
| 	if (_touchPress
 | |
| 		&& !_touchMove
 | |
| 		&& ((globalPosition - _touchStart).manhattanLength()
 | |
| 			>= QApplication::startDragDistance())) {
 | |
| 		_touchMove = true;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::touchFinish() {
 | |
| 	if (!_touchPress) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto weak = MakeWeak(this);
 | |
| 	if (!_touchMove && window()) {
 | |
| 		QPoint mapped(mapFromGlobal(_touchStart));
 | |
| 
 | |
| 		if (_touchRightButton) {
 | |
| 			QContextMenuEvent contextEvent(
 | |
| 				QContextMenuEvent::Mouse,
 | |
| 				mapped,
 | |
| 				_touchStart);
 | |
| 			contextMenuEvent(&contextEvent);
 | |
| 		} else {
 | |
| 			QGuiApplication::inputMethod()->show();
 | |
| 		}
 | |
| 	}
 | |
| 	if (weak) {
 | |
| 		_touchTimer.cancel();
 | |
| 		_touchPress
 | |
| 			= _touchMove
 | |
| 			= _touchRightButton
 | |
| 			= _mousePressedInTouch = false;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::paintSurrounding(
 | |
| 		QPainter &p,
 | |
| 		QRect clip,
 | |
| 		float64 errorDegree,
 | |
| 		float64 focusedDegree) {
 | |
| 	if (_st.borderRadius > 0) {
 | |
| 		paintRoundSurrounding(p, clip, errorDegree, focusedDegree);
 | |
| 	} else {
 | |
| 		paintFlatSurrounding(p, clip, errorDegree, focusedDegree);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::paintRoundSurrounding(
 | |
| 		QPainter &p,
 | |
| 		QRect clip,
 | |
| 		float64 errorDegree,
 | |
| 		float64 focusedDegree) {
 | |
| 	const auto divide = _st.borderDenominator ? _st.borderDenominator : 1;
 | |
| 	const auto border = _st.border / float64(divide);
 | |
| 	const auto borderHalf = border / 2.;
 | |
| 	auto pen = anim::pen(_st.borderFg, _st.borderFgActive, focusedDegree);
 | |
| 	pen.setWidthF(border);
 | |
| 	p.setPen(pen);
 | |
| 	p.setBrush(anim::brush(_st.textBg, _st.textBgActive, focusedDegree));
 | |
| 
 | |
| 	PainterHighQualityEnabler hq(p);
 | |
| 	const auto radius = _st.borderRadius - borderHalf;
 | |
| 	p.drawRoundedRect(
 | |
| 		QRectF(0, 0, width(), height()).marginsRemoved(
 | |
| 			QMarginsF(borderHalf, borderHalf, borderHalf, borderHalf)),
 | |
| 		radius,
 | |
| 		radius);
 | |
| }
 | |
| 
 | |
| void InputField::paintFlatSurrounding(
 | |
| 		QPainter &p,
 | |
| 		QRect clip,
 | |
| 		float64 errorDegree,
 | |
| 		float64 focusedDegree) {
 | |
| 	if (_st.textBg->c.alphaF() > 0.) {
 | |
| 		p.fillRect(clip, _st.textBg);
 | |
| 	}
 | |
| 	if (_st.border) {
 | |
| 		p.fillRect(0, height() - _st.border, width(), _st.border, _st.borderFg);
 | |
| 	}
 | |
| 	const auto borderShownDegree = _a_borderShown.value(1.);
 | |
| 	const auto borderOpacity = _a_borderOpacity.value(_borderVisible ? 1. : 0.);
 | |
| 	if (_st.borderActive && (borderOpacity > 0.)) {
 | |
| 		auto borderStart = std::clamp(_borderAnimationStart, 0, width());
 | |
| 		auto borderFrom = qRound(borderStart * (1. - borderShownDegree));
 | |
| 		auto borderTo = borderStart + qRound((width() - borderStart) * borderShownDegree);
 | |
| 		if (borderTo > borderFrom) {
 | |
| 			auto borderFg = anim::brush(_st.borderFgActive, _st.borderFgError, errorDegree);
 | |
| 			p.setOpacity(borderOpacity);
 | |
| 			p.fillRect(borderFrom, height() - _st.borderActive, borderTo - borderFrom, _st.borderActive, borderFg);
 | |
| 			p.setOpacity(1);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::paintEvent(QPaintEvent *e) {
 | |
| 	auto p = QPainter(this);
 | |
| 
 | |
| 	const auto r = rect().intersected(e->rect());
 | |
| 	const auto errorDegree = _a_error.value(_error ? 1. : 0.);
 | |
| 	const auto focusedDegree = _a_focused.value(_focused ? 1. : 0.);
 | |
| 	paintSurrounding(p, r, errorDegree, focusedDegree);
 | |
| 
 | |
| 	const auto skip = int(base::SafeRound(_inner->document()->documentMargin()));
 | |
| 	const auto margins = _st.textMargins
 | |
| 		+ _st.placeholderMargins
 | |
| 		+ QMargins(skip, skip + _placeholderCustomFontSkip, skip, 0)
 | |
| 		+ _additionalMargins
 | |
| 		+ _customFontMargins;
 | |
| 
 | |
| 	if (_st.placeholderScale > 0. && !_placeholderPath.isEmpty()) {
 | |
| 		auto placeholderShiftDegree = _a_placeholderShifted.value(_placeholderShifted ? 1. : 0.);
 | |
| 		p.save();
 | |
| 		p.setClipRect(r);
 | |
| 
 | |
| 		auto placeholderTop = anim::interpolate(0, _st.placeholderShift, placeholderShiftDegree);
 | |
| 
 | |
| 		QRect r(rect().marginsRemoved(margins));
 | |
| 		r.moveTop(r.top() + placeholderTop);
 | |
| 		if (style::RightToLeft()) r.moveLeft(width() - r.left() - r.width());
 | |
| 
 | |
| 		auto placeholderScale = 1. - (1. - _st.placeholderScale) * placeholderShiftDegree;
 | |
| 		auto placeholderFg = anim::color(_st.placeholderFg, _st.placeholderFgActive, focusedDegree);
 | |
| 		placeholderFg = anim::color(placeholderFg, _st.placeholderFgError, errorDegree);
 | |
| 
 | |
| 		PainterHighQualityEnabler hq(p);
 | |
| 		p.setPen(Qt::NoPen);
 | |
| 		p.setBrush(placeholderFg);
 | |
| 		p.translate(r.topLeft());
 | |
| 		p.scale(placeholderScale, placeholderScale);
 | |
| 		p.drawPath(_placeholderPath);
 | |
| 
 | |
| 		p.restore();
 | |
| 	} else if (!_placeholder.isEmpty()) {
 | |
| 		const auto placeholderHiddenDegree = _a_placeholderShifted.value(_placeholderShifted ? 1. : 0.);
 | |
| 		if (placeholderHiddenDegree < 1.) {
 | |
| 			p.setOpacity(1. - placeholderHiddenDegree);
 | |
| 			p.save();
 | |
| 			p.setClipRect(r);
 | |
| 
 | |
| 			const auto placeholderLeft = anim::interpolate(0, -_st.placeholderShift, placeholderHiddenDegree);
 | |
| 
 | |
| 			p.setFont(_st.placeholderFont);
 | |
| 			p.setPen(anim::pen(_st.placeholderFg, _st.placeholderFgActive, focusedDegree));
 | |
| 			if (_st.placeholderAlign == style::al_topleft && _placeholderAfterSymbols > 0) {
 | |
| 				const auto skipWidth = placeholderSkipWidth();
 | |
| 				p.drawText(
 | |
| 					margins.left() + skipWidth,
 | |
| 					margins.top() + _st.placeholderFont->ascent,
 | |
| 					_placeholder);
 | |
| 			} else {
 | |
| 				auto r = rect().marginsRemoved(margins);
 | |
| 				r.moveLeft(r.left() + placeholderLeft);
 | |
| 				if (style::RightToLeft()) r.moveLeft(width() - r.left() - r.width());
 | |
| 				p.drawText(r, _placeholder, _st.placeholderAlign);
 | |
| 			}
 | |
| 
 | |
| 			p.restore();
 | |
| 		}
 | |
| 	}
 | |
| 	RpWidget::paintEvent(e);
 | |
| }
 | |
| 
 | |
| int InputField::placeholderSkipWidth() const {
 | |
| 	if (!_placeholderAfterSymbols) {
 | |
| 		return 0;
 | |
| 	}
 | |
| 	const auto &text = getTextWithTags().text;
 | |
| 	auto result = _st.font->width(text.mid(0, _placeholderAfterSymbols));
 | |
| 	if (_placeholderAfterSymbols > text.size()) {
 | |
| 		result += _st.font->spacew;
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| void InputField::startBorderAnimation() {
 | |
| 	auto borderVisible = (_error || _focused);
 | |
| 	if (_borderVisible != borderVisible) {
 | |
| 		_borderVisible = borderVisible;
 | |
| 		if (_borderVisible) {
 | |
| 			if (_a_borderOpacity.animating()) {
 | |
| 				_a_borderOpacity.start([this] { update(); }, 0., 1., _st.duration);
 | |
| 			} else {
 | |
| 				_a_borderShown.start([this] { update(); }, 0., 1., _st.duration);
 | |
| 			}
 | |
| 		} else {
 | |
| 			_a_borderOpacity.start([this] { update(); }, 1., 0., _st.duration);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::focusInEvent(QFocusEvent *e) {
 | |
| 	_borderAnimationStart = (e->reason() == Qt::MouseFocusReason)
 | |
| 		? mapFromGlobal(QCursor::pos()).x()
 | |
| 		: (width() / 2);
 | |
| 	InvokeQueued(this, [=] {
 | |
| 		if (QWidget::hasFocus()) {
 | |
| 			focusInner();
 | |
| 		}
 | |
| 	});
 | |
| }
 | |
| 
 | |
| void InputField::mousePressEvent(QMouseEvent *e) {
 | |
| 	_borderAnimationStart = e->pos().x();
 | |
| 	InvokeQueued(this, [=] { focusInner(); });
 | |
| }
 | |
| 
 | |
| void InputField::mousePressEventInner(QMouseEvent *e) {
 | |
| 	if (_touchPress && e->button() == Qt::LeftButton) {
 | |
| 		_mousePressedInTouch = true;
 | |
| 		_touchStart = e->globalPos();
 | |
| 	}
 | |
| 	_inner->QTextEdit::mousePressEvent(e);
 | |
| }
 | |
| 
 | |
| void InputField::mouseReleaseEventInner(QMouseEvent *e) {
 | |
| 	if (_mousePressedInTouch) {
 | |
| 		touchFinish();
 | |
| 	} else {
 | |
| 		_inner->QTextEdit::mouseReleaseEvent(e);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::mouseMoveEventInner(QMouseEvent *e) {
 | |
| 	if (_mousePressedInTouch) {
 | |
| 		touchUpdate(e->globalPos());
 | |
| 	}
 | |
| 	_inner->QTextEdit::mouseMoveEvent(e);
 | |
| }
 | |
| 
 | |
| void InputField::focusInner() {
 | |
| 	auto borderStart = _borderAnimationStart;
 | |
| 	_inner->setFocus();
 | |
| 	_borderAnimationStart = borderStart;
 | |
| }
 | |
| 
 | |
| int InputField::borderAnimationStart() const {
 | |
| 	return _borderAnimationStart;
 | |
| }
 | |
| 
 | |
| void InputField::contextMenuEvent(QContextMenuEvent *e) {
 | |
| 	_inner->contextMenuEvent(e);
 | |
| }
 | |
| 
 | |
| void InputField::focusInEventInner(QFocusEvent *e) {
 | |
| 	_borderAnimationStart = (e->reason() == Qt::MouseFocusReason)
 | |
| 		? mapFromGlobal(QCursor::pos()).x()
 | |
| 		: (width() / 2);
 | |
| 	setFocused(true);
 | |
| 	_inner->QTextEdit::focusInEvent(e);
 | |
| 	_focusedChanges.fire(true);
 | |
| }
 | |
| 
 | |
| void InputField::focusOutEventInner(QFocusEvent *e) {
 | |
| 	setFocused(false);
 | |
| 	_inner->QTextEdit::focusOutEvent(e);
 | |
| 	_focusedChanges.fire(false);
 | |
| }
 | |
| 
 | |
| void InputField::setFocused(bool focused) {
 | |
| 	if (_focused != focused) {
 | |
| 		_focused = focused;
 | |
| 		_a_focused.start([this] { update(); }, _focused ? 0. : 1., _focused ? 1. : 0., _st.duration);
 | |
| 		startPlaceholderAnimation();
 | |
| 		startBorderAnimation();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| QSize InputField::sizeHint() const {
 | |
| 	return geometry().size();
 | |
| }
 | |
| 
 | |
| QSize InputField::minimumSizeHint() const {
 | |
| 	return geometry().size();
 | |
| }
 | |
| 
 | |
| bool InputField::hasText() const {
 | |
| 	const auto document = _inner->document();
 | |
| 	const auto from = document->begin();
 | |
| 	const auto till = document->end();
 | |
| 
 | |
| 	if (from == till) {
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	for (auto item = from.begin(); !item.atEnd(); ++item) {
 | |
| 		const auto fragment = item.fragment();
 | |
| 		if (!fragment.isValid()) {
 | |
| 			continue;
 | |
| 		} else if (!fragment.text().isEmpty()) {
 | |
| 			return true;
 | |
| 		}
 | |
| 	}
 | |
| 	return (from.next() != till);
 | |
| }
 | |
| 
 | |
| InputField::TextPart InputField::getTextPart(
 | |
| 		int start,
 | |
| 		int end,
 | |
| 		TagList &outTagsList,
 | |
| 		bool &outTagsChanged,
 | |
| 		std::vector<MarkdownTag> *outMarkdownTags) const {
 | |
| 	Expects((start == 0 && end < 0) || outMarkdownTags == nullptr);
 | |
| 
 | |
| 	if (end >= 0 && end <= start) {
 | |
| 		outTagsChanged = !outTagsList.isEmpty();
 | |
| 		outTagsList.clear();
 | |
| 		return { QString(), 0 };
 | |
| 	}
 | |
| 
 | |
| 	if (start < 0) {
 | |
| 		start = 0;
 | |
| 	}
 | |
| 	const auto full = (start == 0 && end < 0);
 | |
| 
 | |
| 	auto lastTag = QString();
 | |
| 	TagAccumulator tagAccumulator(outTagsList);
 | |
| 	MarkdownTagAccumulator markdownTagAccumulator(outMarkdownTags);
 | |
| 	const auto newline = outMarkdownTags ? QString(1, '\n') : QString();
 | |
| 
 | |
| 	const auto document = _inner->document();
 | |
| 	const auto from = full ? document->begin() : document->findBlock(start);
 | |
| 	auto till = (end < 0) ? document->end() : document->findBlock(end);
 | |
| 	if (till.isValid()) {
 | |
| 		till = till.next();
 | |
| 	}
 | |
| 
 | |
| 	auto possibleLength = 0;
 | |
| 	for (auto block = from; block != till; block = block.next()) {
 | |
| 		possibleLength += block.length();
 | |
| 	}
 | |
| 	auto result = QString();
 | |
| 	result.reserve(possibleLength);
 | |
| 	if (!full && end < 0) {
 | |
| 		end = possibleLength;
 | |
| 	}
 | |
| 	auto textSizeWithoutSurrogatePairsCount = document->characterCount() - 1;
 | |
| 
 | |
| 	for (auto block = from; block != till;) {
 | |
| 		for (auto item = block.begin(); !item.atEnd(); ++item) {
 | |
| 			const auto fragment = item.fragment();
 | |
| 			if (!fragment.isValid()) {
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			const auto fragmentPosition = full ? 0 : fragment.position();
 | |
| 			const auto fragmentEnd = full
 | |
| 				? 0
 | |
| 				: (fragmentPosition + fragment.length());
 | |
| 			const auto format = fragment.charFormat();
 | |
| 			if (!full) {
 | |
| 				if (fragmentPosition == end) {
 | |
| 					tagAccumulator.feed(
 | |
| 						format.property(kTagProperty).toString(),
 | |
| 						result.size());
 | |
| 					break;
 | |
| 				} else if (fragmentPosition > end) {
 | |
| 					break;
 | |
| 				} else if (fragmentEnd <= start) {
 | |
| 					continue;
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			struct EmojiEntry final {
 | |
| 				QString text;
 | |
| 				uint8 surrogatePairs = 0;
 | |
| 			};
 | |
| 
 | |
| 			const auto emojiEntry = [&]() -> EmojiEntry {
 | |
| 				if (format.isImageFormat()) {
 | |
| 					const auto imageName = format.toImageFormat().name();
 | |
| 					if (const auto emoji = Emoji::FromUrl(imageName)) {
 | |
| 						return { emoji->text(), emoji->surrogatePairs() };
 | |
| 					}
 | |
| 				}
 | |
| 				return { format.property(kCustomEmojiText).toString(), 0 };
 | |
| 			}();
 | |
| 			auto text = [&] {
 | |
| 				const auto result = fragment.text();
 | |
| 				if (!full) {
 | |
| 					if (fragmentPosition < start) {
 | |
| 						return result.mid(start - fragmentPosition, end - start);
 | |
| 					} else if (fragmentEnd > end) {
 | |
| 						return result.mid(0, end - fragmentPosition);
 | |
| 					}
 | |
| 				}
 | |
| 				return result;
 | |
| 			}();
 | |
| 
 | |
| 			if (full || !text.isEmpty()) {
 | |
| 				lastTag = format.property(kTagProperty).toString();
 | |
| 				tagAccumulator.feed(lastTag, result.size());
 | |
| 			}
 | |
| 
 | |
| 			auto begin = text.data();
 | |
| 			auto ch = begin;
 | |
| 			auto adjustedLength = text.size();
 | |
| 			for (const auto end = begin + text.size(); ch != end; ++ch) {
 | |
| 				if (IsNewline(*ch) && ch->unicode() != '\r') {
 | |
| 					*ch = QLatin1Char('\n');
 | |
| 				} else switch (ch->unicode()) {
 | |
| 				case QChar::ObjectReplacementCharacter: {
 | |
| 					if (ch > begin) {
 | |
| 						result.append(begin, ch - begin);
 | |
| 					}
 | |
| 					const auto size = emojiEntry.text.size() - 1;
 | |
| 					adjustedLength += size;
 | |
| 					if (!emojiEntry.text.isEmpty()) {
 | |
| 						result.append(emojiEntry.text);
 | |
| 						textSizeWithoutSurrogatePairsCount += size
 | |
| 							- emojiEntry.surrogatePairs;
 | |
| 					}
 | |
| 					begin = ch + 1;
 | |
| 				} break;
 | |
| 				}
 | |
| 			}
 | |
| 			if (ch > begin) {
 | |
| 				result.append(begin, ch - begin);
 | |
| 			}
 | |
| 
 | |
| 			if (full || !text.isEmpty()) {
 | |
| 				markdownTagAccumulator.feed(text, adjustedLength, lastTag);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		block = block.next();
 | |
| 		if (block != till) {
 | |
| 			tagAccumulator.feed(
 | |
| 				TagWithoutCustomEmoji(
 | |
| 					block.charFormat().property(kTagProperty).toString()),
 | |
| 				result.size());
 | |
| 			result.append('\n');
 | |
| 			markdownTagAccumulator.feed(newline, 1, lastTag);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	tagAccumulator.feed(QString(), result.size());
 | |
| 	tagAccumulator.finish();
 | |
| 	markdownTagAccumulator.finish();
 | |
| 
 | |
| 	outTagsChanged = tagAccumulator.changed();
 | |
| 	return { result, textSizeWithoutSurrogatePairsCount };
 | |
| }
 | |
| 
 | |
| bool InputField::isUndoAvailable() const {
 | |
| 	return _undoAvailable;
 | |
| }
 | |
| 
 | |
| bool InputField::isRedoAvailable() const {
 | |
| 	return _redoAvailable;
 | |
| }
 | |
| 
 | |
| void InputField::processFormatting(int insertPosition, int insertEnd) {
 | |
| 	// Tilde formatting.
 | |
| 	const auto tildeFormatting = (_st.font->f.pixelSize() * style::DevicePixelRatio() == 13)
 | |
| 		&& (_st.font->f.family() == qstr("Open Sans"));
 | |
| 	auto isTildeFragment = false;
 | |
| 	auto tildeFixedFont = _st.font->semibold()->f;
 | |
| 
 | |
| 	// First tag handling (the one we inserted text to).
 | |
| 	bool startTagFound = false;
 | |
| 	bool breakTagOnNotLetter = false;
 | |
| 
 | |
| 	auto document = _inner->document();
 | |
| 
 | |
| 	// Apply inserted tags.
 | |
| 	const auto insertedTagsProcessor = _insertedTagsAreFromMime
 | |
| 		? (_tagMimeProcessor ? _tagMimeProcessor : DefaultTagMimeProcessor)
 | |
| 		: nullptr;
 | |
| 	const auto breakTagOnNotLetterTill = ProcessInsertedTags(
 | |
| 		_st,
 | |
| 		document,
 | |
| 		insertPosition,
 | |
| 		insertEnd,
 | |
| 		_insertedTags,
 | |
| 		insertedTagsProcessor);
 | |
| 	using ActionType = FormattingAction::Type;
 | |
| 	while (true) {
 | |
| 		FormattingAction action;
 | |
| 
 | |
| 		auto checkedTill = insertPosition;
 | |
| 		auto fromBlock = document->findBlock(insertPosition);
 | |
| 		auto tillBlock = document->findBlock(insertEnd);
 | |
| 		if (tillBlock.isValid()) tillBlock = tillBlock.next();
 | |
| 
 | |
| 		for (auto block = fromBlock; block != tillBlock; block = block.next()) {
 | |
| 			if (block.blockFormat().lineHeightType() != QTextBlockFormat::FixedHeight) {
 | |
| 				action.type = ActionType::FixLineHeight;
 | |
| 				break;
 | |
| 			}
 | |
| 			for (auto fragmentIt = block.begin(); !fragmentIt.atEnd(); ++fragmentIt) {
 | |
| 				auto fragment = fragmentIt.fragment();
 | |
| 				Assert(fragment.isValid());
 | |
| 
 | |
| 				const auto fragmentPosition = fragment.position();
 | |
| 				const auto fragmentEnd = fragmentPosition + fragment.length();
 | |
| 				if (insertPosition > fragmentEnd) {
 | |
| 					// In case insertPosition == fragmentEnd we still
 | |
| 					// need to fill startTagFound / breakTagOnNotLetter.
 | |
| 					// This can happen if we inserted a newline after
 | |
| 					// a text fragment with some formatting tag, like Bold.
 | |
| 					continue;
 | |
| 				}
 | |
| 				int changedPositionInFragment = insertPosition - fragmentPosition; // Can be negative.
 | |
| 				int changedEndInFragment = insertEnd - fragmentPosition;
 | |
| 				if (changedEndInFragment <= 0) {
 | |
| 					break;
 | |
| 				}
 | |
| 
 | |
| 				auto format = fragment.charFormat();
 | |
| 				if (!format.hasProperty(kTagProperty)) {
 | |
| 					action.type = ActionType::RemoveTag;
 | |
| 					action.intervalStart = fragmentPosition;
 | |
| 					action.intervalEnd = fragmentPosition + fragment.length();
 | |
| 					break;
 | |
| 				}
 | |
| 				if (tildeFormatting) {
 | |
| 					const auto formatFont = format.font();
 | |
| 					if (!tildeFixedFont.styleName().isEmpty()
 | |
| 						&& formatFont.styleName().isEmpty()) {
 | |
| 						tildeFixedFont.setStyleName(QString());
 | |
| 					}
 | |
| 					isTildeFragment = (format.font() == tildeFixedFont);
 | |
| 				}
 | |
| 
 | |
| 				auto fragmentText = fragment.text();
 | |
| 				auto *textStart = fragmentText.constData();
 | |
| 				auto *textEnd = textStart + fragmentText.size();
 | |
| 
 | |
| 				if (_customEmojiObject
 | |
| 					&& format.objectType() == kCustomEmojiFormat) {
 | |
| 					if (fragmentText == kObjectReplacement) {
 | |
| 						checkedTill = fragmentEnd;
 | |
| 						continue;
 | |
| 					}
 | |
| 					action.type = ActionType::InsertCustomEmoji;
 | |
| 					action.intervalStart = fragmentPosition;
 | |
| 					action.intervalEnd = fragmentPosition
 | |
| 						+ fragmentText.size();
 | |
| 					action.customEmojiText = fragmentText;
 | |
| 					action.customEmojiLink = format.property(
 | |
| 						kCustomEmojiLink).toString();
 | |
| 					break;
 | |
| 				}
 | |
| 
 | |
| 				const auto with = format.property(kInstantReplaceWithId);
 | |
| 				if (with.isValid()) {
 | |
| 					const auto string = with.toString();
 | |
| 					if (fragmentText != string) {
 | |
| 						action.type = ActionType::ClearInstantReplace;
 | |
| 						action.intervalStart = fragmentPosition
 | |
| 							+ (fragmentText.startsWith(string)
 | |
| 								? string.size()
 | |
| 								: 0);
 | |
| 						action.intervalEnd = fragmentPosition
 | |
| 							+ fragmentText.size();
 | |
| 						break;
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				if (format.hasProperty(kCustomEmojiLink)
 | |
| 					&& !format.property(kCustomEmojiLink).toString().isEmpty()) {
 | |
| 					action.type = ActionType::RemoveCustomEmoji;
 | |
| 					action.existingTags = format.property(kTagProperty).toString();
 | |
| 					action.intervalStart = fragmentPosition;
 | |
| 					action.intervalEnd = fragmentPosition
 | |
| 						+ fragmentText.size();
 | |
| 					break;
 | |
| 				}
 | |
| 				if (!startTagFound) {
 | |
| 					startTagFound = true;
 | |
| 					auto tagName = format.property(kTagProperty).toString();
 | |
| 					if (!tagName.isEmpty()) {
 | |
| 						breakTagOnNotLetter = WasInsertTillTheEndOfTag(
 | |
| 							block,
 | |
| 							fragmentIt,
 | |
| 							insertEnd);
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				auto *ch = textStart + qMax(changedPositionInFragment, 0);
 | |
| 				for (; ch < textEnd; ++ch) {
 | |
| 					const auto removeNewline = (_mode != Mode::MultiLine)
 | |
| 						&& IsNewline(*ch);
 | |
| 					if (removeNewline) {
 | |
| 						if (action.type == ActionType::Invalid) {
 | |
| 							action.type = ActionType::RemoveNewline;
 | |
| 							action.intervalStart = fragmentPosition + (ch - textStart);
 | |
| 							action.intervalEnd = action.intervalStart + 1;
 | |
| 						}
 | |
| 						break;
 | |
| 					}
 | |
| 
 | |
| 					auto emojiLength = 0;
 | |
| 					if (const auto emoji = Emoji::Find(ch, textEnd, &emojiLength)) {
 | |
| 						// Replace emoji if no current action is prepared.
 | |
| 						if (action.type == ActionType::Invalid) {
 | |
| 							action.type = ActionType::InsertEmoji;
 | |
| 							action.emoji = emoji;
 | |
| 							action.intervalStart = fragmentPosition + (ch - textStart);
 | |
| 							action.intervalEnd = action.intervalStart + emojiLength;
 | |
| 						}
 | |
| 						if (emojiLength > 1) {
 | |
| 							_emojiSurrogateAmount += emojiLength - 1;
 | |
| 						}
 | |
| 						break;
 | |
| 					}
 | |
| 
 | |
| 					if (breakTagOnNotLetter && !ch->isLetterOrNumber()) {
 | |
| 						// Remove tag name till the end if no current action is prepared.
 | |
| 						if (action.type != ActionType::Invalid) {
 | |
| 							break;
 | |
| 						}
 | |
| 						breakTagOnNotLetter = false;
 | |
| 						if (fragmentPosition + (ch - textStart) < breakTagOnNotLetterTill) {
 | |
| 							action.type = ActionType::RemoveTag;
 | |
| 							action.intervalStart = fragmentPosition + (ch - textStart);
 | |
| 							action.intervalEnd = breakTagOnNotLetterTill;
 | |
| 							break;
 | |
| 						}
 | |
| 					}
 | |
| 					if (tildeFormatting) { // Tilde symbol fix in OpenSans.
 | |
| 						bool tilde = (ch->unicode() == '~');
 | |
| 						if ((tilde && !isTildeFragment) || (!tilde && isTildeFragment)) {
 | |
| 							if (action.type == ActionType::Invalid) {
 | |
| 								action.type = ActionType::TildeFont;
 | |
| 								action.intervalStart = fragmentPosition + (ch - textStart);
 | |
| 								action.intervalEnd = action.intervalStart + 1;
 | |
| 								action.tildeTag = format.property(kTagProperty).toString();
 | |
| 								action.isTilde = tilde;
 | |
| 							} else {
 | |
| 								++action.intervalEnd;
 | |
| 							}
 | |
| 						} else if (action.type == ActionType::TildeFont) {
 | |
| 							break;
 | |
| 						}
 | |
| 					}
 | |
| 
 | |
| 					if (ch + 1 < textEnd && ch->isHighSurrogate() && (ch + 1)->isLowSurrogate()) {
 | |
| 						++ch;
 | |
| 					}
 | |
| 				}
 | |
| 				if (action.type != ActionType::Invalid) {
 | |
| 					break;
 | |
| 				}
 | |
| 				checkedTill = fragmentEnd;
 | |
| 			}
 | |
| 			if (action.type != ActionType::Invalid) {
 | |
| 				break;
 | |
| 			} else if (_mode != Mode::MultiLine
 | |
| 				&& block.next() != document->end()) {
 | |
| 				action.type = ActionType::RemoveNewline;
 | |
| 				action.intervalStart = block.next().position() - 1;
 | |
| 				action.intervalEnd = action.intervalStart + 1;
 | |
| 				break;
 | |
| 			} else if (breakTagOnNotLetter) {
 | |
| 				// In case we need to break on not letter and we didn't
 | |
| 				// find any non letter symbol, we found it here - a newline.
 | |
| 				breakTagOnNotLetter = false;
 | |
| 				if (checkedTill < breakTagOnNotLetterTill) {
 | |
| 					action.type = ActionType::RemoveTag;
 | |
| 					action.intervalStart = checkedTill;
 | |
| 					action.intervalEnd = breakTagOnNotLetterTill;
 | |
| 					break;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		if (action.type != ActionType::Invalid) {
 | |
| 			PrepareFormattingOptimization(document);
 | |
| 
 | |
| 			auto cursor = QTextCursor(document);
 | |
| 			if (action.type == ActionType::FixLineHeight) {
 | |
| 				cursor.select(QTextCursor::Document);
 | |
| 				cursor.setBlockFormat(_defaultBlockFormat);
 | |
| 				continue;
 | |
| 			}
 | |
| 			cursor.setPosition(action.intervalStart);
 | |
| 			cursor.setPosition(action.intervalEnd, QTextCursor::KeepAnchor);
 | |
| 			if (action.type == ActionType::InsertEmoji
 | |
| 				|| action.type == ActionType::InsertCustomEmoji) {
 | |
| 				if (action.type == ActionType::InsertEmoji) {
 | |
| 					InsertEmojiAtCursor(cursor, action.emoji);
 | |
| 				} else {
 | |
| 					InsertCustomEmojiAtCursor(
 | |
| 						this,
 | |
| 						cursor,
 | |
| 						action.customEmojiText,
 | |
| 						action.customEmojiLink);
 | |
| 				}
 | |
| 				insertPosition = action.intervalStart + 1;
 | |
| 				if (insertEnd >= action.intervalEnd) {
 | |
| 					insertEnd -= action.intervalEnd
 | |
| 						- action.intervalStart
 | |
| 						- 1;
 | |
| 				}
 | |
| 			} else if (action.type == ActionType::RemoveTag) {
 | |
| 				RemoveDocumentTags(
 | |
| 					_st,
 | |
| 					document,
 | |
| 					action.intervalStart,
 | |
| 					action.intervalEnd);
 | |
| 			} else if (action.type == ActionType::RemoveCustomEmoji) {
 | |
| 				RemoveCustomEmojiTag(
 | |
| 					_st,
 | |
| 					document,
 | |
| 					action.existingTags,
 | |
| 					action.intervalStart,
 | |
| 					action.intervalEnd);
 | |
| 			} else if (action.type == ActionType::TildeFont) {
 | |
| 				auto format = QTextCharFormat();
 | |
| 				format.setFont(action.isTilde
 | |
| 					? tildeFixedFont
 | |
| 					: PrepareTagFormat(_st, action.tildeTag).font());
 | |
| 				cursor.mergeCharFormat(format);
 | |
| 				insertPosition = action.intervalEnd;
 | |
| 			} else if (action.type == ActionType::ClearInstantReplace) {
 | |
| 				auto format = _defaultCharFormat;
 | |
| 				ApplyTagFormat(format, cursor.charFormat());
 | |
| 				cursor.setCharFormat(format);
 | |
| 			} else if (action.type == ActionType::RemoveNewline) {
 | |
| 				cursor.removeSelectedText();
 | |
| 				insertPosition = action.intervalStart;
 | |
| 				if (insertEnd >= action.intervalEnd) {
 | |
| 					insertEnd -= action.intervalEnd - action.intervalStart;
 | |
| 				}
 | |
| 			}
 | |
| 		} else {
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::forceProcessContentsChanges() {
 | |
| 	PostponeCall(this, [=] {
 | |
| 		handleContentsChanged();
 | |
| 	});
 | |
| }
 | |
| 
 | |
| void InputField::documentContentsChanged(
 | |
| 		int position,
 | |
| 		int charsRemoved,
 | |
| 		int charsAdded) {
 | |
| 	if (_correcting) {
 | |
| 		return;
 | |
| 	}
 | |
| 	// In case of input method events Qt emits
 | |
| 	// document content change signals for a whole
 | |
| 	// text block where the even took place.
 | |
| 	// This breaks our wysiwyg markup, so we adjust
 | |
| 	// the parameters to match the real change.
 | |
| 	if (_inputMethodCommit.has_value()
 | |
| 		&& charsAdded > _inputMethodCommit->size()
 | |
| 		&& charsRemoved > 0) {
 | |
| 		const auto inBlockBefore = charsAdded - _inputMethodCommit->size();
 | |
| 		if (charsRemoved >= inBlockBefore) {
 | |
| 			charsAdded -= inBlockBefore;
 | |
| 			charsRemoved -= inBlockBefore;
 | |
| 			position += inBlockBefore;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	const auto document = _inner->document();
 | |
| 
 | |
| 	// Qt bug workaround https://bugreports.qt.io/browse/QTBUG-49062
 | |
| 	if (!position) {
 | |
| 		auto cursor = QTextCursor(document);
 | |
| 		cursor.movePosition(QTextCursor::End);
 | |
| 		if (position + charsAdded > cursor.position()) {
 | |
| 			const auto delta = position + charsAdded - cursor.position();
 | |
| 			if (charsRemoved >= delta) {
 | |
| 				charsAdded -= delta;
 | |
| 				charsRemoved -= delta;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	const auto insertPosition = (_realInsertPosition >= 0)
 | |
| 		? _realInsertPosition
 | |
| 		: position;
 | |
| 	const auto insertLength = (_realInsertPosition >= 0)
 | |
| 		? _realCharsAdded
 | |
| 		: charsAdded;
 | |
| 
 | |
| 	_correcting = true;
 | |
| 	QTextCursor(document).joinPreviousEditBlock();
 | |
| 	const auto guard = gsl::finally([&] {
 | |
| 		_correcting = false;
 | |
| 		QTextCursor(document).endEditBlock();
 | |
| 		handleContentsChanged();
 | |
| 		const auto added = charsAdded - _emojiSurrogateAmount;
 | |
| 		_documentContentsChanges.fire({ position, charsRemoved, added });
 | |
| 		_emojiSurrogateAmount = 0;
 | |
| 	});
 | |
| 
 | |
| 	chopByMaxLength(insertPosition, insertLength);
 | |
| 
 | |
| 	if (document->availableRedoSteps() == 0 && insertLength > 0) {
 | |
| 		const auto pageSize = document->pageSize();
 | |
| 		processFormatting(insertPosition, insertPosition + insertLength);
 | |
| 		if (document->pageSize() != pageSize) {
 | |
| 			document->setPageSize(pageSize);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::chopByMaxLength(int insertPosition, int insertLength) {
 | |
| 	Expects(_correcting);
 | |
| 
 | |
| 	if (_maxLength < 0) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	auto cursor = QTextCursor(document());
 | |
| 	cursor.movePosition(QTextCursor::End);
 | |
| 	const auto fullSize = cursor.position();
 | |
| 	const auto toRemove = fullSize - _maxLength;
 | |
| 	if (toRemove > 0) {
 | |
| 		if (toRemove > insertLength) {
 | |
| 			if (insertLength) {
 | |
| 				cursor.setPosition(insertPosition);
 | |
| 				cursor.setPosition(
 | |
| 					(insertPosition + insertLength),
 | |
| 					QTextCursor::KeepAnchor);
 | |
| 				cursor.removeSelectedText();
 | |
| 			}
 | |
| 			cursor.setPosition(fullSize - (toRemove - insertLength));
 | |
| 			cursor.setPosition(fullSize, QTextCursor::KeepAnchor);
 | |
| 			cursor.removeSelectedText();
 | |
| 		} else {
 | |
| 			cursor.setPosition(
 | |
| 				insertPosition + (insertLength - toRemove));
 | |
| 			cursor.setPosition(
 | |
| 				insertPosition + insertLength,
 | |
| 				QTextCursor::KeepAnchor);
 | |
| 			cursor.removeSelectedText();
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::handleContentsChanged() {
 | |
| 	setErrorShown(false);
 | |
| 
 | |
| 	auto tagsChanged = false;
 | |
| 	const auto [currentText, currentTextSize] = getTextPart(
 | |
| 		0,
 | |
| 		-1,
 | |
| 		_lastTextWithTags.tags,
 | |
| 		tagsChanged,
 | |
| 		_markdownEnabled ? &_lastMarkdownTags : nullptr);
 | |
| 
 | |
| 	//highlightMarkdown();
 | |
| 
 | |
| 	_lastTextSizeWithoutSurrogatePairsCount = currentTextSize;
 | |
| 	if (tagsChanged || (_lastTextWithTags.text != currentText)) {
 | |
| 		_lastTextWithTags.text = currentText;
 | |
| 		const auto weak = MakeWeak(this);
 | |
| 		_changes.fire({});
 | |
| 		if (!weak) {
 | |
| 			return;
 | |
| 		}
 | |
| 		checkContentHeight();
 | |
| 	}
 | |
| 	startPlaceholderAnimation();
 | |
| 	if (_lastTextWithTags.text.isEmpty()) {
 | |
| 		if (const auto object = _customEmojiObject.get()) {
 | |
| 			object->clear();
 | |
| 		}
 | |
| 	}
 | |
| 	Integration::Instance().textActionsUpdated();
 | |
| }
 | |
| 
 | |
| void InputField::highlightMarkdown() {
 | |
| 	// Highlighting may interfere with markdown parsing -> inaccurate.
 | |
| 	// For debug.
 | |
| 	auto from = 0;
 | |
| 	auto applyColor = [&](int a, int b, QColor color) {
 | |
| 		auto cursor = textCursor();
 | |
| 		cursor.setPosition(a);
 | |
| 		cursor.setPosition(b, QTextCursor::KeepAnchor);
 | |
| 		auto format = QTextCharFormat();
 | |
| 		format.setForeground(color);
 | |
| 		cursor.mergeCharFormat(format);
 | |
| 		from = b;
 | |
| 	};
 | |
| 	for (const auto &tag : _lastMarkdownTags) {
 | |
| 		if (tag.internalStart > from) {
 | |
| 			applyColor(from, tag.internalStart, QColor(0, 0, 0));
 | |
| 		} else if (tag.internalStart < from) {
 | |
| 			continue;
 | |
| 		}
 | |
| 		applyColor(
 | |
| 			tag.internalStart,
 | |
| 			tag.internalStart + tag.internalLength,
 | |
| 			(tag.closed
 | |
| 				? QColor(0, 128, 0)
 | |
| 				: QColor(128, 0, 0)));
 | |
| 	}
 | |
| 	auto cursor = textCursor();
 | |
| 	cursor.movePosition(QTextCursor::End);
 | |
| 	if (const auto till = cursor.position(); till > from) {
 | |
| 		applyColor(from, till, QColor(0, 0, 0));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::setDisplayFocused(bool focused) {
 | |
| 	setFocused(focused);
 | |
| 	finishAnimating();
 | |
| }
 | |
| 
 | |
| void InputField::selectAll() {
 | |
| 	auto cursor = _inner->textCursor();
 | |
| 	cursor.setPosition(0);
 | |
| 	cursor.movePosition(QTextCursor::End, QTextCursor::KeepAnchor);
 | |
| 	_inner->setTextCursor(cursor);
 | |
| }
 | |
| 
 | |
| void InputField::finishAnimating() {
 | |
| 	_a_focused.stop();
 | |
| 	_a_error.stop();
 | |
| 	_a_placeholderShifted.stop();
 | |
| 	_a_borderShown.stop();
 | |
| 	_a_borderOpacity.stop();
 | |
| 	update();
 | |
| }
 | |
| 
 | |
| void InputField::setPlaceholderHidden(bool forcePlaceholderHidden) {
 | |
| 	_forcePlaceholderHidden = forcePlaceholderHidden;
 | |
| 	startPlaceholderAnimation();
 | |
| }
 | |
| 
 | |
| void InputField::startPlaceholderAnimation() {
 | |
| 	const auto textLength = [&] {
 | |
| 		return getTextWithTags().text.size() + _lastPreEditText.size();
 | |
| 	};
 | |
| 	const auto placeholderShifted = _forcePlaceholderHidden
 | |
| 		|| (_focused && _st.placeholderScale > 0.)
 | |
| 		|| (textLength() > _placeholderAfterSymbols);
 | |
| 	if (_placeholderShifted != placeholderShifted) {
 | |
| 		_placeholderShifted = placeholderShifted;
 | |
| 		_a_placeholderShifted.start(
 | |
| 			[=] { update(); },
 | |
| 			_placeholderShifted ? 0. : 1.,
 | |
| 			_placeholderShifted ? 1. : 0.,
 | |
| 			_st.duration);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| QMimeData *InputField::createMimeDataFromSelectionInner() const {
 | |
| 	const auto cursor = _inner->textCursor();
 | |
| 	const auto start = cursor.selectionStart();
 | |
| 	const auto end = cursor.selectionEnd();
 | |
| 	return TextUtilities::MimeDataFromText((end > start)
 | |
| 		? getTextWithTagsPart(start, end)
 | |
| 		: TextWithTags()
 | |
| 	).release();
 | |
| }
 | |
| 
 | |
| void InputField::customUpDown(bool isCustom) {
 | |
| 	_customUpDown = isCustom;
 | |
| }
 | |
| 
 | |
| void InputField::customTab(bool isCustom) {
 | |
| 	_customTab = isCustom;
 | |
| }
 | |
| 
 | |
| void InputField::setSubmitSettings(SubmitSettings settings) {
 | |
| 	_submitSettings = settings;
 | |
| }
 | |
| 
 | |
| not_null<QTextDocument*> InputField::document() {
 | |
| 	return _inner->document();
 | |
| }
 | |
| 
 | |
| not_null<const QTextDocument*> InputField::document() const {
 | |
| 	return _inner->document();
 | |
| }
 | |
| 
 | |
| void InputField::setTextCursor(const QTextCursor &cursor) {
 | |
| 	return _inner->setTextCursor(cursor);
 | |
| }
 | |
| 
 | |
| QTextCursor InputField::textCursor() const {
 | |
| 	return _inner->textCursor();
 | |
| }
 | |
| 
 | |
| void InputField::setCursorPosition(int pos) {
 | |
| 	auto cursor = _inner->textCursor();
 | |
| 	cursor.setPosition(pos);
 | |
| 	_inner->setTextCursor(cursor);
 | |
| }
 | |
| 
 | |
| void InputField::setText(const QString &text) {
 | |
| 	setTextWithTags({ text, {} });
 | |
| }
 | |
| 
 | |
| void InputField::setTextWithTags(
 | |
| 		const TextWithTags &textWithTags,
 | |
| 		HistoryAction historyAction) {
 | |
| 	_insertedTags = textWithTags.tags;
 | |
| 	_insertedTagsAreFromMime = false;
 | |
| 	_realInsertPosition = 0;
 | |
| 	_realCharsAdded = textWithTags.text.size();
 | |
| 	const auto document = _inner->document();
 | |
| 	auto cursor = QTextCursor(document);
 | |
| 	cursor.setBlockFormat(_defaultBlockFormat);
 | |
| 	if (historyAction == HistoryAction::Clear) {
 | |
| 		document->setUndoRedoEnabled(false);
 | |
| 		cursor.beginEditBlock();
 | |
| 	} else if (historyAction == HistoryAction::MergeEntry) {
 | |
| 		cursor.joinPreviousEditBlock();
 | |
| 	} else {
 | |
| 		cursor.beginEditBlock();
 | |
| 	}
 | |
| 	cursor.movePosition(QTextCursor::End, QTextCursor::KeepAnchor);
 | |
| 	cursor.insertText(textWithTags.text);
 | |
| 	cursor.movePosition(QTextCursor::End);
 | |
| 	cursor.endEditBlock();
 | |
| 	if (historyAction == HistoryAction::Clear) {
 | |
| 		document->setUndoRedoEnabled(true);
 | |
| 	}
 | |
| 	_insertedTags.clear();
 | |
| 	_realInsertPosition = -1;
 | |
| 	finishAnimating();
 | |
| }
 | |
| 
 | |
| TextWithTags InputField::getTextWithTagsPart(int start, int end) const {
 | |
| 	auto changed = false;
 | |
| 	auto result = TextWithTags();
 | |
| 	result.text = getTextPart(start, end, result.tags, changed).text;
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| TextWithTags InputField::getTextWithAppliedMarkdown() const {
 | |
| 	if (!_markdownEnabled || _lastMarkdownTags.empty()) {
 | |
| 		return getTextWithTags();
 | |
| 	}
 | |
| 	const auto &originalText = _lastTextWithTags.text;
 | |
| 	const auto &originalTags = _lastTextWithTags.tags;
 | |
| 
 | |
| 	// Ignore tags that partially intersect some http-links.
 | |
| 	// This will allow sending http://test.com/__test__/test correctly.
 | |
| 	const auto links = TextUtilities::ParseEntities(
 | |
| 		originalText,
 | |
| 		0).entities;
 | |
| 
 | |
| 	auto result = TextWithTags();
 | |
| 	result.text.reserve(originalText.size());
 | |
| 	result.tags.reserve(originalTags.size() + _lastMarkdownTags.size());
 | |
| 	auto removed = 0;
 | |
| 	auto originalTag = originalTags.begin();
 | |
| 	const auto originalTagsEnd = originalTags.end();
 | |
| 	const auto addOriginalTagsUpTill = [&](int offset) {
 | |
| 		while (originalTag != originalTagsEnd
 | |
| 			&& originalTag->offset + originalTag->length <= offset) {
 | |
| 			result.tags.push_back(*originalTag++);
 | |
| 			result.tags.back().offset -= removed;
 | |
| 		}
 | |
| 	};
 | |
| 	auto from = 0;
 | |
| 	const auto addOriginalTextUpTill = [&](int offset) {
 | |
| 		if (offset > from) {
 | |
| 			result.text.append(base::StringViewMid(originalText, from, offset - from));
 | |
| 		}
 | |
| 	};
 | |
| 	auto link = links.begin();
 | |
| 	const auto linksEnd = links.end();
 | |
| 	for (const auto &tag : _lastMarkdownTags) {
 | |
| 		const auto tagLength = int(tag.tag.size());
 | |
| 		if (!tag.closed || tag.adjustedStart < from) {
 | |
| 			continue;
 | |
| 		}
 | |
| 		auto entityLength = tag.adjustedLength - 2 * tagLength;
 | |
| 		if (entityLength <= 0) {
 | |
| 			continue;
 | |
| 		}
 | |
| 		addOriginalTagsUpTill(tag.adjustedStart);
 | |
| 		const auto tagAdjustedEnd = tag.adjustedStart + tag.adjustedLength;
 | |
| 		if (originalTag != originalTagsEnd
 | |
| 			&& originalTag->offset < tagAdjustedEnd) {
 | |
| 			continue;
 | |
| 		}
 | |
| 		while (link != linksEnd
 | |
| 			&& link->offset() + link->length() <= tag.adjustedStart) {
 | |
| 			++link;
 | |
| 		}
 | |
| 		if (link != linksEnd
 | |
| 			&& link->offset() < tagAdjustedEnd
 | |
| 			&& (link->offset() + link->length() > tagAdjustedEnd
 | |
| 				|| link->offset() < tag.adjustedStart)) {
 | |
| 			continue;
 | |
| 		}
 | |
| 		addOriginalTextUpTill(tag.adjustedStart);
 | |
| 
 | |
| 		auto tagId = tag.tag;
 | |
| 		auto entityStart = tag.adjustedStart + tagLength;
 | |
| 		if (tagId == kTagPre) {
 | |
| 			// Remove redundant newlines for pre.
 | |
| 			// If ``` is on a separate line add only one newline.
 | |
| 			const auto languageName = ReadPreLanguageName(
 | |
| 				originalText,
 | |
| 				entityStart,
 | |
| 				entityLength);
 | |
| 			if (!languageName.isEmpty()) {
 | |
| 				// ```language-name{\n}code
 | |
| 				entityStart += languageName.size() + 1;
 | |
| 				entityLength -= languageName.size() + 1;
 | |
| 				tagId += languageName;
 | |
| 			} else if (IsNewline(originalText[entityStart])
 | |
| 				&& (result.text.isEmpty()
 | |
| 					|| IsNewline(result.text[result.text.size() - 1]))) {
 | |
| 				++entityStart;
 | |
| 				--entityLength;
 | |
| 			}
 | |
| 			const auto entityEnd = entityStart + entityLength;
 | |
| 			if (IsNewline(originalText[entityEnd - 1])
 | |
| 				&& (originalText.size() <= entityEnd + tagLength
 | |
| 					|| IsNewline(originalText[entityEnd + tagLength]))) {
 | |
| 				--entityLength;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (entityLength > 0) {
 | |
| 			// Add tag text and entity.
 | |
| 			result.tags.push_back(TextWithTags::Tag{
 | |
| 				int(result.text.size()),
 | |
| 				entityLength,
 | |
| 				tagId });
 | |
| 			result.text.append(base::StringViewMid(
 | |
| 				originalText,
 | |
| 				entityStart,
 | |
| 				entityLength));
 | |
| 		}
 | |
| 
 | |
| 		from = tag.adjustedStart + tag.adjustedLength;
 | |
| 		removed += (tag.adjustedLength - entityLength);
 | |
| 	}
 | |
| 	addOriginalTagsUpTill(originalText.size());
 | |
| 	addOriginalTextUpTill(originalText.size());
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| void InputField::clear() {
 | |
| 	_inner->clear();
 | |
| 	startPlaceholderAnimation();
 | |
| 	if (const auto object = _customEmojiObject.get()) {
 | |
| 		object->clear();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool InputField::hasFocus() const {
 | |
| 	return _inner->hasFocus();
 | |
| }
 | |
| 
 | |
| void InputField::setFocus() {
 | |
| 	_inner->setFocus();
 | |
| }
 | |
| 
 | |
| void InputField::clearFocus() {
 | |
| 	_inner->clearFocus();
 | |
| }
 | |
| 
 | |
| void InputField::ensureCursorVisible() {
 | |
| 	_inner->ensureCursorVisible();
 | |
| }
 | |
| 
 | |
| not_null<QTextEdit*> InputField::rawTextEdit() {
 | |
| 	return _inner.get();
 | |
| }
 | |
| 
 | |
| not_null<const QTextEdit*> InputField::rawTextEdit() const {
 | |
| 	return _inner.get();
 | |
| }
 | |
| 
 | |
| bool InputField::ShouldSubmit(
 | |
| 		SubmitSettings settings,
 | |
| 		Qt::KeyboardModifiers modifiers) {
 | |
| 	const auto shift = modifiers.testFlag(Qt::ShiftModifier);
 | |
| 	const auto ctrl = modifiers.testFlag(Qt::ControlModifier)
 | |
| 		|| modifiers.testFlag(Qt::MetaModifier);
 | |
| 	return (ctrl && shift)
 | |
| 		|| (ctrl
 | |
| 			&& settings != SubmitSettings::None
 | |
| 			&& settings != SubmitSettings::Enter)
 | |
| 		|| (!ctrl
 | |
| 			&& !shift
 | |
| 			&& settings != SubmitSettings::None
 | |
| 			&& settings != SubmitSettings::CtrlEnter);
 | |
| }
 | |
| 
 | |
| void InputField::keyPressEventInner(QKeyEvent *e) {
 | |
| 	const auto shift = e->modifiers().testFlag(Qt::ShiftModifier);
 | |
| 	const auto alt = e->modifiers().testFlag(Qt::AltModifier);
 | |
| 	const auto macmeta = Platform::IsMac()
 | |
| 		&& e->modifiers().testFlag(Qt::ControlModifier)
 | |
| 		&& !e->modifiers().testFlag(Qt::MetaModifier)
 | |
| 		&& !e->modifiers().testFlag(Qt::AltModifier);
 | |
| 	const auto ctrl = e->modifiers().testFlag(Qt::ControlModifier)
 | |
| 		|| e->modifiers().testFlag(Qt::MetaModifier);
 | |
| 	const auto enterSubmit = (_mode != Mode::MultiLine)
 | |
| 		|| ShouldSubmit(_submitSettings, e->modifiers());
 | |
| 	const auto enter = (e->key() == Qt::Key_Enter || e->key() == Qt::Key_Return);
 | |
| 	const auto backspace = (e->key() == Qt::Key_Backspace);
 | |
| 	if (e->key() == Qt::Key_Left
 | |
| 		|| e->key() == Qt::Key_Right
 | |
| 		|| e->key() == Qt::Key_Up
 | |
| 		|| e->key() == Qt::Key_Down
 | |
| 		|| e->key() == Qt::Key_Home
 | |
| 		|| e->key() == Qt::Key_End) {
 | |
| 		_reverseMarkdownReplacement = false;
 | |
| 	}
 | |
| 
 | |
| 	if (macmeta && backspace) {
 | |
| 		QTextCursor tc(textCursor()), start(tc);
 | |
| 		start.movePosition(QTextCursor::StartOfLine);
 | |
| 		tc.setPosition(start.position(), QTextCursor::KeepAnchor);
 | |
| 		tc.removeSelectedText();
 | |
| 	} else if (backspace
 | |
| 		&& e->modifiers() == 0
 | |
| 		&& revertFormatReplace()) {
 | |
| 		e->accept();
 | |
| 	} else if (enter && enterSubmit) {
 | |
| 		_submits.fire(e->modifiers());
 | |
| 	} else if (e->key() == Qt::Key_Escape) {
 | |
| 		e->ignore();
 | |
| 		_cancelled.fire({});
 | |
| 	} else if (e->key() == Qt::Key_Tab || e->key() == Qt::Key_Backtab) {
 | |
| 		if (alt || ctrl) {
 | |
| 			e->ignore();
 | |
| 		} else if (_customTab) {
 | |
| 			_tabbed.fire({});
 | |
| 		} else if (!focusNextPrevChild(e->key() == Qt::Key_Tab && !shift)) {
 | |
| 			e->ignore();
 | |
| 		}
 | |
| 	} else if (e->key() == Qt::Key_Search || e == QKeySequence::Find) {
 | |
| 		e->ignore();
 | |
| 	} else if (handleMarkdownKey(e)) {
 | |
| 		e->accept();
 | |
| 	} else if (_customUpDown && (e->key() == Qt::Key_Up || e->key() == Qt::Key_Down || e->key() == Qt::Key_PageUp || e->key() == Qt::Key_PageDown)) {
 | |
| 		e->ignore();
 | |
| #ifdef Q_OS_MAC
 | |
| 	} else if (e->key() == Qt::Key_E && e->modifiers().testFlag(Qt::ControlModifier)) {
 | |
| 		const auto cursor = textCursor();
 | |
| 		const auto start = cursor.selectionStart();
 | |
| 		const auto end = cursor.selectionEnd();
 | |
| 		if (end > start) {
 | |
| 			QGuiApplication::clipboard()->setText(
 | |
| 				getTextWithTagsPart(start, end).text,
 | |
| 				QClipboard::FindBuffer);
 | |
| 		}
 | |
| #endif // Q_OS_MAC
 | |
| 	} else {
 | |
| 		const auto text = e->text();
 | |
| 		const auto old = textCursor();
 | |
| 		const auto oldPosition = old.position();
 | |
| 		const auto oldSelection = old.hasSelection();
 | |
| 		const auto oldModifiers = e->modifiers();
 | |
| 		const auto allowedModifiers = (enter && ctrl)
 | |
| 			? (~Qt::ControlModifier)
 | |
| 			: (enter && shift)
 | |
| 			? (~Qt::ShiftModifier)
 | |
| 			: oldModifiers;
 | |
| 		const auto changeModifiers = (oldModifiers & ~allowedModifiers) != 0;
 | |
| 		if (changeModifiers) {
 | |
| 			e->setModifiers(oldModifiers & allowedModifiers);
 | |
| 		}
 | |
| 		if (e == QKeySequence::InsertParagraphSeparator) {
 | |
| 			// qtbase commit dbb9579566f3accd8aa5fe61db9692991117afd3 introduced
 | |
| 			// special logic for repeated 'Enter' key presses, which drops the
 | |
| 			// block format instead of inserting a newline in case the block format
 | |
| 			// is non-trivial. For custom fonts we use non-trivial block formats
 | |
| 			// always for the entire QTextEdit, so we revert that logic and simply
 | |
| 			// insert a newline as it was before Qt 6.X.Y where this was added.
 | |
| 			textCursor().insertBlock();
 | |
| 			_inner->ensureCursorVisible();
 | |
| 			e->accept();
 | |
| 		} else {
 | |
| 			_inner->QTextEdit::keyPressEvent(e);
 | |
| 		}
 | |
| 		if (changeModifiers) {
 | |
| 			e->setModifiers(oldModifiers);
 | |
| 		}
 | |
| 		auto cursor = textCursor();
 | |
| 		if (cursor.position() == oldPosition) {
 | |
| 			bool check = false;
 | |
| 			if (e->key() == Qt::Key_PageUp || e->key() == Qt::Key_Up) {
 | |
| 				cursor.movePosition(QTextCursor::Start, e->modifiers().testFlag(Qt::ShiftModifier) ? QTextCursor::KeepAnchor : QTextCursor::MoveAnchor);
 | |
| 				check = true;
 | |
| 			} else if (e->key() == Qt::Key_PageDown || e->key() == Qt::Key_Down) {
 | |
| 				cursor.movePosition(QTextCursor::End, e->modifiers().testFlag(Qt::ShiftModifier) ? QTextCursor::KeepAnchor : QTextCursor::MoveAnchor);
 | |
| 				check = true;
 | |
| 			} else if (!oldSelection
 | |
| 				&& (e->key() == Qt::Key_Left
 | |
| 					|| e->key() == Qt::Key_Right
 | |
| 					|| e->key() == Qt::Key_Backspace)) {
 | |
| 				e->ignore();
 | |
| 			}
 | |
| 			if (check) {
 | |
| 				if (oldPosition == cursor.position()) {
 | |
| 					e->ignore();
 | |
| 				} else {
 | |
| 					setTextCursor(cursor);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		if (!processMarkdownReplaces(text)) {
 | |
| 			processInstantReplaces(text);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| TextWithTags InputField::getTextWithTagsSelected() const {
 | |
| 	const auto cursor = textCursor();
 | |
| 	const auto start = cursor.selectionStart();
 | |
| 	const auto end = cursor.selectionEnd();
 | |
| 	return (end > start) ? getTextWithTagsPart(start, end) : TextWithTags();
 | |
| }
 | |
| 
 | |
| bool InputField::handleMarkdownKey(QKeyEvent *e) {
 | |
| 	if (!_markdownEnabled) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	const auto modifiers = e->modifiers()
 | |
| 		& ~(Qt::KeypadModifier | Qt::GroupSwitchModifier);
 | |
| 	const auto matches = [&](const QKeySequence &sequence) {
 | |
| 		const auto events = QKeySequence(modifiers | e->key());
 | |
| 		return sequence.matches(events) == QKeySequence::ExactMatch;
 | |
| 	};
 | |
| 	const auto matchesCtrlShiftDot = [&] {
 | |
| 		// We can't match ctrl+shift+. with QKeySequence because
 | |
| 		// shift+. gives us '>' and ctrl+shift+> is not the same.
 | |
| 		// So we check with native code instead.
 | |
| #ifdef Q_OS_WIN
 | |
| 		return (modifiers == (Qt::ControlModifier | Qt::ShiftModifier))
 | |
| 			&& (e->nativeVirtualKey() == VK_OEM_PERIOD);
 | |
| #elif !defined DESKTOP_APP_DISABLE_X11_INTEGRATION // Q_OS_WIN
 | |
| 		if (!_inner->_xcbKeySymbols) {
 | |
| 			return false;
 | |
| 		}
 | |
| 		const auto keysym = xcb_key_symbols_get_keysym(
 | |
| 			_inner->_xcbKeySymbols.get(),
 | |
| 			e->nativeScanCode(),
 | |
| 			0);
 | |
| 		return (modifiers == (Qt::ControlModifier | Qt::ShiftModifier))
 | |
| 			&& (keysym == XKB_KEY_period);
 | |
| #else // !Q_OS_WIN && !DESKTOP_APP_DISABLE_X11_INTEGRATION
 | |
| 		return false;
 | |
| #endif // !Q_OS_WIN && DESKTOP_APP_DISABLE_X11_INTEGRATION
 | |
| 	};
 | |
| 	if (e == QKeySequence::Bold) {
 | |
| 		toggleSelectionMarkdown(kTagBold);
 | |
| 	} else if (e == QKeySequence::Italic) {
 | |
| 		toggleSelectionMarkdown(kTagItalic);
 | |
| 	} else if (e == QKeySequence::Underline) {
 | |
| 		toggleSelectionMarkdown(kTagUnderline);
 | |
| 	} else if (matches(kStrikeOutSequence)) {
 | |
| 		toggleSelectionMarkdown(kTagStrikeOut);
 | |
| 	} else if (matches(kMonospaceSequence)) {
 | |
| 		toggleSelectionMarkdown(kTagCode);
 | |
| 	} else if (matches(kBlockquoteSequence) || matchesCtrlShiftDot()) {
 | |
| 		toggleSelectionMarkdown(kTagBlockquote);
 | |
| 	}  else if (matches(kSpoilerSequence)) {
 | |
| 		toggleSelectionMarkdown(kTagSpoiler);
 | |
| 	} else if (matches(kClearFormatSequence)) {
 | |
| 		clearSelectionMarkdown();
 | |
| 	} else if (matches(kEditLinkSequence) && _editLinkCallback) {
 | |
| 		const auto cursor = textCursor();
 | |
| 		editMarkdownLink({
 | |
| 			cursor.selectionStart(),
 | |
| 			cursor.selectionEnd()
 | |
| 		});
 | |
| 	} else {
 | |
| 		return false;
 | |
| 	}
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| auto InputField::selectionEditLinkData(EditLinkSelection selection) const
 | |
| -> EditLinkData {
 | |
| 	Expects(_editLinkCallback != nullptr);
 | |
| 
 | |
| 	const auto position = (selection.from == selection.till
 | |
| 		&& selection.from > 0)
 | |
| 		? (selection.from - 1)
 | |
| 		: selection.from;
 | |
| 	const auto link = [&] {
 | |
| 		return (position != selection.till)
 | |
| 			? CheckFullTextTag(
 | |
| 				getTextWithTagsPart(position, selection.till),
 | |
| 				kTagCheckLinkMeta)
 | |
| 			: QString();
 | |
| 	}();
 | |
| 	const auto simple = EditLinkData{
 | |
| 		selection.from,
 | |
| 		selection.till,
 | |
| 		QString()
 | |
| 	};
 | |
| 	if (!_editLinkCallback(selection, {}, link, EditLinkAction::Check)) {
 | |
| 		return simple;
 | |
| 	}
 | |
| 	Assert(!link.isEmpty());
 | |
| 
 | |
| 	struct State {
 | |
| 		QTextBlock block;
 | |
| 		QTextBlock::iterator i;
 | |
| 	};
 | |
| 	const auto document = _inner->document();
 | |
| 	const auto skipInvalid = [&](State &state) {
 | |
| 		if (state.block == document->end()) {
 | |
| 			return false;
 | |
| 		}
 | |
| 		while (state.i.atEnd()) {
 | |
| 			state.block = state.block.next();
 | |
| 			if (state.block == document->end()) {
 | |
| 				return false;
 | |
| 			}
 | |
| 			state.i = state.block.begin();
 | |
| 		}
 | |
| 		return true;
 | |
| 	};
 | |
| 	const auto moveToNext = [&](State &state) {
 | |
| 		Expects(state.block != document->end());
 | |
| 		Expects(!state.i.atEnd());
 | |
| 
 | |
| 		++state.i;
 | |
| 	};
 | |
| 	const auto moveToPrevious = [&](State &state) {
 | |
| 		Expects(state.block != document->end());
 | |
| 		Expects(!state.i.atEnd());
 | |
| 
 | |
| 		while (state.i == state.block.begin()) {
 | |
| 			if (state.block == document->begin()) {
 | |
| 				state.block = document->end();
 | |
| 				return false;
 | |
| 			}
 | |
| 			state.block = state.block.previous();
 | |
| 			state.i = state.block.end();
 | |
| 		}
 | |
| 		--state.i;
 | |
| 		return true;
 | |
| 	};
 | |
| 	const auto stateTag = [&](const State &state) {
 | |
| 		const auto format = state.i.fragment().charFormat();
 | |
| 		return format.property(kTagProperty).toString();
 | |
| 	};
 | |
| 	const auto stateTagHasLink = [&](const State &state) {
 | |
| 		const auto tag = stateTag(state);
 | |
| 		return (tag == link)
 | |
| 			|| TextUtilities::SplitTags(tag).contains(QStringView(link));
 | |
| 	};
 | |
| 	const auto stateStart = [&](const State &state) {
 | |
| 		return state.i.fragment().position();
 | |
| 	};
 | |
| 	const auto stateEnd = [&](const State &state) {
 | |
| 		const auto fragment = state.i.fragment();
 | |
| 		return fragment.position() + fragment.length();
 | |
| 	};
 | |
| 	auto state = State{ document->findBlock(position) };
 | |
| 	if (state.block != document->end()) {
 | |
| 		state.i = state.block.begin();
 | |
| 	}
 | |
| 	for (; skipInvalid(state); moveToNext(state)) {
 | |
| 		const auto fragmentStart = stateStart(state);
 | |
| 		const auto fragmentEnd = stateEnd(state);
 | |
| 		if (fragmentEnd <= position) {
 | |
| 			continue;
 | |
| 		} else if (fragmentStart >= selection.till) {
 | |
| 			break;
 | |
| 		}
 | |
| 		if (stateTagHasLink(state)) {
 | |
| 			auto start = fragmentStart;
 | |
| 			auto finish = fragmentEnd;
 | |
| 			auto copy = state;
 | |
| 			while (moveToPrevious(copy) && stateTagHasLink(copy)) {
 | |
| 				start = stateStart(copy);
 | |
| 			}
 | |
| 			while (skipInvalid(state) && stateTagHasLink(state)) {
 | |
| 				finish = stateEnd(state);
 | |
| 				moveToNext(state);
 | |
| 			}
 | |
| 			return { start, finish, link };
 | |
| 		}
 | |
| 	}
 | |
| 	return simple;
 | |
| }
 | |
| 
 | |
| auto InputField::editLinkSelection(QContextMenuEvent *e) const
 | |
| -> EditLinkSelection {
 | |
| 	const auto cursor = textCursor();
 | |
| 	if (!cursor.hasSelection() && e->reason() == QContextMenuEvent::Mouse) {
 | |
| 		const auto clickCursor = _inner->cursorForPosition(
 | |
| 			_inner->viewport()->mapFromGlobal(e->globalPos()));
 | |
| 		if (!clickCursor.isNull() && !clickCursor.hasSelection()) {
 | |
| 			return {
 | |
| 				clickCursor.position(),
 | |
| 				clickCursor.position()
 | |
| 			};
 | |
| 		}
 | |
| 	}
 | |
| 	return {
 | |
| 		cursor.selectionStart(),
 | |
| 		cursor.selectionEnd()
 | |
| 	};
 | |
| }
 | |
| 
 | |
| void InputField::editMarkdownLink(EditLinkSelection selection) {
 | |
| 	if (!_editLinkCallback) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto data = selectionEditLinkData(selection);
 | |
| 	_editLinkCallback(
 | |
| 		selection,
 | |
| 		getTextWithTagsPart(data.from, data.till).text,
 | |
| 		data.link,
 | |
| 		EditLinkAction::Edit);
 | |
| }
 | |
| 
 | |
| void InputField::inputMethodEventInner(QInputMethodEvent *e) {
 | |
| 	const auto preedit = e->preeditString();
 | |
| 	if (_lastPreEditText != preedit) {
 | |
| 		_lastPreEditText = preedit;
 | |
| 		startPlaceholderAnimation();
 | |
| 	}
 | |
| 	_inputMethodCommit = e->commitString();
 | |
| 
 | |
| 	const auto weak = MakeWeak(this);
 | |
| 	_inner->QTextEdit::inputMethodEvent(e);
 | |
| 
 | |
| 	if (weak && _inputMethodCommit.has_value()) {
 | |
| 		const auto text = *base::take(_inputMethodCommit);
 | |
| 		if (!processMarkdownReplaces(text)) {
 | |
| 			processInstantReplaces(text);
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| const InstantReplaces &InputField::instantReplaces() const {
 | |
| 	return _mutableInstantReplaces;
 | |
| }
 | |
| 
 | |
| // Disable markdown instant replacement.
 | |
| bool InputField::processMarkdownReplaces(const QString &appended) {
 | |
| 	//if (appended.size() != 1 || !_markdownEnabled) {
 | |
| 	//	return false;
 | |
| 	//}
 | |
| 	//const auto ch = appended[0];
 | |
| 	//if (ch == '`') {
 | |
| 	//	return processMarkdownReplace(kTagCode)
 | |
| 	//		|| processMarkdownReplace(kTagPre);
 | |
| 	//} else if (ch == '*') {
 | |
| 	//	return processMarkdownReplace(kTagBold);
 | |
| 	//} else if (ch == '_') {
 | |
| 	//	return processMarkdownReplace(kTagItalic);
 | |
| 	//}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| //bool InputField::processMarkdownReplace(const QString &tag) {
 | |
| //	const auto position = textCursor().position();
 | |
| //	const auto tagLength = tag.size();
 | |
| //	const auto start = [&] {
 | |
| //		for (const auto &possible : _lastMarkdownTags) {
 | |
| //			const auto end = possible.start + possible.length;
 | |
| //			if (possible.start + 2 * tagLength >= position) {
 | |
| //				return MarkdownTag();
 | |
| //			} else if (end >= position || end + tagLength == position) {
 | |
| //				if (possible.tag == tag) {
 | |
| //					return possible;
 | |
| //				}
 | |
| //			}
 | |
| //		}
 | |
| //		return MarkdownTag();
 | |
| //	}();
 | |
| //	if (start.tag.isEmpty()) {
 | |
| //		return false;
 | |
| //	}
 | |
| //	return commitMarkdownReplacement(start.start, position, tag, tag);
 | |
| //}
 | |
| 
 | |
| void InputField::processInstantReplaces(const QString &appended) {
 | |
| 	const auto &replaces = instantReplaces();
 | |
| 	if (appended.size() != 1
 | |
| 		|| !_instantReplacesEnabled
 | |
| 		|| !replaces.maxLength) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto it = replaces.reverseMap.tail.find(appended[0]);
 | |
| 	if (it == end(replaces.reverseMap.tail)) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto position = textCursor().position();
 | |
| 	for (const auto &tag : _lastMarkdownTags) {
 | |
| 		if (tag.internalStart < position
 | |
| 			&& tag.internalStart + tag.internalLength >= position
 | |
| 			&& (tag.tag == kTagCode || IsTagPre(tag.tag))) {
 | |
| 			return;
 | |
| 		}
 | |
| 	}
 | |
| 	const auto typed = getTextWithTagsPart(
 | |
| 		std::max(position - replaces.maxLength, 0),
 | |
| 		position - 1).text;
 | |
| 	auto node = &it->second;
 | |
| 	auto i = typed.size();
 | |
| 	do {
 | |
| 		if (!node->text.isEmpty()) {
 | |
| 			applyInstantReplace(typed.mid(i) + appended, node->text);
 | |
| 			return;
 | |
| 		} else if (!i) {
 | |
| 			return;
 | |
| 		}
 | |
| 		const auto it = node->tail.find(typed[--i]);
 | |
| 		if (it == end(node->tail)) {
 | |
| 			return;
 | |
| 		}
 | |
| 		node = &it->second;
 | |
| 	} while (true);
 | |
| }
 | |
| 
 | |
| void InputField::applyInstantReplace(
 | |
| 		const QString &what,
 | |
| 		const QString &with) {
 | |
| 	const auto length = int(what.size());
 | |
| 	const auto cursor = textCursor();
 | |
| 	const auto position = cursor.position();
 | |
| 	if (cursor.hasSelection()) {
 | |
| 		return;
 | |
| 	} else if (position < length) {
 | |
| 		return;
 | |
| 	}
 | |
| 	commitInstantReplacement(
 | |
| 		position - length,
 | |
| 		position,
 | |
| 		with,
 | |
| 		QString(),
 | |
| 		what,
 | |
| 		true);
 | |
| }
 | |
| 
 | |
| void InputField::commitInstantReplacement(
 | |
| 		int from,
 | |
| 		int till,
 | |
| 		const QString &with,
 | |
| 		const QString &customEmojiData) {
 | |
| 	commitInstantReplacement(
 | |
| 		from,
 | |
| 		till,
 | |
| 		with,
 | |
| 		customEmojiData,
 | |
| 		std::nullopt,
 | |
| 		false);
 | |
| }
 | |
| 
 | |
| void InputField::commitInstantReplacement(
 | |
| 		int from,
 | |
| 		int till,
 | |
| 		const QString &with,
 | |
| 		const QString &customEmojiData,
 | |
| 		std::optional<QString> checkOriginal,
 | |
| 		bool checkIfInMonospace) {
 | |
| 	const auto original = getTextWithTagsPart(from, till).text;
 | |
| 	if (checkOriginal
 | |
| 		&& checkOriginal->compare(original, Qt::CaseInsensitive) != 0) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	auto cursor = textCursor();
 | |
| 	if (checkIfInMonospace) {
 | |
| 		const auto currentTag = cursor.charFormat().property(
 | |
| 			kTagProperty
 | |
| 		).toString();
 | |
| 		for (const auto &tag : TextUtilities::SplitTags(currentTag)) {
 | |
| 			if (tag == kTagCode || IsTagPre(tag)) {
 | |
| 				return;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	cursor.setPosition(from);
 | |
| 	cursor.setPosition(till, QTextCursor::KeepAnchor);
 | |
| 
 | |
| 	const auto link = customEmojiData.isEmpty()
 | |
| 		? QString()
 | |
| 		: CustomEmojiLink(customEmojiData);
 | |
| 	const auto unique = link.isEmpty()
 | |
| 		? QString()
 | |
| 		: MakeUniqueCustomEmojiLink(link);
 | |
| 	auto format = [&]() -> QTextCharFormat {
 | |
| 		auto emojiLength = 0;
 | |
| 		const auto emoji = Emoji::Find(with, &emojiLength);
 | |
| 		if (!emoji || with.size() != emojiLength) {
 | |
| 			return _defaultCharFormat;
 | |
| 		} else if (!customEmojiData.isEmpty()) {
 | |
| 			auto result = QTextCharFormat();
 | |
| 			result.setObjectType(kCustomEmojiFormat);
 | |
| 			result.setProperty(kCustomEmojiText, with);
 | |
| 			result.setProperty(kCustomEmojiLink, unique);
 | |
| 			result.setProperty(kCustomEmojiId, CustomEmojiIdFromLink(link));
 | |
| 			result.setVerticalAlignment(QTextCharFormat::AlignTop);
 | |
| 			return result;
 | |
| 		}
 | |
| 		const auto use = Integration::Instance().defaultEmojiVariant(
 | |
| 			emoji);
 | |
| 		return PrepareEmojiFormat(use, _st.font);
 | |
| 	}();
 | |
| 	const auto replacement = (format.isImageFormat()
 | |
| 		|| format.objectType() == kCustomEmojiFormat)
 | |
| 		? kObjectReplacement
 | |
| 		: with;
 | |
| 	format.setProperty(kInstantReplaceWhatId, original);
 | |
| 	format.setProperty(kInstantReplaceWithId, replacement);
 | |
| 	format.setProperty(
 | |
| 		kInstantReplaceRandomId,
 | |
| 		base::RandomValue<uint32>());
 | |
| 	ApplyTagFormat(format, cursor.charFormat());
 | |
| 	if (!unique.isEmpty()) {
 | |
| 		format.setProperty(kTagProperty, TextUtilities::TagWithAdded(
 | |
| 			format.property(kTagProperty).toString(),
 | |
| 			unique));
 | |
| 	}
 | |
| 	cursor.insertText(replacement, format);
 | |
| }
 | |
| 
 | |
| #if 0
 | |
| bool InputField::commitMarkdownReplacement(
 | |
| 		int from,
 | |
| 		int till,
 | |
| 		const QString &tag,
 | |
| 		const QString &edge) {
 | |
| 	const auto end = [&] {
 | |
| 		auto cursor = QTextCursor(document());
 | |
| 		cursor.movePosition(QTextCursor::End);
 | |
| 		return cursor.position();
 | |
| 	}();
 | |
| 
 | |
| 	// In case of 'pre' tag extend checked text by one symbol.
 | |
| 	// So that we'll know if we need to insert additional newlines.
 | |
| 	// "Test ```test``` Test" should become three-line text.
 | |
| 	const auto blocktag = (tag == kTagPre);
 | |
| 	const auto extendLeft = (blocktag && from > 0) ? 1 : 0;
 | |
| 	const auto extendRight = (blocktag && till < end) ? 1 : 0;
 | |
| 	const auto extended = getTextWithTagsPart(
 | |
| 		from - extendLeft,
 | |
| 		till + extendRight).text;
 | |
| 	const auto outer = base::StringViewMid(
 | |
| 		extended,
 | |
| 		extendLeft,
 | |
| 		extended.size() - extendLeft - extendRight);
 | |
| 	if ((outer.size() <= 2 * edge.size())
 | |
| 		|| (!edge.isEmpty()
 | |
| 			&& !(outer.startsWith(edge) && outer.endsWith(edge)))) {
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	// In case of 'pre' tag check if we need to remove one of two newlines.
 | |
| 	// "Test\n```\ntest\n```" should become two-line text + newline.
 | |
| 	const auto innerRight = edge.size();
 | |
| 	const auto checkIfTwoNewlines = blocktag
 | |
| 		&& (extendLeft > 0)
 | |
| 		&& IsNewline(extended[0]);
 | |
| 	const auto innerLeft = [&] {
 | |
| 		const auto simple = edge.size();
 | |
| 		if (!checkIfTwoNewlines) {
 | |
| 			return simple;
 | |
| 		}
 | |
| 		const auto last = outer.size() - innerRight;
 | |
| 		for (auto check = simple; check != last; ++check) {
 | |
| 			const auto ch = outer.at(check);
 | |
| 			if (IsNewline(ch)) {
 | |
| 				return check + 1;
 | |
| 			} else if (!Text::IsSpace(ch)) {
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 		return simple;
 | |
| 	}();
 | |
| 	const auto innerLength = outer.size() - innerLeft - innerRight;
 | |
| 
 | |
| 	// Prepare the final "insert" replacement for the "outer" text part.
 | |
| 	const auto newlineleft = blocktag
 | |
| 		&& (extendLeft > 0)
 | |
| 		&& !IsNewline(extended[0])
 | |
| 		&& !IsNewline(outer.at(innerLeft));
 | |
| 	const auto newlineright = blocktag
 | |
| 		&& (!extendRight || !IsNewline(extended[extended.size() - 1]))
 | |
| 		&& !IsNewline(outer.at(outer.size() - innerRight - 1));
 | |
| 	const auto insert = (newlineleft ? "\n" : "")
 | |
| 		+ outer.mid(innerLeft, innerLength).toString()
 | |
| 		+ (newlineright ? "\n" : "");
 | |
| 
 | |
| 	// Trim inserted tag, so that all newlines are left outside.
 | |
| 	_insertedTags.clear();
 | |
| 	auto tagFrom = newlineleft ? 1 : 0;
 | |
| 	auto tagTill = insert.size() - (newlineright ? 1 : 0);
 | |
| 	for (; tagFrom != tagTill; ++tagFrom) {
 | |
| 		const auto ch = insert.at(tagFrom);
 | |
| 		if (!IsNewline(ch)) {
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 	for (; tagTill != tagFrom; --tagTill) {
 | |
| 		const auto ch = insert.at(tagTill - 1);
 | |
| 		if (!IsNewline(ch)) {
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 	if (tagTill > tagFrom) {
 | |
| 		_insertedTags.push_back({
 | |
| 			tagFrom,
 | |
| 			int(tagTill - tagFrom),
 | |
| 			tag,
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	// Replace.
 | |
| 	auto cursor = _inner->textCursor();
 | |
| 	cursor.setPosition(from);
 | |
| 	cursor.setPosition(till, QTextCursor::KeepAnchor);
 | |
| 	auto format = _defaultCharFormat;
 | |
| 	if (!edge.isEmpty()) {
 | |
| 		format.setProperty(kReplaceTagId, edge);
 | |
| 		_reverseMarkdownReplacement = true;
 | |
| 	}
 | |
| 	_insertedTagsAreFromMime = false;
 | |
| 	cursor.insertText(insert, format);
 | |
| 	_insertedTags.clear();
 | |
| 
 | |
| 	cursor.setCharFormat(_defaultCharFormat);
 | |
| 	_inner->setTextCursor(cursor);
 | |
| 
 | |
| 	// Fire the tag to the spellchecker.
 | |
| 	_markdownTagApplies.fire({ from, till, -1, -1, false, tag });
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| void InputField::addMarkdownTag(
 | |
| 		int from,
 | |
| 		int till,
 | |
| 		const QString &tag) {
 | |
| 	const auto current = getTextWithTagsPart(from, till);
 | |
| 	const auto currentLength = int(current.text.size());
 | |
| 
 | |
| 	// #TODO Trim inserted tag, so that all newlines are left outside.
 | |
| 	auto tags = TagList();
 | |
| 	auto filled = 0;
 | |
| 	const auto add = [&](const TextWithTags::Tag &existing) {
 | |
| 		const auto id = TextUtilities::TagWithAdded(existing.id, tag);
 | |
| 		tags.push_back({ existing.offset, existing.length, id });
 | |
| 		filled = std::clamp(
 | |
| 			existing.offset + existing.length,
 | |
| 			filled,
 | |
| 			currentLength);
 | |
| 	};
 | |
| 	if (!TextUtilities::IsSeparateTag(tag)) {
 | |
| 		for (const auto &existing : current.tags) {
 | |
| 			if (existing.offset >= currentLength) {
 | |
| 				break;
 | |
| 			} else if (existing.offset > filled) {
 | |
| 				add({ filled, existing.offset - filled, tag });
 | |
| 			}
 | |
| 			add(existing);
 | |
| 		}
 | |
| 	}
 | |
| 	if (filled < currentLength) {
 | |
| 		add({ filled, currentLength - filled, tag });
 | |
| 	}
 | |
| 
 | |
| 	finishMarkdownTagChange(from, till, { current.text, tags });
 | |
| 
 | |
| 	// Fire the tag to the spellchecker.
 | |
| 	_markdownTagApplies.fire({ from, till, -1, -1, false, tag });
 | |
| }
 | |
| 
 | |
| void InputField::removeMarkdownTag(
 | |
| 		int from,
 | |
| 		int till,
 | |
| 		const QString &tag) {
 | |
| 	const auto current = getTextWithTagsPart(from, till);
 | |
| 
 | |
| 	auto tags = TagList();
 | |
| 	for (const auto &existing : current.tags) {
 | |
| 		const auto id = TextUtilities::TagWithRemoved(existing.id, tag);
 | |
| 		const auto additional = (tag == kTagPre)
 | |
| 			? kTagCode
 | |
| 			: (tag == kTagCode)
 | |
| 			? kTagPre
 | |
| 			: QString();
 | |
| 		const auto use = additional.isEmpty()
 | |
| 			? id
 | |
| 			: TextUtilities::TagWithRemoved(id, additional);
 | |
| 		if (!use.isEmpty()) {
 | |
| 			tags.push_back({ existing.offset, existing.length, use });
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	finishMarkdownTagChange(from, till, { current.text, tags });
 | |
| }
 | |
| 
 | |
| void InputField::finishMarkdownTagChange(
 | |
| 		int from,
 | |
| 		int till,
 | |
| 		const TextWithTags &textWithTags) {
 | |
| 	auto cursor = _inner->textCursor();
 | |
| 	cursor.setPosition(from);
 | |
| 	cursor.setPosition(till, QTextCursor::KeepAnchor);
 | |
| 	_insertedTags = textWithTags.tags;
 | |
| 	_insertedTagsAreFromMime = false;
 | |
| 	cursor.insertText(textWithTags.text, _defaultCharFormat);
 | |
| 	_insertedTags.clear();
 | |
| 
 | |
| 	cursor.setCharFormat(_defaultCharFormat);
 | |
| 	_inner->setTextCursor(cursor);
 | |
| }
 | |
| 
 | |
| bool InputField::IsValidMarkdownLink(QStringView link) {
 | |
| 	return ::Ui::IsValidMarkdownLink(link) && !::Ui::IsCustomEmojiLink(link);
 | |
| }
 | |
| 
 | |
| bool InputField::IsCustomEmojiLink(QStringView link) {
 | |
| 	return ::Ui::IsCustomEmojiLink(link);
 | |
| }
 | |
| 
 | |
| QString InputField::CustomEmojiLink(QStringView entityData) {
 | |
| 	return MakeUniqueCustomEmojiLink(u"%1%2"_q
 | |
| 		.arg(kCustomEmojiTagStart)
 | |
| 		.arg(entityData));
 | |
| }
 | |
| 
 | |
| QString InputField::CustomEmojiEntityData(QStringView link) {
 | |
| 	const auto match = qthelp::regex_match(
 | |
| 		"^(\\d+)(\\?|$)",
 | |
| 		base::StringViewMid(link, kCustomEmojiTagStart.size()));
 | |
| 	return match ? match->captured(1) : QString();
 | |
| }
 | |
| 
 | |
| void InputField::commitMarkdownLinkEdit(
 | |
| 		EditLinkSelection selection,
 | |
| 		const QString &text,
 | |
| 		const QString &link) {
 | |
| 	if (text.isEmpty()
 | |
| 		|| !IsValidMarkdownLink(link)
 | |
| 		|| !_editLinkCallback) {
 | |
| 		return;
 | |
| 	}
 | |
| 	_insertedTags.clear();
 | |
| 	_insertedTags.push_back({ 0, int(text.size()), link });
 | |
| 
 | |
| 	auto cursor = textCursor();
 | |
| 	const auto editData = selectionEditLinkData(selection);
 | |
| 	cursor.setPosition(editData.from);
 | |
| 	cursor.setPosition(editData.till, QTextCursor::KeepAnchor);
 | |
| 	auto format = _defaultCharFormat;
 | |
| 	_insertedTagsAreFromMime = false;
 | |
| 	cursor.insertText(
 | |
| 		(editData.from == editData.till) ? (text + QChar(' ')) : text,
 | |
| 		_defaultCharFormat);
 | |
| 	_insertedTags.clear();
 | |
| 
 | |
| 	_reverseMarkdownReplacement = false;
 | |
| 	_correcting = true;
 | |
| 	cursor.joinPreviousEditBlock();
 | |
| 	cursor.setCharFormat(_defaultCharFormat);
 | |
| 	cursor.setBlockFormat(_defaultBlockFormat);
 | |
| 	cursor.endEditBlock();
 | |
| 	_inner->setTextCursor(cursor);
 | |
| 	_correcting = false;
 | |
| }
 | |
| 
 | |
| void InputField::toggleSelectionMarkdown(const QString &tag) {
 | |
| 	_reverseMarkdownReplacement = false;
 | |
| 	const auto cursor = textCursor();
 | |
| 	auto position = cursor.position();
 | |
| 	auto from = cursor.selectionStart();
 | |
| 	auto till = cursor.selectionEnd();
 | |
| 	if (from == till) {
 | |
| 		return;
 | |
| 	}
 | |
| 	if (tag.isEmpty()) {
 | |
| 		RemoveDocumentTags(_st, document(), from, till);
 | |
| 	} else if (HasFullTextTag(getTextWithTagsSelected(), tag)) {
 | |
| 		removeMarkdownTag(from, till, tag);
 | |
| 	} else {
 | |
| 		const auto leftForBlock = [&] {
 | |
| 			if (!from) {
 | |
| 				return true;
 | |
| 			}
 | |
| 			const auto text = getTextWithTagsPart(
 | |
| 				from - 1,
 | |
| 				from + 1
 | |
| 			).text;
 | |
| 			return text.isEmpty()
 | |
| 				|| IsNewline(text[0])
 | |
| 				|| IsNewline(text[text.size() - 1]);
 | |
| 		}();
 | |
| 		const auto rightForBlock = [&] {
 | |
| 			const auto text = getTextWithTagsPart(
 | |
| 				till - 1,
 | |
| 				till + 1
 | |
| 			).text;
 | |
| 			return text.isEmpty()
 | |
| 				|| IsNewline(text[0])
 | |
| 				|| IsNewline(text[text.size() - 1]);
 | |
| 		}();
 | |
| 
 | |
| 		const auto useTag = (tag != kTagCode)
 | |
| 			? tag
 | |
| 			: (leftForBlock && rightForBlock)
 | |
| 			? kTagPre
 | |
| 			: kTagCode;
 | |
| 		if (tag == kTagBlockquote) {
 | |
| 			QTextCursor(document()).beginEditBlock();
 | |
| 			if (!leftForBlock) {
 | |
| 				auto copy = textCursor();
 | |
| 				copy.setPosition(from);
 | |
| 				copy.insertText(u"\n"_q);
 | |
| 				++position;
 | |
| 				++from;
 | |
| 				++till;
 | |
| 			}
 | |
| 			if (!rightForBlock) {
 | |
| 				auto copy = textCursor();
 | |
| 				copy.setPosition(till);
 | |
| 				copy.insertText(u"\n"_q);
 | |
| 			}
 | |
| 			QTextCursor(document()).endEditBlock();
 | |
| 		}
 | |
| 		addMarkdownTag(from, till, useTag);
 | |
| 	}
 | |
| 	auto restorePosition = textCursor();
 | |
| 	restorePosition.setPosition((position == till) ? from : till);
 | |
| 	restorePosition.setPosition(position, QTextCursor::KeepAnchor);
 | |
| 	setTextCursor(restorePosition);
 | |
| }
 | |
| 
 | |
| void InputField::clearSelectionMarkdown() {
 | |
| 	toggleSelectionMarkdown(QString());
 | |
| }
 | |
| 
 | |
| bool InputField::revertFormatReplace() {
 | |
| 	const auto cursor = textCursor();
 | |
| 	const auto position = cursor.position();
 | |
| 	if (position <= 0 || cursor.hasSelection()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	const auto inside = position - 1;
 | |
| 	const auto document = _inner->document();
 | |
| 	const auto block = document->findBlock(inside);
 | |
| 	if (block == document->end()) {
 | |
| 		return false;
 | |
| 	}
 | |
| 	for (auto i = block.begin(); !i.atEnd(); ++i) {
 | |
| 		const auto fragment = i.fragment();
 | |
| 		const auto fragmentStart = fragment.position();
 | |
| 		const auto fragmentEnd = fragmentStart + fragment.length();
 | |
| 		if (fragmentEnd <= inside) {
 | |
| 			continue;
 | |
| 		} else if (fragmentStart > inside || fragmentEnd != position) {
 | |
| 			return false;
 | |
| 		}
 | |
| 		const auto current = fragment.charFormat();
 | |
| 		if (current.hasProperty(kInstantReplaceWithId)) {
 | |
| 			const auto with = current.property(kInstantReplaceWithId);
 | |
| 			const auto string = with.toString();
 | |
| 			if (fragment.text() != string) {
 | |
| 				return false;
 | |
| 			}
 | |
| 			auto replaceCursor = cursor;
 | |
| 			replaceCursor.setPosition(fragmentStart);
 | |
| 			replaceCursor.setPosition(fragmentEnd, QTextCursor::KeepAnchor);
 | |
| 			const auto what = current.property(kInstantReplaceWhatId);
 | |
| 			auto format = _defaultCharFormat;
 | |
| 			ApplyTagFormat(format, current);
 | |
| 			replaceCursor.insertText(what.toString(), format);
 | |
| 			return true;
 | |
| 		} else if (_reverseMarkdownReplacement
 | |
| 			&& current.hasProperty(kReplaceTagId)) {
 | |
| 			const auto tag = current.property(kReplaceTagId).toString();
 | |
| 			if (tag.isEmpty()) {
 | |
| 				return false;
 | |
| 			} else if (auto test = i; !(++test).atEnd()) {
 | |
| 				const auto format = test.fragment().charFormat();
 | |
| 				if (format.property(kReplaceTagId).toString() == tag) {
 | |
| 					return false;
 | |
| 				}
 | |
| 			} else if (auto test = block; test.next() != document->end()) {
 | |
| 				const auto begin = test.begin();
 | |
| 				if (begin != test.end()) {
 | |
| 					const auto format = begin.fragment().charFormat();
 | |
| 					if (format.property(kReplaceTagId).toString() == tag) {
 | |
| 						return false;
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			const auto first = [&] {
 | |
| 				auto checkBlock = block;
 | |
| 				auto checkLast = i;
 | |
| 				while (true) {
 | |
| 					for (auto j = checkLast; j != checkBlock.begin();) {
 | |
| 						--j;
 | |
| 						const auto format = j.fragment().charFormat();
 | |
| 						if (format.property(kReplaceTagId) != tag) {
 | |
| 							return ++j;
 | |
| 						}
 | |
| 					}
 | |
| 					if (checkBlock == document->begin()) {
 | |
| 						return checkBlock.begin();
 | |
| 					}
 | |
| 					checkBlock = checkBlock.previous();
 | |
| 					checkLast = checkBlock.end();
 | |
| 				}
 | |
| 			}();
 | |
| 			const auto from = first.fragment().position();
 | |
| 			const auto till = fragmentEnd;
 | |
| 			auto replaceCursor = cursor;
 | |
| 			replaceCursor.setPosition(from);
 | |
| 			replaceCursor.setPosition(till, QTextCursor::KeepAnchor);
 | |
| 			replaceCursor.insertText(
 | |
| 				tag + getTextWithTagsPart(from, till).text + tag,
 | |
| 				_defaultCharFormat);
 | |
| 			return true;
 | |
| 		}
 | |
| 		return false;
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| void InputField::contextMenuEventInner(QContextMenuEvent *e, QMenu *m) {
 | |
| 	if (const auto menu = m ? m : _inner->createStandardContextMenu()) {
 | |
| 		addMarkdownActions(menu, e);
 | |
| 		_contextMenu = base::make_unique_q<PopupMenu>(this, menu, _st.menu);
 | |
| 		QObject::connect(_contextMenu.get(), &QObject::destroyed, [=] {
 | |
| 			_menuShownChanges.fire(false);
 | |
| 		});
 | |
| 		_menuShownChanges.fire(true);
 | |
| 		_contextMenu->popup(e->globalPos());
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::addMarkdownActions(
 | |
| 		not_null<QMenu*> menu,
 | |
| 		QContextMenuEvent *e) {
 | |
| 	if (!_markdownEnabled) {
 | |
| 		return;
 | |
| 	}
 | |
| 	auto &integration = Integration::Instance();
 | |
| 
 | |
| 	const auto formatting = new QAction(
 | |
| 		integration.phraseFormattingTitle(),
 | |
| 		menu);
 | |
| 	addMarkdownMenuAction(menu, formatting);
 | |
| 
 | |
| 	const auto submenu = new QMenu(menu);
 | |
| 	formatting->setMenu(submenu);
 | |
| 
 | |
| 	const auto textWithTags = getTextWithTagsSelected();
 | |
| 	const auto &text = textWithTags.text;
 | |
| 	const auto &tags = textWithTags.tags;
 | |
| 	const auto hasText = !text.isEmpty();
 | |
| 	const auto hasTags = !tags.isEmpty();
 | |
| 	const auto disabled = (!_editLinkCallback && !hasText);
 | |
| 	formatting->setDisabled(disabled);
 | |
| 	if (disabled) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto add = [&](
 | |
| 			const QString &base,
 | |
| 			QKeySequence sequence,
 | |
| 			bool disabled,
 | |
| 			auto callback) {
 | |
| 		const auto add = sequence.isEmpty()
 | |
| 			? QString()
 | |
| 			: QChar('\t') + sequence.toString(QKeySequence::NativeText);
 | |
| 		const auto action = new QAction(base + add, submenu);
 | |
| 		connect(action, &QAction::triggered, this, callback);
 | |
| 		action->setDisabled(disabled);
 | |
| 		submenu->addAction(action);
 | |
| 	};
 | |
| 	const auto addtag = [&](
 | |
| 			const QString &base,
 | |
| 			QKeySequence sequence,
 | |
| 			const QString &tag) {
 | |
| 		const auto disabled = !hasText;
 | |
| 		add(base, sequence, disabled, [=] {
 | |
| 			toggleSelectionMarkdown(tag);
 | |
| 		});
 | |
| 	};
 | |
| 	const auto addlink = [&] {
 | |
| 		const auto selection = editLinkSelection(e);
 | |
| 		const auto data = selectionEditLinkData(selection);
 | |
| 		const auto base = data.link.isEmpty()
 | |
| 			? integration.phraseFormattingLinkCreate()
 | |
| 			: integration.phraseFormattingLinkEdit();
 | |
| 		add(base, kEditLinkSequence, false, [=] {
 | |
| 			editMarkdownLink(selection);
 | |
| 		});
 | |
| 	};
 | |
| 	const auto addclear = [&] {
 | |
| 		const auto disabled = !hasText || !hasTags;
 | |
| 		add(integration.phraseFormattingClear(), kClearFormatSequence, disabled, [=] {
 | |
| 			clearSelectionMarkdown();
 | |
| 		});
 | |
| 	};
 | |
| 
 | |
| 	addtag(integration.phraseFormattingBold(), QKeySequence::Bold, kTagBold);
 | |
| 	addtag(integration.phraseFormattingItalic(), QKeySequence::Italic, kTagItalic);
 | |
| 	addtag(integration.phraseFormattingUnderline(), QKeySequence::Underline, kTagUnderline);
 | |
| 	addtag(integration.phraseFormattingStrikeOut(), kStrikeOutSequence, kTagStrikeOut);
 | |
| 	addtag(integration.phraseFormattingBlockquote(), kBlockquoteSequence, kTagBlockquote);
 | |
| 	addtag(integration.phraseFormattingMonospace(), kMonospaceSequence, kTagCode);
 | |
| 	addtag(integration.phraseFormattingSpoiler(), kSpoilerSequence, kTagSpoiler);
 | |
| 
 | |
| 	if (_editLinkCallback) {
 | |
| 		submenu->addSeparator();
 | |
| 		addlink();
 | |
| 	}
 | |
| 
 | |
| 	submenu->addSeparator();
 | |
| 	addclear();
 | |
| }
 | |
| 
 | |
| void InputField::addMarkdownMenuAction(
 | |
| 		not_null<QMenu*> menu,
 | |
| 		not_null<QAction*> action) {
 | |
| 	const auto actions = menu->actions();
 | |
| 	const auto before = [&] {
 | |
| 		auto seenAfter = false;
 | |
| 		for (const auto action : actions) {
 | |
| 			if (seenAfter) {
 | |
| 				return action;
 | |
| 			} else if (action->objectName() == qstr("edit-delete")) {
 | |
| 				seenAfter = true;
 | |
| 			}
 | |
| 		}
 | |
| 		return (QAction*)nullptr;
 | |
| 	}();
 | |
| 	menu->insertSeparator(before);
 | |
| 	menu->insertAction(before, action);
 | |
| }
 | |
| 
 | |
| void InputField::dropEventInner(QDropEvent *e) {
 | |
| 	_inDrop = true;
 | |
| 	_inner->QTextEdit::dropEvent(e);
 | |
| 	_inDrop = false;
 | |
| 	_insertedTags.clear();
 | |
| 	_realInsertPosition = -1;
 | |
| 	window()->raise();
 | |
| 	window()->activateWindow();
 | |
| }
 | |
| 
 | |
| bool InputField::canInsertFromMimeDataInner(const QMimeData *source) const {
 | |
| 	if (source
 | |
| 		&& _mimeDataHook
 | |
| 		&& _mimeDataHook(source, MimeAction::Check)) {
 | |
| 		return true;
 | |
| 	}
 | |
| 	return _inner->QTextEdit::canInsertFromMimeData(source);
 | |
| }
 | |
| 
 | |
| void InputField::insertFromMimeDataInner(const QMimeData *source) {
 | |
| 	if (source
 | |
| 		&& _mimeDataHook
 | |
| 		&& _mimeDataHook(source, MimeAction::Insert)) {
 | |
| 		return;
 | |
| 	}
 | |
| 	const auto text = [&] {
 | |
| 		const auto textMime = TextUtilities::TagsTextMimeType();
 | |
| 		const auto tagsMime = TextUtilities::TagsMimeType();
 | |
| 		if (!source->hasFormat(textMime) || !source->hasFormat(tagsMime)) {
 | |
| 			_insertedTags.clear();
 | |
| 			return source->text();
 | |
| 		}
 | |
| 		auto result = QString::fromUtf8(source->data(textMime));
 | |
| 		_insertedTags = TextUtilities::DeserializeTags(
 | |
| 			source->data(tagsMime),
 | |
| 			result.size());
 | |
| 		_insertedTagsAreFromMime = true;
 | |
| 		return result;
 | |
| 	}();
 | |
| 	auto cursor = textCursor();
 | |
| 	_realInsertPosition = cursor.selectionStart();
 | |
| 	_realCharsAdded = text.size();
 | |
| 	if (_realCharsAdded > 0) {
 | |
| 		cursor.insertFragment(QTextDocumentFragment::fromPlainText(text));
 | |
| 	}
 | |
| 	ensureCursorVisible();
 | |
| 	if (!_inDrop) {
 | |
| 		_insertedTags.clear();
 | |
| 		_realInsertPosition = -1;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::resizeEvent(QResizeEvent *e) {
 | |
| 	refreshPlaceholder(_placeholderFull.current());
 | |
| 	_inner->setGeometry(rect().marginsRemoved(
 | |
| 		_st.textMargins + _additionalMargins + _customFontMargins));
 | |
| 	_borderAnimationStart = width() / 2;
 | |
| 	RpWidget::resizeEvent(e);
 | |
| 	checkContentHeight();
 | |
| }
 | |
| 
 | |
| void InputField::refreshPlaceholder(const QString &text) {
 | |
| 	const auto margins = _st.textMargins
 | |
| 		+ _st.placeholderMargins
 | |
| 		+ _additionalMargins
 | |
| 		+ _customFontMargins;
 | |
| 	const auto availableWidth = rect().marginsRemoved(margins).width();
 | |
| 	if (_st.placeholderScale > 0.) {
 | |
| 		auto placeholderFont = _st.placeholderFont->f;
 | |
| 		placeholderFont.setStyleStrategy(QFont::PreferMatch);
 | |
| 		const auto metrics = QFontMetrics(placeholderFont);
 | |
| 		_placeholder = metrics.elidedText(text, Qt::ElideRight, availableWidth);
 | |
| 		_placeholderPath = QPainterPath();
 | |
| 		if (!_placeholder.isEmpty()) {
 | |
| 			const auto result = style::FindAdjustResult(placeholderFont);
 | |
| 			const auto ascent = result ? result->iascent : metrics.ascent();
 | |
| 			_placeholderPath.addText(0, ascent, placeholderFont, _placeholder);
 | |
| 		}
 | |
| 	} else {
 | |
| 		_placeholder = _st.placeholderFont->elided(text, availableWidth);
 | |
| 	}
 | |
| 	update();
 | |
| }
 | |
| 
 | |
| void InputField::setPlaceholder(
 | |
| 		rpl::producer<QString> placeholder,
 | |
| 		int afterSymbols) {
 | |
| 	_placeholderFull = std::move(placeholder);
 | |
| 	if (_placeholderAfterSymbols != afterSymbols) {
 | |
| 		_placeholderAfterSymbols = afterSymbols;
 | |
| 		startPlaceholderAnimation();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::setEditLinkCallback(
 | |
| 	Fn<bool(
 | |
| 		EditLinkSelection selection,
 | |
| 		QString text,
 | |
| 		QString link,
 | |
| 		EditLinkAction action)> callback) {
 | |
| 	_editLinkCallback = std::move(callback);
 | |
| }
 | |
| 
 | |
| void InputField::showError() {
 | |
| 	showErrorNoFocus();
 | |
| 	if (!hasFocus()) {
 | |
| 		_inner->setFocus();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void InputField::showErrorNoFocus() {
 | |
| 	setErrorShown(true);
 | |
| }
 | |
| 
 | |
| void InputField::hideError() {
 | |
| 	setErrorShown(false);
 | |
| }
 | |
| 
 | |
| void InputField::setErrorShown(bool error) {
 | |
| 	if (_error != error) {
 | |
| 		_error = error;
 | |
| 		_a_error.start([this] { update(); }, _error ? 0. : 1., _error ? 1. : 0., _st.duration);
 | |
| 		startBorderAnimation();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| rpl::producer<> InputField::heightChanges() const {
 | |
| 	return _heightChanges.events();
 | |
| }
 | |
| 
 | |
| rpl::producer<bool> InputField::focusedChanges() const {
 | |
| 	return _focusedChanges.events();
 | |
| }
 | |
| 
 | |
| rpl::producer<> InputField::tabbed() const {
 | |
| 	return _tabbed.events();
 | |
| }
 | |
| 
 | |
| rpl::producer<> InputField::cancelled() const {
 | |
| 	return _cancelled.events();
 | |
| }
 | |
| 
 | |
| rpl::producer<> InputField::changes() const {
 | |
| 	return _changes.events();
 | |
| }
 | |
| 
 | |
| rpl::producer<Qt::KeyboardModifiers> InputField::submits() const {
 | |
| 	return _submits.events();
 | |
| }
 | |
| 
 | |
| InputField::~InputField() = default;
 | |
| 
 | |
| // Optimization: with null page size document does not re-layout
 | |
| // on each insertText / mergeCharFormat.
 | |
| void PrepareFormattingOptimization(not_null<QTextDocument*> document) {
 | |
| 	if (!document->pageSize().isNull()) {
 | |
| 		document->setPageSize(QSizeF(0, 0));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int FieldCharacterCount(not_null<InputField*> field) {
 | |
| 	// This method counts emoji properly.
 | |
| 	return field->lastTextSizeWithoutSurrogatePairsCount();
 | |
| }
 | |
| 
 | |
| void AddLengthLimitLabel(not_null<InputField*> field, int limit) {
 | |
| 	struct State {
 | |
| 		rpl::variable<int> length;
 | |
| 	};
 | |
| 	const auto state = field->lifetime().make_state<State>();
 | |
| 	state->length = rpl::single(
 | |
| 		rpl::empty
 | |
| 	) | rpl::then(field->changes()) | rpl::map([=] {
 | |
| 		return int(field->getLastText().size());
 | |
| 	});
 | |
| 	const auto allowExceed = std::max(limit / 2, 9);
 | |
| 	field->setMaxLength(limit + allowExceed);
 | |
| 	const auto threshold = std::min(limit / 2, 9);
 | |
| 	auto warningText = state->length.value() | rpl::map([=](int count) {
 | |
| 		const auto left = limit - count;
 | |
| 		return (left < threshold) ? QString::number(left) : QString();
 | |
| 	});
 | |
| 	const auto warning = CreateChild<FlatLabel>(
 | |
| 		field.get(),
 | |
| 		std::move(warningText),
 | |
| 		st::defaultInputFieldLimit);
 | |
| 
 | |
| 	const auto maxSize = st::defaultInputFieldLimit.style.font->width(
 | |
| 		QString::number(-allowExceed));
 | |
| 	const auto add = std::max(maxSize - field->st().textMargins.right(), 0);
 | |
| 	if (add) {
 | |
| 		field->setAdditionalMargins({ 0, 0, add, 0 });
 | |
| 	}
 | |
| 	state->length.value() | rpl::map(
 | |
| 		rpl::mappers::_1 > limit
 | |
| 	) | rpl::start_with_next([=](bool exceeded) {
 | |
| 		warning->setTextColorOverride(exceeded
 | |
| 			? st::attentionButtonFg->c
 | |
| 			: std::optional<QColor>());
 | |
| 	}, warning->lifetime());
 | |
| 	rpl::combine(
 | |
| 		field->sizeValue(),
 | |
| 		warning->sizeValue()
 | |
| 	) | rpl::start_with_next([=] {
 | |
| 		// Baseline alignment.
 | |
| 		const auto top = field->st().textMargins.top()
 | |
| 			+ field->st().font->ascent
 | |
| 			- st::defaultInputFieldLimit.style.font->ascent;
 | |
| 		warning->moveToRight(0, top);
 | |
| 	}, warning->lifetime());
 | |
| 	warning->setAttribute(Qt::WA_TransparentForMouseEvents);
 | |
| }
 | |
| 
 | |
| } // namespace Ui
 |