/* 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-2017 John Preston, https://desktop.telegram.org */ #pragma once #include "ui/twidget.h" #include "ui/abstract_button.h" #include "ui/effects/panel_animation.h" #include "mtproto/sender.h" #include "inline_bots/inline_bot_layout_item.h" #include "auth_session.h" namespace InlineBots { namespace Layout { class ItemBase; } // namespace Layout class Result; } // namespace InlineBots namespace Ui { class PlainShadow; class ScrollArea; class IconButton; class LinkButton; class RoundButton; class RippleAnimation; class SettingsSlider; } // namesapce Ui namespace ChatHelpers { class EmojiListWidget; class StickersListWidget; class GifsListWidget; using InlineResult = InlineBots::Result; using InlineResults = std::vector>; using InlineItem = InlineBots::Layout::ItemBase; struct InlineCacheEntry { QString nextOffset; QString switchPmText, switchPmStartToken; InlineResults results; }; struct StickerIcon { StickerIcon(uint64 setId) : setId(setId) { } StickerIcon(uint64 setId, DocumentData *sticker, int32 pixw, int32 pixh) : setId(setId), sticker(sticker), pixw(pixw), pixh(pixh) { } uint64 setId = 0; DocumentData *sticker = nullptr; int pixw = 0; int pixh = 0; }; class EmojiPanel : public TWidget, private MTP::Sender { Q_OBJECT public: EmojiPanel(QWidget *parent); void setMinTop(int minTop); void setMinBottom(int minBottom); void moveBottom(int bottom); void hideFast(); bool hiding() const { return _hiding || _hideTimer.isActive(); } void step_icons(TimeMs ms, bool timer); void leaveToChildEvent(QEvent *e, QWidget *child) override; void stickersInstalled(uint64 setId); void queryInlineBot(UserData *bot, PeerData *peer, QString query); void clearInlineBot(); bool overlaps(const QRect &globalRect) const; bool ui_isInlineItemBeingChosen(); void showAnimated(); void hideAnimated(); ~EmojiPanel(); class Inner; protected: void enterEventHook(QEvent *e) override; void leaveEventHook(QEvent *e) override; void otherEnter(); void otherLeave(); void mousePressEvent(QMouseEvent *e) override; void mouseMoveEvent(QMouseEvent *e) override; void mouseReleaseEvent(QMouseEvent *e) override; void paintEvent(QPaintEvent *e) override; bool event(QEvent *e) override; bool eventFilter(QObject *obj, QEvent *e) override; public slots: void refreshStickers(); private slots: void hideByTimerOrLeave(); void refreshSavedGifs(); void onWndActiveChanged(); void onScroll(); void onDisplaySet(quint64 setId); void onInstallSet(quint64 setId); void onRemoveSet(quint64 setId); void onDelayedHide(); void onRefreshIcons(bool scrollAnimation); void onSaveConfig(); void onSaveConfigDelayed(int delay); void onInlineRequest(); void onEmptyInlineRows(); signals: void emojiSelected(EmojiPtr emoji); void stickerSelected(DocumentData *sticker); void photoSelected(PhotoData *photo); void inlineResultSelected(InlineBots::Result *result, UserData *bot); void updateStickers(); private: using TabType = EmojiPanelTab; class Tab { public: static constexpr auto kCount = 3; Tab(TabType type, object_ptr widget); object_ptr takeWidget(); void returnWidget(object_ptr widget); TabType type() const { return _type; } gsl::not_null widget() const { return _weak; } void saveScrollTop(); void saveScrollTop(int scrollTop) { _scrollTop = scrollTop; } int getScrollTop() const { return _scrollTop; } private: TabType _type = TabType::Emoji; object_ptr _widget = { nullptr }; QPointer _weak; int _scrollTop = 0; }; int marginTop() const; int marginBottom() const; int countBottom() const; void moveByBottom(); void paintSlideFrame(Painter &p, TimeMs ms); void paintContent(Painter &p); style::margins innerPadding() const; // Rounded rect which has shadow around it. QRect innerRect() const; // Inner rect with removed st::buttonRadius from top and bottom. // This one is allowed to be not rounded. QRect horizontalRect() const; // Inner rect with removed st::buttonRadius from left and right. // This one is allowed to be not rounded. QRect verticalRect() const; enum class GrabType { Panel, Slide, }; QImage grabForComplexAnimation(GrabType type); void startShowAnimation(); void startOpacityAnimation(bool hiding); void prepareCache(); void opacityAnimationCallback(); void hideFinished(); void showStarted(); bool preventAutoHide() const; void setActiveSection(DBIEmojiSection section); void setCurrentSectionIcon(DBIEmojiSection section); void paintStickerSettingsIcon(Painter &p) const; void paintFeaturedStickerSetsBadge(Painter &p, int iconLeft) const; enum class ValidateIconAnimations { Full, Scroll, None, }; void validateSelectedIcon(ValidateIconAnimations animations); void updateContentHeight(); void updateSelected(); void updateIcons(); void prepareSection(int &left, int top, int _width, Ui::IconButton *sectionIcon, DBIEmojiSection section); void showAll(); void hideForSliding(); void setWidgetToScrollArea(); void createTabsSlider(); void switchTab(); gsl::not_null getTab(TabType type) { return &_tabs[static_cast(type)]; } gsl::not_null getTab(TabType type) const { return &_tabs[static_cast(type)]; } gsl::not_null currentTab() { return getTab(_currentTabType); } gsl::not_null currentTab() const { return getTab(_currentTabType); } gsl::not_null emoji() const; gsl::not_null stickers() const; gsl::not_null gifs() const; int _minTop = 0; int _minBottom = 0; int _contentMaxHeight = 0; int _contentHeight = 0; bool _horizontal = false; int _width = 0; int _height = 0; int _bottom = 0; std::unique_ptr _showAnimation; Animation _a_show; bool _hiding = false; bool _hideAfterSlide = false; QPixmap _cache; Animation _a_opacity; QTimer _hideTimer; bool _inComplrexGrab = false; class SlideAnimation; std::unique_ptr _slideAnimation; Animation _a_slide; object_ptr _recent; object_ptr _people; object_ptr _nature; object_ptr _food; object_ptr _activity; object_ptr _travel; object_ptr _objects; object_ptr _symbols; QList _icons; int _iconOver = -1; int _iconSel = 0; int _iconDown = -1; bool _iconsDragging = false; BasicAnimation _a_icons; QPoint _iconsMousePos, _iconsMouseDown; int _iconsLeft = 0; int _iconsTop = 0; int _iconsStartX = 0; int _iconsMax = 0; anim::value _iconsX; anim::value _iconSelX; TimeMs _iconsStartAnim = 0; object_ptr _tabsSlider = { nullptr }; object_ptr _topShadow; object_ptr _bottomShadow; object_ptr _scroll; std::array _tabs; TabType _currentTabType = TabType::Emoji; uint64 _displayingSetId = 0; uint64 _removingSetId = 0; QTimer _saveConfigTimer; // inline bots std::map> _inlineCache; QTimer _inlineRequestTimer; void inlineBotChanged(); int32 showInlineRows(bool newResults); bool refreshInlineRows(int32 *added = 0); UserData *_inlineBot = nullptr; PeerData *_inlineQueryPeer = nullptr; QString _inlineQuery, _inlineNextQuery, _inlineNextOffset; mtpRequestId _inlineRequestId = 0; void inlineResultsDone(const MTPmessages_BotResults &result); }; class EmojiPanel::Inner : public TWidget { Q_OBJECT public: Inner(QWidget *parent); void setVisibleTopBottom(int visibleTop, int visibleBottom) override; int getVisibleTop() const { return _visibleTop; } int getVisibleBottom() const { return _visibleBottom; } virtual void refreshRecent() = 0; virtual void preloadImages() { } virtual void hideFinish(bool completely) = 0; virtual void clearSelection() = 0; virtual object_ptr createController() = 0; signals: void scrollToY(int y); void disableScroll(bool disabled); void saveConfigDelayed(int delay); protected: virtual int countHeight() = 0; private: int _visibleTop = 0; int _visibleBottom = 0; }; } // namespace ChatHelpers