// Copyright (C) 2020 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only #ifndef QQUICKPALETTEPROVIDERPRIVATEBASE_H #define QQUICKPALETTEPROVIDERPRIVATEBASE_H // // W A R N I N G // ------------- // // This file is not part of the Qt API. It exists purely as an // implementation detail. This header file may change from version to // version without notice, or even be removed. // // We mean it. // #include #include #include #include QT_BEGIN_NAMESPACE class QWindow; class QQuickWindow; class QQuickWindowPrivate; class QQuickItem; class QQuickItemPrivate; class QQuickPopup; class QQuickPopupPrivate; /*! \internal Implements all required operations with palette. I -- is interface class (e.g. QQuickItem). Impl -- is implementation class (e.g. QQuickItemPrivate). To use this class you need to inherit implementation class from it. */ template class QQuickPaletteProviderPrivateBase : public QQuickAbstractPaletteProvider { static_assert(std::is_base_of{}, "The interface class must inherit QObject"); public: virtual ~QQuickPaletteProviderPrivateBase() = default; /*! \internal Get current palette. \note Palette might be lazily allocated. Signal \p paletteCreated() will be emitted by an object of interface class in this case. \note This function doesn't ask an object of interface class to emit paletteChanged() signal in order to avoid problems with property bindigns. */ virtual QQuickPalette *palette() const; /*! \internal Set new palette. Doesn't transfer ownership. */ virtual void setPalette(QQuickPalette *p); /*! \internal Reset palette to the default one. */ virtual void resetPalette(); /*! \internal Check if everything is internally allocated and palette exists. Use before call \p palette() to avoid unnecessary allocations. */ virtual bool providesPalette() const; /*! \internal The default palette for this component. */ QPalette defaultPalette() const override; /*! \internal The parent palette for this component. Can be null. */ QPalette parentPalette(const QPalette &fallbackPalette) const override; /*! \internal Inherit from \p parentPalette. This function is also called when either parent or window of this item is changed. */ void inheritPalette(const QPalette &parentPalette); /*! \internal Updates children palettes. The default implementation invokes inheritPalette for all visual children. This function is also called when palette is changed (signal changed() is emitted). */ virtual void updateChildrenPalettes(const QPalette &parentPalette); protected: void setCurrentColorGroup(); private: using PalettePtr = std::unique_ptr; using Self = QQuickPaletteProviderPrivateBase; void registerPalette(PalettePtr palette); bool isValidPalette(const QQuickPalette *palette) const; QQuickPalette *windowPalette() const; void connectItem(); const I *itemWithPalette() const; I *itemWithPalette(); QQuickPalette *paletteData() const; QPalette toQPalette() const; private: PalettePtr m_palette; }; template QQuickPalette *QQuickPaletteProviderPrivateBase::palette() const { if (!providesPalette()) { // It's required to create a new palette without parent, // because this method can be called from the rendering thread const_cast(this)->registerPalette(std::make_unique()); Q_EMIT const_cast(this)->itemWithPalette()->paletteCreated(); } return paletteData(); } template bool QQuickPaletteProviderPrivateBase::isValidPalette(const QQuickPalette *palette) const { if (!palette) { qWarning("Palette cannot be null."); return false; } if (providesPalette() && paletteData() == palette) { qWarning("Self assignment makes no sense."); return false; } return true; } template void QQuickPaletteProviderPrivateBase::setPalette(QQuickPalette *p) { if (isValidPalette(p)) { palette()->fromQPalette(p->toQPalette()); } } template void QQuickPaletteProviderPrivateBase::resetPalette() { paletteData()->reset(); } template bool QQuickPaletteProviderPrivateBase::providesPalette() const { return !!m_palette; } template QPalette QQuickPaletteProviderPrivateBase::defaultPalette() const { return QPalette(); } template inline constexpr bool isRootWindow() { return std::is_base_of_v; } template void QQuickPaletteProviderPrivateBase::registerPalette(PalettePtr palette) { if constexpr (!isRootWindow()) { // Connect item only once, before initial data allocation if (!providesPalette()) { connectItem(); } } m_palette = std::move(palette); m_palette->setPaletteProvider(this); m_palette->inheritPalette(parentPalette(defaultPalette())); setCurrentColorGroup(); // In order to avoid extra noise, we should connect // the following signals only after everything is already setup I::connect(paletteData(), &QQuickPalette::changed, itemWithPalette(), &I::paletteChanged); I::connect(paletteData(), &QQuickPalette::changed, itemWithPalette(), [this]{ updateChildrenPalettes(toQPalette()); }); } template struct dependent_false : std::false_type {}; template decltype(auto) getPrivateImpl(I &t) { return Impl::get(&t); } template decltype(auto) getPrivate(T &item) { if constexpr (std::is_base_of_v) { return getPrivateImpl(item); } else if constexpr (std::is_base_of_v) { return getPrivateImpl(item); } else { static_assert (dependent_false::value, "Extend please."); } } template QQuickPalette *QQuickPaletteProviderPrivateBase::windowPalette() const { if constexpr (!isRootWindow()) { if (auto window = itemWithPalette()->window()) { if (getPrivate(*window)->providesPalette()) { return getPrivate(*window)->palette(); } } } return nullptr; } template QPalette QQuickPaletteProviderPrivateBase::parentPalette(const QPalette &fallbackPalette) const { if constexpr (!isRootWindow()) { // Popups should always inherit from their window, even child popups: QTBUG-115707. if (!std::is_base_of_v) { for (auto parentItem = itemWithPalette()->parentItem(); parentItem; parentItem = parentItem->parentItem()) { // Don't allocate a new palette here. Use only if it's already pre allocated if (parentItem && getPrivate(*parentItem)->providesPalette()) { return getPrivate(*parentItem)->palette()->toQPalette(); } } } if (auto wp = windowPalette()) { return wp->toQPalette(); } } return fallbackPalette; } template const QQuickItem* rootItem(const I &item) { if constexpr (isRootWindow()) { return item.contentItem(); } else if constexpr (std::is_base_of_v) { return nullptr; } else { return &item; } } template void QQuickPaletteProviderPrivateBase::inheritPalette(const QPalette &parentPalette) { if (providesPalette()) { // If palette is changed, then this function will be invoked // for all children because of connection with signal changed() palette()->inheritPalette(parentPalette); } else { // Otherwise, just propagate parent palette to all children updateChildrenPalettes(parentPalette); } } template void QQuickPaletteProviderPrivateBase::setCurrentColorGroup() { if constexpr (!isRootWindow()) { if (providesPalette()) { const bool enabled = itemWithPalette()->isEnabled(); const auto window = itemWithPalette()->window(); const bool active = window ? window->isActive() : true; palette()->setCurrentGroup(enabled ? (active ? QPalette::Active : QPalette::Inactive) : QPalette::Disabled); } } } template void QQuickPaletteProviderPrivateBase::updateChildrenPalettes(const QPalette &parentPalette) { if constexpr (std::is_same_v && std::is_same_v) { /* QQuickWindowPrivate instantiates this template, but does not include QQuickItemPrivate * This causes an error with the QQuickItemPrivate::inheritPalette call below on MSVC in * static builds, as QQuickItemPrivate is incomplete. To work around this situation, we do * nothing in this instantiation of updateChildrenPalettes and instead add an override in * QQuickWindowPrivate, which does the correct thing. */ Q_UNREACHABLE_RETURN(); // You are not supposed to call this function } else { if (auto root = rootItem(*itemWithPalette())) { for (auto &&child : root->childItems()) { if (Q_LIKELY(child)) { getPrivate(*child)->inheritPalette(parentPalette); } } } } } template void QQuickPaletteProviderPrivateBase::connectItem() { Q_ASSERT(itemWithPalette()); if constexpr (!isRootWindow()) { // Item with palette has the same lifetime as its implementation that inherits this class I::connect(itemWithPalette(), &I::parentChanged, itemWithPalette(), [this]() { inheritPalette(parentPalette(defaultPalette())); }); I::connect(itemWithPalette(), &I::windowChanged, itemWithPalette(), [this]() { inheritPalette(parentPalette(defaultPalette())); }); I::connect(itemWithPalette(), &I::enabledChanged, itemWithPalette(), [this]() { setCurrentColorGroup(); }); } } template const I *QQuickPaletteProviderPrivateBase::itemWithPalette() const { static_assert(std::is_base_of{}, "The Impl class must inherit QObjectData"); return static_cast(static_cast(this)->q_ptr); } template I *QQuickPaletteProviderPrivateBase::itemWithPalette() { return const_cast(const_cast(this)->itemWithPalette()); } template QQuickPalette *QQuickPaletteProviderPrivateBase::paletteData() const { Q_ASSERT(m_palette); return m_palette.get(); } template QPalette QQuickPaletteProviderPrivateBase::toQPalette() const { return palette()->toQPalette(); } QT_END_NAMESPACE #endif // QQUICKPALETTEPROVIDERPRIVATEBASE_H