Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qtgradientutils.cpp
Go to the documentation of this file.
1// Copyright (C) 2016 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3
6
7#include <QtGui/qbrush.h>
8#include <QtGui/qpainter.h>
9
10#include <QtXml/qdom.h>
11
12#include <QtCore/qdebug.h>
13
15
16using namespace Qt::StringLiterals;
17
18static QString gradientTypeToString(QGradient::Type type)
19{
20 if (type == QGradient::LinearGradient)
21 return "LinearGradient"_L1;
22 if (type == QGradient::RadialGradient)
23 return "RadialGradient"_L1;
24 if (type == QGradient::ConicalGradient)
25 return "ConicalGradient"_L1;
26 return "NoGradient"_L1;
27}
28
29static QGradient::Type stringToGradientType(const QString &name)
30{
31 if (name == "LinearGradient"_L1)
32 return QGradient::LinearGradient;
33 if (name == "RadialGradient"_L1)
34 return QGradient::RadialGradient;
35 if (name == "ConicalGradient"_L1)
36 return QGradient::ConicalGradient;
37 return QGradient::NoGradient;
38}
39
40static QString gradientSpreadToString(QGradient::Spread spread)
41{
42 if (spread == QGradient::PadSpread)
43 return "PadSpread"_L1;
44 if (spread == QGradient::RepeatSpread)
45 return "RepeatSpread"_L1;
46 if (spread == QGradient::ReflectSpread)
47 return "ReflectSpread"_L1;
48 return "PadSpread"_L1;
49}
50
51static QGradient::Spread stringToGradientSpread(const QString &name)
52{
53 if (name == "PadSpread"_L1)
54 return QGradient::PadSpread;
55 if (name == "RepeatSpread"_L1)
56 return QGradient::RepeatSpread;
57 if (name == "ReflectSpread"_L1)
58 return QGradient::ReflectSpread;
59 return QGradient::PadSpread;
60}
61
62static QString gradientCoordinateModeToString(QGradient::CoordinateMode mode)
63{
64 if (mode == QGradient::LogicalMode)
65 return "LogicalMode"_L1;
66 if (mode == QGradient::StretchToDeviceMode)
67 return "StretchToDeviceMode"_L1;
68 if (mode == QGradient::ObjectBoundingMode)
69 return "ObjectBoundingMode"_L1;
70 return "StretchToDeviceMode"_L1;
71}
72
74{
75 if (name == "LogicalMode"_L1)
76 return QGradient::LogicalMode;
77 if (name == "StretchToDeviceMode"_L1)
78 return QGradient::StretchToDeviceMode;
79 if (name == "ObjectBoundingMode"_L1)
80 return QGradient::ObjectBoundingMode;
81 return QGradient::StretchToDeviceMode;
82}
83
84static QDomElement saveColor(QDomDocument &doc, QColor color)
85{
86 QDomElement colorElem = doc.createElement("colorData"_L1);
87
88 colorElem.setAttribute("r"_L1, QString::number(color.red()));
89 colorElem.setAttribute("g"_L1, QString::number(color.green()));
90 colorElem.setAttribute("b"_L1, QString::number(color.blue()));
91 colorElem.setAttribute("a"_L1, QString::number(color.alpha()));
92
93 return colorElem;
94}
95
96static QDomElement saveGradientStop(QDomDocument &doc, const QGradientStop &stop)
97{
98 QDomElement stopElem = doc.createElement("stopData"_L1);
99
100 stopElem.setAttribute("position"_L1, QString::number(stop.first));
101
102 const QDomElement colorElem = saveColor(doc, stop.second);
103 stopElem.appendChild(colorElem);
104
105 return stopElem;
106}
107
108static QDomElement saveGradient(QDomDocument &doc, const QGradient &gradient)
109{
110 QDomElement gradElem = doc.createElement("gradientData"_L1);
111
112 const QGradient::Type type = gradient.type();
113 gradElem.setAttribute("type"_L1, gradientTypeToString(type));
114 gradElem.setAttribute("spread"_L1, gradientSpreadToString(gradient.spread()));
115 gradElem.setAttribute("coordinateMode"_L1, gradientCoordinateModeToString(gradient.coordinateMode()));
116
117 const QGradientStops stops = gradient.stops();
118 for (const QGradientStop &stop : stops)
119 gradElem.appendChild(saveGradientStop(doc, stop));
120
121 if (type == QGradient::LinearGradient) {
122 const QLinearGradient &g = *static_cast<const QLinearGradient *>(&gradient);
123 gradElem.setAttribute("startX"_L1, QString::number(g.start().x()));
124 gradElem.setAttribute("startY"_L1, QString::number(g.start().y()));
125 gradElem.setAttribute("endX"_L1, QString::number(g.finalStop().x()));
126 gradElem.setAttribute("endY"_L1, QString::number(g.finalStop().y()));
127 } else if (type == QGradient::RadialGradient) {
128 const QRadialGradient &g = *static_cast<const QRadialGradient *>(&gradient);
129 gradElem.setAttribute("centerX"_L1, QString::number(g.center().x()));
130 gradElem.setAttribute("centerY"_L1, QString::number(g.center().y()));
131 gradElem.setAttribute("focalX"_L1, QString::number(g.focalPoint().x()));
132 gradElem.setAttribute("focalY"_L1, QString::number(g.focalPoint().y()));
133 gradElem.setAttribute("radius"_L1, QString::number(g.radius()));
134 } else if (type == QGradient::ConicalGradient) {
135 const QConicalGradient &g = *static_cast<const QConicalGradient*>(&gradient);
136 gradElem.setAttribute("centerX"_L1, QString::number(g.center().x()));
137 gradElem.setAttribute("centerY"_L1, QString::number(g.center().y()));
138 gradElem.setAttribute("angle"_L1, QString::number(g.angle()));
139 }
140
141 return gradElem;
142}
143
144static QColor loadColor(const QDomElement &elem)
145{
146 if (elem.tagName() != "colorData"_L1)
147 return QColor();
148
149 return QColor(elem.attribute("r"_L1).toInt(),
150 elem.attribute("g"_L1).toInt(),
151 elem.attribute("b"_L1).toInt(),
152 elem.attribute("a"_L1).toInt());
153}
154
155static QGradientStop loadGradientStop(const QDomElement &elem)
156{
157 if (elem.tagName() != "stopData"_L1)
158 return QGradientStop();
159
160 const qreal pos = static_cast<qreal>(elem.attribute("position"_L1).toDouble());
161 return std::make_pair(pos, loadColor(elem.firstChild().toElement()));
162}
163
164static QGradient loadGradient(const QDomElement &elem)
165{
166 if (elem.tagName() != "gradientData"_L1)
167 return QLinearGradient();
168
169 const QGradient::Type type = stringToGradientType(elem.attribute("type"_L1));
170 const QGradient::Spread spread = stringToGradientSpread(elem.attribute("spread"_L1));
171 const QGradient::CoordinateMode mode = stringToGradientCoordinateMode(elem.attribute("coordinateMode"_L1));
172
173 QGradient gradient = QLinearGradient();
174
175 if (type == QGradient::LinearGradient) {
176 QLinearGradient g;
177 g.setStart(elem.attribute("startX"_L1).toDouble(), elem.attribute("startY"_L1).toDouble());
178 g.setFinalStop(elem.attribute("endX"_L1).toDouble(), elem.attribute("endY"_L1).toDouble());
179 gradient = g;
180 } else if (type == QGradient::RadialGradient) {
181 QRadialGradient g;
182 g.setCenter(elem.attribute("centerX"_L1).toDouble(), elem.attribute("centerY"_L1).toDouble());
183 g.setFocalPoint(elem.attribute("focalX"_L1).toDouble(), elem.attribute("focalY"_L1).toDouble());
184 g.setRadius(elem.attribute("radius"_L1).toDouble());
185 gradient = g;
186 } else if (type == QGradient::ConicalGradient) {
187 QConicalGradient g;
188 g.setCenter(elem.attribute("centerX"_L1).toDouble(), elem.attribute("centerY"_L1).toDouble());
189 g.setAngle(elem.attribute("angle"_L1).toDouble());
190 gradient = g;
191 }
192
193 QDomElement stopElem = elem.firstChildElement();
194 while (!stopElem.isNull()) {
195 QGradientStop stop = loadGradientStop(stopElem);
196
197 gradient.setColorAt(stop.first, stop.second);
198
199 stopElem = stopElem.nextSiblingElement();
200 }
201
202 gradient.setSpread(spread);
203 gradient.setCoordinateMode(mode);
204
205 return gradient;
206}
207
208QString QtGradientUtils::saveState(const QtGradientManager *manager)
209{
210 QDomDocument doc;
211
212 QDomElement rootElem = doc.createElement("gradients"_L1);
213
214 QMap<QString, QGradient> grads = manager->gradients();
215 for (auto itGrad = grads.cbegin(), end = grads.cend(); itGrad != end; ++itGrad) {
216 QDomElement idElem = doc.createElement("gradient"_L1);
217 idElem.setAttribute("name"_L1, itGrad.key());
218 QDomElement gradElem = saveGradient(doc, itGrad.value());
219 idElem.appendChild(gradElem);
220
221 rootElem.appendChild(idElem);
222 }
223
224 doc.appendChild(rootElem);
225
226 return doc.toString();
227}
228
229void QtGradientUtils::restoreState(QtGradientManager *manager, const QString &state)
230{
231 manager->clear();
232
233 QDomDocument doc;
234 doc.setContent(state);
235
236 QDomElement rootElem = doc.documentElement();
237
238 QDomElement gradElem = rootElem.firstChildElement();
239 while (!gradElem.isNull()) {
240 const QString name = gradElem.attribute("name"_L1);
241 const QGradient gradient = loadGradient(gradElem.firstChildElement());
242
243 manager->addGradient(name, gradient);
244 gradElem = gradElem.nextSiblingElement();
245 }
246}
247
248QPixmap QtGradientUtils::gradientPixmap(const QGradient &gradient, QSize size,
249 bool checkeredBackground)
250{
251 QImage image(size, QImage::Format_ARGB32);
252 QPainter p(&image);
253 p.setCompositionMode(QPainter::CompositionMode_Source);
254
255 if (checkeredBackground) {
256 int pixSize = 20;
257 QPixmap pm(2 * pixSize, 2 * pixSize);
258
259 QPainter pmp(&pm);
260 pmp.fillRect(0, 0, pixSize, pixSize, Qt::lightGray);
261 pmp.fillRect(pixSize, pixSize, pixSize, pixSize, Qt::lightGray);
262 pmp.fillRect(0, pixSize, pixSize, pixSize, Qt::darkGray);
263 pmp.fillRect(pixSize, 0, pixSize, pixSize, Qt::darkGray);
264
265 p.setBrushOrigin((size.width() % pixSize + pixSize) / 2, (size.height() % pixSize + pixSize) / 2);
266 p.fillRect(0, 0, size.width(), size.height(), pm);
267 p.setBrushOrigin(0, 0);
268 p.setCompositionMode(QPainter::CompositionMode_SourceOver);
269 }
270
271 const qreal scaleFactor = 0.999999;
272 p.scale(scaleFactor, scaleFactor);
273 QGradient grad = gradient;
274 grad.setCoordinateMode(QGradient::StretchToDeviceMode);
275 p.fillRect(QRect(0, 0, size.width(), size.height()), grad);
276 p.drawRect(QRect(0, 0, size.width() - 1, size.height() - 1));
277
278 return QPixmap::fromImage(image);
279}
280
281static QString styleSheetFillName(const QGradient &gradient)
282{
283 QString result;
284
285 switch (gradient.type()) {
286 case QGradient::LinearGradient:
287 result += "qlineargradient"_L1;
288 break;
289 case QGradient::RadialGradient:
290 result += "qradialgradient"_L1;
291 break;
292 case QGradient::ConicalGradient:
293 result += "qconicalgradient"_L1;
294 break;
295 default:
296 qWarning() << "QtGradientUtils::styleSheetFillName(): gradient type" << gradient.type() << "not supported!";
297 break;
298 }
299
300 return result;
301}
302
303static QStringList styleSheetParameters(const QGradient &gradient)
304{
305 QStringList result;
306
307 if (gradient.type() != QGradient::ConicalGradient) {
308 QString spread;
309 switch (gradient.spread()) {
310 case QGradient::PadSpread:
311 spread = "pad"_L1;
312 break;
313 case QGradient::ReflectSpread:
314 spread = "reflect"_L1;
315 break;
316 case QGradient::RepeatSpread:
317 spread = "repeat"_L1;
318 break;
319 default:
320 qWarning() << "QtGradientUtils::styleSheetParameters(): gradient spread" << gradient.spread() << "not supported!";
321 break;
322 }
323 result << "spread:"_L1 + spread;
324 }
325
326 switch (gradient.type()) {
327 case QGradient::LinearGradient: {
328 const QLinearGradient *linearGradient = static_cast<const QLinearGradient*>(&gradient);
329 result << "x1:"_L1 + QString::number(linearGradient->start().x())
330 << "y1:"_L1 + QString::number(linearGradient->start().y())
331 << "x2:"_L1 + QString::number(linearGradient->finalStop().x())
332 << "y2:"_L1 + QString::number(linearGradient->finalStop().y());
333 break;
334 }
335 case QGradient::RadialGradient: {
336 const QRadialGradient *radialGradient = static_cast<const QRadialGradient*>(&gradient);
337 result << "cx:"_L1 + QString::number(radialGradient->center().x())
338 << "cy:"_L1 + QString::number(radialGradient->center().y())
339 << "radius:"_L1 + QString::number(radialGradient->radius())
340 << "fx:"_L1 + QString::number(radialGradient->focalPoint().x())
341 << "fy:"_L1 + QString::number(radialGradient->focalPoint().y());
342 break;
343 }
344 case QGradient::ConicalGradient: {
345 const QConicalGradient *conicalGradient = static_cast<const QConicalGradient*>(&gradient);
346 result << "cx:"_L1 + QString::number(conicalGradient->center().x())
347 << "cy:"_L1 + QString::number(conicalGradient->center().y())
348 << "angle:"_L1 + QString::number(conicalGradient->angle());
349 break;
350 }
351 default:
352 qWarning() << "QtGradientUtils::styleSheetParameters(): gradient type" << gradient.type() << "not supported!";
353 break;
354 }
355
356 return result;
357}
358
359static QStringList styleSheetStops(const QGradient &gradient)
360{
361 QStringList result;
362 const QGradientStops &stops = gradient.stops();
363 for (const QGradientStop &stop : stops) {
364 const QColor color = stop.second;
365
366 const QString stopDescription = "stop:"_L1 + QString::number(stop.first) + " rgba("_L1
367 + QString::number(color.red()) + ", "_L1
368 + QString::number(color.green()) + ", "_L1
369 + QString::number(color.blue()) + ", "_L1
370 + QString::number(color.alpha()) + QLatin1Char(')');
371 result << stopDescription;
372 }
373
374 return result;
375}
376
377QString QtGradientUtils::styleSheetCode(const QGradient &gradient)
378{
379 QStringList gradientParameters;
380 gradientParameters << styleSheetParameters(gradient) << styleSheetStops(gradient);
381
382 return styleSheetFillName(gradient) + QLatin1Char('(') + gradientParameters.join(", "_L1) + QLatin1Char(')');
383}
384
385QT_END_NAMESPACE
Combined button and popup list for selecting options.
static QString gradientSpreadToString(QGradient::Spread spread)
static QStringList styleSheetStops(const QGradient &gradient)
static QDomElement saveColor(QDomDocument &doc, QColor color)
static QGradient loadGradient(const QDomElement &elem)
static QGradient::Spread stringToGradientSpread(const QString &name)
static QString styleSheetFillName(const QGradient &gradient)
static QString gradientCoordinateModeToString(QGradient::CoordinateMode mode)
static QStringList styleSheetParameters(const QGradient &gradient)
static QColor loadColor(const QDomElement &elem)
static QGradient::Type stringToGradientType(const QString &name)
static QGradient::CoordinateMode stringToGradientCoordinateMode(const QString &name)
static QGradientStop loadGradientStop(const QDomElement &elem)
static QString gradientTypeToString(QGradient::Type type)
static QDomElement saveGradient(QDomDocument &doc, const QGradient &gradient)
static QDomElement saveGradientStop(QDomDocument &doc, const QGradientStop &stop)