Chromatic Adaptation


I find chromatic adaptation mathematically interesting. This vignette is a longwinded formal approach that delays code to the end. Featured functions in this vignette are: CAT() and adaptXYZ().



Analogy of Proportion

Chromatic adaptation can be viewed as an Aristotelian analogy of proportion, see [1]. A general analogy of this type is usually written A : B = C : D and read as “A is to B as C is to D”. In our case the expression A : I is interpreted as “the color appearance of A in a viewing environment with illuminant I after the viewer is adapted to I”, or more simply “the appearance of A under illuminant I”. It is better to think of A not as an object color, but as a self-luminous color. The analogy A : I = B : J can be read as “the appearance of A under illuminant I is the same as the appearance of B under illuminant J”. This phenomenon is called chromatic adaptation and takes place in the human eye and brain. Of course, the exact values of A and B vary depending on the individual, so it only makes sense to think of averages, for a standard observer. All of these variables are typically XYZ tristimulus vectors, and that is what we will assume from now on. We require that all components of I and J are positive, i.e. the illuminants are in the positive octant.

We will adopt these axioms throughout this vignette: In A2) the first equality is equality of appearance, and the second is equality of vectors.

In most cases A, I, and J are known, and B is not. Solving the chromatic adaptation problem is solving the analogy A : I = X : J for X, where A is the source color, I is the the source illuminant (also called the test illuminant), and J is the target illuminant (also called the reference illuminant). By axiom A2) the target color solution X is unique.


Chromatic Adaptation Transforms

In most practical cases, illuminants I and J are fixed, and the source color A is allowed to vary widely. Recall that X is uniquely determined by A, so in the analogy A : I = X : J it makes sense to think of X as a transform T(A) of A. Define the ideal Chromatic Adaption Transform T from I to J by the analogy:

If T(A) : J = T(B) : J then A : I = B : I and so A = B, by (1.1). This shows that T is injective.

It is convenient to abbreviate Chromatic Adaption Transform by the acronym CAT. In theory the exactly accurate ideal CAT from I to J is unique, but in practice there are so many confounding variables and complexities that we cannot hope to find it. So from now on we open things up and look for approximations to T using various methods (see the next section).

There are 2 special cases where we can assert something exactly. Since I : I = J : J by axiom A1), we can say that T(I) = J exactly. And since A : I = A : I, we can say that if J = I, then T is the identity transform. In both theory and practice, we obviously want these 2 properties to be true, so we require that all approximations have them.


CAT Methods

A CAT method assigns to every pair of illuminants I and J a CAT from I to J, which we denote by TIJ. It is understood that TIJ is an approximation of the ideal. We require that TIJ(I) = J (see previous paragraph). A CAT is usually an instance of a CAT method. Consider these 3 properties for a CAT method:
  1. identity. If J = I, TII is the identity transform
  2. inverse. TJI is the inverse of TIJ
  3. commutative triangle. Adapting from I to J, and then from J to K, is the same as adapting from I to K. In symbols: TJK ∘ TIJ = TIK (where denotes transform composition).

Property 1 (identity) is required by the last paragraph of the previous section. We now argue that properties 2 and 3 are desirable.

Property 2 (inverse) is true for an ideal CAT method, by the following argument: Swapping I and J gives TIJ ∘ TJI = identity, and this proves property 2. According to Hunt [4] p. 591, the non-linear version of the CMCCAT97 method does NOT have property 2. The degree to which property 2 fails is therefore a measure of the accuracy of the appearance matching of the CMCCAT97 CATs. In Burns [2] is a CAT method which in the original and direct form does not have the inverse property, but can be slightly modified to a “symmetric” version that does enjoy the property.

Property 3 (commutative triangle) is desirable because it makes it possible to use the intermediate Profile Connection Space (PCS with Illuminant D50) in ICC color profiles with no loss of accuracy. In property 3, J represents the PCS in the middle, and the endpoints I and K are viewing illuminants for devices. If property 3 fails, then the 2-step adaptation may be sub-optimal.

