Perceptual Color

rgbcolorspace_p.h
1// SPDX-FileCopyrightText: Lukas Sommer <sommerluk@gmail.com>
2// SPDX-License-Identifier: BSD-2-Clause OR MIT
3
4#ifndef RGBCOLORSPACE_P_H
5#define RGBCOLORSPACE_P_H
6
7// Include the header of the public class of this private implementation.
8// #include "rgbcolorspace.h"
9
10#include "cielchd50values.h"
11#include "constpropagatingrawpointer.h"
12#include "helperconstants.h"
13#include "oklchvalues.h"
14#include <lcms2.h>
15#include <optional>
16#include <qcontainerfwd.h>
17#include <qdatetime.h>
18#include <qglobal.h>
19#include <qlist.h>
20#include <qmap.h>
21#include <qstring.h>
22#include <qversionnumber.h>
23
24namespace PerceptualColor
25{
26class RgbColorSpace;
27
28/** @internal
29 *
30 * @brief Private implementation within the <em>Pointer to
31 * implementation</em> idiom */
32class RgbColorSpacePrivate final
33{
34private:
35 [[nodiscard]] static QMap<cmsUInt32Number, QString> getIntentList();
36
37public:
38 explicit RgbColorSpacePrivate(RgbColorSpace *backLink);
39 /** @brief Default destructor
40 *
41 * The destructor is non-<tt>virtual</tt> because
42 * the class as a whole is <tt>final</tt>. */
43 ~RgbColorSpacePrivate() noexcept = default;
44
45 // Data members:
46 /** @brief The darkest in-gamut point on the L* axis.
47 * @sa whitepointL
48 *
49 * @internal
50 *
51 * @todo Use cmsDetectBlackPoint? But “our” “blackpoint” is always on
52 * the grey axis, but the real blackpoint not? Document this? */
53 qreal m_cielabD50BlackpointL = 0;
54 /** @brief The lightest in-gamut point on the L* axis.
55 * @sa blackpointL() */
56 qreal m_cielabD50WhitepointL = 100;
57 /** @brief The darkest in-gamut point on the L* axis.
58 * @sa whitepointL
59 *
60 * @internal
61 *
62 * @todo Use cmsDetectBlackPoint? But “our” “blackpoint” is always on
63 * the grey axis, but the real blackpoint not? Document this? */
64 qreal m_oklabBlackpointL = 0;
65 /** @brief The lightest in-gamut point on the L* axis.
66 * @sa blackpointL() */
67 qreal m_oklabWhitepointL = 1;
68 /** @brief Internal storage for property
69 * @ref RgbColorSpace::profileAbsoluteFilePath */
70 QString m_profileAbsoluteFilePath;
71 /** @brief Internal storage for property
72 * @ref RgbColorSpace::profileClass */
73 cmsProfileClassSignature m_profileClass;
74 /** @brief Internal storage for property
75 * @ref RgbColorSpace::profileColorModel */
76 cmsColorSpaceSignature m_profileColorModel;
77 /** @brief Internal storage for property
78 * @ref RgbColorSpace::profileCopyright */
79 QString m_profileCopyright;
80 /** @brief Internal storage for property
81 * @ref RgbColorSpace::profileCreationDateTime */
82 QDateTime m_profileCreationDateTime;
83 /** @brief Internal storage for property
84 * @ref RgbColorSpace::profileFileSize */
85 qint64 m_profileFileSize = -1;
86 /** @brief Internal storage for property
87 * @ref RgbColorSpace::profileHasClut */
88 bool m_profileHasClut = false;
89 /** @brief Internal storage for property
90 * @ref RgbColorSpace::profileHasMatrixShaper */
91 bool m_profileHasMatrixShaper = false;
92 /** @brief Internal storage for property
93 * @ref RgbColorSpace::profileIccVersion */
94 QVersionNumber m_profileIccVersion;
95 /** @brief Internal storage for property
96 * @ref RgbColorSpace::profileManufacturer */
97 QString m_profileManufacturer;
98 /** @brief Internal storage for property
99 * @ref RgbColorSpace::profileMaximumCielchD50Chroma */
100 double m_profileMaximumCielchD50Chroma = CielchD50Values::maximumChroma;
101 /** @brief Internal storage for property
102 * @ref RgbColorSpace::profileMaximumOklchChroma */
103 double m_profileMaximumOklchChroma = OklchValues::maximumChroma;
104 /** @brief Internal storage for property
105 * @ref RgbColorSpace::profileModel */
106 QString m_profileModel;
107 /** @brief Internal storage for property
108 * @ref RgbColorSpace::profileName */
109 QString m_profileName;
110 /** @brief Internal storage for property
111 * @ref RgbColorSpace::profilePcsColorModel */
112 cmsColorSpaceSignature m_profilePcsColorModel;
113 /** @brief Internal storage for property
114 * @ref RgbColorSpace::profileTagBlackpoint */
115 std::optional<cmsCIEXYZ> m_profileTagBlackpoint;
116 /** @brief Internal storage for property
117 * @ref RgbColorSpace::profileTagBluePrimary */
118 std::optional<cmsCIEXYZ> m_profileTagBluePrimary;
119 /** @brief Internal storage for property
120 * @ref RgbColorSpace::profileTagGreenPrimary */
121 std::optional<cmsCIEXYZ> m_profileTagGreenPrimary;
122 /** @brief Internal storage for property
123 * @ref RgbColorSpace::profileTagRedPrimary */
124 std::optional<cmsCIEXYZ> m_profileTagRedPrimary;
125 /** @brief Internal storage for property
126 * @ref RgbColorSpace::profileTagSignatures */
127 QStringList m_profileTagSignatures;
128 /** @brief Internal storage for property
129 * @ref RgbColorSpace::profileTagWhitepoint */
130 std::optional<cmsCIEXYZ> m_profileTagWhitepoint;
131 /** @brief A handle to a LittleCMS transform. */
132 cmsHTRANSFORM m_transformCielabD50ToRgb16Handle = nullptr;
133 /** @brief A handle to a LittleCMS transform. */
134 cmsHTRANSFORM m_transformCielabD50ToRgbHandle = nullptr;
135 /** @brief A handle to a LittleCMS transform. */
136 cmsHTRANSFORM m_transformRgbToCielabD50Handle = nullptr;
137
138 // Functions:
139 static void deleteTransform(cmsHTRANSFORM *transformHandle);
140 [[nodiscard]] double detectMaximumCielchD50Chroma() const;
141 [[nodiscard]] double detectMaximumOklchChroma() const;
142 [[nodiscard]] bool initialize(cmsHPROFILE rgbProfileHandle);
143 [[nodiscard]] static QDateTime profileCreationDateTime(cmsHPROFILE profileHandle);
144 [[nodiscard]] static QVersionNumber profileIccVersion(cmsHPROFILE profileHandle);
145 [[nodiscard]] static QString profileInformation(cmsHPROFILE profileHandle, cmsInfoType infoType, const QString &languageTerritory);
146 [[nodiscard]] static std::optional<cmsCIEXYZ> profileReadCmsciexyzTag(cmsHPROFILE profileHandle, cmsTagSignature signature);
147 [[nodiscard]] static QStringList profileTagSignatures(cmsHPROFILE profileHandle);
148
149 /** @brief The rendering intents supported by the LittleCMS library.
150 *
151 * Contains all rendering intents supported by the LittleCMS library
152 * against which this we are currently linking. Each entry contains
153 * the code and the (english-language) description just as provided
154 * by LittleCMS.
155 *
156 * Note that LittleCMS supports as built-in intents the four official
157 * ICC intents and also some other, non-ICC intents. Furthermore,
158 * LittleCMS plugins can provide even more intents. As of LittleCMS 2.13
159 * the following built-in intents are available:
160 *
161 * | Type | Macro name | Code |
162 * | :------ | :-------------------------------------------- | ---: |
163 * | ICC | INTENT_PERCEPTUAL | 0 |
164 * | ICC | INTENT_RELATIVE_COLORIMETRIC | 1 |
165 * | ICC | INTENT_SATURATION | 2 |
166 * | ICC | INTENT_ABSOLUTE_COLORIMETRIC | 3 |
167 * | Non-ICC | INTENT_PRESERVE_K_ONLY_PERCEPTUAL | 10 |
168 * | Non-ICC | INTENT_PRESERVE_K_ONLY_RELATIVE_COLORIMETRIC | 11 |
169 * | Non-ICC | INTENT_PRESERVE_K_ONLY_SATURATION | 12 |
170 * | Non-ICC | INTENT_PRESERVE_K_PLANE_PERCEPTUAL | 13 |
171 * | Non-ICC | INTENT_PRESERVE_K_PLANE_RELATIVE_COLORIMETRIC | 14 |
172 * | Non-ICC | INTENT_PRESERVE_K_PLANE_SATURATION | 15 |
173 *
174 * @todo Either actually <em>use</em> this code or <em>remove</em> this
175 * code. */
176 static inline const QMap<cmsUInt32Number, QString> intentList = getIntentList();
177
178 /** @brief Precision of HSV hue during maximum-chroma detection.
179 *
180 * @todo A value smaller than 0.001 does not make sense
181 * currently, because QColor has only a limited precision for
182 * HSV conversions. Furthermore, since Qt6 it’s floating point interface
183 * has been defined with “float”. For a more exact solution, we would
184 * have to implement our own HSV conversion first. */
185 static constexpr double chromaDetectionHuePrecision = gamutPrecisionCielab;
186 /** @brief Increment factor for the maximum-chroma detection.
187 *
188 * The maximum-chroma detection, regardless of the precision,
189 * might always return a value that is a bit too small. However,
190 * we want to have @ref RgbColorSpace::profileMaximumCielchD50Chroma and
191 * @ref RgbColorSpace::profileMaximumOklchChroma values that are equal
192 * or slightly bigger than the actual maximum-chroma, to make sure to
193 * not exclude valid values. Therefore, @ref detectMaximumCielchD50Chroma()
194 * and @ref detectMaximumOklchChroma use this increment factor to
195 * slightly increment the outcome of the chroma detection relative to
196 * the original value, as a safety margin. Note that additionally,
197 * an absolute increment should also be added, because of limited
198 * precision in floating point operations. */
199 static constexpr double chromaDetectionIncrementFactor = 1.02;
200 /** @brief For detecting CIELab in-gamut or out-of-gamut colors.
201 *
202 * For gamut detection, a roundtrip conversion is performed: Lab values
203 * are converted to an RGB color space and backwards. If the distance
204 * in euclidean space between the the original Lab value and the result
205 * of the roundtrip is smaller than a certain value, it is considered
206 * as an in-gamut value.
207 *
208 * This deviation limit should be as small as possible for a more correct
209 * gamut boundary. But it must unfortunately also be big enough to ignore
210 * rounding errors. The current value was chosen by trial-and-error. */
211 static constexpr qreal cielabDeviationLimit = 0.5;
212 /** @brief For detecting Oklab in-gamut or out-of-gamut colors.
213 *
214 * For gamut detection, a roundtrip conversion is performed: Lab values
215 * are converted to an RGB color space and backwards. If the distance
216 * in euclidean space between the the original Lab value and the result
217 * of the roundtrip is smaller than a certain value, it is considered
218 * as an in-gamut value.
219 *
220 * This deviation limit should be as small as possible for a more correct
221 * gamut boundary. But it must unfortunately also be big enough to ignore
222 * rounding errors. The current value was chosen by trial-and-error. */
223 static constexpr qreal oklabDeviationLimit = 0.001;
224
225private:
226 Q_DISABLE_COPY(RgbColorSpacePrivate)
227
228 /** @brief Pointer to the object from which <em>this</em> object
229 * is the private implementation. */
230 ConstPropagatingRawPointer<RgbColorSpace> q_pointer;
231};
232
233} // namespace PerceptualColor
234
235#endif // RGBCOLORSPACE_P_H
The namespace of this library.
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.