sectionOptimalColors {colorSpec} | R Documentation |
compute sections of an optimal color surface by hyperplanes
Description
Consider a colorSpec object x
with type
equal to 'responsivity.material'
.
The set of all possible material reflectance functions (or transmittance functions)
is convex, closed, and bounded (in fact they form a cube),
and this implies that the set of all possible output responses
from x
is also convex, closed, and bounded.
The latter set is called the object-color solid or Rösch Farbkörper for x
.
If the dimension of the response of x
is 2,
this solid is a convex polygon
that is centrally symmetric - a zonogon.
If the dimension of the response of x
is 3 (e.g. RGB or XYZ),
this solid is a special type of centrally symmetric convex polyhedron
called a zonohedron, see Centore.
This function only supports dimensions 2 and 3.
Denote this object-color solid by Z.
A color on the boundary of Z is called an optimal color. Consider the intersection of a hyperplane with the boundary of Z. Let the equation of the hyperplane be given by:
<v,normal> = \beta
where normal
is orthogonal to the hyperplane,
and \beta
is the plane constant, and v
is a variable vector.
The purpose of the function sectionOptimalColors()
is to compute the intersection set.
In dimension 2 this hyperplane is a line, and the intersection is generically 2 points, and 1 point if the line only intersects the boundary (we ignore the special case when the intersection is an edge of the polygon).
In dimension 3 this hyperplane is a 2D plane, and the intersection is generically a polygon, and 1 point if the line only intersects the boundary (we ignore the special case when the intersection is a face of the zonohedron).
Of course, the intersection can also be empty.
Usage
## S3 method for class 'colorSpec'
sectionOptimalColors( x, normal, beta )
Arguments
x |
a colorSpec object with |
normal |
a nonzero vector of dimension M, that is the normal to a hyperplane |
beta |
a vector of numbers of positive length.
The number |
.
Details
Consider first the case that the dimension of x
is 3,
so that Z is a zonohedron.
In the preprocessing phase the zonohedral representation is calculated.
The faces of Z are either parallelograms,
or compound faces that are partitioned into parallelograms.
The centers of all these parallelograms are computed,
along with their extent in direction normal
.
For a given plane <v,normal>=\beta
,
the parallelograms that intersect the plane are extracted.
The boundary of each parallelogram intersects the plane in 2 points (in general)
and one of those points is computed.
The set of all these points is then sorted into proper order around the boundary.
In the case that the dimension of x
is 2,
so that Z is a zonogon,
the parallelograms are replaced by line segments (edges),
and the processing is much easier.
Value
The function returns a list with an item for each value in vector beta
.
Each item in the output is a list with these items:
beta |
the value of the plane constant |
section |
an NxM matrix, where N is the number of points in the section,
and M is the dimension of |
In case of global error, the function returns NULL
.
WARNING
The preprocessing calculation of the zonohedron dominates the total time.
And this time goes up rapidly with the number of wavelengths.
We recommend using a wavelength step of 5nm, as in the Examples.
For best results, batch a lot of beta
s into a single function call
and then process the output.
Moreover, the preprocessing time is dominated by the partitioning
of the compound faces into parallelograms.
This is made worse by an x
whose spectral responses have little overlap,
as in scanner.ACES
.
In these cases, try a larger step size, and then reduce.
Optimizing these compound faces is a possible topic for the future.
References
Centore, Paul. A Zonohedral Approach to Optimal Colours. Color Research & Application. Vol. 38. No. 2. pp. 110-119. April 2013.
Logvinenko, A. D.
An object-color space.
Journal of Vision.
9(11):5, 1-23, (2009).
https://jov.arvojournals.org/article.aspx?articleid=2203976
.
doi:10.1167/9.11.5.
See Also
vignette Plotting Chromaticity Loci of Optimal Colors,
probeOptimalColors()
Examples
wave = seq(420,680,by=5)
Flea2.scanner = product( A.1nm, "material", Flea2.RGB, wavelength=wave )
seclist = sectionOptimalColors( Flea2.scanner, normal=c(0,1,0), beta=10 )
length( seclist[[1]]$section )
seclist[[1]]$section[ 1:5, ]
## [1] 207 # the polygon has 207 vertices, and the first 5 are:
## Red Green Blue
## [1,] 109.2756 10 3.5391342
## [2,] 109.5729 10 2.5403628
## [3,] 109.8078 10 1.7020526
## [4,] 109.9942 10 1.0111585
## [5,] 110.1428 10 0.4513051