Remark 3.1 Assuming that each TII is invertible (a very mild condition), properties 1. and 2. follow from 3. Letting J = K = I in property 3 we get TII ∘ TII = TII. Since TII is invertible we conclude TII = identity and this proves property 1. Letting K = I in property 3 we get Swapping I and J gives and this proves property 2. Note that property 3 is similar to the cocycle condition in the construction of fiber bundles, see [9] and p. 14 of [8]. That property 3 implies properties 1 and 2 is on p. 8 of [8].

Remark 3.2 Property 1 does NOT follow from property 2. Here is a CAT method that is a counter-example. First suppose that the vectors I and J have the same length. Define TIJ to be the transform that first rotates space π radians around I and then rotates by the most direct rotation from I to J. If I and J do not have the same length, then rotate I to the line spanned by J and then follow this by a uniform scaling so the result expands or shrinks to J. By “most direct rotation” we mean the rotation around the axis I × J (the 3D cross product). This TIJ is perfectly well-defined (unless J = −I which is impossible since both are in the positive octant), and maps I to J. It is not hard to show that TJI ∘ TIJ = TIJ ∘ TJI = identity, so property 2 is true. But TII is rotation of π radians around I, so property 1 is false.



The Simplest CAT Method - XYZ Scaling

A CAT method is called linear if and only if every TIJ is a linear map (from 3 to 3). From this point on in the vignette all CAT methods are linear. TIJ now denotes a 3x3 matrix. In ICC v4.0 color profiles this matrix is saved in the chromaticAdaptationTag or chad tag, and converts from device white to PCS white, see [5]. To avoid confusion between matrices and vectors, we shift notation and replace I, J, and K by 3-vectors u, v, and w (since lower-case i, j, and k look too much like integers). We also use the bold 1 := (1, 1, 1) for the 3-vector of all 1s, which is also a valid whitepoint.

The 3 properties in the previous section now take this form:
  1. identity. Tuu = I
  2. inverse. Tvu = Tvu−1
  3. commutative triangle. TvwTuv = Tuw

The rest of this section follows Lindbloom [7]. Given 3-vectors u and v with all entries positive, define the XYZ scaling CAT method by: Here and through the rest of this vignette, the vector division v/u is component by component, i.e. the Hadamard division, see [10]. One easily verifies that: so this CAT method satisfies property 3. Since Tuu is trivially invertible, it also satisfies properties 1 and 2 by Remark 3.1. Each channel in XYZ is scaled independently. When this technique is used in electronic RGB cameras, and applied to the RGB channels independently, it is often called white-balancing. This CAT method is what one would get by transforming to XYZ to Lab using u as the whitepoint, and then from Lab to XYZ using v as the whitepoint (though this view hides the linearity of Tuv). This CAT method is sometimes called the “Wrong von Kries”, see [3].



Von-Kries-Based CAT Methods

All the linear CAT methods in common use are von-Kries-based which is now defined, loosely following Lindbloom [7]. Let Ma be an invertible 3x3 matrix that does not move 1 too much, i.e. Ma1 ≈ 1. First define One easily checks that Tu1u = 1 and T1u1 = u and these matrices are inverses of each other. Now define a von-Kries-based CAT method in general by: One easily checks that so the method satisfies property 3. By Remark 3.1 it also satisfies properties 1 and 2.

The argument for property 3 can be visualized in categorical terms by this diagram:


Figure 5.1 - Commutative Diagram of CATs


Since the 3 inner triangles commute, the outer triangle commutes too. The symbol u3 denotes the space of all possible XYZs under illuminant u. We are being sloppy here because a CAT makes no sense when one of the XYZs is negative; but since the CATs in the figure are linear maps, they can still be defined mathematically even if they make no physical sense.

