Perceptual Color

helperconversion.h
1// SPDX-FileCopyrightText: Lukas Sommer <sommerluk@gmail.com>
2// SPDX-License-Identifier: BSD-2-Clause OR MIT
3
4#ifndef HELPERCONVERSION_H
5#define HELPERCONVERSION_H
6
7#include "genericcolor.h"
8#include "helperqttypes.h"
9#include <lcms2.h>
10#include <qcolor.h>
11#include <qglobal.h>
12#include <type_traits>
13
14#if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0))
15#include <qtmetamacros.h>
16#endif
17
18/** @internal
19 *
20 * @file
21 *
22 * Provides type conversions. */
23
24namespace PerceptualColor
25{
26
27Q_NAMESPACE
28
29/** @brief Identifiers for color models.
30 *
31 * @internal
32 *
33 * @note Maybe
34 * <a href="https://doc-snapshots.qt.io/qt6-dev/qcolorspace.html#ColorModel-enum">
35 * <tt>enum QColorSpace::NamedColorSpace</tt></a> might also be useful in
36 * the future.
37 */
38enum class ColorModel {
39 CielabD50, /**< The absolute Cielab color space using a D50 illuminant.
40 Lightness: [0, 100].<br/>
41 a: unbound.<br/>
42 b: unbound. */
43 CielchD50, /**< The absolute Cielch color space using a D50 illuminant.
44 Lightness: [0, 100].<br/>
45 Chroma: unbound.<br/>
46 Hue: [0, 360[. */
47 Hsl360_1_1, /**< Any color space using the HSL color model, which is a
48 transformation of the RGB color model. Color values only have a
49 well-defined absolute meaning when associated with a corresponding
50 RGB color profile.
51
52 Hue: [0, 360[.<br/>
53 Saturation: [0, 1].<br/>
54 Lightness: [0, 1]. */
55 Hwb360_1_1, /**< Any color space using the HSL color model, which is a
56 transformation of the HWB color model. Color values only have a
57 well-defined absolute meaning when associated with a corresponding
58 RGB color profile.
59
60 Hue: [0, 360[.<br/>
61 Whiteness: [0, 1].<br/>
62 Blackness: [0, 1]. */
63 Invalid, /**< Represents invalid data. */
64 OklabD65, /**< The absolute Oklab color space, which by definition always
65 and exclusively uses a D65 illuminant.
66
67 Lightness: [0, 1].<br/>
68 a: unbound.<br/>
69 b: unbound. */
70 OklchD65, /**< The absolute Oklch color space, which by definition always
71 and exclusively uses a D65 illuminant.
72
73 Lightness: [0, 1].<br/>
74 Chroma: unbound.<br/>
75 Hue: [0, 360[. */
76 Rgb1, /**< Any color space using the RGB color model. Color values only
77 have a well-defined absolute meaning when associated with a
78 corresponding RGB color profile.
79
80 R: [0, 1].<br/>
81 G: [0, 1].<br/>
82 B: [0, 1]. */
83 XyzD50, /**< The absolute XYZ color space assuming chromatic adaption
84 for the D50 illuminant.
85
86 X: unbound.<br/>
87 Y: [0, 1]. Diffuse white has a luminance (Y) of 1.0<br/>
88 Z: unbound. */
89 XyzD65 /**< The absolute XYZ color space assuming chromatic adaption
90 for the D65 illuminant.
91
92 X: unbound.<br/>
93 Y: [0, 1]. Diffuse white has a luminance (Y) of 1.0<br/>
94 Z: unbound. */
95};
96Q_ENUM_NS(ColorModel)
97
98#if (QT_VERSION < QT_VERSION_CHECK(6, 0, 0))
99uint qHash(const ColorModel t, uint seed = 0); // clazy:exclude=qt6-qhash-signature
100#endif
101
102/** @internal
103 *
104 * @brief Converts from <tt>[0, 1]</tt> to <tt>[0, 255]</tt>.
105 *
106 * @param original A value on a scale <tt>[0, 1]</tt>.
107 *
108 * @returns Value converted to the scale <tt>[0, 255]</tt>, applying correct
109 * rounding. Out-of-range values are bound to the valid range. */
110template<typename T>
111[[nodiscard]] constexpr quint8 fromFloatingToEightBit(const T &original)
112{
113 static_assert( //
114 std::is_floating_point<T>::value, //
115 "Template fromFloatingToEightBit() only works with floating point types");
116 const int rounded = qRound(original * 255);
117 const auto bounded = qBound<int>(0, rounded, 255);
118 return static_cast<quint8>(bounded);
119}
120
121/** @internal
122 *
123 * @brief Like <tt>QColor::fromRgbF</tt> but for all floating point types.
124 *
125 * @param red Red component. Range: <tt>[0, 1]</tt>
126 * @param green See above.
127 * @param blue See above.
128 * @returns A corresponding <tt>QColor</tt> object. */
129template<typename T>
130[[nodiscard]] QColor qColorFromRgbDouble(T red, T green, T blue)
131{
132 static_assert( //
133 std::is_floating_point<T>::value, //
134 "Template fromFloatingToEightBit() only works with floating point types");
135 return QColor::fromRgbF(static_cast<QColorFloatType>(red), //
136 static_cast<QColorFloatType>(green), //
137 static_cast<QColorFloatType>(blue));
138}
139
140[[nodiscard]] cmsCIELab toCmsLab(const cmsCIELCh &value);
141
142[[nodiscard]] GenericColor toGenericColorCielabD50(const cmsCIELab &value);
143
144/** @internal
145 *
146 * @brief White point D65 for 2°-observer.
147 *
148 * According to
149 * <a href="https://en.wikipedia.org/w/index.php?title=Illuminant_D65&oldid=1100467073#Definition">
150 * Wikipedia</a>:
151 *
152 * > “Using the standard 2° observer […] of D65 […] Normalizing for
153 * > relative luminance (i.e. set Y=100), the XYZ tristimulus
154 * > values are:<br/>
155 * > X = 95.047<br/>
156 * > Y = 100<br/>
157 * > Z = 108.883”
158 *
159 * Normalizing this to Y = 1 as expected by LittleCMS, gives this value. */
160// TODO xxx This seems to be not longer used!?
161constexpr cmsCIEXYZ whitePointD65TwoDegree{0.95047, 1.00000, 1.08883};
162
163} // namespace PerceptualColor
164
165#endif // HELPERCONVERSION_H
The namespace of this library.
ColorModel
Identifiers for color models.
@ OklabD65
The absolute Oklab color space, which by definition always and exclusively uses a D65 illuminant.
@ XyzD50
The absolute XYZ color space assuming chromatic adaption for the D50 illuminant.
@ CielabD50
The absolute Cielab color space using a D50 illuminant.
@ OklchD65
The absolute Oklch color space, which by definition always and exclusively uses a D65 illuminant.
@ Hsl360_1_1
Any color space using the HSL color model, which is a transformation of the RGB color model.
@ Invalid
Represents invalid data.
@ XyzD65
The absolute XYZ color space assuming chromatic adaption for the D65 illuminant.
@ Rgb1
Any color space using the RGB color model.
@ CielchD50
The absolute Cielch color space using a D50 illuminant.
@ Hwb360_1_1
Any color space using the HSL color model, which is a transformation of the HWB color model.
QColor fromRgbF(float r, float g, float b, float a)
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.