// 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 #include #include class TWidgetPrivate : public QWidgetPrivate { public: #if QT_VERSION >= QT_VERSION_CHECK(6, 4, 0) QPlatformBackingStoreRhiConfig rhiConfig() const override { const auto q = static_cast(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(*(new TWidgetPrivate), parent, {}) { [[maybe_unused]] static const auto Once = [] { auto format = QSurfaceFormat::defaultFormat(); format.setSwapInterval(0); QSurfaceFormat::setDefaultFormat(format); return true; }(); } namespace Ui { namespace { [[nodiscard]] std::vector> GetChildWidgets( not_null widget) { const auto &children = widget->children(); auto result = std::vector>(); result.reserve(children.size()); for (const auto child : children) { if (child && child->isWidgetType()) { result.push_back(static_cast(child)); } } return result; } } // namespace void ToggleChildrenVisibility(not_null widget, bool visible) { for (const auto &child : GetChildWidgets(widget)) { if (child) { child->setVisible(visible); } } } void ResizeFitChild( not_null parent, not_null 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> RpWidgetWrap::events() const { auto &stream = eventStreams().events; return stream.events(); } rpl::producer RpWidgetWrap::geometryValue() const { auto &stream = eventStreams().geometry; return stream.events_starting_with_copy(rpWidget()->geometry()); } rpl::producer RpWidgetWrap::sizeValue() const { return geometryValue() | rpl::map([](QRect &&value) { return value.size(); }) | rpl::distinct_until_changed(); } rpl::producer RpWidgetWrap::heightValue() const { return geometryValue() | rpl::map([](QRect &&value) { return value.height(); }) | rpl::distinct_until_changed(); } rpl::producer RpWidgetWrap::widthValue() const { return geometryValue() | rpl::map([](QRect &&value) { return value.width(); }) | rpl::distinct_until_changed(); } rpl::producer RpWidgetWrap::positionValue() const { return geometryValue() | rpl::map([](QRect &&value) { return value.topLeft(); }) | rpl::distinct_until_changed(); } rpl::producer RpWidgetWrap::leftValue() const { return geometryValue() | rpl::map([](QRect &&value) { return value.left(); }) | rpl::distinct_until_changed(); } rpl::producer RpWidgetWrap::topValue() const { return geometryValue() | rpl::map([](QRect &&value) { return value.top(); }) | rpl::distinct_until_changed(); } rpl::producer RpWidgetWrap::desiredHeightValue() const { return heightValue(); } rpl::producer RpWidgetWrap::shownValue() const { auto &stream = eventStreams().shown; return stream.events_starting_with(!rpWidget()->isHidden()); } rpl::producer 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(); #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(); 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(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(); } return *_eventStreams; } } // namespace Ui