443 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			443 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| This file is part of Telegram Desktop,
 | |
| the official desktop application for the Telegram messaging service.
 | |
| 
 | |
| For license and copyright information please follow this link:
 | |
| https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
 | |
| */
 | |
| #pragma once
 | |
| 
 | |
| #include "api/api_common.h"
 | |
| #include "chat_helpers/compose/compose_features.h"
 | |
| #include "ui/rp_widget.h"
 | |
| #include "ui/effects/animations.h"
 | |
| #include "ui/effects/message_sending_animation_common.h"
 | |
| #include "ui/effects/panel_animation.h"
 | |
| #include "ui/cached_round_corners.h"
 | |
| #include "mtproto/sender.h"
 | |
| #include "base/object_ptr.h"
 | |
| 
 | |
| namespace InlineBots {
 | |
| struct ResultSelected;
 | |
| } // namespace InlineBots
 | |
| 
 | |
| namespace Main {
 | |
| class Session;
 | |
| } // namespace Main
 | |
| 
 | |
| namespace Ui {
 | |
| class PlainShadow;
 | |
| class PopupMenu;
 | |
| class ScrollArea;
 | |
| class SettingsSlider;
 | |
| class FlatLabel;
 | |
| class BoxContent;
 | |
| class TabbedSearch;
 | |
| } // namespace Ui
 | |
| 
 | |
| namespace SendMenu {
 | |
| enum class Type;
 | |
| } // namespace SendMenu
 | |
| 
 | |
| namespace style {
 | |
| struct EmojiPan;
 | |
| } // namespace style
 | |
| 
 | |