Note that the dependence of the transforms on Ma is suppressed in this notation. In the von Kries theory, Ma transforms from XYZ to the cone response domain. In practice Ma is calculated from a large number of pairs of experimentally measured corresponding colors. For example the popular Bradford Ma was calculated from 58 pairs, see [6]. Ma is called the cone response matrix for the von-Kries-based CAT method. In ArgyllCMS ICC color profiles, Ma is saved in the SigAbsToRelTransSpace or arts private ICC tag, see [3].

In the special case Ma = I the transforms become: which is just the XYZ scaling method in the previous section. There is more going on. In words, equation (5.1) says: transform u and v by Ma and form the diagonal matrix one would get from XYZ scaling. To adapt color A from u to v, transform A by Ma, and then by the diagonal matrix, and then transform back again by Ma−1. A fancy way to say the same thing is: a CAT matrix is von-Kries-based if and only if it is linearly conjugate to an XYZ scaling.

Since Ma has 9 parameters, it appears that these von-Kries-based CAT methods have 9 degrees of freedom. However, some changes of Ma do not change the transforms. Let D be a diagonal 3x3 matrix with positive entries on the diagonal, and let Ma := DMa. Then so changing Ma to DMa leaves the CAT unchanged. Setting D = diag (Ma1)−1 implies that Ma1 = 1. The matrix diag (Ma1) is invertible because Ma1 ≈ 1 as we assumed above. Thus, any acceptable cone response matrix can be normalized so that Ma1 = 1 exactly, i.e. the row sums are all 1. So these von-Kries-based CAT methods have 9-3 = 6 degrees of freedom. Geometrically, Ma leaves the line generated by 1 pointwise fixed, so we think of it as a “twist” around the line (more general than a rotation around the line). If {1, b2, b3} is a basis of 3, then Ma can map b2 and b3 to 3 almost arbitrarily, and this gives the 3+3 = 6 degrees of freedom.



Other Linear CAT Methods

This section is a detour that asks: “Are there are linear CAT methods that are NOT von-Kries-based ?” Note from (5.1) that if Tuv is a von-Kries-based matrix, then Tuv is diagonalizable. So if we can construct a linear CAT method whose matrices are not diagonalizable, then it is not von-Kries-based.

As an example, define Tuv to be the most direct rotation of u to the ray generated by v, followed by uniform scaling to take u to v. Since all the non-trivial Tuv are products of rotations and scalings, they are not diagonalizable. Note that this CAT method has properties 1 and 2, but NOT property 3. It shows that properties 1 and 2 do NOT imply property 3.

Taking an idea from the previous section, we can modify this method to have property 3. Define Tu1 to be the most direct rotation of u to the ray generated by 1, then followed by a uniform scaling that makes u map to 1. Define T1u to be the inverse of Tu1. Now define Tuv := T1vTu1 as before. All three properties in section CAT Methods are satisfied by the same arguments from the previous section. This CAT method may perform poorly. In Burns [2] the problem of negative tristimulus values is discussed, and this type of CAT method might make make the problem worse. I am sure that many other interesting (but impractical) examples can be constructed.



CAT Methods in Package spacesXYZ

Finally, we explore the CAT methods available in software.

library( spacesXYZ )

There is an S3 class CAT with constructor CAT(). The constructor takes arguments the source and target illuminant XYZ (denoted u and v above), and the CAT method. All the available non-trivial methods - Bradford, MCAT02, vonKries, and Bianco-Schettini - are von-Kries-based. For MCAT02 only the simple linear variant is used. The CAT object is a list with cone response matrix Ma (denoted Ma above), the adaptation matrix M (denoted Tuv above), and other things, see the CAT() man page.

Ma = CAT( source='A', target='D65', method='bradford' )$Ma ;  Ma
##         X       Y       Z
## L  0.8951  0.2664 -0.1614
## M -0.7502  1.7135  0.0367
## S  0.0389 -0.0685  1.0296

This is the famous Bradford cone-response-matrix, appearing in Lam [6] p. 3-46.

rowSums( Ma )
##      L      M      S 
## 1.0001 1.0000 1.0000

It appears that an attempt was made to normalize the row sums to 1, but roundoff made the last digit in the first row off by 1. There is no practical effect. The actual adaptation matrix is easily inspected and tested too:

