/* 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 "styles/style_widgets.h" #include "ui/rp_widget.h" #include "ui/effects/animations.h" #include "base/object_ptr.h" namespace Ui { class InputField; class CrossButton; class ScrollArea; class MultiSelect : public RpWidget { public: MultiSelect( QWidget *parent, const style::MultiSelect &st, rpl::producer placeholder = nullptr); QString getQuery() const; void setInnerFocus(); void clearQuery(); void setQueryChangedCallback(Fn callback); void setSubmittedCallback(Fn callback); void setCancelledCallback(Fn callback); void setResizedCallback(Fn callback); enum class AddItemWay { Default, SkipAnimation, }; using PaintRoundImage = Fn; void addItem(uint64 itemId, const QString &text, style::color color, PaintRoundImage paintRoundImage, AddItemWay way = AddItemWay::Default); void addItemInBunch(uint64 itemId, const QString &text, style::color color, PaintRoundImage paintRoundImage); void finishItemsBunch(); void setItemText(uint64 itemId, const QString &text); void setItemRemovedCallback(Fn callback); void removeItem(uint64 itemId); int getItemsCount() const; QVector getItems() const; bool hasItem(uint64 itemId) const; class Item; protected: int resizeGetHeight(int newWidth) override; bool eventFilter(QObject *o, QEvent *e) override; private: void scrollTo(int activeTop, int activeBottom); const style::MultiSelect &_st; object_ptr _scroll; class Inner; QPointer _inner; Fn _resizedCallback; Fn _queryChangedCallback; }; // This class is hold in header because it requires Qt preprocessing. class MultiSelect::Inner : public TWidget { public: using ScrollCallback = Fn; Inner( QWidget *parent, const style::MultiSelect &st, rpl::producer placeholder, ScrollCallback callback); QString getQuery() const; bool setInnerFocus(); void clearQuery(); void setQueryChangedCallback(Fn callback); void setSubmittedCallback(Fn callback); void setCancelledCallback(Fn callback); void addItemInBunch(std::unique_ptr item); void finishItemsBunch(AddItemWay way); void setItemText(uint64 itemId, const QString &text); void setItemRemovedCallback(Fn callback); void removeItem(uint64 itemId); int getItemsCount() const; QVector getItems() const; bool hasItem(uint64 itemId) const; void setResizedCallback(Fn callback); ~Inner(); protected: int resizeGetHeight(int newWidth) override; void paintEvent(QPaintEvent *e) override; void leaveEventHook(QEvent *e) override; void mouseMoveEvent(QMouseEvent *e) override; void mousePressEvent(QMouseEvent *e) override; void keyPressEvent(QKeyEvent *e) override; private: void submitted(Qt::KeyboardModifiers modifiers); void cancelled(); void queryChanged(); void fieldFocused(); void computeItemsGeometry(int newWidth); void updateItemsGeometry(); void updateFieldGeometry(); void updateHasAnyItems(bool hasAnyItems); void updateSelection(QPoint mousePosition); void clearSelection() { updateSelection(QPoint(-1, -1)); } void updateCursor(); void updateHeightStep(); void finishHeightAnimation(); enum class ChangeActiveWay { Default, SkipSetFocus, }; void setActiveItem(int active, ChangeActiveWay skipSetFocus = ChangeActiveWay::Default); void setActiveItemPrevious(); void setActiveItemNext(); QMargins itemPaintMargins() const; const style::MultiSelect &_st; Ui::Animations::Simple _iconOpacity; ScrollCallback _scrollCallback; std::set _idsMap; std::vector> _items; std::set> _removingItems; int _selected = -1; int _active = -1; bool _overDelete = false; int _fieldLeft = 0; int _fieldTop = 0; int _fieldWidth = 0; object_ptr _field; object_ptr _cancel; int _newHeight = 0; Ui::Animations::Simple _height; Fn _queryChangedCallback; Fn _submittedCallback; Fn _cancelledCallback; Fn _itemRemovedCallback; Fn _resizedCallback; }; class MultiSelect::Item { public: Item(const style::MultiSelectItem &st, uint64 id, const QString &text, style::color color, PaintRoundImage &&paintRoundImage); uint64 id() const { return _id; } int getWidth() const { return _width; } QRect rect() const { return QRect(_x, _y, _width, _st.height); } bool isOverDelete() const { return _overDelete; } void setActive(bool active) { _active = active; } void setPosition(int x, int y, int outerWidth, int maxVisiblePadding); QRect paintArea(int outerWidth) const; void setUpdateCallback(Fn updateCallback) { _updateCallback = updateCallback; } void setText(const QString &text); void paint(Painter &p, int outerWidth); void mouseMoveEvent(QPoint point); void leaveEvent(); void showAnimated() { setVisibleAnimated(true); } void hideAnimated() { setVisibleAnimated(false); } bool hideFinished() const { return (_hiding && !_visibility.animating()); } private: void setOver(bool over); void paintOnce(Painter &p, int x, int y, int outerWidth); void paintDeleteButton(Painter &p, int x, int y, int outerWidth, float64 overOpacity); bool paintCached(Painter &p, int x, int y, int outerWidth); void prepareCache(); void setVisibleAnimated(bool visible); const style::MultiSelectItem &_st; uint64 _id; struct SlideAnimation { SlideAnimation(Fn updateCallback, int fromX, int toX, int y, float64 duration) : fromX(fromX) , toX(toX) , y(y) { x.start(updateCallback, fromX, toX, duration); } Ui::Animations::Simple x; int fromX, toX; int y; }; std::vector _copies; int _x = -1; int _y = -1; int _width = 0; Text::String _text; style::color _color; bool _over = false; QPixmap _cache; Ui::Animations::Simple _visibility; Ui::Animations::Simple _overOpacity; bool _overDelete = false; bool _active = false; PaintRoundImage _paintRoundImage; Fn _updateCallback; bool _hiding = false; }; } // namespace Ui