Source code for colour.models.osa_ucs

"""
Optical Society of America Uniform Colour Scales (OSA UCS)
==========================================================

Define the *OSA UCS* colourspace transformations.

-   :func:`colour.XYZ_to_OSA_UCS`
-   :func:`colour.OSA_UCS_to_XYZ`

References
----------
-   :cite:`Cao2013` : Cao, R., Trussell, H. J., & Shamey, R. (2013). Comparison
    of the performance of inverse transformation methods from OSA-UCS to
    CIEXYZ. Journal of the Optical Society of America A, 30(8), 1508.
    doi:10.1364/JOSAA.30.001508
-   :cite:`Moroney2003` : Moroney, N. (2003). A Radial Sampling of the OSA
    Uniform Color Scales. Color and Imaging Conference, 2003(1), 175-180.
    ISSN:2166-9635
-   :cite:`Schlomer2019` : Schlömer, N. (2019). On the conversion from OSA-UCS
    to CIEXYZ (Version 2). arXiv. doi:10.48550/ARXIV.1911.08323
"""

from __future__ import annotations

import typing

import numpy as np

from colour.algebra import sdiv, sdiv_mode, spow, vecmul

if typing.TYPE_CHECKING:
    from colour.hints import NDArrayFloat

from colour.hints import (  # noqa: TC001
    Domain100,
    NDArrayFloat,
    Range100,
)
from colour.models import XYZ_to_xyY
from colour.utilities import (
    from_range_100,
    to_domain_100,
    tsplit,
    tstack,
)

__author__ = "Colour Developers"
__copyright__ = "Copyright 2013 Colour Developers"
__license__ = "BSD-3-Clause - https://opensource.org/licenses/BSD-3-Clause"
__maintainer__ = "Colour Developers"
__email__ = "colour-developers@colour-science.org"
__status__ = "Production"

__all__ = [
    "XYZ_to_OSA_UCS",
    "OSA_UCS_to_XYZ",
]

MATRIX_XYZ_TO_RGB_OSA_UCS: NDArrayFloat = np.array(
    [
        [0.799, 0.4194, -0.1648],
        [-0.4493, 1.3265, 0.0927],
        [-0.1149, 0.3394, 0.717],
    ]
)
"""
*OSA UCS* matrix converting from *CIE XYZ* tristimulus values to *RGB*
colourspace.
"""

MATRIX_RGB_TO_XYZ_OSA_UCS: NDArrayFloat = np.linalg.inv(MATRIX_XYZ_TO_RGB_OSA_UCS)
"""
*OSA UCS* matrix converting from *RGB* colourspace to *CIE XYZ* tristimulus
values (inverse of MATRIX_XYZ_TO_RGB_OSA_UCS).
"""


