295 lines
		
	
	
	
		
			7.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			295 lines
		
	
	
	
		
			7.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
// This file is part of Desktop App Toolkit,
 | 
						|
// a set of libraries for developing nice desktop applications.
 | 
						|
//
 | 
						|
// For license and copyright information please follow this link:
 | 
						|
// https://github.com/desktop-app/legal/blob/master/LEGAL
 | 
						|
//
 | 
						|
#pragma once
 | 
						|
 | 
						|
#include "base/timer.h"
 | 
						|
#include "base/unique_qptr.h"
 | 
						|
#include "ui/rp_widget.h"
 | 
						|
#include "ui/wrap/padding_wrap.h"
 | 
						|
#include "ui/text/text.h"
 | 
						|
#include "ui/click_handler.h"
 | 
						|
#include "ui/widgets/box_content_divider.h"
 | 
						|
#include "styles/style_widgets.h"
 | 
						|
 | 
						|
class QTouchEvent;
 | 
						|
 | 
						|
namespace Ui {
 | 
						|
 | 
						|
class PopupMenu;
 | 
						|
class BoxContentDivider;
 | 
						|
 | 
						|
class CrossFadeAnimation {
 | 
						|
public:
 | 
						|
	struct Data {
 | 
						|
		QImage full;
 | 
						|
		std::vector<int> lineWidths;
 | 
						|
		QPoint position;
 | 
						|
		style::align align;
 | 
						|
		style::font font;
 | 
						|
		style::margins margin;
 | 
						|
		int lineHeight = 0;
 | 
						|
		int lineAddTop = 0;
 | 
						|
	};
 | 
						|
 | 
						|
	CrossFadeAnimation(style::color bg, Data &&from, Data &&to);
 | 
						|
 | 
						|
	struct Part {
 | 
						|
		QPixmap snapshot;
 | 
						|
		QPoint position;
 | 
						|
	};
 | 
						|
	void addLine(Part was, Part now);
 | 
						|
 | 
						|
	void paintFrame(QPainter &p, float64 dt);
 | 
						|
	void paintFrame(
 | 
						|
		QPainter &p,
 | 
						|
		float64 positionReady,
 | 
						|
		float64 alphaWas,
 | 
						|
		float64 alphaNow);
 | 
						|
 | 
						|
private:
 | 
						|
	struct Line {
 | 
						|
		Line(Part was, Part now) : was(std::move(was)), now(std::move(now)) {
 | 
						|
		}
 | 
						|
		Part was;
 | 
						|
		Part now;
 | 
						|
	};
 | 
						|
	void paintLine(
 | 
						|
		QPainter &p,
 | 
						|
		const Line &line,
 | 
						|
		float64 positionReady,
 | 
						|
		float64 alphaWas,
 | 
						|
		float64 alphaNow);
 | 
						|
 | 
						|
	style::color _bg;
 | 
						|
	QList<Line> _lines;
 | 
						|
 | 
						|
};
 | 
						|
 | 
						|
class LabelSimple : public RpWidget {
 | 
						|
public:
 | 
						|
	LabelSimple(
 | 
						|
		QWidget *parent,
 | 
						|
		const style::LabelSimple &st = st::defaultLabelSimple,
 | 
						|
		const QString &value = QString());
 | 
						|
 | 
						|
	// This method also resizes the label.
 | 
						|
	void setText(const QString &newText, bool *outTextChanged = nullptr);
 | 
						|
 | 
						|
protected:
 | 
						|
	void paintEvent(QPaintEvent *e) override;
 | 
						|
 | 
						|
private:
 | 
						|
	QString _fullText;
 | 
						|
	int _fullTextWidth;
 | 
						|
 | 
						|
	QString _text;
 | 
						|
	int _textWidth;
 | 
						|
 | 
						|
	const style::LabelSimple &_st;
 | 
						|
 | 
						|
};
 | 
						|
 | 
						|
class FlatLabel : public RpWidget, public ClickHandlerHost {
 | 
						|
public:
 | 
						|
	FlatLabel(
 | 
						|
		QWidget *parent,
 | 
						|
		const style::FlatLabel &st = st::defaultFlatLabel,
 | 
						|
		const style::PopupMenu &stMenu = st::defaultPopupMenu);
 | 
						|
 | 
						|
	FlatLabel(
 | 
						|
		QWidget *parent,
 | 
						|
		const QString &text,
 | 
						|
		const style::FlatLabel &st = st::defaultFlatLabel,
 | 
						|
		const style::PopupMenu &stMenu = st::defaultPopupMenu);
 | 
						|
 | 
						|
	FlatLabel(
 | 
						|
		QWidget *parent,
 | 
						|
		rpl::producer<QString> &&text,
 | 
						|
		const style::FlatLabel &st = st::defaultFlatLabel,
 | 
						|
		const style::PopupMenu &stMenu = st::defaultPopupMenu);
 | 
						|
	FlatLabel(
 | 
						|
		QWidget *parent,
 | 
						|
		rpl::producer<TextWithEntities> &&text,
 | 
						|
		const style::FlatLabel &st = st::defaultFlatLabel,
 | 
						|
		const style::PopupMenu &stMenu = st::defaultPopupMenu);
 | 
						|
 | 
						|
	[[nodiscard]] const style::FlatLabel &st() const {
 | 
						|
		return _st;
 | 
						|
	}
 | 
						|
 | 
						|
	void setOpacity(float64 o);
 | 
						|
	void setTextColorOverride(std::optional<QColor> color);
 | 
						|
 | 
						|
	void setText(const QString &text);
 | 
						|
	void setMarkedText(
 | 
						|
		const TextWithEntities &textWithEntities,
 | 
						|
		const std::any &context = {});
 | 
						|
	void setSelectable(bool selectable);
 | 
						|
	void setDoubleClickSelectsParagraph(bool doubleClickSelectsParagraph);
 | 
						|
	void setContextCopyText(const QString ©Text);
 | 
						|
	void setBreakEverywhere(bool breakEverywhere);
 | 
						|
	void setTryMakeSimilarLines(bool tryMakeSimilarLines);
 | 
						|
	enum class WhichAnimationsPaused {
 | 
						|
		None,
 | 
						|
		CustomEmoji,
 | 
						|
		Spoiler,
 | 
						|
		All,
 | 
						|
	};
 | 
						|
	void setAnimationsPausedCallback(Fn<WhichAnimationsPaused()> callback) {
 | 
						|
		_animationsPausedCallback = std::move(callback);
 | 
						|
	}
 | 
						|
 | 
						|
	[[nodiscard]] int textMaxWidth() const;
 | 
						|
	int naturalWidth() const override;
 | 
						|
	QMargins getMargins() const override;
 | 
						|
 | 
						|
	void setLink(uint16 index, const ClickHandlerPtr &lnk);
 | 
						|
	void setLinksTrusted();
 | 
						|
 | 
						|
	using ClickHandlerFilter = Fn<bool(const ClickHandlerPtr&, Qt::MouseButton)>;
 | 
						|
	void setClickHandlerFilter(ClickHandlerFilter &&filter);
 | 
						|
	void overrideLinkClickHandler(Fn<void()> handler);
 | 
						|
	void overrideLinkClickHandler(Fn<void(QString url)> handler);
 | 
						|
 | 
						|
	struct ContextMenuRequest {
 | 
						|
		not_null<PopupMenu*> menu;
 | 
						|
		ClickHandlerPtr link;
 | 
						|
		bool hasSelection = false;
 | 
						|
		bool uponSelection = false;
 | 
						|
		bool fullSelection = false;
 | 
						|
	};
 | 
						|
	void setContextMenuHook(Fn<void(ContextMenuRequest)> hook);
 | 
						|
	void fillContextMenu(ContextMenuRequest request);
 | 
						|
 | 
						|
	// ClickHandlerHost interface
 | 
						|
	void clickHandlerActiveChanged(const ClickHandlerPtr &action, bool active) override;
 | 
						|
	void clickHandlerPressedChanged(const ClickHandlerPtr &action, bool pressed) override;
 | 
						|
 | 
						|
	[[nodiscard]] CrossFadeAnimation::Data crossFadeData(
 | 
						|
		style::color bg,
 | 
						|
		QPoint basePosition = QPoint());
 | 
						|
 | 
						|
	static std::unique_ptr<CrossFadeAnimation> CrossFade(
 | 
						|
		not_null<FlatLabel*> from,
 | 
						|
		not_null<FlatLabel*> to,
 | 
						|
		style::color bg,
 | 
						|
		QPoint fromPosition = QPoint(),
 | 
						|
		QPoint toPosition = QPoint());
 | 
						|
 | 
						|
protected:
 | 
						|
	void paintEvent(QPaintEvent *e) override;
 | 
						|
	void mouseMoveEvent(QMouseEvent *e) override;
 | 
						|
	void mousePressEvent(QMouseEvent *e) override;
 | 
						|
	void mouseReleaseEvent(QMouseEvent *e) override;
 | 
						|
	void mouseDoubleClickEvent(QMouseEvent *e) override;
 | 
						|
	void enterEventHook(QEnterEvent *e) override;
 | 
						|
	void leaveEventHook(QEvent *e) override;
 | 
						|
	void focusOutEvent(QFocusEvent *e) override;
 | 
						|
	void focusInEvent(QFocusEvent *e) override;
 | 
						|
	void keyPressEvent(QKeyEvent *e) override;
 | 
						|
	void contextMenuEvent(QContextMenuEvent *e) override;
 | 
						|
	bool eventHook(QEvent *e) override; // calls touchEvent when necessary
 | 
						|
	void touchEvent(QTouchEvent *e);
 | 
						|
 | 
						|
	int resizeGetHeight(int newWidth) override;
 | 
						|
 | 
						|
	void copySelectedText();
 | 
						|
	void copyContextText();
 | 
						|
 | 
						|
	void touchSelect();
 | 
						|
 | 
						|
	void executeDrag();
 | 
						|
 | 
						|
private:
 | 
						|
	void init();
 | 
						|
	void textUpdated();
 | 
						|
 | 
						|
	Text::StateResult dragActionUpdate();
 | 
						|
	Text::StateResult dragActionStart(const QPoint &p, Qt::MouseButton button);
 | 
						|
	Text::StateResult dragActionFinish(const QPoint &p, Qt::MouseButton button);
 | 
						|
	void updateHover(const Text::StateResult &state);
 | 
						|
	Text::StateResult getTextState(const QPoint &m) const;
 | 
						|
	void refreshCursor(bool uponSymbol);
 | 
						|
 | 
						|
	int countTextWidth() const;
 | 
						|
	int countTextHeight(int textWidth);
 | 
						|
	void refreshSize();
 | 
						|
 | 
						|
	enum class ContextMenuReason {
 | 
						|
		FromEvent,
 | 
						|
		FromTouch,
 | 
						|
	};
 | 
						|
	void showContextMenu(QContextMenuEvent *e, ContextMenuReason reason);
 | 
						|
 | 
						|
	Text::String _text;
 | 
						|
	const style::FlatLabel &_st;
 | 
						|
	const style::PopupMenu &_stMenu;
 | 
						|
	std::optional<QColor> _textColorOverride;
 | 
						|
	float64 _opacity = 1.;
 | 
						|
 | 
						|
	int _allowedWidth = 0;
 | 
						|
	int _textWidth = 0;
 | 
						|
	int _fullTextHeight = 0;
 | 
						|
	bool _breakEverywhere = false;
 | 
						|
	bool _tryMakeSimilarLines = false;
 | 
						|
 | 
						|
	style::cursor _cursor = style::cur_default;
 | 
						|
	bool _selectable = false;
 | 
						|
	TextSelection _selection, _savedSelection;
 | 
						|
	TextSelectType _selectionType = TextSelectType::Letters;
 | 
						|
	bool _doubleClickSelectsParagraph = false;
 | 
						|
 | 
						|
	enum DragAction {
 | 
						|
		NoDrag = 0x00,
 | 
						|
		PrepareDrag = 0x01,
 | 
						|
		Dragging = 0x02,
 | 
						|
		Selecting = 0x04,
 | 
						|
	};
 | 
						|
	DragAction _dragAction = NoDrag;
 | 
						|
	QPoint _dragStartPosition;
 | 
						|
	uint16 _dragSymbol = 0;
 | 
						|
	bool _dragWasInactive = false;
 | 
						|
 | 
						|
	QPoint _lastMousePos;
 | 
						|
 | 
						|
	QPoint _trippleClickPoint;
 | 
						|
	base::Timer _trippleClickTimer;
 | 
						|
 | 
						|
	base::unique_qptr<PopupMenu> _contextMenu;
 | 
						|
	Fn<void(ContextMenuRequest)> _contextMenuHook;
 | 
						|
	QString _contextCopyText;
 | 
						|
 | 
						|
	ClickHandlerFilter _clickHandlerFilter;
 | 
						|
	Fn<WhichAnimationsPaused()> _animationsPausedCallback;
 | 
						|
 | 
						|
	// text selection and context menu by touch support (at least Windows Surface tablets)
 | 
						|
	bool _touchSelect = false;
 | 
						|
	bool _touchInProgress = false;
 | 
						|
	QPoint _touchStart, _touchPrevPos, _touchPos;
 | 
						|
	base::Timer _touchSelectTimer;
 | 
						|
 | 
						|
};
 | 
						|
 | 
						|
class DividerLabel : public PaddingWrap<> {
 | 
						|
public:
 | 
						|
	DividerLabel(
 | 
						|
		QWidget *parent,
 | 
						|
		object_ptr<RpWidget> &&child,
 | 
						|
		const style::margins &padding,
 | 
						|
		RectParts parts = RectPart::Top | RectPart::Bottom);
 | 
						|
 | 
						|
	int naturalWidth() const override;
 | 
						|
 | 
						|
protected:
 | 
						|
	void resizeEvent(QResizeEvent *e) override;
 | 
						|
 | 
						|
private:
 | 
						|
	object_ptr<BoxContentDivider> _background;
 | 
						|
 | 
						|
};
 | 
						|
 | 
						|
} // namespace Ui
 |