Reputation: 15
I wanna make something like Material.accent, where I can change in the parent and children getting the parent property definition.
Here is the way I did at this time, but I can't find any information about it in the documentation. I know it is possible, Material Style uses this method and other things like font property too.
class MyThemeAttached : public QObject {
Q_OBJECT
Q_PROPERTY(QColor color READ color WRITE setColor NOTIFY colorChanged)
QML_ANONYMOUS
public:
explicit MyThemeAttached(QObject* parent = nullptr)
: QObject(parent)
, m_color("#FF0000"){}
QColor color() const;
void setColor(const QColor color);
signals:
void backgroundChanged(QColor background);
private:
QColor m_color;
};
class MyTheme : public QObject
{
Q_OBJECT
QML_ATTACHED(MyThemeAttached)
QML_ELEMENT
public:
explicit MyTheme(QObject *parent = nullptr);
static MyThemeAttached *qmlAttachedProperties(QObject *object) {
return new MyThemeAttached(object);
}
};
ApplicationWindow {
id: root
visible: true
width: 800
height: 600
title: qsTr("Window")
MyCustomProperty.color: "orange"
Rectangle {
color: MyCustomProperty.color
}
}
Upvotes: 0
Views: 848
Reputation: 24386
In Qt 6.5 you can use QQuickAttachedPropertyPropagator for this. From the documentation:
To use QQuickAttachedPropertyPropagator:
- Derive from it
- Call initialize() in the constructor
- Define set/inherit/propagate/reset functions for each property as needed
- Reimplement attachedParentChange() to handle property inheritance
For reference, the relevant files from the example are below.
// Copyright (C) 2022 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
#ifndef MYSTYLE_H
#define MYSTYLE_H
#include <QColor>
#include <QQmlEngine>
#include <QQuickAttachedPropertyPropagator>
#include "mystyle_export.h"
class MYSTYLE_EXPORT MyStyle : public QQuickAttachedPropertyPropagator
{
Q_OBJECT
// Provide a RESET function in order to allow an item to set MyStyle.theme to undefined
// in order to use its parent's (or global) theme after one was explicitly set on it.
Q_PROPERTY(Theme theme READ theme WRITE setTheme RESET resetTheme NOTIFY themeChanged FINAL)
// As the values of these properties only depend on the theme, they can all use the theme
// property's change signal.
Q_PROPERTY(QColor windowColor READ windowColor NOTIFY themeChanged FINAL)
Q_PROPERTY(QColor windowTextColor READ windowTextColor NOTIFY themeChanged FINAL)
Q_PROPERTY(QColor buttonColor READ buttonColor NOTIFY themeChanged FINAL)
Q_PROPERTY(QColor buttonTextColor READ buttonTextColor NOTIFY themeChanged FINAL)
Q_PROPERTY(QColor toolBarColor READ toolBarColor NOTIFY themeChanged FINAL)
Q_PROPERTY(QColor popupColor READ popupColor NOTIFY themeChanged FINAL)
Q_PROPERTY(QColor popupBorderColor READ popupBorderColor NOTIFY themeChanged FINAL)
Q_PROPERTY(QColor backgroundDimColor READ backgroundDimColor NOTIFY themeChanged FINAL)
QML_ELEMENT
QML_ATTACHED(MyStyle)
QML_UNCREATABLE("")
QML_ADDED_IN_VERSION(1, 0)
public:
enum Theme {
Light,
Dark
};
Q_ENUM(Theme)
explicit MyStyle(QObject *parent = nullptr);
static MyStyle *qmlAttachedProperties(QObject *object);
Theme theme() const;
void setTheme(Theme theme);
void inheritTheme(Theme theme);
void propagateTheme();
void resetTheme();
void themeChange();
QColor windowColor() const;
QColor windowTextColor() const;
QColor buttonColor() const;
QColor buttonTextColor() const;
QColor toolBarColor() const;
QColor popupColor() const;
QColor popupBorderColor() const;
QColor backgroundDimColor() const;
Q_SIGNALS:
void themeChanged();
protected:
void attachedParentChange(QQuickAttachedPropertyPropagator *newParent, QQuickAttachedPropertyPropagator *oldParent) override;
private:
// Whether a color value was explicitly set on the specific object that this attached style object represents.
bool m_explicitTheme = false;
// The actual values for this item, whether explicit, inherited or globally set.
Theme m_theme = Light;
};
#endif // MYSTYLE_H
// Copyright (C) 2022 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
#include "mystyle.h"
// If no value was inherited from a parent or explicitly set, the "global" values are used.
static MyStyle::Theme globalTheme = MyStyle::Light;
MyStyle::MyStyle(QObject *parent)
: QQuickAttachedPropertyPropagator(parent)
, m_theme(globalTheme)
{
// A static function could be called here that reads globalTheme from a
// settings file once at startup. That value would override the global
// value. This is similar to what the Imagine and Material styles do, for
// example.
initialize();
}
MyStyle *MyStyle::qmlAttachedProperties(QObject *object)
{
return new MyStyle(object);
}
MyStyle::Theme MyStyle::theme() const
{
return m_theme;
}
void MyStyle::setTheme(Theme theme)
{
// When this function is called, we know that the user has explicitly
// set a theme on this attached object. We set this to true even if
// the effective theme didn't change, because it's important that
// the user's specified value is respected (and not inherited from
// from the parent).
m_explicitTheme = true;
if (m_theme == theme)
return;
m_theme = theme;
propagateTheme();
themeChange();
}
void MyStyle::inheritTheme(Theme theme)
{
if (m_explicitTheme || m_theme == theme)
return;
m_theme = theme;
propagateTheme();
themeChange();
}
void MyStyle::propagateTheme()
{
const auto styles = attachedChildren();
for (QQuickAttachedPropertyPropagator *child : styles) {
MyStyle *myStyle = qobject_cast<MyStyle *>(child);
if (myStyle)
myStyle->inheritTheme(m_theme);
}
}
void MyStyle::resetTheme()
{
if (!m_explicitTheme)
return;
m_explicitTheme = false;
MyStyle *myStyle = qobject_cast<MyStyle *>(attachedParent());
inheritTheme(myStyle ? myStyle->theme() : globalTheme);
}
void MyStyle::themeChange()
{
emit themeChanged();
// Emit any other change signals for properties that depend on the theme here...
}
QColor MyStyle::windowColor() const
{
return m_theme == Light ? QColor::fromRgb(0xf0f0f0) : QColor::fromRgb(0x303030);
}
QColor MyStyle::windowTextColor() const
{
return m_theme == Light ? QColor::fromRgb(0x5c5c5c) : QColor::fromRgb(0xe0e0e0);
}
QColor MyStyle::buttonColor() const
{
return m_theme == Light ? QColor::fromRgb(0xc2e1ff) : QColor::fromRgb(0x74bbff);
}
QColor MyStyle::buttonTextColor() const
{
return m_theme == Light ? QColor::fromRgb(0x5c5c5c) : QColor::fromRgb(0xffffff);
}
QColor MyStyle::toolBarColor() const
{
return m_theme == Light ? QColor::fromRgb(0x4da6ff) : QColor::fromRgb(0x0066cc);
}
QColor MyStyle::popupColor() const
{
return windowColor().lighter(120);
}
QColor MyStyle::popupBorderColor() const
{
const QColor winColor = windowColor();
return m_theme == Light ? winColor.darker(140) : winColor.lighter(140);
}
QColor MyStyle::backgroundDimColor() const
{
const QColor winColor = windowColor().darker();
return QColor::fromRgb(winColor.red(), winColor.green(), winColor.blue(), 100);
}
void MyStyle::attachedParentChange(QQuickAttachedPropertyPropagator *newParent, QQuickAttachedPropertyPropagator *oldParent)
{
Q_UNUSED(oldParent);
MyStyle *attachedParentStyle = qobject_cast<MyStyle *>(newParent);
if (attachedParentStyle) {
inheritTheme(attachedParentStyle->theme());
// Do any other inheriting here...
}
}
Upvotes: 1
Reputation: 4168
Why not look at the code that Material has? I introduce you to Woboq.org.
Here you can see that the Material theme actually pro-actively pushes the theme on the children:
void QQuickMaterialStyle::propagateTheme()
{
const auto styles = attachedChildren();
for (QQuickAttachedObject *child : styles) {
QQuickMaterialStyle *material = qobject_cast<QQuickMaterialStyle *>(child);
if (material)
material->inheritTheme(m_theme);
}
}
Upvotes: 1