theCAT  = CAT( source='A', target='D65', method='bradford' )
A  =  standardXYZ('A')
A %*% t(theCAT$M) - standardXYZ('D65')
##              X Y Z
## A 1.110223e-16 0 0

So M maps the XYZ of Illuminant A to that of D65 as required. Using explicit matrix multiplication is OK, but the function adaptXYZ() is preferred:

identical(  adaptXYZ( theCAT, A ), A %*% t(theCAT$M) )
## [1] FALSE

We can also inspect the row sums for method MCAT02.

rowSums( CAT( source='A', target='D65', method='MCAT02' )$Ma )
## L M S 
## 1 1 1

So for MCAT02 the normalization was more careful about roundoff. And for vonKries we have:

rowSums( CAT( source='A', target='D65', method='vonKries' )$Ma )
##       L       M       S 
## 1.02703 0.98472 0.91822

So for vonKries there was no normalization.





References

[1]
BROWN, William R. Two traditions of analogy. Informal Logic. 1989, 11(3), 161–172.
[2]
BURNS, Scott A. Chromatic adaptation transform by spectral reconstruction. Color Research & Application. 2019, 44(5), 682–693.
[3]
GILL, Graeme. ArgyllCMS’s ’arts’ (Absolute to media Relative Transform Space matrix) ICC tag (V1.0) [online]. no date. Available at: https://www.argyllcms.com/doc/ArgyllCMS_arts_tag.html
[4]
HUNT, R. W. G. The Reproduction of Colour, 6th Edition. B.m.: John Wiley & Sons, 2005. The Wiley-IS&T Series in Imaging Science and Technology. ISBN 9780470024263.
[5]
INTERNATIONAL COLOR CONSORTIUM. File Format for Color Profiles (version 4.1.0) [online]. Standard. B.m.: International Color Consortium. 2003. Available at: https://www.color.org/ICC1-V41.pdf
[6]
LAM, King Man. Metamerism and Colour Constancy. B.m.: University of Bradford, 1985.
[7]
LINDBLOOM, Bruce. Chromatic Adaptation [online]. no date. Available at: http://brucelindbloom.com/index.html?Eqn_ChromAdapt.html
[8]
STEENROD, Norman Earl. The topology of fibre bundles. B.m.: Princeton University Press, 1951. Princeton landmarks in mathematics and physics series. ISBN 9780691080550.
[9]
WIKIPEDIA CONTRIBUTORS. Fiber bundle construction theorem — Wikipedia, the free encyclopedia [online]. 2019. Available at: https://en.wikipedia.org/w/index.php?title=Fiber_bundle_construction_theorem. [Online; accessed 29-February-2020]
[10]
WIKIPEDIA CONTRIBUTORS. Hadamard product (matrices) — Wikipedia, the free encyclopedia [online]. 2020. Available at: https://en.wikipedia.org/w/index.php?title=Hadamard_product_(matrices). [Online; accessed 29-February-2020]



Appendix A - Recovery of Ma

Suppose we know that a chromatic adaption matrix Tuv is von-Kries-based and we also know the white points u and v. Can we recover the cone response matrix Ma ? This appendix presents a recipe to do that. We saw earlier that Ma is not unique; its rows are only defined up to a constant. So our solution will follow common practice and scale the rows to have sum 1.

Rearrange equation (5.1) to the form: We see that the columns of Ma - equivalently the rows of Ma - are the eigenvectors of Tuv. Amazingly, the eigenvectors of Tuv do not depend on u and v ! The eigenvectors are also only defined up to a constant, so rescaling them to have sum 1 is easy. But the eigenvectors (and eigenvalues) are also only defined up to a permutation; finding the right permutation is a little harder and is where u and v are used.

Here is a worked out example.

whiteA = standardXYZ("A")[1, ]  ;  whiteB = standardXYZ("B")[1, ]
theCAT = CAT( whiteA, whiteB, method='MCAT02' )
T   = theCAT$M ; Ma = theCAT$Ma
res = eigen( t(T) )
X   = t(res$vectors)  ;  X = diag( 1 / rowSums(X) ) %*% X  # X is 'first cut' at the unknown Ma

