Perceptual Color

colorpatch.cpp
1// SPDX-FileCopyrightText: Lukas Sommer <sommerluk@gmail.com>
2// SPDX-License-Identifier: BSD-2-Clause OR MIT
3
4// We actually delete the PIMPL's copy constructor in the private header using
5// Q_DISABLE_COPY(ColorPatchPrivate)
6// Nevertheless, cppcheck worries about a default copy constructor:
7// "Class 'ColorPatchPrivate' does not have a copy constructor which is
8// recommended since it has dynamic memory/resource allocation(s).
9// (CWE-398)"
10// So we suppress this warning:
11// cppcheck-suppress-file noCopyConstructor
12
13// Own headers
14// First the interface, which forces the header to be self-contained.
15#include "colorpatch.h"
16// Second, the private implementation.
17#include "colorpatch_p.h" // IWYU pragma: associated
18
19#include "constpropagatinguniquepointer.h"
20#include "helper.h"
21#include <algorithm>
22#include <qapplication.h>
23#include <qbrush.h>
24#include <qdrag.h>
25#include <qevent.h>
26#include <qfont.h>
27#include <qframe.h>
28#include <qimage.h>
29#include <qlabel.h>
30#include <qmath.h>
31#include <qmimedata.h>
32#include <qnamespace.h>
33#include <qpainter.h>
34#include <qpalette.h>
35#include <qpen.h>
36#include <qpixmap.h>
37#include <qpoint.h>
38#include <qrect.h>
39#include <qsizepolicy.h>
40#include <qstyle.h>
41#include <qstyleoption.h>
42#include <qvariant.h>
43class QWidget;
44
45namespace PerceptualColor
46{
47/** @brief Constructor
48 * @param parent The parent of the widget, if any */
51 , d_pointer(new ColorPatchPrivate(this))
52{
53 setAcceptDrops(true);
55 d_pointer->updatePixmap();
56}
57
58/** @brief Destructor */
60{
61}
62
63/** @brief Constructor
64 *
65 * @param backLink Pointer to the object from which <em>this</em> object
66 * is the private implementation. */
67ColorPatchPrivate::ColorPatchPrivate(ColorPatch *backLink)
68 : m_label(new QLabel(backLink))
69 , q_pointer(backLink)
70{
71 m_label->setFrameShape(QFrame::StyledPanel);
72 m_label->setFrameShadow(QFrame::Sunken);
73 m_label->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored);
74 m_label->setGeometry(0, 0, backLink->width(), backLink->height());
75 // The following alignment is mirrored by Qt on right-to-left layouts:
76 constexpr Qt::Alignment myAlignment{Qt::AlignLeading, Qt::AlignTop};
77 m_label->setAlignment(myAlignment);
78}
79
80/** @brief Provide the size hint.
81 *
82 * Reimplemented from base class.
83 *
84 * @returns the size hint
85 *
86 * @sa @ref minimumSizeHint() */
88{
89 return minimumSizeHint();
90}
91
92/** @brief Provide the minimum size hint.
93 *
94 * Reimplemented from base class.
95 *
96 * @returns the minimum size hint
97 *
98 * @sa @ref sizeHint() */
100{
101 // Use a size similar to a QToolButton with an icon (and without text)
104 option.initFrom(this);
105 option.font = font();
106 const int iconSize = style()->pixelMetric( //
108 nullptr,
109 this);
110 option.iconSize = QSize(iconSize, iconSize);
111 return style()->sizeFromContents( //
113 &option,
114 option.iconSize,
115 this);
116}
117
118/** @brief Updates the pixmap in @ref m_label and its alignment. */
119void ColorPatchPrivate::updatePixmap()
120{
121 const QRect qLabelContentsRect = m_label->contentsRect();
122 const QPixmap pixmap = renderPixmap(qLabelContentsRect.width(), //
123 qLabelContentsRect.height());
124 // NOTE Kvantum was mistakenly scaling the pixmap (even though
125 // QLabel::hasScaledContents() == false) for versions ≤ 1.0.2. This bug
126 // has been fixed: https://github.com/tsujan/Kvantum/issues/804.
127 m_label->setPixmap(pixmap);
128 // There were rendering artefacts under certain QStyle (Breeze, Plastik,
129 // Windows): When selecting in the color dialog a new color with the
130 // screen color picker using “Portal” under 125% scaling, the left and
131 // the top border of the QLabel show a thin line of the previous color.
132 // We can work around this by simply updating the whole widget:
133 m_label->update();
134}
135
136/** @brief Handle resize events.
137 *
138 * Reimplemented from base class.
139 *
140 * @param event The corresponding event */
142{
143 d_pointer->m_label->resize(event->size());
144
145 // NOTE It would be more efficient not to always update the pixmap,
146 // but only when either the height or the width of the new pixmap to
147 // be calculated are larger than those of the current pixmap available
148 // under d_pointer->updatePixmap(). After all, a pixmap that is too
149 // large does not disturb the drawing, while one that is too small does.
150 // Unfortunately, however, resizing QLabel (at least with high-DPI and
151 // RTL layout at the same time) causes the correct alignment (here
152 // Qt::AlignLeading and Qt::AlignTop) to be lost and the image to be
153 // shifted. This error can be worked around by actually each time a new
154 // pixmap is assigned, which is not identical to the old one:
155 d_pointer->updatePixmap();
156}
157
158// No documentation here (documentation of properties
159// and its getters are in the header)
161{
162 return d_pointer->m_color;
163}
164
165/** @brief Setter for the @ref color property.
166 * @param newColor the new color */
167void ColorPatch::setColor(const QColor &newColor)
168{
169 if (newColor != d_pointer->m_color) {
170 d_pointer->m_color = newColor;
171 d_pointer->updatePixmap();
172 Q_EMIT colorChanged(newColor);
173 }
174}
175
176/** @brief Renders the image to be displayed.
177 *
178 * @param width of the requested image, measured in device-independent pixels.
179 *
180 * @param height of the requested image, measured in device-independent pixels.
181 *
182 * @returns An image containing the color of @ref m_color. If the color is
183 * transparent or semi-transparent, background with small gray squares is
184 * visible. If @ref ColorPatch has RTL layout, the image is mirrored. The
185 * device-pixel-ratio is set accordingly to @ref ColorPatch. The size of
186 * the image is equal or (if rounding has to be done because of fractional
187 * scale factors) slightly bigger than necessary to paint the whole
188 * @ref ColorPatch surface at the given device-pixel-ratio. As @ref m_label
189 * does <em>not</em> scale the image by default, it will be displayed with
190 * the correct aspect ratio, while guaranteeing to be big enough whatever
191 * QLabel’s frame size is with the currently used QStyle. */
192QImage ColorPatchPrivate::renderImage(const int width, const int height)
193{
194 // Initialization
195 // Round up to the next integer to be sure to have a big-enough image:
196 const qreal imageWidthF = width * q_pointer->devicePixelRatioF();
197 const int imageWidth = qCeil(imageWidthF);
198 const qreal imageHeightF = height * q_pointer->devicePixelRatioF();
199 const int imageHeight = qCeil(imageHeightF);
200 QImage myImage(imageWidth, //
201 imageHeight, //
203 if ((imageWidth <= 0) || (imageHeight <= 0)) {
204 // Initializing a QPainter on an image of zero size would print
205 // errors. Therefore, returning immediately:
206 myImage.setDevicePixelRatio(q_pointer->devicePixelRatioF());
207 return QImage();
208 }
210 opt.initFrom(q_pointer); // Sets also QStyle::State_MouseOver if appropriate
211
212 // Draw content of an invalid color (and return)
213 if (!m_color.isValid()) {
214 const QPalette::ColorGroup myColorGroup = //
215 (q_pointer->isEnabled()) //
216 ? QPalette::ColorGroup::Normal //
217 : QPalette::ColorGroup::Disabled;
218 myImage.fill( //
220 // An alternative value might be:
221 // q_pointer->palette().color(myColorGroup, QPalette::Window)
222 // but this integrates less nice with styles like QtCurve who
223 // might have background decorations that cover all widgets.
224 // Ultimately, however, it is a matter of taste.
225 );
226 QPen pen( //
227 q_pointer->palette().color(myColorGroup, QPalette::WindowText));
228
229 const int defaultFrameWidth = qMax( //
230 q_pointer->style()->pixelMetric(QStyle::PM_DefaultFrameWidth, &opt),
231 1);
232 const auto lineWidthF = //
233 defaultFrameWidth * q_pointer->devicePixelRatioF();
234 pen.setWidthF(lineWidthF);
235 pen.setCapStyle(Qt::PenCapStyle::SquareCap);
236 {
237 QPainter painter{&myImage};
238 // Because Qt::PenCapStyle::SquareCap will extends beyond the line
239 // end by half the line width, we can use an offset and the line
240 // will still touch the corner pixels of the image. It is a good
241 // idea to do so, because on widgets with an extreme aspect ratio
242 // (for example width 400, height 40, which is a realistic value in
243 // ColorDialog), the lines seem to “shift out of the image”. Using
244 // an offset, it looks nicer. How big should the offset be? To keep
245 // it simple, we use the same offset for both, x and y. The
246 // distance from the offset point to the point where the line
247 // touches the border depends on the angle of the line. The worst
248 // case (that means, the biggest distance) is for 45°. With
249 // Pythagoras, we have, for the offset “a” (identical for x and y):
250 // a² + a² = (½ linewidth)²
251 // 2 a² = ¼ linewidth²
252 // a² = ⅛ linewidth²
253 // a = 1 ÷ (√8) linewidth
254 // a ≈ 0.35 linewidth (Rounding down to be safe)
255 const qreal offset = static_cast<qreal>(lineWidthF * 0.35);
256 const qreal &left = offset; // alias for “offset”
257 const qreal &top = offset; // alias for “offset”
258 const qreal bottom = imageHeightF - offset;
259 const qreal right = imageWidthF - offset;
260 painter.setPen(pen);
261 painter.setRenderHint(QPainter::Antialiasing, true);
262 painter.drawLine(QPointF(left, top), //
263 QPointF(right, bottom));
264 painter.drawLine(QPointF(left, bottom), //
265 QPointF(right, top));
266 }
267 myImage.setDevicePixelRatio(q_pointer->devicePixelRatioF());
268 return myImage;
269 }
270
271 // Draw content of a valid color
272 if (m_color.alphaF() < 1) {
273 // Prepare the image with (semi-)transparent color
274 // Background for colors that are not fully opaque
275 QImage tempBackground = transparencyBackground( //
276 q_pointer->devicePixelRatioF());
277 // Paint the color above
278 QPainter(&tempBackground).fillRect(tempBackground.rect(), m_color);
279 {
280 // Fill a given rectangle with tiles. (QBrush will ignore
281 // the devicePixelRatioF of the image of the tile.)
282 QPainter painter{&myImage};
283 painter.setRenderHint(QPainter::Antialiasing, false);
284 painter.fillRect(myImage.rect(), QBrush(tempBackground));
285 }
286 if (q_pointer->layoutDirection() == Qt::RightToLeft) {
287 // Horizontally mirrored image for right-to-left layout,
288 // so that the “nice” part is the first you see in reading
289 // direction.
290 myImage = myImage.mirrored(true, // horizontally mirrored
291 false // vertically mirrored
292 );
293 }
294 } else {
295 // Prepare the image with plain color
296 myImage.fill(m_color);
297 }
298 myImage.setDevicePixelRatio(q_pointer->devicePixelRatioF());
299 return myImage;
300}
301
302/** @brief Renders the image to be displayed.
303 *
304 * @param width of the requested image, measured in logical pixels.
305 *
306 * @param height of the requested image, measured in logical pixels.
307 *
308 * @returns Same as @ref renderImage but as QPixmap. */
309QPixmap ColorPatchPrivate::renderPixmap(const int width, const int height)
310{
311 QPixmap pixmap = QPixmap::fromImage(renderImage(width, height));
312 pixmap.setDevicePixelRatio(q_pointer->devicePixelRatioF());
313 return pixmap;
314}
315
316/** @brief React on a mouse move event.
317 *
318 * Reimplemented from base class.
319 *
320 * @param event The corresponding mouse event */
322{
323 if (event->button() == Qt::LeftButton)
324 d_pointer->dragStartPosition = event->pos();
326}
327
328/** @brief React on a mouse press event.
329 *
330 * Reimplemented from base class.
331 *
332 * @param event The corresponding mouse event */
334{
335 if (event->buttons() & Qt::LeftButton) {
336 // Distance since the left mouse buttons was originally clicked.
337 const auto vector = event->pos() - d_pointer->dragStartPosition;
338 const auto distanceSquare = vector.x() * vector.x() //
339 + vector.y() * vector.y();
340 const auto refSquare = QApplication::startDragDistance() //
342 if (d_pointer->m_color.isValid() && (distanceSquare >= refSquare)) {
343 QDrag *drag = new QDrag(this); // Mandatory on heap and with parent
344 QMimeData *mimeData = new QMimeData;
345 mimeData->setColorData(d_pointer->m_color);
346 drag->setMimeData(mimeData); // Takes ownership of mime data
347 const auto finalSize = std::max({30, //
348 minimumSizeHint().width(), //
350 drag->setPixmap(d_pointer->renderPixmap(finalSize, finalSize));
351 drag->exec(Qt::CopyAction);
352 }
353 }
354 // NOTE Intentionally not calling the parent’s class’ implementation to
355 // avoid that on Breeze style, instead of drag-and-drop, sometimes
356 // the window gets moved.
357}
358
359/** @brief Accepts drag events for colors.
360 *
361 * Reimplemented from base class.
362 *
363 * @param event The corresponding event */
365{
366 if (event->mimeData()->hasColor()) {
367 const QColor colorToDrop = qvariant_cast<QColor>( //
368 event->mimeData()->colorData());
369 if (colorToDrop.isValid()) {
370 event->acceptProposedAction();
371 return;
372 }
373 }
374}
375
376/** @brief Accepts drag events for colors.
377 *
378 * Reimplemented from base class.
379 *
380 * @param event The corresponding event */
382{
383 if (event->mimeData()->hasColor()) {
384 const QColor colorToDrop = qvariant_cast<QColor>( //
385 event->mimeData()->colorData());
386 if (colorToDrop.isValid()) {
387 setColor(colorToDrop);
388 event->acceptProposedAction();
389 return;
390 }
391 }
392}
393
394} // namespace PerceptualColor
Q_INVOKABLE AbstractDiagram(QWidget *parent=nullptr)
The constructor.
A color display widget.
Definition colorpatch.h:70
virtual QSize minimumSizeHint() const override
Provide the minimum size hint.
virtual void mouseMoveEvent(QMouseEvent *event) override
React on a mouse press event.
virtual void mousePressEvent(QMouseEvent *event) override
React on a mouse move event.
virtual void dragEnterEvent(QDragEnterEvent *event) override
Accepts drag events for colors.
virtual void resizeEvent(QResizeEvent *event) override
Handle resize events.
void colorChanged(const QColor &color)
Notify signal for property color.
Q_INVOKABLE ColorPatch(QWidget *parent=nullptr)
Constructor.
virtual ~ColorPatch() noexcept override
Destructor.
void setColor(const QColor &newColor)
Setter for the color property.
virtual void dropEvent(QDropEvent *event) override
Accepts drag events for colors.
QColor color
The color that is displayed.
Definition colorpatch.h:90
virtual QSize sizeHint() const override
Provide the size hint.
The namespace of this library.
bool isValid() const const
Qt::DropAction exec(Qt::DropActions supportedActions)
void setMimeData(QMimeData *data)
void setPixmap(const QPixmap &pixmap)
Format_ARGB32_Premultiplied
QRect rect() const const
void setPixmap(const QPixmap &)
void setColorData(const QVariant &color)
Q_EMITQ_EMIT
QObject * parent() const const
QPixmap fromImage(QImage &&image, Qt::ImageConversionFlags flags)
void setDevicePixelRatio(qreal scaleFactor)
int height() const const
int width() const const
int height() const const
int width() const const
PM_ButtonIconSize
virtual int pixelMetric(PixelMetric metric, const QStyleOption *option, const QWidget *widget) const const=0
virtual QSize sizeFromContents(ContentsType type, const QStyleOption *option, const QSize &contentsSize, const QWidget *widget) const const=0
void initFrom(const QWidget *widget)
typedef Alignment
CopyAction
transparent
RightToLeft
LeftButton
QTextStream & left(QTextStream &stream)
QTextStream & right(QTextStream &stream)
QWidget(QWidget *parent, Qt::WindowFlags f)
void setAcceptDrops(bool on)
QRect contentsRect() const const
void ensurePolished() const const
virtual bool event(QEvent *event) override
virtual void mousePressEvent(QMouseEvent *event)
void setSizePolicy(QSizePolicy)
QStyle * style() const const
void update()
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Apr 4 2025 11:54:42 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.