// 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/widgets/popup_menu.h" #include "ui/text/text.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 #include #include #include #include #include #include #include #include #ifdef Q_OS_WIN #include #include #elif !defined DESKTOP_APP_DISABLE_X11_INTEGRATION // Q_OS_WIN #include "base/platform/linux/base_linux_xcb_utilities.h" #include #include #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); // 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 _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(Ui::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 = Ui::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::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(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 _checked = { { false, false } }; }; const std::vector &TagStartExpressions() { static auto cached = std::vector { { 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 &TagIndices() { static auto cached = std::map { { 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 *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 &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 *_tags = nullptr; const std::vector &_expressions; const std::map &_tagIndices; std::vector _items; int _currentTag = 0; int _currentFreeTag = 0; int _currentInternalLength = 0; int _currentAdjustedLength = 0; }; template 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, const QFont &font) { const auto factor = style::DevicePixelRatio(); const auto size = Emoji::GetSizeNormal(); const auto width = size + st::emojiPadding * factor * 2; const auto height = std::max(QFontMetrics(font).height() * factor, size); auto result = QTextImageFormat(); result.setWidth(width / factor); result.setHeight(height / factor); result.setName(emoji->toUrl()); result.setVerticalAlignment(QTextCharFormat::AlignBottom); return result; } void RemoveDocumentTags( const style::InputField &st, not_null 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(); auto bg = std::optional(); 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)); } 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 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 document, int changedPosition, int changedEnd, const TextWithTags::Tags &tags, Fn 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, }; 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 // 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 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 outer() const { return static_cast(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(); auto format = PrepareEmojiFormat(emoji, currentFormat.font()); ApplyTagFormat(format, currentFormat); cursor.insertText(kObjectReplacement, format); } void InsertCustomEmojiAtCursor( not_null 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::AlignBottom); 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()); } 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)); return result; }(); return result; } CustomEmojiObject::CustomEmojiObject(Factory factory, Fn paused) : _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()) { return static_cast(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 font = format.toCharFormat().font(); const auto height = std::min(QFontMetrics(font).height() * 1., size); if (!_skip) { const auto emoji = Ui::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 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 placeholder, const QString &value) : InputField( parent, st, mode, std::move(placeholder), { value, {} }) { } InputField::InputField( QWidget *parent, const style::InputField &st, Mode mode, rpl::producer placeholder, const TextWithTags &value) : RpWidget(parent) , _st(st) , _mode(mode) , _minHeight(st.heightMin) , _maxHeight(st.heightMax) , _inner(std::make_unique(this)) , _lastTextWithTags(value) , _placeholderFull(std::move(placeholder)) { _inner->setDocument(CreateChild(_inner.get(), _st)); _inner->setAcceptRichText(false); resize(_st.width, _minHeight); 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()); _defaultCharFormat = _inner->textCursor().charFormat(); updatePalette(); _inner->textCursor().setCharFormat(_defaultCharFormat); _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); 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 (!_lastTextWithTags.text.isEmpty()) { setTextWithTags(_lastTextWithTags, HistoryAction::Clear); } startBorderAnimation(); startPlaceholderAnimation(); finishAnimating(); } const rpl::variable &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 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(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 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::setInstantReplacesEnabled(rpl::producer enabled) { std::move( enabled ) | rpl::start_with_next([=](bool value) { _instantReplacesEnabled = value; }, lifetime()); } void InputField::setMarkdownReplacesEnabled(rpl::producer 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 processor) { _tagMimeProcessor = std::move(processor); } void InputField::setCustomEmojiFactory( CustomEmojiFactory factory, Fn paused) { _customEmojiObject = std::make_unique([=]( 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); 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(_st.textMargins + _st.placeholderMargins)); 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( _st.textMargins.left() + _st.placeholderMargins.left() + skipWidth, _st.textMargins.top() + _st.placeholderMargins.top() + _st.placeholderFont->ascent, _placeholder); } else { auto r = rect().marginsRemoved(_st.textMargins + _st.placeholderMargins); 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 (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); } QString InputField::getTextPart( int start, int end, TagList &outTagsList, bool &outTagsChanged, std::vector *outMarkdownTags) const { Expects((start == 0 && end < 0) || outMarkdownTags == nullptr); if (end >= 0 && end <= start) { outTagsChanged = !outTagsList.isEmpty(); outTagsList.clear(); return QString(); } 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; } 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; } } const auto emojiText = [&] { if (format.isImageFormat()) { const auto imageName = format.toImageFormat().name(); if (const auto emoji = Emoji::FromUrl(imageName)) { return emoji->text(); } } return format.property(kCustomEmojiText).toString(); }(); 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); } adjustedLength += (emojiText.size() - 1); if (!emojiText.isEmpty()) { result.append(emojiText); } 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; } 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("DAOpenSansRegular")); 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()) { 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); 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 = getTextPart( 0, -1, _lastTextWithTags.tags, tagsChanged, _markdownEnabled ? &_lastMarkdownTags : nullptr); //highlightMarkdown(); 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 InputField::document() { return _inner->document(); } not_null 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); 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); 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 InputField::rawTextEdit() { return _inner.get(); } not_null 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 oldPosition = textCursor().position(); const auto oldModifiers = e->modifiers(); const auto allowedModifiers = (enter && ctrl) ? (~Qt::ControlModifier) : (enter && shift) ? (~Qt::ShiftModifier) // Qt bug workaround https://bugreports.qt.io/browse/QTBUG-49771 : (backspace && Platform::IsX11()) ? (Qt::ControlModifier) : oldModifiers; const auto changeModifiers = (oldModifiers & ~allowedModifiers) != 0; if (changeModifiers) { e->setModifiers(oldModifiers & allowedModifiers); } _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 (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 = Ui::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 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::AlignBottom); 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()); 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; cursor.setCharFormat(_defaultCharFormat); _inner->setTextCursor(cursor); } 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(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 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 menu, not_null 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)); _borderAnimationStart = width() / 2; RpWidget::resizeEvent(e); checkContentHeight(); } void InputField::refreshPlaceholder(const QString &text) { const auto availableWidth = width() - _st.textMargins.left() - _st.textMargins.right() - _st.placeholderMargins.left() - _st.placeholderMargins.right(); 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()) { _placeholderPath.addText(0, QFontMetrics(placeholderFont).ascent(), placeholderFont, _placeholder); } } else { _placeholder = _st.placeholderFont->elided(text, availableWidth); } update(); } void InputField::setPlaceholder( rpl::producer placeholder, int afterSymbols) { _placeholderFull = std::move(placeholder); if (_placeholderAfterSymbols != afterSymbols) { _placeholderAfterSymbols = afterSymbols; startPlaceholderAnimation(); } } void InputField::setEditLinkCallback( Fn 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 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 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 document) { if (!document->pageSize().isNull()) { document->setPageSize(QSizeF(0, 0)); } } int FieldCharacterCount(not_null field) { // This method counts emoji properly. return field->document()->characterCount() - 1; } } // namespace Ui