mirror of
https://github.com/telegramdesktop/tdesktop
synced 2025-01-11 09:19:35 +00:00
2bdce7dce6
This was causing an assertion violation in Ui::PostponeCall. - Add a generic Core::QtSignalProducer to convert Qt signals to rpl::producer. - Track event loop nesting inside QtSignalProducer. - Use QtSignalProducer for QWindow::activeChanged tracking.
237 lines
5.3 KiB
C++
237 lines
5.3 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 <rpl/event_stream.h>
|
|
#include <rpl/map.h>
|
|
#include <rpl/distinct_until_changed.h>
|
|
#include "base/unique_qptr.h"
|
|
#include "core/qt_signal_producer.h"
|
|
|
|
namespace Ui {
|
|
namespace details {
|
|
|
|
struct ForwardTag {
|
|
};
|
|
|
|
struct InPlaceTag {
|
|
};
|
|
|
|
template <typename Value>
|
|
class AttachmentOwner : public QObject {
|
|
public:
|
|
template <typename OtherValue>
|
|
AttachmentOwner(QObject *parent, const ForwardTag&, OtherValue &&value)
|
|
: QObject(parent)
|
|
, _value(std::forward<OtherValue>(value)) {
|
|
}
|
|
|
|
template <typename ...Args>
|
|
AttachmentOwner(QObject *parent, const InPlaceTag&, Args &&...args)
|
|
: QObject(parent)
|
|
, _value(std::forward<Args>(args)...) {
|
|
}
|
|
|
|
not_null<Value*> value() {
|
|
return &_value;
|
|
}
|
|
|
|
private:
|
|
Value _value;
|
|
|
|
};
|
|
|
|
} // namespace details
|
|
|
|
class RpWidget;
|
|
|
|
template <typename Widget, typename ...Args>
|
|
inline base::unique_qptr<Widget> CreateObject(Args &&...args) {
|
|
return base::make_unique_q<Widget>(
|
|
nullptr,
|
|
std::forward<Args>(args)...);
|
|
}
|
|
|
|
template <typename Value, typename Parent, typename ...Args>
|
|
inline Value *CreateChild(
|
|
Parent *parent,
|
|
Args &&...args) {
|
|
Expects(parent != nullptr);
|
|
|
|
if constexpr (std::is_base_of_v<QObject, Value>) {
|
|
return new Value(parent, std::forward<Args>(args)...);
|
|
} else {
|
|
return CreateChild<details::AttachmentOwner<Value>>(
|
|
parent,
|
|
details::InPlaceTag{},
|
|
std::forward<Args>(args)...)->value();
|
|
}
|
|
}
|
|
|
|
inline void DestroyChild(QWidget *child) {
|
|
delete child;
|
|
}
|
|
|
|
void ResizeFitChild(
|
|
not_null<RpWidget*> parent,
|
|
not_null<RpWidget*> child);
|
|
|
|
template <typename Value>
|
|
inline not_null<std::decay_t<Value>*> AttachAsChild(
|
|
not_null<QObject*> parent,
|
|
Value &&value) {
|
|
return CreateChild<details::AttachmentOwner<std::decay_t<Value>>>(
|
|
parent.get(),
|
|
details::ForwardTag{},
|
|
std::forward<Value>(value))->value();
|
|
}
|
|
|
|
template <typename Widget>
|
|
using RpWidgetParent = std::conditional_t<
|
|
std::is_same_v<Widget, QWidget>,
|
|
TWidget,
|
|
TWidgetHelper<Widget>>;
|
|
|
|
template <typename Widget>
|
|
class RpWidgetWrap;
|
|
|
|
class RpWidgetMethods {
|
|
public:
|
|
rpl::producer<QRect> geometryValue() const;
|
|
rpl::producer<QSize> sizeValue() const;
|
|
rpl::producer<int> heightValue() const;
|
|
rpl::producer<int> widthValue() const;
|
|
rpl::producer<QPoint> positionValue() const;
|
|
rpl::producer<int> leftValue() const;
|
|
rpl::producer<int> topValue() const;
|
|
virtual rpl::producer<int> desiredHeightValue() const;
|
|
rpl::producer<bool> shownValue() const;
|
|
rpl::producer<QRect> paintRequest() const;
|
|
rpl::producer<> alive() const;
|
|
|
|
template <typename Error, typename Generator>
|
|
void showOn(rpl::producer<bool, Error, Generator> &&shown) {
|
|
std::move(
|
|
shown
|
|
) | rpl::start_with_next([this](bool visible) {
|
|
callSetVisible(visible);
|
|
}, lifetime());
|
|
}
|
|
|
|
rpl::lifetime &lifetime();
|
|
|
|
virtual ~RpWidgetMethods() = default;
|
|
|
|
protected:
|
|
bool handleEvent(QEvent *event);
|
|
virtual bool eventHook(QEvent *event) = 0;
|
|
|
|
private:
|
|
template <typename Widget>
|
|
friend class RpWidgetWrap;
|
|
|
|
struct EventStreams {
|
|
rpl::event_stream<QRect> geometry;
|
|
rpl::event_stream<QRect> paint;
|
|
rpl::event_stream<bool> shown;
|
|
rpl::event_stream<> alive;
|
|
};
|
|
struct Initer {
|
|
Initer(QWidget *parent);
|
|
};
|
|
|
|
virtual void callSetVisible(bool visible) = 0;
|
|
virtual QPointer<QObject> callCreateWeak() = 0;
|
|
virtual QRect callGetGeometry() const = 0;
|
|
virtual bool callIsHidden() const = 0;
|
|
|
|
void visibilityChangedHook(bool wasVisible, bool nowVisible);
|
|
EventStreams &eventStreams() const;
|
|
|
|
mutable std::unique_ptr<EventStreams> _eventStreams;
|
|
rpl::lifetime _lifetime;
|
|
|
|
};
|
|
|
|
template <typename Widget>
|
|
class RpWidgetWrap
|
|
: public RpWidgetParent<Widget>
|
|
, public RpWidgetMethods {
|
|
using Self = RpWidgetWrap<Widget>;
|
|
using Parent = RpWidgetParent<Widget>;
|
|
|
|
public:
|
|
using Parent::Parent;
|
|
|
|
void setVisible(bool visible) final override {
|
|
auto wasVisible = !this->isHidden();
|
|
setVisibleHook(visible);
|
|
visibilityChangedHook(wasVisible, !this->isHidden());
|
|
}
|
|
|
|
auto windowDeactivateEvents() const {
|
|
Expects(Widget::window()->windowHandle() != nullptr);
|
|
|
|
const auto window = Widget::window()->windowHandle();
|
|
return Core::QtSignalProducer(
|
|
window,
|
|
&QWindow::activeChanged
|
|
) | rpl::filter([=] {
|
|
return !window->isActive();
|
|
});
|
|
}
|
|
auto macWindowDeactivateEvents() const {
|
|
#ifdef Q_OS_MAC
|
|
return windowDeactivateEvents();
|
|
#else // Q_OS_MAC
|
|
return rpl::never<rpl::empty_value>();
|
|
#endif // Q_OS_MAC
|
|
}
|
|
|
|
~RpWidgetWrap() {
|
|
base::take(_lifetime);
|
|
base::take(_eventStreams);
|
|
}
|
|
|
|
protected:
|
|
bool event(QEvent *event) final override {
|
|
return handleEvent(event);
|
|
}
|
|
bool eventHook(QEvent *event) override {
|
|
return Parent::event(event);
|
|
}
|
|
virtual void setVisibleHook(bool visible) {
|
|
Parent::setVisible(visible);
|
|
}
|
|
|
|
private:
|
|
void callSetVisible(bool visible) override {
|
|
Self::setVisible(visible);
|
|
}
|
|
QPointer<QObject> callCreateWeak() override {
|
|
return QPointer<QObject>((QObject*)this);
|
|
}
|
|
QRect callGetGeometry() const override {
|
|
return this->geometry();
|
|
}
|
|
bool callIsHidden() const override {
|
|
return this->isHidden();
|
|
}
|
|
|
|
Initer _initer = { this };
|
|
|
|
};
|
|
|
|
class RpWidget : public RpWidgetWrap<QWidget> {
|
|
public:
|
|
using RpWidgetWrap<QWidget>::RpWidgetWrap;
|
|
|
|
};
|
|
|
|
} // namespace Ui
|