tdesktop/Telegram/SourceFiles/ui/rp_widget.h
John Preston 2bdce7dce6 Fix window activations handling without event loop nesting.
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.
2019-04-05 14:13:54 +04:00

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