21struct SurfaceFormatSelectionResult {
23 std::optional<KisSurfaceColorimetry::SurfaceDescription> requestedDescription;
32 QScopedPointer<KisSurfaceColorManagerInterface>
interface;
40 SurfaceFormatSelectionResult
52 m_d->currentConfig.setOptions(options);
53 m_d->currentConfig.isHDR =
false;
59 if (
m_d->interface->isReady()) {
73 if (!
m_d->interface->isReady()) {
74 str <<
"WARNING: surface color management interface is not ready!" << Qt::endl;
83 if (!
m_d->lastErrorString.isEmpty()) {
84 str.noquote().nospace()
85 <<
"ERROR: Failed to set up color management for the surface: "
86 <<
m_d->lastErrorString
91 str <<
"Configured mode:" <<
m_d->surfaceMode << Qt::endl;
93 RenderIntent preferredIntent = Private::calculateConfigIntent(
m_d->currentConfig.options());
94 str <<
"Configured intent:" << preferredIntent <<
"supported:" <<
m_d->interface->supportsRenderIntent(preferredIntent) << Qt::endl;
96 str <<
"Actual intent:";
97 if (
m_d->interface->renderingIntent()) {
98 str << *
m_d->interface->renderingIntent() << Qt::endl;
100 str <<
"<none>" << Qt::endl;
104 str <<
"Active surface description:";
105 if (
m_d->interface->surfaceDescription()) {
107 str.noquote() <<
m_d->interface->surfaceDescription()->makeTextReport() << Qt::endl;
109 str <<
"<none>" << Qt::endl;
113 str <<
"Selected Profile:";
114 if (
m_d->currentConfig.profile) {
115 auto profile =
m_d->currentConfig.profile;
117 str << profile->name() << Qt::endl;
123 auto colVec = profile->getColorantsxyY();
129 str <<
" red: " << colR << Qt::endl;
130 str <<
" green:" << colG << Qt::endl;
131 str <<
" blue: " << colB << Qt::endl;
135 auto whiteVec = profile->getWhitePointxyY();
139 str <<
" white: " << white << Qt::endl;
144 str <<
"<none>" << Qt::endl;
147 str <<
"Compositor preferred surface description:";
148 if (
m_d->interface->preferredSurfaceDescription()) {
150 str.noquote() <<
m_d->interface->preferredSurfaceDescription()->makeTextReport() << Qt::endl;
152 str <<
"<none>" << Qt::endl;
164 if (!
m_d->interface->isReady()) {
165 str <<
"WARNING: surface color management interface is not ready!" << Qt::endl;
169 if (
m_d->interface->preferredSurfaceDescription()) {
171 str.noquote() <<
m_d->interface->preferredSurfaceDescription()->makeTextReport() << Qt::endl;
173 str <<
"<none>" << Qt::endl;
183 RenderIntent intent = RenderIntent::render_intent_perceptual;
185 switch (options.first) {
192 RenderIntent::render_intent_relative_bpc :
193 RenderIntent::render_intent_relative;
197 RenderIntent::render_intent_saturation;
201 RenderIntent::render_intent_absolute;
209 if (!
m_d->interface->isReady())
return;
210 if (
surfaceMode ==
m_d->surfaceMode && options ==
m_d->currentConfig.options())
return;
236SurfaceFormatSelectionResult
237KisCanvasSurfaceColorSpaceManager::Private::
246 std::optional<SurfaceDescription> requestedDescription = SurfaceDescription();
251 auto makeKritaRec2020PQLuminance = []() {
253 luminance.minLuminance = 0;
254 luminance.referenceLuminance = 80;
255 luminance.maxLuminance = 10000;
264 requestedDescription->colorSpace.
primaries = NamedPrimaries::primaries_bt2020;
265 requestedDescription->colorSpace.transferFunction = NamedTransferFunction::transfer_function_st2084_pq;
269 requestedDescription->colorSpace = compositorPreferred.
colorSpace;
272 if (std::holds_alternative<NamedTransferFunction>(requestedDescription->colorSpace.transferFunction)
273 && std::get<NamedTransferFunction>(requestedDescription->colorSpace.transferFunction)
274 == NamedTransferFunction::transfer_function_st2084_pq) {
275 requestedDescription->colorSpace.
luminance = makeKritaRec2020PQLuminance();
278 requestedDescription->colorSpace.primaries = NamedPrimaries::primaries_bt2020;
279 requestedDescription->colorSpace.transferFunction = NamedTransferFunction::transfer_function_st2084_pq;
280 requestedDescription->colorSpace.luminance = makeKritaRec2020PQLuminance();
282 requestedDescription->colorSpace.primaries = NamedPrimaries::primaries_srgb;
283 requestedDescription->colorSpace.transferFunction = NamedTransferFunction::transfer_function_gamma22;
286 requestedDescription->colorSpace.luminance = compositorPreferred.
colorSpace.
luminance->clipToSdr();
289 requestedDescription->colorSpace.primaries = NamedPrimaries::primaries_srgb;
290 requestedDescription->colorSpace.transferFunction = NamedTransferFunction::transfer_function_ext_linear;
304 requestedDescription->colorSpace.luminance = compositorPreferred.
colorSpace.
luminance->clipToSdr();
308 if (std::holds_alternative<NamedPrimaries>(requestedDescription->colorSpace.primaries)
309 && std::get<NamedPrimaries>(requestedDescription->colorSpace.primaries) == NamedPrimaries::primaries_unknown) {
310 requestedDescription->colorSpace.primaries = NamedPrimaries::primaries_srgb;
313 if (std::holds_alternative<NamedTransferFunction>(requestedDescription->colorSpace.transferFunction)
314 && std::get<NamedTransferFunction>(requestedDescription->colorSpace.transferFunction)
315 == NamedTransferFunction::transfer_function_unknown) {
316 requestedDescription->colorSpace.transferFunction = NamedTransferFunction::transfer_function_gamma22;
319 if (!this->interface->supportsSurfaceDescription(*requestedDescription)) {
321 requestedDescription->colorSpace.primaries = NamedPrimaries::primaries_srgb;
322 requestedDescription->colorSpace.transferFunction = NamedTransferFunction::transfer_function_srgb;
324 if (!this->interface->supportsSurfaceDescription(*requestedDescription)) {
325 requestedDescription->colorSpace.transferFunction = NamedTransferFunction::transfer_function_gamma22;
327 if (!this->interface->supportsSurfaceDescription(*requestedDescription)) {
328 QString errorMessage =
"failed to find a suitable surface format for the compositor";
329 qWarning().nospace().noquote() <<
"ERROR: " << errorMessage;
337 if (request.isValid()) {
339 request.colorPrimariesType,
340 request.transferFunction);
346 requestedDescription->colorSpace.transferFunction = NamedTransferFunction::transfer_function_gamma22;
347 if (this->interface->supportsSurfaceDescription(*requestedDescription)) {
349 if (request.isValid()) {
351 request.colorPrimariesType,
352 request.transferFunction);
359 requestedDescription->colorSpace.
primaries = NamedPrimaries::primaries_srgb;
360 if (this->interface->supportsSurfaceDescription(*requestedDescription)) {
362 if (request.isValid()) {
364 request.colorPrimariesType,
365 request.transferFunction);
372 requestedDescription->colorSpace.transferFunction = NamedTransferFunction::transfer_function_srgb;
373 if (this->interface->supportsSurfaceDescription(*requestedDescription)) {
375 if (request.isValid()) {
377 request.colorPrimariesType,
378 request.transferFunction);
384 QString errorMessage =
"failed to create a profile for the compositor's preferred color space";
385 qWarning().nospace().noquote() <<
"ERROR: " << errorMessage;
386 qWarning() <<
" " <<
ppVar(compositorPreferred);
387 qWarning() <<
" " <<
ppVar(*requestedDescription);
391 return {profile, requestedDescription, {}};
401 RenderIntent preferredIntent =
402 Private::calculateConfigIntent(newOptions);
404 if (!
m_d->interface->supportsRenderIntent(preferredIntent)) {
405 qWarning() <<
"WARNING: failed to set user preferred rendering"
406 <<
"intent for the surface, intent \""
407 << preferredIntent <<
"\" is unsupported, falling back to \"perceptual\"";
409 preferredIntent = RenderIntent::render_intent_perceptual;
415 std::optional<SurfaceDescription> requestedDescription;
420 m_d->lastErrorString.clear();
422 const auto compositorPreferred =
m_d->interface->preferredSurfaceDescription();
425 auto result =
m_d->selectSurfaceDescription(
m_d->surfaceMode, *compositorPreferred);
426 profile = result.profile;
427 requestedDescription = result.requestedDescription;
428 m_d->lastErrorString = result.errorMessage;
434 if (
m_d->interface->surfaceDescription() != requestedDescription ||
435 m_d->interface->renderingIntent() != preferredIntent) {
437 if (requestedDescription) {
438 auto future =
m_d->interface->setSurfaceDescription(*requestedDescription, preferredIntent);
440 if (!result.isValid() || !result.result()) {
441 QString errorMessage =
"failed to set color space for the surface, setSurfaceDescription() returned false";
442 m_d->lastErrorString = errorMessage;
443 qWarning().nospace().noquote() <<
"ERROR: " << errorMessage;
447 m_d->interface->unsetSurfaceDescription();
451 const bool requestedDescriptionIsHDR = requestedDescription && requestedDescription->colorSpace.isHDR();
454 newDisplayConfig.
profile = profile;
456 newDisplayConfig.
isHDR = requestedDescriptionIsHDR;
462 if (
m_d->currentConfig != newDisplayConfig) {
463 m_d->currentConfig = newDisplayConfig;
470 return m_d->interface->isReady();
475 return m_d->currentConfig;
KisDisplayConfig This class keeps track of the color management configuration for image to display....
void setOptions(const Options &options)
std::pair< KoColorConversionTransformation::Intent, KoColorConversionTransformation::ConversionFlags > Options
const KoColorProfile * profile
void sigReadyChanged(bool value)
void sigPreferredSurfaceDescriptionChanged(const KisSurfaceColorimetry::SurfaceDescription &desc)
#define KIS_SAFE_ASSERT_RECOVER(cond)
#define KIS_SAFE_ASSERT_RECOVER_RETURN_VALUE(cond, val)
#define KIS_SAFE_ASSERT_RECOVER_RETURN(cond)
#define INTENT_ABSOLUTE_COLORIMETRIC
#define INTENT_PERCEPTUAL
#define INTENT_RELATIVE_COLORIMETRIC
#define INTENT_SATURATION
PigmentProfileRequest colorSpaceToRequest(ColorSpace cs)
QScopedPointer< KisSurfaceColorManagerInterface > interface
KisConfig::CanvasSurfaceMode surfaceMode
KisCanvasSurfaceColorSpaceManager(KisSurfaceColorManagerInterface *interface, const KisConfig::CanvasSurfaceMode surfaceMode, const KisDisplayConfig::Options &options, QObject *parent=nullptr)
void sigDisplayConfigChanged(const KisDisplayConfig &config)
void slotInterfacePreferredDescriptionChanged()
KisDisplayConfig currentConfig
QString osPreferredColorSpaceReport() const
Private(KisSurfaceColorManagerInterface *interface)
QScopedPointer< Private > m_d
std::optional< KisSurfaceColorimetry::RenderIntent > proofingIntentOverride
void reinitializeSurfaceDescription(const KisDisplayConfig::Options &newOptions)
SurfaceFormatSelectionResult selectSurfaceDescription(KisConfig::CanvasSurfaceMode surfaceMode, const KisSurfaceColorimetry::SurfaceDescription &compositorPreferred)
static KisSurfaceColorimetry::RenderIntent calculateConfigIntent(const KisDisplayConfig::Options &options)
void slotInterfaceReadyChanged(bool isReady)
KisDisplayConfig displayConfig() const
QString colorManagementReport() const
~KisCanvasSurfaceColorSpaceManager()
void setDisplayConfigOptions(const KisConfig::CanvasSurfaceMode surfaceMode, const KisDisplayConfig::Options &options)
std::optional< Luminance > luminance
static QString getTransferCharacteristicName(TransferCharacteristics curve)
getTransferCharacteristicName
static QString getColorPrimariesName(ColorPrimaries primaries)
getColorPrimariesName
const KoColorProfile * profileFor(const QVector< double > &colorants, ColorPrimaries colorPrimaries, TransferCharacteristics transferFunction) const
profileFor tries to find the profile that matches these characteristics, if no such profile is found,...
static KoColorSpaceRegistry * instance()
const KoColorProfile * p709SRGBProfile() const