1 | // Copyright (C) 2022 The Qt Company Ltd. |
2 | // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only |
3 | |
4 | #include "qquick3dparticlescaleaffector_p.h" |
5 | #include <qmath.h> |
6 | |
7 | QT_BEGIN_NAMESPACE |
8 | |
9 | /*! |
10 | \qmltype ScaleAffector3D |
11 | \inherits Affector3D |
12 | \inqmlmodule QtQuick3D.Particles3D |
13 | \brief Particle scale affector. |
14 | \since 6.4 |
15 | |
16 | Scale affector scales the particle size based on its lifetime and parameters. |
17 | */ |
18 | |
19 | QQuick3DParticleScaleAffector::QQuick3DParticleScaleAffector(QQuick3DNode *parent) |
20 | : QQuick3DParticleAffector(parent) |
21 | { |
22 | |
23 | } |
24 | |
25 | /*! |
26 | \qmlproperty real ScaleAffector3D::minSize |
27 | |
28 | This property holds the minimum size the affector can scale the particle. |
29 | The default is 1.0. |
30 | */ |
31 | float QQuick3DParticleScaleAffector::minSize() const |
32 | { |
33 | return m_minSize; |
34 | } |
35 | |
36 | /*! |
37 | \qmlproperty real ScaleAffector3D::maxSize |
38 | |
39 | This property holds the maximum size the affector can scale the particle. |
40 | The default is 1.0. |
41 | */ |
42 | float QQuick3DParticleScaleAffector::maxSize() const |
43 | { |
44 | return m_maxSize; |
45 | } |
46 | |
47 | /*! |
48 | \qmlproperty int ScaleAffector3D::duration |
49 | |
50 | This property holds the duration of scaling cycle in milliseconds. |
51 | The default is 1000. |
52 | */ |
53 | int QQuick3DParticleScaleAffector::duration() const |
54 | { |
55 | return m_duration; |
56 | } |
57 | |
58 | /*! |
59 | \qmlproperty enumeration ScaleAffector3D::ScalingType |
60 | |
61 | Defines the scaling type of the affector. |
62 | |
63 | \value ScaleAffector3D.Linear |
64 | The scale is calculated using the easing curve to interpolate between minimum and maximum |
65 | scale size between duration milliseconds and then continues from the minimum size. |
66 | \value ScaleAffector3D.SewSaw |
67 | The scale is calculated using the easing curve to interpolate between minimum and maximum |
68 | scale size between duration milliseconds on a rising edge then continues from maximum to minimum |
69 | on a falling edge. |
70 | \value ScaleAffector3D.SineWave |
71 | The scale follows the sine wave. Easing curve is not used. |
72 | \value ScaleAffector3D.AbsSineWave |
73 | The scale follows the sine wave except negative values are inverted. Easing curve is not used. |
74 | \value ScaleAffector3D.Step |
75 | The scale stays at minimum size until half of the duration milliseconds have passed then steps directly |
76 | to the maximum size. Easing curve is not used. |
77 | \value ScaleAffector3D.SmoothStep |
78 | The scale smootly transitions from minimum to maximum size. Easing curve is not used. |
79 | */ |
80 | |
81 | /*! |
82 | \qmlproperty ScalingType ScaleAffector3D::type |
83 | |
84 | This property holds the scaling type of the affector. The default value is \c Linear. |
85 | */ |
86 | QQuick3DParticleScaleAffector::ScalingType QQuick3DParticleScaleAffector::type() const |
87 | { |
88 | return m_type; |
89 | } |
90 | |
91 | /*! |
92 | \qmlproperty EasingCurve ScaleAffector3D::easingCurve |
93 | |
94 | This property holds the \l {QtQuick::PropertyAnimation::easing}{easing curve} providing |
95 | more fine tuned control on how the scaling occurs. The easing curve is used with \c Linear |
96 | and \c SewSaw scaling types. The default easing curve provides linear value between [0, 1]. |
97 | */ |
98 | QEasingCurve QQuick3DParticleScaleAffector::easingCurve() const |
99 | { |
100 | return m_easing; |
101 | } |
102 | |
103 | void QQuick3DParticleScaleAffector::setMinSize(float size) |
104 | { |
105 | if (qFuzzyCompare(p1: size, p2: m_minSize)) |
106 | return; |
107 | m_minSize = size; |
108 | Q_EMIT minSizeChanged(); |
109 | } |
110 | |
111 | void QQuick3DParticleScaleAffector::setMaxSize(float size) |
112 | { |
113 | if (qFuzzyCompare(p1: size, p2: m_maxSize)) |
114 | return; |
115 | m_maxSize = size; |
116 | Q_EMIT maxSizeChanged(); |
117 | } |
118 | |
119 | void QQuick3DParticleScaleAffector::setDuration(int duration) |
120 | { |
121 | duration = qMax(a: 0, b: duration); |
122 | if (duration == m_duration) |
123 | return; |
124 | m_duration = duration; |
125 | Q_EMIT durationChanged(); |
126 | } |
127 | |
128 | void QQuick3DParticleScaleAffector::setType(ScalingType type) |
129 | { |
130 | if (m_type == type) |
131 | return; |
132 | m_type = type; |
133 | Q_EMIT typeChanged(); |
134 | } |
135 | |
136 | void QQuick3DParticleScaleAffector::setEasingCurve(const QEasingCurve &curve) |
137 | { |
138 | if (m_easing == curve) |
139 | return; |
140 | m_easing = curve; |
141 | Q_EMIT easingCurveChanged(); |
142 | } |
143 | |
144 | void QQuick3DParticleScaleAffector::prepareToAffect() |
145 | { |
146 | |
147 | } |
148 | |
149 | void QQuick3DParticleScaleAffector::affectParticle(const QQuick3DParticleData &, QQuick3DParticleDataCurrent *d, float time) |
150 | { |
151 | float scale = 1.0f; |
152 | |
153 | const auto fract = [](const float v) -> float { |
154 | return v - qFloor(v); |
155 | }; |
156 | const auto lerp = [](const float a, const float b, const float f) -> float { |
157 | return a + (b - a) * f; |
158 | }; |
159 | const auto smoothstep = [](const float a, const float b, const float f) -> float { |
160 | return a + (b - a) * f * f * (3.0f - 2.0f * f); |
161 | }; |
162 | |
163 | float pos = fract(time / float(m_duration * 0.001f)); |
164 | switch (m_type) { |
165 | case Linear: |
166 | scale = lerp(m_minSize, m_maxSize, m_easing.valueForProgress(progress: pos)); |
167 | scale = qMax(a: scale, b: 0.0f); |
168 | break; |
169 | case SewSaw: |
170 | if (pos < 0.5f) |
171 | scale = lerp(m_minSize, m_maxSize, m_easing.valueForProgress(progress: pos * 2.0f)); |
172 | else |
173 | scale = lerp(m_maxSize, m_minSize, m_easing.valueForProgress(progress: (pos - 0.5) * 2.0f)); |
174 | scale = qMax(a: scale, b: 0.0f); |
175 | break; |
176 | case SineWave: |
177 | scale = m_minSize + (m_maxSize - m_minSize) * (1.0f + qSin(v: 2.0f * M_PI * pos)) * 0.5f; |
178 | break; |
179 | case AbsSineWave: |
180 | scale = m_minSize + (m_maxSize - m_minSize) * qAbs(t: qSin(v: 2.0f * M_PI * pos)); |
181 | break; |
182 | case Step: |
183 | if (pos < 0.5f) |
184 | scale = m_minSize; |
185 | else |
186 | scale = m_maxSize; |
187 | break; |
188 | case SmoothStep: |
189 | scale = smoothstep(m_minSize, m_maxSize, pos); |
190 | break; |
191 | } |
192 | |
193 | d->scale *= scale; |
194 | } |
195 | |
196 | QT_END_NAMESPACE |
197 | |