/* 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 "base/flags.h" class TWidget; template class object_ptr; inline QPoint rtlpoint(int x, int y, int outerw) { return QPoint(rtl() ? (outerw - x) : x, y); } inline QPoint rtlpoint(const QPoint &p, int outerw) { return rtl() ? QPoint(outerw - p.x(), p.y()) : p; } inline QPointF rtlpoint(const QPointF &p, int outerw) { return rtl() ? QPointF(outerw - p.x(), p.y()) : p; } inline QRect rtlrect(int x, int y, int w, int h, int outerw) { return QRect(rtl() ? (outerw - x - w) : x, y, w, h); } inline QRect rtlrect(const QRect &r, int outerw) { return rtl() ? QRect(outerw - r.x() - r.width(), r.y(), r.width(), r.height()) : r; } inline QRectF rtlrect(const QRectF &r, int outerw) { return rtl() ? QRectF(outerw - r.x() - r.width(), r.y(), r.width(), r.height()) : r; } namespace Ui { inline bool InFocusChain(not_null widget) { if (auto top = widget->window()) { if (auto focused = top->focusWidget()) { return !widget->isHidden() && (focused == widget || widget->isAncestorOf(focused)); } } return false; } template inline ChildWidget *AttachParentChild( not_null parent, const object_ptr &child) { if (auto raw = child.data()) { raw->setParent(parent); raw->show(); return raw; } return nullptr; } void SendPendingMoveResizeEvents(not_null target); QPixmap GrabWidget( not_null target, QRect rect = QRect(), QColor bg = QColor(255, 255, 255, 0)); QImage GrabWidgetToImage( not_null target, QRect rect = QRect(), QColor bg = QColor(255, 255, 255, 0)); void RenderWidget( QPainter &painter, not_null source, const QPoint &targetOffset = QPoint(), const QRegion &sourceRegion = QRegion(), QWidget::RenderFlags renderFlags = QWidget::DrawChildren | QWidget::IgnoreMask); void ForceFullRepaint(not_null widget); void PostponeCall(FnMut &&callable); template < typename Guard, typename Callable, typename GuardTraits = crl::guard_traits>, typename = std::enable_if_t< sizeof(GuardTraits) != crl::details::dependent_zero>> inline void PostponeCall(Guard &&object, Callable &&callable) { return PostponeCall(crl::guard( std::forward(object), std::forward(callable))); } } // namespace Ui enum class RectPart { None = 0, TopLeft = (1 << 0), Top = (1 << 1), TopRight = (1 << 2), Left = (1 << 3), Center = (1 << 4), Right = (1 << 5), BottomLeft = (1 << 6), Bottom = (1 << 7), BottomRight = (1 << 8), FullTop = TopLeft | Top | TopRight, NoTopBottom = Left | Center | Right, FullBottom = BottomLeft | Bottom | BottomRight, NoTop = NoTopBottom | FullBottom, NoBottom = FullTop | NoTopBottom, FullLeft = TopLeft | Left | BottomLeft, NoLeftRight = Top | Center | Bottom, FullRight = TopRight | Right | BottomRight, NoLeft = NoLeftRight | FullRight, NoRight = FullLeft | NoLeftRight, AllCorners = TopLeft | TopRight | BottomLeft | BottomRight, AllSides = Top | Bottom | Left | Right, Full = FullTop | NoTop, }; using RectParts = base::flags; inline constexpr auto is_flag_type(RectPart) { return true; }; inline bool IsTopCorner(RectPart corner) { return (corner == RectPart::TopLeft) || (corner == RectPart::TopRight); } inline bool IsBottomCorner(RectPart corner) { return (corner == RectPart::BottomLeft) || (corner == RectPart::BottomRight); } inline bool IsLeftCorner(RectPart corner) { return (corner == RectPart::TopLeft) || (corner == RectPart::BottomLeft); } inline bool IsRightCorner(RectPart corner) { return (corner == RectPart::TopRight) || (corner == RectPart::BottomRight); } template class TWidgetHelper : public Base { public: using Base::Base; virtual QMargins getMargins() const { return QMargins(); } bool inFocusChain() const { return Ui::InFocusChain(this); } void hideChildren() { for (auto child : Base::children()) { if (child->isWidgetType()) { static_cast(child)->hide(); } } } void showChildren() { for (auto child : Base::children()) { if (child->isWidgetType()) { static_cast(child)->show(); } } } void moveToLeft(int x, int y, int outerw = 0) { auto margins = getMargins(); x -= margins.left(); y -= margins.top(); Base::move(rtl() ? ((outerw > 0 ? outerw : Base::parentWidget()->width()) - x - Base::width()) : x, y); } void moveToRight(int x, int y, int outerw = 0) { auto margins = getMargins(); x -= margins.right(); y -= margins.top(); Base::move(rtl() ? x : ((outerw > 0 ? outerw : Base::parentWidget()->width()) - x - Base::width()), y); } void setGeometryToLeft(int x, int y, int w, int h, int outerw = 0) { auto margins = getMargins(); x -= margins.left(); y -= margins.top(); w -= margins.left() - margins.right(); h -= margins.top() - margins.bottom(); Base::setGeometry(rtl() ? ((outerw > 0 ? outerw : Base::parentWidget()->width()) - x - w) : x, y, w, h); } void setGeometryToRight(int x, int y, int w, int h, int outerw = 0) { auto margins = getMargins(); x -= margins.right(); y -= margins.top(); w -= margins.left() - margins.right(); h -= margins.top() - margins.bottom(); Base::setGeometry(rtl() ? x : ((outerw > 0 ? outerw : Base::parentWidget()->width()) - x - w), y, w, h); } QPoint myrtlpoint(int x, int y) const { return rtlpoint(x, y, Base::width()); } QPoint myrtlpoint(const QPoint point) const { return rtlpoint(point, Base::width()); } QRect myrtlrect(int x, int y, int w, int h) const { return rtlrect(x, y, w, h, Base::width()); } QRect myrtlrect(const QRect &rect) const { return rtlrect(rect, Base::width()); } void rtlupdate(const QRect &rect) { Base::update(myrtlrect(rect)); } void rtlupdate(int x, int y, int w, int h) { Base::update(myrtlrect(x, y, w, h)); } QPoint mapFromGlobal(const QPoint &point) const { return Base::mapFromGlobal(point); } QPoint mapToGlobal(const QPoint &point) const { return Base::mapToGlobal(point); } QRect mapFromGlobal(const QRect &rect) const { return QRect(mapFromGlobal(rect.topLeft()), rect.size()); } QRect mapToGlobal(const QRect &rect) const { return QRect(mapToGlobal(rect.topLeft()), rect.size()); } protected: void enterEvent(QEvent *e) final override { if (auto parent = tparent()) { parent->leaveToChildEvent(e, this); } return enterEventHook(e); } virtual void enterEventHook(QEvent *e) { return Base::enterEvent(e); } void leaveEvent(QEvent *e) final override { if (auto parent = tparent()) { parent->enterFromChildEvent(e, this); } return leaveEventHook(e); } virtual void leaveEventHook(QEvent *e) { return Base::leaveEvent(e); } // e - from enterEvent() of child TWidget virtual void leaveToChildEvent(QEvent *e, QWidget *child) { } // e - from leaveEvent() of child TWidget virtual void enterFromChildEvent(QEvent *e, QWidget *child) { } private: TWidget *tparent() { return qobject_cast(Base::parentWidget()); } const TWidget *tparent() const { return qobject_cast(Base::parentWidget()); } template friend class TWidgetHelper; }; class TWidget : public TWidgetHelper { // The Q_OBJECT meta info is used for qobject_cast! Q_OBJECT public: TWidget(QWidget *parent = nullptr) : TWidgetHelper(parent) { } // Get the size of the widget as it should be. // Negative return value means no default width. virtual int naturalWidth() const { return -1; } // Count new height for width=newWidth and resize to it. void resizeToWidth(int newWidth) { auto margins = getMargins(); auto fullWidth = margins.left() + newWidth + margins.right(); auto fullHeight = margins.top() + resizeGetHeight(newWidth) + margins.bottom(); auto newSize = QSize(fullWidth, fullHeight); if (newSize != size()) { resize(newSize); update(); } } // Resize to minimum of natural width and available width. void resizeToNaturalWidth(int newWidth) { auto maxWidth = naturalWidth(); resizeToWidth((maxWidth >= 0) ? qMin(newWidth, maxWidth) : newWidth); } QRect rectNoMargins() const { return rect().marginsRemoved(getMargins()); } int widthNoMargins() const { return rectNoMargins().width(); } int heightNoMargins() const { return rectNoMargins().height(); } int bottomNoMargins() const { auto rectWithoutMargins = rectNoMargins(); return y() + rectWithoutMargins.y() + rectWithoutMargins.height(); } QSize sizeNoMargins() const { return rectNoMargins().size(); } // Updates the area that is visible inside the scroll container. void setVisibleTopBottom(int visibleTop, int visibleBottom) { auto max = height(); visibleTopBottomUpdated( snap(visibleTop, 0, max), snap(visibleBottom, 0, max)); } signals: // Child widget is responsible for emitting this signal. void heightUpdated(); protected: void setChildVisibleTopBottom( TWidget *child, int visibleTop, int visibleBottom) { if (child) { auto top = child->y(); child->setVisibleTopBottom( visibleTop - top, visibleBottom - top); } } // Resizes content and counts natural widget height for the desired width. virtual int resizeGetHeight(int newWidth) { return heightNoMargins(); } virtual void visibleTopBottomUpdated( int visibleTop, int visibleBottom) { } }; template QPointer make_weak(Widget *object) { return QPointer(object); } template QPointer make_weak(const Widget *object) { return QPointer(object); } template QPointer make_weak(not_null object) { return QPointer(object.get()); } template QPointer make_weak(not_null object) { return QPointer(object.get()); } class SingleQueuedInvokation : public QObject { public: SingleQueuedInvokation(Fn callback) : _callback(callback) { } void call() { if (_pending.testAndSetAcquire(0, 1)) { InvokeQueued(this, [this] { if (_pending.testAndSetRelease(1, 0)) { _callback(); } }); } } private: Fn _callback; QAtomicInt _pending = { 0 }; }; // Smart pointer for QObject*, has move semantics, destroys object if it doesn't have a parent. template class object_ptr { public: object_ptr(std::nullptr_t) noexcept { } // No default constructor, but constructors with at least // one argument are simply make functions. template explicit object_ptr(Parent &&parent, Args&&... args) : _object(new Object(std::forward(parent), std::forward(args)...)) { } static object_ptr fromRaw(Object *value) noexcept { object_ptr result = { nullptr }; result._object = value; return result; } object_ptr(const object_ptr &other) = delete; object_ptr &operator=(const object_ptr &other) = delete; object_ptr(object_ptr &&other) noexcept : _object(base::take(other._object)) { } object_ptr &operator=(object_ptr &&other) noexcept { auto temp = std::move(other); destroy(); std::swap(_object, temp._object); return *this; } template < typename OtherObject, typename = std::enable_if_t< std::is_base_of_v>> object_ptr(object_ptr &&other) noexcept : _object(base::take(other._object)) { } template < typename OtherObject, typename = std::enable_if_t< std::is_base_of_v>> object_ptr &operator=(object_ptr &&other) noexcept { _object = base::take(other._object); return *this; } object_ptr &operator=(std::nullptr_t) noexcept { _object = nullptr; return *this; } // So we can pass this pointer to methods like connect(). Object *data() const noexcept { return static_cast(_object.data()); } operator Object*() const noexcept { return data(); } explicit operator bool() const noexcept { return _object != nullptr; } Object *operator->() const noexcept { return data(); } Object &operator*() const noexcept { return *data(); } // Use that instead "= new Object(parent, ...)" template Object *create(Parent &&parent, Args&&... args) { destroy(); _object = new Object( std::forward(parent), std::forward(args)...); return data(); } void destroy() noexcept { delete base::take(_object); } void destroyDelayed() { if (_object) { if (auto widget = base::up_cast(data())) { widget->hide(); } base::take(_object)->deleteLater(); } } ~object_ptr() noexcept { if (auto pointer = _object) { if (!pointer->parent()) { destroy(); } } } template friend object_ptr static_object_cast( object_ptr source); private: template friend class object_ptr; QPointer _object; }; template inline object_ptr static_object_cast( object_ptr source) { auto result = object_ptr(nullptr); result._object = static_cast( base::take(source._object).data()); return std::move(result); } void sendSynteticMouseEvent( QWidget *widget, QEvent::Type type, Qt::MouseButton button, const QPoint &globalPoint); inline void sendSynteticMouseEvent( QWidget *widget, QEvent::Type type, Qt::MouseButton button) { return sendSynteticMouseEvent(widget, type, button, QCursor::pos()); }