258 lines
6.6 KiB
C++
258 lines
6.6 KiB
C++
// This file is part of Desktop App Toolkit,
|
|
// a set of libraries for developing nice desktop applications.
|
|
//
|
|
// For license and copyright information please follow this link:
|
|
// https://github.com/desktop-app/legal/blob/master/LEGAL
|
|
//
|
|
#include "ui/rp_widget.h"
|
|
|
|
#include "base/platform/base_platform_info.h"
|
|
#include "base/qt_signal_producer.h"
|
|
#include "ui/gl/gl_detection.h"
|
|
|
|
#include <QtGui/QWindow>
|
|
#include <QtGui/QtEvents>
|
|
#include <QtGui/QColorSpace>
|
|
#include <private/qwidget_p.h>
|
|
|
|
// Patching out this code without patching out all other private API usage
|
|
// and the Qt::{Core,Gui,Widgets}Private cmake dependency is asking
|
|
// for memory corruption
|
|
class TWidgetPrivate : public QWidgetPrivate {
|
|
public:
|
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 4, 0)
|
|
QPlatformBackingStoreRhiConfig rhiConfig() const override {
|
|
const auto q = static_cast<TWidget*>(q_ptr);
|
|
if (!q->testAttribute(Qt::WA_WState_Created)) {
|
|
return QWidgetPrivate::rhiConfig();
|
|
}
|
|
if (const auto config = q->rhiConfig()) {
|
|
return *config;
|
|
}
|
|
if (::Platform::IsMac10_14OrGreater()) {
|
|
return { QPlatformBackingStoreRhiConfig::Metal };
|
|
}
|
|
// We can't specify the widget here as q_evaluateRhiConfig is called
|
|
// in QWidgetWindow constructor, while windowHandle is set right after
|
|
// the constructor is completed
|
|
if (::Platform::IsWayland() // old versions of mutter produce flicker without OpenGL
|
|
&& Ui::GL::ChooseBackendDefault(
|
|
Ui::GL::CheckCapabilities(nullptr))
|
|
== Ui::GL::Backend::OpenGL) {
|
|
return { QPlatformBackingStoreRhiConfig::OpenGL };
|
|
}
|
|
return QWidgetPrivate::rhiConfig();
|
|
}
|
|
#endif // Qt >= 6.4.0
|
|
};
|
|
|
|
TWidget::TWidget(QWidget *parent)
|
|
: TWidgetHelper<QWidget>(*(new TWidgetPrivate), parent, {}) {
|
|
[[maybe_unused]] static const auto Once = [] {
|
|
auto format = QSurfaceFormat::defaultFormat();
|
|
format.setSwapInterval(0);
|
|
#ifdef Q_OS_MAC
|
|
format.setColorSpace(QColorSpace::SRgb);
|
|
#endif // Q_OS_MAC
|
|
QSurfaceFormat::setDefaultFormat(format);
|
|
return true;
|
|
}();
|
|
}
|
|
|
|
namespace Ui {
|
|
namespace {
|
|
|
|
[[nodiscard]] std::vector<QPointer<QWidget>> GetChildWidgets(
|
|
not_null<QWidget*> widget) {
|
|
const auto &children = widget->children();
|
|
auto result = std::vector<QPointer<QWidget>>();
|
|
result.reserve(children.size());
|
|
for (const auto child : children) {
|
|
if (child && child->isWidgetType()) {
|
|
result.push_back(static_cast<QWidget*>(child));
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
void ToggleChildrenVisibility(not_null<QWidget*> widget, bool visible) {
|
|
for (const auto &child : GetChildWidgets(widget)) {
|
|
if (child) {
|
|
child->setVisible(visible);
|
|
}
|
|
}
|
|
}
|
|
|
|
void ResizeFitChild(
|
|
not_null<RpWidget*> parent,
|
|
not_null<RpWidget*> child,
|
|
int heightMin) {
|
|
parent->widthValue(
|
|
) | rpl::start_with_next([=](int width) {
|
|
child->resizeToWidth(width);
|
|
}, child->lifetime());
|
|
|
|
child->heightValue(
|
|
) | rpl::start_with_next([=](int height) {
|
|
parent->resize(parent->width(), std::max(height, heightMin));
|
|
}, child->lifetime());
|
|
}
|
|
|
|
rpl::producer<not_null<QEvent*>> RpWidgetWrap::events() const {
|
|
auto &stream = eventStreams().events;
|
|
return stream.events();
|
|
}
|
|
|
|
rpl::producer<QRect> RpWidgetWrap::geometryValue() const {
|
|
auto &stream = eventStreams().geometry;
|
|
return stream.events_starting_with_copy(rpWidget()->geometry());
|
|
}
|
|
|
|
rpl::producer<QSize> RpWidgetWrap::sizeValue() const {
|
|
return geometryValue()
|
|
| rpl::map([](QRect &&value) { return value.size(); })
|
|
| rpl::distinct_until_changed();
|
|
}
|
|
|
|
rpl::producer<int> RpWidgetWrap::heightValue() const {
|
|
return geometryValue()
|
|
| rpl::map([](QRect &&value) { return value.height(); })
|
|
| rpl::distinct_until_changed();
|
|
}
|
|
|
|
rpl::producer<int> RpWidgetWrap::widthValue() const {
|
|
return geometryValue()
|
|
| rpl::map([](QRect &&value) { return value.width(); })
|
|
| rpl::distinct_until_changed();
|
|
}
|
|
|
|
rpl::producer<QPoint> RpWidgetWrap::positionValue() const {
|
|
return geometryValue()
|
|
| rpl::map([](QRect &&value) { return value.topLeft(); })
|
|
| rpl::distinct_until_changed();
|
|
}
|
|
|
|
rpl::producer<int> RpWidgetWrap::leftValue() const {
|
|
return geometryValue()
|
|
| rpl::map([](QRect &&value) { return value.left(); })
|
|
| rpl::distinct_until_changed();
|
|
}
|
|
|
|
rpl::producer<int> RpWidgetWrap::topValue() const {
|
|
return geometryValue()
|
|
| rpl::map([](QRect &&value) { return value.top(); })
|
|
| rpl::distinct_until_changed();
|
|
}
|
|
|
|
rpl::producer<int> RpWidgetWrap::desiredHeightValue() const {
|
|
return heightValue();
|
|
}
|
|
|
|
rpl::producer<bool> RpWidgetWrap::shownValue() const {
|
|
auto &stream = eventStreams().shown;
|
|
return stream.events_starting_with(!rpWidget()->isHidden());
|
|
}
|
|
|
|
rpl::producer<QRect> RpWidgetWrap::paintRequest() const {
|
|
return eventStreams().paint.events();
|
|
}
|
|
|
|
rpl::producer<> RpWidgetWrap::alive() const {
|
|
return eventStreams().alive.events();
|
|
}
|
|
|
|
rpl::producer<> RpWidgetWrap::windowDeactivateEvents() const {
|
|
const auto window = rpWidget()->window()->windowHandle();
|
|
Assert(window != nullptr);
|
|
|
|
return base::qt_signal_producer(
|
|
window,
|
|
&QWindow::activeChanged
|
|
) | rpl::filter([=] {
|
|
return !window->isActive();
|
|
});
|
|
}
|
|
|
|
rpl::producer<> RpWidgetWrap::macWindowDeactivateEvents() const {
|
|
#ifdef Q_OS_MAC
|
|
return windowDeactivateEvents();
|
|
#else // Q_OS_MAC
|
|
return rpl::never<rpl::empty_value>();
|
|
#endif // Q_OS_MAC
|
|
}
|
|
|
|
rpl::lifetime &RpWidgetWrap::lifetime() {
|
|
return _lifetime;
|
|
}
|
|
|
|
bool RpWidgetWrap::handleEvent(QEvent *event) {
|
|
Expects(event != nullptr);
|
|
|
|
auto streams = _eventStreams.get();
|
|
if (!streams) {
|
|
return eventHook(event);
|
|
}
|
|
auto that = QPointer<QWidget>();
|
|
const auto allAreObserved = streams->events.has_consumers();
|
|
if (allAreObserved) {
|
|
that = rpWidget();
|
|
streams->events.fire_copy(event);
|
|
if (!that) {
|
|
return true;
|
|
}
|
|
}
|
|
switch (event->type()) {
|
|
case QEvent::Move:
|
|
case QEvent::Resize:
|
|
if (streams->geometry.has_consumers()) {
|
|
if (!allAreObserved) {
|
|
that = rpWidget();
|
|
}
|
|
streams->geometry.fire_copy(rpWidget()->geometry());
|
|
if (!that) {
|
|
return true;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case QEvent::Paint:
|
|
if (streams->paint.has_consumers()) {
|
|
if (!allAreObserved) {
|
|
that = rpWidget();
|
|
}
|
|
const auto rect = static_cast<QPaintEvent*>(event)->rect();
|
|
streams->paint.fire_copy(rect);
|
|
if (!that) {
|
|
return true;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
return eventHook(event);
|
|
}
|
|
|
|
RpWidgetWrap::Initer::Initer(QWidget *parent, bool setZeroGeometry) {
|
|
if (setZeroGeometry) {
|
|
parent->setGeometry(0, 0, 0, 0);
|
|
}
|
|
}
|
|
|
|
void RpWidgetWrap::visibilityChangedHook(bool wasVisible, bool nowVisible) {
|
|
if (nowVisible != wasVisible) {
|
|
if (auto streams = _eventStreams.get()) {
|
|
streams->shown.fire_copy(nowVisible);
|
|
}
|
|
}
|
|
}
|
|
|
|
auto RpWidgetWrap::eventStreams() const -> EventStreams& {
|
|
if (!_eventStreams) {
|
|
_eventStreams = std::make_unique<EventStreams>();
|
|
}
|
|
return *_eventStreams;
|
|
}
|
|
|
|
} // namespace Ui
|