| namespace ChatHelpers {
 | |
| 
 | |
| class Show;
 | |
| class EmojiListWidget;
 | |
| class StickersListWidget;
 | |
| class GifsListWidget;
 | |
| enum class PauseReason;
 | |
| 
 | |
| enum class SelectorTab {
 | |
| 	Emoji,
 | |
| 	Stickers,
 | |
| 	Gifs,
 | |
| 	Masks,
 | |
| };
 | |
| 
 | |
| struct FileChosen {
 | |
| 	not_null<DocumentData*> document;
 | |
| 	Api::SendOptions options;
 | |
| 	Ui::MessageSendingAnimationFrom messageSendingFrom;
 | |
| };
 | |
| 
 | |
| struct PhotoChosen {
 | |
| 	not_null<PhotoData*> photo;
 | |
| 	Api::SendOptions options;
 | |
| };
 | |
| 
 | |
| struct EmojiChosen {
 | |
| 	EmojiPtr emoji;
 | |
| 	Ui::MessageSendingAnimationFrom messageSendingFrom;
 | |
| };
 | |
| 
 | |
| using InlineChosen = InlineBots::ResultSelected;
 | |
| 
 | |
| enum class TabbedSelectorMode {
 | |
| 	Full,
 | |
| 	EmojiOnly,
 | |
| 	MediaEditor,
 | |
| 	EmojiStatus,
 | |
| 	BackgroundEmoji,
 | |
| 	FullReactions,
 | |
| 	RecentReactions,
 | |
| };
 | |
| 
 | |
| struct TabbedSelectorDescriptor {
 | |
| 	std::shared_ptr<Show> show;
 | |
| 	const style::EmojiPan &st;
 | |
| 	PauseReason level = {};
 | |
| 	TabbedSelectorMode mode = TabbedSelectorMode::Full;
 | |
| 	Fn<QColor()> customTextColor;
 | |
| 	ComposeFeatures features;
 | |
| };
 | |
| 
 | |
| [[nodiscard]] std::unique_ptr<Ui::TabbedSearch> MakeSearch(
 | |
| 	not_null<Ui::RpWidget*> parent,
 | |
| 	const style::EmojiPan &st,
 | |
| 	Fn<void(std::vector<QString>&&)> callback,
 | |
| 	not_null<Main::Session*> session,
 | |
| 	bool statusCategories = false,
 | |
| 	bool profilePhotoCategories = false);
 | |
| 
 | |
| class TabbedSelector : public Ui::RpWidget {
 | |
| public:
 | |
| 	static constexpr auto kPickCustomTimeId = -1;
 | |
| 	using Mode = TabbedSelectorMode;
 | |
| 	enum class Action {
 | |
| 		Update,
 | |
| 		Cancel,
 | |
| 	};
 | |
| 
 | |
| 	TabbedSelector(
 | |
| 		QWidget *parent,
 | |
| 		std::shared_ptr<Show> show,
 | |
| 		PauseReason level,
 | |
| 		Mode mode = Mode::Full);
 | |
| 	TabbedSelector(
 | |
| 		QWidget *parent,
 | |
| 		TabbedSelectorDescriptor &&descriptor);
 | |
| 	~TabbedSelector();
 | |
| 
 | |
| 	[[nodiscard]] const style::EmojiPan &st() const;
 | |
| 	[[nodiscard]] Main::Session &session() const;
 | |
| 	[[nodiscard]] PauseReason level() const;
 | |
| 
 | |
| 	[[nodiscard]] rpl::producer<EmojiChosen> emojiChosen() const;
 | |
| 	[[nodiscard]] rpl::producer<FileChosen> customEmojiChosen() const;
 | |
| 	[[nodiscard]] rpl::producer<FileChosen> fileChosen() const;
 | |
| 	[[nodiscard]] rpl::producer<PhotoChosen> photoChosen() const;
 | |
| 	[[nodiscard]] rpl::producer<InlineChosen> inlineResultChosen() const;
 | |
| 
 | |
| 	[[nodiscard]] rpl::producer<> cancelled() const;
 | |
| 	[[nodiscard]] rpl::producer<> checkForHide() const;
 | |
| 	[[nodiscard]] rpl::producer<> slideFinished() const;
 | |
| 	[[nodiscard]] rpl::producer<> contextMenuRequested() const;
 | |
| 	[[nodiscard]] rpl::producer<Action> choosingStickerUpdated() const;
 | |
| 
 | |
| 	void setAllowEmojiWithoutPremium(bool allow);
 | |
| 	void setRoundRadius(int radius);
 | |
| 	void refreshStickers();
 | |
| 	void setCurrentPeer(PeerData *peer);
 | |
| 	void provideRecentEmoji(
 | |
| 		const std::vector<DocumentId> &customRecentList);
 | |
| 
 | |
| 	void hideFinished();
 | |
| 	void showStarted();
 | |
| 	void beforeHiding();
 | |
| 	void afterShown();
 | |
| 
 | |
| 	[[nodiscard]] int marginTop() const;
 | |
| 	[[nodiscard]] int marginBottom() const;
 | |
| 	[[nodiscard]] int scrollTop() const;
 | |
| 	[[nodiscard]] int scrollBottom() const;
 | |
| 
 | |
| 	bool preventAutoHide() const;
 | |
| 	bool isSliding() const {
 | |
| 		return _a_slide.animating();
 | |
| 	}
 | |
| 	bool hasMenu() const;
 | |
| 
 | |
| 	void setAfterShownCallback(Fn<void(SelectorTab)> callback) {
 | |
| 		_afterShownCallback = std::move(callback);
 | |
| 	}
 | |
| 	void setBeforeHidingCallback(Fn<void(SelectorTab)> callback) {
 | |
| 		_beforeHidingCallback = std::move(callback);
 | |
| 	}
 | |
| 
 | |
| 	void showMenuWithType(SendMenu::Type type);
 | |
| 	void setDropDown(bool dropDown);
 | |
| 
 | |
| 	// Float player interface.
 | |
| 	bool floatPlayerHandleWheelEvent(QEvent *e);
 | |
| 	QRect floatPlayerAvailableRect() const;
 | |
| 
 | |
| 	auto showRequests() const {
 | |
| 		return _showRequests.events();
 | |
| 	}
 | |
| 
 | |
| 	class Inner;
 | |
| 	class InnerFooter;
 | |
| 
 | |
| protected:
 | |
| 	void paintEvent(QPaintEvent *e) override;
 | |
| 	void resizeEvent(QResizeEvent *e) override;
 | |
| 
 | |
| private:
 | |
| 	class Tab {
 | |
| 	public:
 | |
| 		Tab(SelectorTab type, int index, object_ptr<Inner> widget);
 | |
| 
 | |
| 		object_ptr<Inner> takeWidget();
 | |
| 		void returnWidget(object_ptr<Inner> widget);
 | |
| 
 | |
| 		[[nodiscard]] SelectorTab type() const {
 | |
| 			return _type;
 | |
| 		}
 | |
| 		[[nodiscard]] int index() const {
 | |
| 			return _index;
 | |
| 		}
 | |
| 		[[nodiscard]] Inner *widget() const {
 | |
| 			return _weak;
 | |
| 		}
 | |
| 		[[nodiscard]] bool hasFooter() const {
 | |
| 			return _footer != nullptr;
 | |
| 		}
 | |
| 		[[nodiscard]] not_null<InnerFooter*> footer() const {
 | |
| 			return _footer;
 | |
| 		}
 | |
| 
 | |
| 		void saveScrollTop();
 | |
| 		void saveScrollTop(int scrollTop) {
 | |
| 			_scrollTop = scrollTop;
 | |
| 		}
 | |
| 		[[nodiscard]] int getScrollTop() const {
 | |
| 			return _scrollTop;
 | |
| 		}
 | |
| 
 | |
| 	private:
 | |
| 		const SelectorTab _type;
 | |
| 		const int _index;
 | |
| 		object_ptr<Inner> _widget = { nullptr };
 | |
| 		QPointer<Inner> _weak;
 | |
| 		object_ptr<InnerFooter> _footer;
 | |
| 		int _scrollTop = 0;
 | |
| 
 | |
| 	};
 | |
| 
 | |
| 	bool full() const;
 | |
| 	bool mediaEditor() const;
 | |
| 	bool tabbed() const;
 | |
| 	bool hasEmojiTab() const;
 | |
| 	bool hasStickersTab() const;
 | |
| 	bool hasGifsTab() const;
 | |
| 	bool hasMasksTab() const;
 | |
| 	Tab createTab(SelectorTab type, int index);
 | |
| 
 | |
| 	void paintSlideFrame(QPainter &p);
 | |
| 	void paintBgRoundedPart(QPainter &p);
 | |
| 	void paintContent(QPainter &p);
 | |
| 
 | |
| 	void checkRestrictedPeer();
 | |
| 	bool isRestrictedView();
 | |
| 	void updateRestrictedLabelGeometry();
 | |
| 	void updateScrollGeometry(QSize oldSize);
 | |
| 	void updateFooterGeometry();
 | |
| 	void handleScroll();
 | |
| 
 | |
| 	QImage grabForAnimation();
 | |
| 
 | |
| 	void scrollToY(int y);
 | |
| 
 | |
| 	void showAll();
 | |
| 	void hideForSliding();
 | |
| 
 | |
| 	SelectorTab typeByIndex(int index) const;
 | |
| 	int indexByType(SelectorTab type) const;
 | |
| 
 | |
| 	bool hasSectionIcons() const;
 | |
| 	void setWidgetToScrollArea();
 | |
| 	void createTabsSlider();
 | |
| 	void fillTabsSliderSections();
 | |
| 	void updateTabsSliderGeometry();
 | |
| 	void switchTab();
 | |
| 
 | |
| 	not_null<Tab*> getTab(int index);
 | |
| 	not_null<const Tab*> getTab(int index) const;
 | |
| 	not_null<Tab*> currentTab();
 | |
| 	not_null<const Tab*> currentTab() const;
 | |
| 
 | |
| 	not_null<EmojiListWidget*> emoji() const;
 | |
| 	not_null<StickersListWidget*> stickers() const;
 | |
| 	not_null<GifsListWidget*> gifs() const;
 | |
| 	not_null<StickersListWidget*> masks() const;
 | |
| 
 | |
| 	const style::EmojiPan &_st;
 | |
| 	const ComposeFeatures _features;
 | |
| 	const std::shared_ptr<Show> _show;
 | |
| 	const PauseReason _level = {};
 | |
| 	const Fn<QColor()> _customTextColor;
 | |
| 
 | |
| 	Mode _mode = Mode::Full;
 | |
| 	int _roundRadius = 0;
 | |
| 	int _footerTop = 0;
 | |
| 	bool _noFooter = false;
 | |
| 	Ui::CornersPixmaps _panelRounding;
 | |
| 	Ui::CornersPixmaps _categoriesRounding;
 | |
| 	PeerData *_currentPeer = nullptr;
 | |
| 
 | |
| 	class SlideAnimation;
 | |
| 	std::unique_ptr<SlideAnimation> _slideAnimation;
 | |
| 	Ui::Animations::Simple _a_slide;
 | |
| 
 | |
| 	object_ptr<Ui::SettingsSlider> _tabsSlider = { nullptr };
 | |
| 	object_ptr<Ui::PlainShadow> _topShadow;
 | |
| 	object_ptr<Ui::PlainShadow> _bottomShadow;
 | |
| 	object_ptr<Ui::ScrollArea> _scroll;
 | |
| 	object_ptr<Ui::FlatLabel> _restrictedLabel = { nullptr };
 | |
| 	std::vector<Tab> _tabs;
 | |
| 	SelectorTab _currentTabType = SelectorTab::Emoji;
 | |
| 
 | |
| 	const bool _hasEmojiTab;
 | |
| 	const bool _hasStickersTab;
 | |
| 	const bool _hasGifsTab;
 | |
| 	const bool _hasMasksTab;
 | |
| 	const bool _tabbed;
 | |
| 	bool _dropDown = false;
 | |
| 
 | |
| 	base::unique_qptr<Ui::PopupMenu> _menu;
 | |
| 
 | |
| 	Fn<void(SelectorTab)> _afterShownCallback;
 | |
| 	Fn<void(SelectorTab)> _beforeHidingCallback;
 | |
| 
 | |
| 	rpl::event_stream<> _showRequests;
 | |
| 	rpl::event_stream<> _slideFinished;
 | |
| 
 | |
| };
 | |
| 
 | |
| class TabbedSelector::Inner : public Ui::RpWidget {
 | |
| public:
 | |
| 	Inner(
 | |
| 		QWidget *parent,
 | |
| 		std::shared_ptr<Show> show,
 | |
| 		PauseReason level);
 | |
| 	Inner(
 | |
| 		QWidget *parent,
 | |
| 		const style::EmojiPan &st,
 | |
| 		std::shared_ptr<Show> show,
 | |
| 		Fn<bool()> paused);
 | |
| 
 | |
| 	[[nodiscard]] Main::Session &session() const {
 | |
| 		return *_session;
 | |
| 	}
 | |
| 	[[nodiscard]] const style::EmojiPan &st() const {
 | |
| 		return _st;
 | |
| 	}
 | |
| 	[[nodiscard]] Fn<bool()> pausedMethod() const {
 | |
| 		return _paused;
 | |
| 	}
 | |
| 	[[nodiscard]] bool paused() const {
 | |
| 		return _paused();
 | |
| 	}
 | |
| 
 | |
| 	[[nodiscard]] int getVisibleTop() const {
 | |
| 		return _visibleTop;
 | |
| 	}
 | |
| 	[[nodiscard]] int getVisibleBottom() const {
 | |
| 		return _visibleBottom;
 | |
| 	}
 | |
| 	void setMinimalHeight(int newWidth, int newMinimalHeight);
 | |
| 
 | |
| 	[[nodiscard]] rpl::producer<> checkForHide() const {
 | |
| 		return _checkForHide.events();
 | |
| 	}
 | |
| 	[[nodiscard]] bool preventAutoHide() const {
 | |
| 		return _preventHideWithBox;
 | |
| 	}
 | |
| 
 | |
| 	virtual void refreshRecent() = 0;
 | |
| 	virtual void preloadImages() {
 | |
| 	}
 | |
| 	void hideFinished();
 | |
| 	void panelHideFinished();
 | |
| 	virtual void clearSelection() = 0;
 | |
| 
 | |
| 	virtual void afterShown() {
 | |
| 	}
 | |
| 	virtual void beforeHiding() {
 | |
| 	}
 | |
| 	[[nodiscard]] virtual base::unique_qptr<Ui::PopupMenu> fillContextMenu(
 | |
| 			SendMenu::Type type) {
 | |
| 		return nullptr;
 | |
| 	}
 | |
| 
 | |
| 	rpl::producer<int> scrollToRequests() const;
 | |
| 	rpl::producer<bool> disableScrollRequests() const;
 | |
| 
 | |
| 	virtual object_ptr<InnerFooter> createFooter() = 0;
 | |
| 
 | |
| protected:
 | |
| 	void visibleTopBottomUpdated(
 | |
| 		int visibleTop,
 | |
| 		int visibleBottom) override;
 | |
| 	int minimalHeight() const;
 | |
| 	virtual int defaultMinimalHeight() const;
 | |
| 	int resizeGetHeight(int newWidth) override final;
 | |
| 
 | |
| 	virtual int countDesiredHeight(int newWidth) = 0;
 | |
| 	virtual InnerFooter *getFooter() const = 0;
 | |
| 	virtual void processHideFinished() {
 | |
| 	}
 | |
| 	virtual void processPanelHideFinished() {
 | |
| 	}
 | |
| 
 | |
| 	void scrollTo(int y);
 | |
| 	void disableScroll(bool disabled);
 | |
| 
 | |
| 	void checkHideWithBox(object_ptr<Ui::BoxContent> box);
 | |
| 
 | |
| 	void paintEmptySearchResults(
 | |
| 		Painter &p,
 | |
| 		const style::icon &icon,
 | |
| 		const QString &text) const;
 | |
| 
 | |
| private:
 | |
| 	const style::EmojiPan &_st;
 | |
| 	const std::shared_ptr<Show> _show;
 | |
| 	const not_null<Main::Session*> _session;
 | |
| 	const Fn<bool()> _paused;
 | |
| 
 | |
| 	int _visibleTop = 0;
 | |
| 	int _visibleBottom = 0;
 | |
| 	int _minimalHeight = 0;
 | |
| 
 | |
| 	rpl::event_stream<int> _scrollToRequests;
 | |
| 	rpl::event_stream<bool> _disableScrollRequests;
 | |
| 	rpl::event_stream<> _checkForHide;
 | |
| 
 | |
| 	bool _preventHideWithBox = false;
 | |
| 
 | |
| };
 | |
| 
 | |
| class TabbedSelector::InnerFooter : public Ui::RpWidget {
 | |
| public:
 | |
| 	InnerFooter(QWidget *parent, const style::EmojiPan &st);
 | |
| 
 | |
| 	[[nodiscard]] const style::EmojiPan &st() const;
 | |
| 
 | |
| protected:
 | |
| 	virtual void processHideFinished() {
 | |
| 	}
 | |
| 	virtual void processPanelHideFinished() {
 | |
| 	}
 | |
| 	friend class Inner;
 | |
| 
 | |
| private:
 | |
| 	const style::EmojiPan &_st;
 | |
| 
 | |
| };
 | |
| 
 | |
| } // namespace ChatHelpers
 | 
