230 lines
		
	
	
	
		
			6.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			230 lines
		
	
	
	
		
			6.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| This file is part of Telegram Desktop,
 | |
| the official desktop version of Telegram messaging app, see https://telegram.org
 | |
| 
 | |
| Telegram Desktop is free software: you can redistribute it and/or modify
 | |
| it under the terms of the GNU General Public License as published by
 | |
| the Free Software Foundation, either version 3 of the License, or
 | |
| (at your option) any later version.
 | |
| 
 | |
| It is distributed in the hope that it will be useful,
 | |
| but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | |
| GNU General Public License for more details.
 | |
| 
 | |
| In addition, as a special exception, the copyright holders give permission
 | |
| to link the code of portions of this program with the OpenSSL library.
 | |
| 
 | |
| Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | |
| Copyright (c) 2014-2016 John Preston, https://desktop.telegram.org
 | |
| */
 | |
| #pragma once
 | |
| 
 | |
| #include "animation.h"
 | |
| 
 | |
| class UserData;
 | |
| static UserData * const LookingUpInlineBot = SharedMemoryLocation<UserData, 0>();
 | |
| 
 | |
| class FlatTextarea : public QTextEdit {
 | |
| 	Q_OBJECT
 | |
| 	T_WIDGET
 | |
| 
 | |
| public:
 | |
| 
 | |
| 	FlatTextarea(QWidget *parent, const style::flatTextarea &st, const QString &ph = QString(), const QString &val = QString());
 | |
| 
 | |
| 	void setMaxLength(int32 maxLength);
 | |
| 	void setMinHeight(int32 minHeight);
 | |
| 	void setMaxHeight(int32 maxHeight);
 | |
| 
 | |
| 	const QString &getLastText() const {
 | |
| 		return _oldtext;
 | |
| 	}
 | |
| 
 | |
| 	void setPlaceholder(const QString &ph, int32 afterSymbols = 0);
 | |
| 	void updatePlaceholder();
 | |
| 	void finishPlaceholder();
 | |
| 
 | |
| 	QRect getTextRect() const;
 | |
| 	int32 fakeMargin() const;
 | |
| 
 | |
| 	void step_appearance(float64 ms, bool timer);
 | |
| 
 | |
| 	QSize sizeHint() const override;
 | |
| 	QSize minimumSizeHint() const override;
 | |
| 
 | |
| 	EmojiPtr getSingleEmoji() const;
 | |
| 	QString getMentionHashtagBotCommandPart(bool &start) const;
 | |
| 
 | |
| 	// Get the current inline bot and request string for it.
 | |
| 	// The *outInlineBot can be filled by LookingUpInlineBot shared ptr.
 | |
| 	// In that case the caller should lookup the bot by *outInlineBotUsername.
 | |
| 	QString getInlineBotQuery(UserData **outInlineBot, QString *outInlineBotUsername) const;
 | |
| 
 | |
| 	void removeSingleEmoji();
 | |
| 	bool hasText() const;
 | |
| 
 | |
| 	bool isUndoAvailable() const;
 | |
| 	bool isRedoAvailable() const;
 | |
| 
 | |
| 	void parseLinks();
 | |
| 	QStringList linksList() const;
 | |
| 
 | |
| 	void insertFromMimeData(const QMimeData *source) override;
 | |
| 
 | |
| 	QMimeData *createMimeDataFromSelection() const override;
 | |
| 
 | |
| 	enum class SubmitSettings {
 | |
| 		None,
 | |
| 		Enter,
 | |
| 		CtrlEnter,
 | |
| 		Both,
 | |
| 	};
 | |
| 	void setSubmitSettings(SubmitSettings settings);
 | |
| 
 | |
| 	void setTextFast(const QString &text, bool clearUndoHistory = true);
 | |
| 
 | |
| 	struct Tag {
 | |
| 		int offset, length;
 | |
| 		QString id;
 | |
| 	};
 | |
| 	using TagList = QVector<Tag>;
 | |
| 	const TagList &getLastTags() const {
 | |
| 		return _oldtags;
 | |
| 	}
 | |
| 	void insertTag(const QString &text, QString tagId = QString());
 | |
| 
 | |
| 	// If you need to make some preparations of tags before putting them to QMimeData
 | |
| 	// (and then to clipboard or to drag-n-drop object), here is a strategy for that.
 | |
| 	class TagMimeProcessor {
 | |
| 	public:
 | |
| 		virtual QString mimeTagFromTag(const QString &tagId) = 0;
 | |
| 		virtual QString tagFromMimeTag(const QString &mimeTag) = 0;
 | |
| 	};
 | |
| 	void setTagMimeProcessor(std_::unique_ptr<TagMimeProcessor> &&processor);
 | |
| 
 | |
| public slots:
 | |
| 
 | |
| 	void onTouchTimer();
 | |
| 
 | |
| 	void onDocumentContentsChange(int position, int charsRemoved, int charsAdded);
 | |
| 	void onDocumentContentsChanged();
 | |
| 
 | |
| 	void onUndoAvailable(bool avail);
 | |
| 	void onRedoAvailable(bool avail);
 | |
| 
 | |
| signals:
 | |
| 
 | |
| 	void resized();
 | |
| 	void changed();
 | |
| 	void submitted(bool ctrlShiftEnter);
 | |
| 	void cancelled();
 | |
| 	void tabbed();
 | |
| 	void spacedReturnedPasted();
 | |
| 	void linksChanged();
 | |
| 
 | |
| protected:
 | |
| 
 | |
| 	bool viewportEvent(QEvent *e) override;
 | |
| 	void touchEvent(QTouchEvent *e);
 | |
| 	void paintEvent(QPaintEvent *e) override;
 | |
| 	void focusInEvent(QFocusEvent *e) override;
 | |
| 	void focusOutEvent(QFocusEvent *e) override;
 | |
| 	void keyPressEvent(QKeyEvent *e) override;
 | |
| 	void resizeEvent(QResizeEvent *e) override;
 | |
| 	void mousePressEvent(QMouseEvent *e) override;
 | |
| 	void dropEvent(QDropEvent *e) override;
 | |
| 	void contextMenuEvent(QContextMenuEvent *e) override;
 | |
| 
 | |
| 	virtual void correctValue(const QString &was, QString &now, TagList &nowTags) {
 | |
| 	}
 | |
| 
 | |
| 	void insertEmoji(EmojiPtr emoji, QTextCursor c);
 | |
| 
 | |
| 	QVariant loadResource(int type, const QUrl &name) override;
 | |
| 
 | |
| 	void checkContentHeight();
 | |
| 
 | |
| private:
 | |
| 
 | |
| 	// "start" and "end" are in coordinates of text where emoji are replaced by ObjectReplacementCharacter.
 | |
| 	// If "end" = -1 means get text till the end. "outTagsList" and "outTagsChanged" may be nullptr.
 | |
| 	QString getText(int start, int end, TagList *outTagsList, bool *outTagsChanged) const;
 | |
| 
 | |
| 	void getSingleEmojiFragment(QString &text, QTextFragment &fragment) const;
 | |
| 
 | |
| 	// After any characters added we must postprocess them. This includes:
 | |
| 	// 1. Replacing font family to semibold for ~ characters, if we used Open Sans 13px.
 | |
| 	// 2. Replacing font family from semibold for all non-~ characters, if we used ...
 | |
| 	// 3. Replacing emoji code sequences by ObjectReplacementCharacters with emoji pics.
 | |
| 	// 4. Interrupting tags in which the text was inserted by any char except a letter.
 | |
| 	// 5. Applying tags from "_insertedTags" in case we pasted text with tags, not just text.
 | |
| 	// Rule 4 applies only if we inserted chars not in the middle of a tag (but at the end).
 | |
| 	void processFormatting(int changedPosition, int changedEnd);
 | |
| 
 | |
| 	bool heightAutoupdated();
 | |
| 
 | |
| 	int _minHeight = -1; // < 0 - no autosize
 | |
| 	int _maxHeight = -1;
 | |
| 	int _maxLength = -1;
 | |
| 	SubmitSettings _submitSettings = SubmitSettings::Enter;
 | |
| 
 | |
| 	QString _ph, _phelided, _oldtext;
 | |
| 	TagList _oldtags;
 | |
| 	int _phAfter = 0;
 | |
| 	bool _phVisible;
 | |
| 	anim::ivalue a_phLeft;
 | |
| 	anim::fvalue a_phAlpha;
 | |
| 	anim::cvalue a_phColor;
 | |
| 	Animation _a_appearance;
 | |
| 
 | |
| 	// Tags list which we should apply while setText() call or insert from mime data.
 | |
| 	TagList _insertedTags;
 | |
| 
 | |
| 	// Override insert position and charsAdded from complex text editing
 | |
| 	// (like drag-n-drop in the same text edit field).
 | |
| 	int _realInsertPosition = -1;
 | |
| 	int _realCharsAdded = 0;
 | |
| 
 | |
| 	std_::unique_ptr<TagMimeProcessor> _tagMimeProcessor;
 | |
| 
 | |
| 	style::flatTextarea _st;
 | |
| 
 | |
| 	bool _undoAvailable = false;
 | |
| 	bool _redoAvailable = false;
 | |
| 	bool _inDrop = false;
 | |
| 	bool _inHeightCheck = false;
 | |
| 
 | |
| 	int _fakeMargin = 0;
 | |
| 
 | |
| 	QTimer _touchTimer;
 | |
| 	bool _touchPress = false;
 | |
| 	bool _touchRightButton = false;
 | |
| 	bool _touchMove = false;
 | |
| 	QPoint _touchStart;
 | |
| 
 | |
| 	bool _correcting = false;
 | |
| 
 | |
| 	struct LinkRange {
 | |
| 		int start;
 | |
| 		int length;
 | |
| 	};
 | |
| 	friend bool operator==(const LinkRange &a, const LinkRange &b);
 | |
| 	friend bool operator!=(const LinkRange &a, const LinkRange &b);
 | |
| 	using LinkRanges = QVector<LinkRange>;
 | |
| 	LinkRanges _links;
 | |
| };
 | |
| 
 | |
| inline bool operator==(const FlatTextarea::Tag &a, const FlatTextarea::Tag &b) {
 | |
| 	return (a.offset == b.offset) && (a.length == b.length) && (a.id == b.id);
 | |
| }
 | |
| inline bool operator!=(const FlatTextarea::Tag &a, const FlatTextarea::Tag &b) {
 | |
| 	return !(a == b);
 | |
| }
 | |
| 
 | |
| inline bool operator==(const FlatTextarea::LinkRange &a, const FlatTextarea::LinkRange &b) {
 | |
| 	return (a.start == b.start) && (a.length == b.length);
 | |
| }
 | |
| inline bool operator!=(const FlatTextarea::LinkRange &a, const FlatTextarea::LinkRange &b) {
 | |
| 	return !(a == b);
 | |
| }
 | 