[docs] def XYZ_to_OSA_UCS(XYZ: Domain100) -> Range100: """ Convert from *CIE XYZ* tristimulus values under the *CIE 1964 10 Degree Standard Observer* to *OSA UCS* colourspace. The lightness axis, *L*, is typically in range [-9, 5] and centered around middle gray (Munsell N/6). The yellow-blue axis, *j*, is typically in range [-15, 15]. The red-green axis, *g*, is typically in range [-20, 15]. Parameters ---------- XYZ *CIE XYZ* tristimulus values under the *CIE 1964 10 Degree Standard Observer*. Returns ------- :class:`numpy.ndarray` *OSA UCS* :math:`Ljg` lightness, jaune (yellowness), and greenness. Notes ----- +------------+-----------------------+--------------------+ | **Domain** | **Scale - Reference** | **Scale - 1** | +============+=======================+====================+ | ``XYZ`` | 100 | 1 | +------------+-----------------------+--------------------+ +------------+-----------------------+--------------------+ | **Range** | **Scale - Reference** | **Scale - 1** | +============+=======================+====================+ | ``Ljg`` | 100 | 1 | +------------+-----------------------+--------------------+ - *OSA UCS* uses the *CIE 1964 10 Degree Standard Observer*. References ---------- :cite:`Cao2013`, :cite:`Moroney2003` Examples -------- >>> import numpy as np >>> XYZ = np.array([0.20654008, 0.12197225, 0.05136952]) * 100 >>> XYZ_to_OSA_UCS(XYZ) # doctest: +ELLIPSIS array([-3.0049979..., 2.9971369..., -9.6678423...]) """ XYZ = to_domain_100(XYZ) x, y, Y = tsplit(XYZ_to_xyY(XYZ)) Y_0 = Y * ( 4.4934 * x**2 + 4.3034 * y**2 - 4.276 * x * y - 1.3744 * x - 2.5643 * y + 1.8103 ) o_3 = 1 / 3 Y_0_es = spow(Y_0, o_3) - 2 / 3 # Gracefully handles Y_0 < 30. Y_0_s = Y_0 - 30 Lambda = 5.9 * (Y_0_es + 0.042 * spow(Y_0_s, o_3)) RGB = vecmul(MATRIX_XYZ_TO_RGB_OSA_UCS, XYZ) RGB_3 = spow(RGB, 1 / 3) with sdiv_mode(): C = sdiv(Lambda, 5.9 * Y_0_es) L = (Lambda - 14.4) / spow(2, 1 / 2) j = C * np.dot(RGB_3, np.array([1.7, 8, -9.7])) g = C * np.dot(RGB_3, np.array([-13.7, 17.7, -4])) Ljg = tstack([L, j, g]) return from_range_100(Ljg)
[docs] def OSA_UCS_to_XYZ(Ljg: Domain100, optimisation_kwargs: dict | None = None) -> Range100: """ Convert from *OSA UCS* colourspace to *CIE XYZ* tristimulus values under the *CIE 1964 10 Degree Standard Observer*. The lightness axis, *L*, is typically in range [-9, 5] and centered around middle gray (Munsell N/6). The yellow-blue axis, *j*, is typically in range [-15, 15]. The red-green axis, *g*, is typically in range [-20, 15]. Parameters ---------- Ljg *OSA UCS* :math:`Ljg` lightness, jaune (yellowness), and greenness. optimisation_kwargs Parameters for Newton iteration. Supported parameters: - *iterations_maximum*: Maximum number of iterations (default: 20). - *tolerance*: Convergence tolerance (default: 1e-10). - *epsilon*: Step size for numerical derivative (default: 1e-8). Returns ------- :class:`numpy.ndarray` *CIE XYZ* tristimulus values under the *CIE 1964 10 Degree Standard Observer*. Notes ----- +------------+-----------------------+--------------------+ | **Domain** | **Scale - Reference** | **Scale - 1** | +============+=======================+====================+ | ``Ljg`` | 100 | 1 | +------------+-----------------------+--------------------+ +------------+-----------------------+--------------------+ | **Range** | **Scale - Reference** | **Scale - 1** | +============+=======================+====================+ | ``XYZ`` | 100 | 1 | +------------+-----------------------+--------------------+ - *OSA UCS* uses the *CIE 1964 10 Degree Standard Observer*. - This implementation uses the improved algorithm from :cite:`Schlomer2019` which employs Cardano's formula for solving the cubic equation and Newton's method for the remaining nonlinear system. References ---------- :cite:`Cao2013`, :cite:`Moroney2003`, :cite:`Schlomer2019` Examples -------- >>> import numpy as np >>> Ljg = np.array([-3.00499790, 2.99713697, -9.66784231]) >>> OSA_UCS_to_XYZ(Ljg) # doctest: +ELLIPSIS array([ 20.654008..., 12.197225..., 5.1369520...]) """ Ljg = to_domain_100(Ljg) shape = Ljg.shape Ljg = np.atleast_1d(np.reshape(Ljg, (-1, 3))) # Default optimization settings settings: dict[str, typing.Any] = { "iterations_maximum": 20, "tolerance": 1e-10, "epsilon": 1e-8, } if optimisation_kwargs is not None: settings.update(optimisation_kwargs) L, j, g = tsplit(Ljg) # Step 1: Compute L' from L # Forward: L = (Lambda - 14.4) / sqrt(2) # Backward: Lambda = L * sqrt(2) + 14.4 # But L' = Lambda in the intermediate calculation sqrt_2 = np.sqrt(2) L_prime = L * sqrt_2 + 14.4 # Step 2: Solve for Y0 using Cardano's formula # Equation: 0 = f(t) = (L'/5.9 + 2/3 - t)³ - 0.042^3(t^3 - 30) # where t = Y0^(1/3) u = L_prime / 5.9 + 2.0 / 3.0 v = 0.042**3 # Cubic equation: at^3 + bt^2 + ct + d = 0 a = -(v + 1) b = 3 * u c = -3 * u**2 d = u**3 + 30 * v # Convert to depressed cubic: x³ + px + q = 0 p = (3 * a * c - b**2) / (3 * a**2) q = (2 * b**3 - 9 * a * b * c + 27 * a**2 * d) / (27 * a**3) # Cardano's formula discriminant = (q / 2) ** 2 + (p / 3) ** 3 with sdiv_mode(): t = ( -b / (3 * a) + spow(-q / 2 + np.sqrt(discriminant), 1.0 / 3.0) + spow(-q / 2 - np.sqrt(discriminant), 1.0 / 3.0) ) Y0 = t**3 # Step 3: Compute C, a, b with sdiv_mode(): C = sdiv(L_prime, 5.9 * (t - 2.0 / 3.0)) a_coef = sdiv(g, C) b_coef = sdiv(j, C) # Step 4: Solve for RGB using Newton iteration # Matrix A from equation (4) A = np.array([[-13.7, 17.7, -4.0], [1.7, 8.0, -9.7]]) # Augment A with [1, 0, 0] to make it non-singular (set w = cbrt(R)) A_augmented = np.vstack([A, [1.0, 0.0, 0.0]]) A_inv = np.linalg.inv(A_augmented) # Initial guess for w (corresponds to cbrt(R)) # w0 = cbrt(79.9 + 41.94) from paper w = np.full_like(L, (79.9 + 41.94) ** (1.0 / 3.0)) # Newton iteration for _iteration in range(settings["iterations_maximum"]): # Solve for [cbrt(R), cbrt(G), cbrt(B)] given current w ab_w = np.array([a_coef, b_coef, w]).T RGB_cbrt = np.dot(ab_w, A_inv.T) RGB = RGB_cbrt**3 XYZ = vecmul(MATRIX_RGB_TO_XYZ_OSA_UCS, RGB) X, Y, Z = tsplit(XYZ) with sdiv_mode(): sum_XYZ = X + Y + Z x = sdiv(X, sum_XYZ) y = sdiv(Y, sum_XYZ) K = ( 4.4934 * x**2 + 4.3034 * y**2 - 4.276 * x * y - 1.3744 * x - 2.5643 * y + 1.8103 ) Y0_computed = Y * K error = Y0_computed - Y0 if np.all(np.abs(error) < settings["tolerance"]): break # Newton step: compute derivative and update w # Derivative is computed numerically for robustness epsilon = settings["epsilon"] w_plus = w + epsilon ab_w_plus = np.array([a_coef, b_coef, w_plus]).T RGB_cbrt_plus = np.dot(ab_w_plus, A_inv.T) RGB_plus = RGB_cbrt_plus**3 XYZ_plus = vecmul(MATRIX_RGB_TO_XYZ_OSA_UCS, RGB_plus) X_plus, Y_plus, Z_plus = tsplit(XYZ_plus) with sdiv_mode(): sum_XYZ_plus = X_plus + Y_plus + Z_plus x_plus = sdiv(X_plus, sum_XYZ_plus) y_plus = sdiv(Y_plus, sum_XYZ_plus) K_plus = ( 4.4934 * x_plus**2 + 4.3034 * y_plus**2 - 4.276 * x_plus * y_plus - 1.3744 * x_plus - 2.5643 * y_plus + 1.8103 ) Y0_computed_plus = Y_plus * K_plus with sdiv_mode(): derivative = sdiv(Y0_computed_plus - Y0_computed, epsilon) w = w - sdiv(error, derivative) return from_range_100(np.reshape(XYZ, shape))