Source code for colour.quality.tm3018

"""
ANSI/IES TM-30-18 Colour Fidelity Index
=======================================

Defines the *ANSI/IES TM-30-18 Colour Fidelity Index* (CFI) computation
objects:

- :class:`colour.quality.ColourQuality_Specification_ANSIIESTM3018`
- :func:`colour.quality.colour_fidelity_index_ANSIIESTM3018`

References
----------
-   :cite:`ANSI2018` : ANSI, & IES Color Committee. (2018). ANSI/IES TM-30-18 -
    IES Method for Evaluating Light Source Color Rendition.
    ISBN:978-0-87995-379-9
"""

from __future__ import annotations

import numpy as np
from dataclasses import dataclass

from colour.colorimetry import SpectralDistribution
from colour.hints import (
    ArrayLike,
    Boolean,
    Floating,
    List,
    NDArray,
    Tuple,
    Union,
    cast,
)
from colour.quality import colour_fidelity_index_CIE2017
from colour.quality.cfi2017 import (
    ColourRendering_Specification_CIE2017,
    TCS_ColorimetryData_CIE2017,
    delta_E_to_R_f,
)
from colour.utilities import as_float_array, as_int_scalar


[docs]@dataclass class ColourQuality_Specification_ANSIIESTM3018: """ Define the *ANSI/IES TM-30-18 Colour Fidelity Index* (CFI) colour quality specification. Parameters ---------- name Name of the test spectral distribution. sd_test Spectral distribution of the tested illuminant. sd_reference Spectral distribution of the reference illuminant. R_f *Colour Fidelity Index* (CFI) :math:`R_f`. R_s Individual *colour fidelity indexes* data for each sample. CCT Correlated colour temperature :math:`T_{cp}`. D_uv Distance from the Planckian locus :math:`\\Delta_{uv}`. colorimetry_data Colorimetry data for the test and reference computations. R_g Gamut index :math:`R_g`. bins List of 16 lists, each containing the indexes of colour samples that lie in the respective hue bin. averages_test Averages of *CAM02-UCS* a', b' coordinates for each hue bin for test samples. averages_reference Averages for reference samples. average_norms Distance of averages for reference samples from the origin. R_fs Local colour fidelities for each hue bin. R_cs Local chromaticity shifts for each hue bin, in percents. R_hs Local hue shifts for each hue bin. """ name: str sd_test: SpectralDistribution sd_reference: SpectralDistribution R_f: Floating R_s: NDArray CCT: Floating D_uv: Floating colorimetry_data: Tuple[ Tuple[TCS_ColorimetryData_CIE2017, ...], Tuple[TCS_ColorimetryData_CIE2017, ...], ] R_g: Floating bins: List[List[int]] averages_test: NDArray averages_reference: NDArray average_norms: NDArray R_fs: NDArray R_cs: NDArray R_hs: NDArray
[docs]def colour_fidelity_index_ANSIIESTM3018( sd_test: SpectralDistribution, additional_data: Boolean = False ) -> Union[ Floating, ColourQuality_Specification_ANSIIESTM3018, ColourRendering_Specification_CIE2017, ]: """ Return the *ANSI/IES TM-30-18 Colour Fidelity Index* (CFI) :math:`R_f` of given spectral distribution. Parameters ---------- sd_test Test spectral distribution. additional_data Whether to output additional data. Returns ------- :class:`numpy.floating` or \ :class:`colour.quality.ColourQuality_Specification_ANSIIESTM3018` *ANSI/IES TM-30-18 Colour Fidelity Index* (CFI). References ---------- :cite:`ANSI2018` Examples -------- >>> from colour import SDS_ILLUMINANTS >>> sd = SDS_ILLUMINANTS['FL2'] >>> colour_fidelity_index_ANSIIESTM3018(sd) # doctest: +ELLIPSIS 70.1208254... """ if not additional_data: return colour_fidelity_index_CIE2017(sd_test, False) specification: ( ColourRendering_Specification_CIE2017 ) = colour_fidelity_index_CIE2017( sd_test, True ) # type: ignore[assignment] # Setup bins based on where the reference a'b' points are located. bins: List[List[int]] = [[] for _i in range(16)] for i, sample in enumerate(specification.colorimetry_data[1]): bin_index = as_int_scalar( np.floor(cast(Floating, sample.CAM.h) / 22.5) ) bins[bin_index].append(i) # Per-bin a'b' averages. averages_test = np.empty([16, 2]) averages_reference = np.empty([16, 2]) for i in range(16): apbp_s = [ specification.colorimetry_data[0][j].Jpapbp[[1, 2]] for j in bins[i] ] averages_test[i, :] = np.mean(apbp_s, axis=0) apbp_s = [ specification.colorimetry_data[1][j].Jpapbp[[1, 2]] for j in bins[i] ] averages_reference[i, :] = np.mean(apbp_s, axis=0) # Gamut Index. R_g = 100 * ( averages_area(averages_test) / averages_area(averages_reference) ) # Local colour fidelity indexes, i.e. 16 CFIs for each bin. bin_delta_E_s = [ np.mean([specification.delta_E_s[bins[i]]]) for i in range(16) ] R_fs = as_float_array(delta_E_to_R_f(bin_delta_E_s)) # Angles bisecting the hue bins. angles = (22.5 * np.arange(16) + 11.25) / 180 * np.pi cosines = np.cos(angles) sines = np.sin(angles) average_norms = np.linalg.norm(averages_reference, axis=1) a_deltas = averages_test[:, 0] - averages_reference[:, 0] b_deltas = averages_test[:, 1] - averages_reference[:, 1] # Local chromaticity shifts, multiplied by 100 to obtain percentages. R_cs = 100 * (a_deltas * cosines + b_deltas * sines) / average_norms # Local hue shifts. R_hs = (-a_deltas * sines + b_deltas * cosines) / average_norms return ColourQuality_Specification_ANSIIESTM3018( specification.name, sd_test, specification.sd_reference, specification.R_f, specification.R_s, specification.CCT, specification.D_uv, specification.colorimetry_data, R_g, bins, averages_test, averages_reference, average_norms, R_fs, R_cs, R_hs, )
def averages_area(averages: ArrayLike) -> Floating: """ Compute the area of the polygon formed by the hue bin averages. Parameters ---------- averages Hue bin averages. Returns ------- :class:`numpy.floating` Area of the polygon. """ averages = as_float_array(averages) N = averages.shape[0] triangle_areas = np.empty(N) for i in range(N): u = averages[i, :] v = averages[(i + 1) % N, :] triangle_areas[i] = (u[0] * v[1] - u[1] * v[0]) / 2 return np.sum(triangle_areas)