Compare Ma and X

Ma ; X
##         X      Y       Z
## L  0.7328 0.4296 -0.1624
## M -0.7036 1.6975  0.0061
## S  0.0030 0.0136  0.9834
##         [,1]   [,2]    [,3]
## [1,]  0.0030 0.0136  0.9834
## [2,] -0.7036 1.6975  0.0061
## [3,]  0.7328 0.4296 -0.1624

One can check that the row sums of Ma are 1. Now compare the white ratios and the eigenvalues:

as.numeric(Ma %*% whiteB / Ma %*% whiteA) ; res$values
## [1] 0.8643826 1.0850937 2.3297865
## [1] 2.3297865 1.0850937 0.8643826

So the row orders are reversed, and the eigenvalues too. The two permutations are the same: c(3,2,1).
In practice Ma is the unknown so we do not have it, but we do have the eigenvalues res$values. Since we know that res$values are returned in decreasing order, we can compute the desired permutation like this:

perm = order( Ma %*% whiteB / Ma %*% whiteA,  decreasing=TRUE )  ; perm
## [1] 3 2 1

The permutation perm now takes the ratios to the eigenvalues, but we want the inverse, so invert perm:

perm = order(perm) ; perm   
## [1] 3 2 1
res$values[perm]
## [1] 0.8643826 1.0850937 2.3297865
X = X[perm, ]  ;  X  ;  max( abs(X - Ma) )
##         [,1]   [,2]    [,3]
## [1,]  0.7328 0.4296 -0.1624
## [2,] -0.7036 1.6975  0.0061
## [3,]  0.0030 0.0136  0.9834
## [1] 2.220446e-15

We have recovered Ma with good accuracy.

Now suppose u and v are not known, but Ma is known to be in a small list of candidate matrices. Since there only 6 possible permutations that take matrix X to the right candidate, it will be easy to spot the right one.



Session Information

R version 4.4.2 (2024-10-31)
Platform: x86_64-pc-linux-gnu
Running under: Ubuntu 24.04.1 LTS

Matrix products: default
BLAS:   /usr/lib/x86_64-linux-gnu/openblas-pthread/libblas.so.3 
LAPACK: /usr/lib/x86_64-linux-gnu/openblas-pthread/libopenblasp-r0.3.26.so;  LAPACK version 3.12.0

locale:
 [1] LC_CTYPE=en_US.UTF-8       LC_NUMERIC=C               LC_TIME=en_US.UTF-8       
 [4] LC_COLLATE=C               LC_MONETARY=en_US.UTF-8    LC_MESSAGES=en_US.UTF-8   
 [7] LC_PAPER=en_US.UTF-8       LC_NAME=C                  LC_ADDRESS=C              
[10] LC_TELEPHONE=C             LC_MEASUREMENT=en_US.UTF-8 LC_IDENTIFICATION=C       

time zone: Etc/UTC
tzcode source: system (glibc)

attached base packages:
[1] stats     graphics  grDevices utils     datasets  methods   base     

other attached packages:
[1] spacesXYZ_1.4-0 rmarkdown_2.29 

loaded via a namespace (and not attached):
 [1] digest_0.6.37        R6_2.5.1             microbenchmark_1.5.0 fastmap_1.2.0       
 [5] xfun_0.50            maketools_1.3.1      glue_1.8.0           cachem_1.1.0        
 [9] knitr_1.49           htmltools_0.5.8.1    logger_0.4.0         buildtools_1.0.0    
[13] lifecycle_1.0.4      cli_3.6.3            sass_0.4.9           jquerylib_0.1.4     
[17] compiler_4.4.2       sys_3.4.3            tools_4.4.2          evaluate_1.0.3      
[21] bslib_0.8.0          yaml_2.3.10          jsonlite_1.8.9       rlang_1.1.4