#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
Nayatani (1995) Colour Appearance Model
=======================================
Defines Nayatani (1995) colour appearance model objects:
- :class:`Nayatani95_Specification`
- :func:`XYZ_to_Nayatani95`
See Also
--------
`Nayatani (1995) Colour Appearance Model IPython Notebook
<http://nbviewer.ipython.org/github/colour-science/colour-ipython/blob/master/notebooks/appearance/nayatani95.ipynb>`_ # noqa
References
----------
.. [1] Fairchild, M. D. (2013). The Nayatani et al. Model. In Color
Appearance Models (3rd ed., pp. 4810–5085). Wiley. ASIN:B00DAYO8E2
.. [2] Nayatani, Y., Sobagaki, H., & Yano, K. H. T. (1995). Lightness
dependency of chroma scales of a nonlinear color-appearance model and
its latest formulation. Color Research & Application, 20(3), 156–167.
doi:10.1002/col.5080200305
"""
from __future__ import division, unicode_literals
import numpy as np
from collections import namedtuple
from colour.adaptation.cie1994 import (
CIE1994_XYZ_TO_RGB_MATRIX,
beta_1,
exponential_factors,
intermediate_values)
from colour.models import XYZ_to_xy
from colour.utilities import dot_vector, tsplit, tstack
__author__ = 'Colour Developers'
__copyright__ = 'Copyright (C) 2013 - 2015 - Colour Developers'
__license__ = 'GPL V3.0 - http://www.gnu.org/licenses/'
__maintainer__ = 'Colour Developers'
__email__ = 'colour-science@googlegroups.com'
__status__ = 'Production'
__all__ = ['NAYATANI95_XYZ_TO_RGB_MATRIX',
'Nayatani95_ReferenceSpecification',
'Nayatani95_Specification',
'XYZ_to_Nayatani95',
'illuminance_to_luminance',
'XYZ_to_RGB_Nayatani95',
'scaling_coefficient',
'achromatic_response',
'tritanopic_response',
'protanopic_response',
'brightness_correlate',
'ideal_white_brightness_correlate',
'achromatic_lightness_correlate',
'normalised_achromatic_lightness_correlate',
'hue_angle',
'saturation_components',
'saturation_correlate',
'chroma_components',
'chroma_correlate',
'colourfulness_components',
'colourfulness_correlate',
'chromatic_strength_function']
NAYATANI95_XYZ_TO_RGB_MATRIX = CIE1994_XYZ_TO_RGB_MATRIX
"""
Nayatani (1995) colour appearance model *CIE XYZ* tristimulus values to cone
responses matrix.
NAYATANI95_XYZ_TO_RGB_MATRIX : array_like, (3, 3)
"""
[docs]class Nayatani95_ReferenceSpecification(
namedtuple(
'Nayatani95_ReferenceSpecification',
('Lstar_P', 'C', 'theta', 'S', 'B_r', 'M', 'H', 'H_C', 'Lstar_N'))):
"""
Defines the Nayatani (1995) colour appearance model reference
specification.
This specification has field names consistent with Fairchild (2013)
reference.
Parameters
----------
Lstar_P : numeric or array_like
Correlate of *achromatic Lightness* :math:`L_p^\star`.
C : numeric or array_like
Correlate of *chroma* :math:`C`.
theta : numeric or array_like
*Hue* angle :math:`\\theta` in degrees.
S : numeric or array_like
Correlate of *saturation* :math:`S`.
B_r : numeric or array_like
Correlate of *brightness* :math:`B_r`.
M : numeric or array_like
Correlate of *colourfulness* :math:`M`.
H : numeric or array_like
*Hue* :math:`h` quadrature :math:`H`.
H_C : numeric or array_like
*Hue* :math:`h` composition :math:`H_C`.
Lstar_N : numeric or array_like
Correlate of *normalised achromatic Lightness* :math:`L_n^\star`.
"""
[docs]class Nayatani95_Specification(
namedtuple('Nayatani95_Specification',
('Lstar_P', 'C', 'h', 's', 'Q', 'M', 'H', 'HC', 'Lstar_N'))):
"""
Defines the Nayatani (1995) colour appearance model specification.
This specification has field names consistent with the remaining colour
appearance models in :mod:`colour.appearance` but diverge from Fairchild
(2013) reference.
Parameters
----------
Lstar_P : numeric or array_like
Correlate of *achromatic Lightness* :math:`L_p^\star`.
C : numeric or array_like
Correlate of *chroma* :math:`C`.
h : numeric or array_like
*Hue* angle :math:`\\theta` in degrees.
s : numeric or array_like
Correlate of *saturation* :math:`S`.
Q : numeric or array_like
Correlate of *brightness* :math:`B_r`.
M : numeric or array_like
Correlate of *colourfulness* :math:`M`.
H : numeric or array_like
*Hue* :math:`h` quadrature :math:`H`.
HC : numeric or array_like
*Hue* :math:`h` composition :math:`H_C`.
Lstar_N : numeric or array_like
Correlate of *normalised achromatic Lightness* :math:`L_n^\star`.
"""
[docs]def XYZ_to_Nayatani95(XYZ,
XYZ_n,
Y_o,
E_o,
E_or,
n=1):
"""
Computes the Nayatani (1995) colour appearance model correlates.
Parameters
----------
XYZ : array_like
*CIE XYZ* tristimulus values of test sample / stimulus in domain
[0, 100].
XYZ_n : array_like
*CIE XYZ* tristimulus values of reference white in domain [0, 100].
Y_o : numeric or array_like
Luminance factor :math:`Y_o` of achromatic background as percentage in
domain [0.18, 1.0]
E_o : numeric or array_like
Illuminance :math:`E_o` of the viewing field in lux.
E_or : numeric or array_like
Normalising illuminance :math:`E_{or}` in lux usually in domain
[1000, 3000]
n : numeric or array_like, optional
Noise term used in the non linear chromatic adaptation model.
Returns
-------
Nayatani95_Specification
Nayatani (1995) colour appearance model specification.
Warning
-------
The input domain of that definition is non standard!
Notes
-----
- Input *CIE XYZ* tristimulus values are in domain [0, 100].
- Input *CIE XYZ_n* tristimulus values are in domain [0, 100].
Examples
--------
>>> XYZ = np.array([19.01, 20.00, 21.78])
>>> XYZ_n = np.array([95.05, 100.00, 108.88])
>>> Y_o = 20.0
>>> E_o = 5000.0
>>> E_or = 1000.0
>>> XYZ_to_Nayatani95(XYZ, XYZ_n, Y_o, E_o, E_or) # doctest: +ELLIPSIS
Nayatani95_Specification(Lstar_P=49.9998829..., C=0.0133550..., h=257.5232268..., s=0.0133550..., Q=62.6266734..., M=0.0167262..., H=None, HC=None, Lstar_N=50.0039154...)
"""
Y_o = np.asarray(Y_o)
E_o = np.asarray(E_o)
E_or = np.asarray(E_or)
# Computing adapting luminance :math:`L_o` and normalising luminance
# :math:`L_{or}` in in :math:`cd/m^2`.
# L_o = illuminance_to_luminance(E_o, Y_o)
L_or = illuminance_to_luminance(E_or, Y_o)
# Computing :math:`\xi`, :math:`\eta`, :math:`\zeta` values.
xez = intermediate_values(XYZ_to_xy(XYZ_n))
xi, eta, zeta = tsplit(xez)
# Computing adapting field cone responses.
RGB_o = (((Y_o[..., np.newaxis] * E_o[..., np.newaxis]) /
(100 * np.pi)) * xez)
# Computing stimulus cone responses.
RGB = XYZ_to_RGB_Nayatani95(XYZ)
R, G, B = tsplit(RGB)
# Computing exponential factors of the chromatic adaptation.
bRGB_o = exponential_factors(RGB_o)
bL_or = beta_1(L_or)
# Computing scaling coefficients :math:`e(R)` and :math:`e(G)`
eR = scaling_coefficient(R, xi)
eG = scaling_coefficient(G, eta)
# Computing opponent colour dimensions.
# Computing achromatic response :math:`Q`:
Q_response = achromatic_response(RGB, bRGB_o, xez, bL_or, eR, eG, n)
# Computing tritanopic response :math:`t`:
t_response = tritanopic_response(RGB, bRGB_o, xez, n)
# Computing protanopic response :math:`p`:
p_response = protanopic_response(RGB, bRGB_o, xez, n)
# -------------------------------------------------------------------------
# Computing the correlate of *brightness* :math:`B_r`.
# -------------------------------------------------------------------------
B_r = brightness_correlate(bRGB_o, bL_or, Q_response)
# Computing *brightness* :math:`B_{rw}` of ideal white.
brightness_ideal_white = ideal_white_brightness_correlate(
bRGB_o, xez, bL_or, n)
# -------------------------------------------------------------------------
# Computing the correlate of achromatic *Lightness* :math:`L_p^\star`.
# -------------------------------------------------------------------------
Lstar_P = (
achromatic_lightness_correlate(Q_response))
# -------------------------------------------------------------------------
# Computing the correlate of normalised achromatic *Lightness*
# :math:`L_n^\star`.
# -------------------------------------------------------------------------
Lstar_N = (
normalised_achromatic_lightness_correlate(B_r, brightness_ideal_white))
# -------------------------------------------------------------------------
# Computing the *hue* angle :math:`\\theta`.
# -------------------------------------------------------------------------
theta = hue_angle(p_response, t_response)
# TODO: Implement hue quadrature & composition computation.
# -------------------------------------------------------------------------
# Computing the correlate of *saturation* :math:`S`.
# -------------------------------------------------------------------------
S_RG, S_YB = tsplit(saturation_components(
theta, bL_or, t_response, p_response))
S = saturation_correlate(S_RG, S_YB)
# -------------------------------------------------------------------------
# Computing the correlate of *chroma* :math:`C`.
# -------------------------------------------------------------------------
# C_RG, C_YB = tsplit(chroma_components(Lstar_P, S_RG, S_YB))
C = chroma_correlate(Lstar_P, S)
# -------------------------------------------------------------------------
# Computing the correlate of *colourfulness* :math:`M`.
# -------------------------------------------------------------------------
# TODO: Investigate components usage.
# M_RG, M_YB = tsplit(colourfulness_components(C_RG, C_YB,
# brightness_ideal_white))
M = colourfulness_correlate(C, brightness_ideal_white)
return Nayatani95_Specification(Lstar_P,
C,
theta,
S,
B_r,
M,
None,
None,
Lstar_N)
[docs]def illuminance_to_luminance(E, Y_f):
"""
Converts given *illuminance* :math:`E` value in lux to *luminance* in
:math:`cd/m^2`.
Parameters
----------
E : numeric or array_like
*Illuminance* :math:`E` in lux.
Y_f : numeric or array_like
*Luminance* factor :math:`Y_f` in :math:`cd/m^2`.
Returns
-------
numeric or ndarray
*Luminance* :math:`Y` in :math:`cd/m^2`.
Examples
--------
>>> illuminance_to_luminance(5000.0, 20.0) # doctest: +ELLIPSIS
318.3098861...
"""
E = np.asarray(E)
Y_f = np.asarray(Y_f)
return Y_f * E / (100 * np.pi)
[docs]def XYZ_to_RGB_Nayatani95(XYZ):
"""
Converts from *CIE XYZ* tristimulus values to cone responses.
Parameters
----------
XYZ : array_like
*CIE XYZ* tristimulus values.
Returns
-------
ndarray
Cone responses.
Examples
--------
>>> XYZ = np.array([19.01, 20.00, 21.78])
>>> XYZ_to_RGB_Nayatani95(XYZ) # doctest: +ELLIPSIS
array([ 20.000520..., 19.999783..., 19.998831...])
"""
return dot_vector(NAYATANI95_XYZ_TO_RGB_MATRIX, XYZ)
[docs]def scaling_coefficient(x, y):
"""
Returns the scaling coefficient :math:`e(R)` or :math:`e(G)`.
Parameters
----------
x: numeric or array_like
Cone response.
y: numeric or array_like
Intermediate value.
Returns
-------
numeric or ndarray
Scaling coefficient :math:`e(R)` or :math:`e(G)`.
Examples
--------
>>> x = 20.000520600000002
>>> y = 1.000042192
>>> scaling_coefficient(x, y)
array(1.0)
"""
x = np.asarray(x)
y = np.asarray(y)
return np.where(x >= (20 * y), 1.758, 1)
[docs]def achromatic_response(RGB, bRGB_o, xez, bL_or, eR, eG, n=1):
"""
Returns the achromatic response :math:`Q` from given stimulus cone
responses.
Parameters
----------
RGB: ndarray
Stimulus cone responses.
bRGB_o: ndarray
Chromatic adaptation exponential factors :math:`\\beta_1(R_o)`,
:math:`\\beta_1(G_o)` and :math:`\\beta_2(B_o)`.
xez: ndarray
Intermediate values :math:`\\xi`, :math:`\eta`, :math:`\zeta`.
bL_or: numeric or array_like
Normalising chromatic adaptation exponential factor
:math:`\\beta_1(B_or)`.
eR: numeric or array_like
Scaling coefficient :math:`e(R)`.
eG: numeric or array_like
Scaling coefficient :math:`e(G)`.
n : numeric or array_like, optional
Noise term used in the non linear chromatic adaptation model.
Returns
-------
numeric or ndarray
Achromatic response :math:`Q`.
Examples
--------
>>> RGB = np.array([20.00052060, 19.99978300, 19.99883160])
>>> bRGB_o = np.array([4.61062223, 4.61058926, 4.65206986])
>>> xez = np.array([1.00004219, 0.99998001, 0.99975794])
>>> bL_or = 3.6810214956040888
>>> eR = 1.0
>>> eG = 1.758
>>> n = 1.0
>>> achromatic_response(RGB, bRGB_o, xez, bL_or, eR, eG, n) # noqa # doctest: +ELLIPSIS
-0.0001169...
"""
R, G, B = tsplit(RGB)
bR_o, bG_o, bB_o = tsplit(bRGB_o)
xi, eta, zeta = tsplit(xez)
bL_or = np.asarray(bL_or)
eR = np.asarray(eR)
eG = np.asarray(eG)
Q = (2 / 3) * bR_o * eR * np.log10((R + n) / (20 * xi + n))
Q += (1 / 3) * bG_o * eG * np.log10((G + n) / (20 * eta + n))
Q *= 41.69 / bL_or
return Q
[docs]def tritanopic_response(RGB, bRGB_o, xez, n):
"""
Returns the tritanopic response :math:`t` from given stimulus cone
responses.
Parameters
----------
RGB: ndarray
Stimulus cone responses.
bRGB_o: ndarray
Chromatic adaptation exponential factors :math:`\\beta_1(R_o)`,
:math:`\\beta_1(G_o)` and :math:`\\beta_2(B_o)`.
xez: ndarray
Intermediate values :math:`\\xi`, :math:`\eta`, :math:`\zeta`.
n : numeric or array_like, optional
Noise term used in the non linear chromatic adaptation model.
Returns
-------
numeric or ndarray
Tritanopic response :math:`t`.
Examples
--------
>>> RGB = np.array([20.00052060, 19.99978300, 19.99883160])
>>> bRGB_o = np.array([4.61062223, 4.61058926, 4.65206986])
>>> xez = np.array([1.00004219, 0.99998001, 0.99975794])
>>> n = 1.0
>>> tritanopic_response(RGB, bRGB_o, xez, n) # doctest: +ELLIPSIS
-1.7703650...e-05
"""
R, G, B = tsplit(RGB)
bR_o, bG_o, bB_o = tsplit(bRGB_o)
xi, eta, zeta = tsplit(xez)
t = (1 / 1) * bR_o * np.log10((R + n) / (20 * xi + n))
t += - (12 / 11) * bG_o * np.log10((G + n) / (20 * eta + n))
t += (1 / 11) * bB_o * np.log10((B + n) / (20 * zeta + n))
return t
[docs]def protanopic_response(RGB, bRGB_o, xez, n):
"""
Returns the protanopic response :math:`p` from given stimulus cone
responses.
Parameters
----------
RGB: ndarray
Stimulus cone responses.
bRGB_o: ndarray
Chromatic adaptation exponential factors :math:`\\beta_1(R_o)`,
:math:`\\beta_1(G_o)` and :math:`\\beta_2(B_o)`.
xez: ndarray
Intermediate values :math:`\\xi`, :math:`\eta`, :math:`\zeta`.
n : numeric or array_like, optional
Noise term used in the non linear chromatic adaptation model.
Returns
-------
numeric or ndarray
Protanopic response :math:`p`.
Examples
--------
>>> RGB = np.array([20.00052060, 19.99978300, 19.99883160])
>>> bRGB_o = np.array([4.61062223, 4.61058926, 4.65206986])
>>> xez = np.array([1.00004219, 0.99998001, 0.99975794])
>>> n = 1.0
>>> protanopic_response(RGB, bRGB_o, xez, n) # doctest: +ELLIPSIS
-8.0021426...e-05
"""
R, G, B = tsplit(RGB)
bR_o, bG_o, bB_o = tsplit(bRGB_o)
xi, eta, zeta = tsplit(xez)
p = (1 / 9) * bR_o * np.log10((R + n) / (20 * xi + n))
p += (1 / 9) * bG_o * np.log10((G + n) / (20 * eta + n))
p += - (2 / 9) * bB_o * np.log10((B + n) / (20 * zeta + n))
return p
[docs]def brightness_correlate(bRGB_o, bL_or, Q):
"""
Returns the *brightness* correlate :math:`B_r`.
Parameters
----------
bRGB_o: ndarray
Chromatic adaptation exponential factors :math:`\\beta_1(R_o)`,
:math:`\\beta_1(G_o)` and :math:`\\beta_2(B_o)`.
bL_or: numeric or array_like
Normalising chromatic adaptation exponential factor
:math:`\\beta_1(B_or)`.
Q : numeric or array_like
Achromatic response :math:`Q`.
Returns
-------
numeric or ndarray
*Brightness* correlate :math:`B_r`.
Examples
--------
>>> bRGB_o = np.array([4.61062223, 4.61058926, 4.65206986])
>>> bL_or = 3.6810214956040888
>>> Q = -0.000117024294955
>>> brightness_correlate(bRGB_o, bL_or, Q) # doctest: +ELLIPSIS
62.6266734...
"""
bR_o, bG_o, bB_o = tsplit(bRGB_o)
bL_or = np.asarray(bL_or)
Q = np.asarray(Q)
B_r = (50 / bL_or) * ((2 / 3) * bR_o + (1 / 3) * bG_o) + Q
return B_r
[docs]def ideal_white_brightness_correlate(bRGB_o, xez, bL_or, n):
"""
Returns the ideal white *brightness* correlate :math:`B_{rw}`.
Parameters
----------
bRGB_o: ndarray
Chromatic adaptation exponential factors :math:`\\beta_1(R_o)`,
:math:`\\beta_1(G_o)` and :math:`\\beta_2(B_o)`.
xez: ndarray
Intermediate values :math:`\\xi`, :math:`\eta`, :math:`\zeta`.
bL_or: numeric or array_like
Normalising chromatic adaptation exponential factor
:math:`\\beta_1(B_or)`.
n : numeric or array_like, optional
Noise term used in the non linear chromatic adaptation model.
Returns
-------
numeric or ndarray
Ideal white *brightness* correlate :math:`B_{rw}`.
Examples
--------
>>> bRGB_o = np.array([4.61062223, 4.61058926, 4.65206986])
>>> xez = np.array([1.00004219, 0.99998001, 0.99975794])
>>> bL_or = 3.6810214956040888
>>> n = 1.0
>>> ideal_white_brightness_correlate(bRGB_o, xez, bL_or, n) # noqa # doctest: +ELLIPSIS
125.2435392...
"""
bR_o, bG_o, bB_o = tsplit(bRGB_o)
xi, eta, zeta = tsplit(xez)
bL_or = np.asarray(bL_or)
B_rw = (2 / 3) * bR_o * 1.758 * np.log10((100 * xi + n) / (20 * xi + n))
B_rw += (1 / 3) * bG_o * 1.758 * np.log10((100 * eta + n) / (20 * eta + n))
B_rw *= 41.69 / bL_or
B_rw += (50 / bL_or) * (2 / 3) * bR_o
B_rw += (50 / bL_or) * (1 / 3) * bG_o
return B_rw
[docs]def achromatic_lightness_correlate(Q):
"""
Returns the *achromatic Lightness* correlate :math:`L_p^\star`.
Parameters
----------
Q : numeric or array_like
Achromatic response :math:`Q`.
Returns
-------
numeric or ndarray
*Achromatic Lightness* correlate :math:`L_p^\star`.
Examples
--------
>>> Q = -0.000117024294955
>>> achromatic_lightness_correlate(Q) # doctest: +ELLIPSIS
49.9998829...
"""
Q = np.asarray(Q)
return Q + 50
[docs]def normalised_achromatic_lightness_correlate(B_r, B_rw):
"""
Returns the *normalised achromatic Lightness* correlate :math:`L_n^\star`.
Parameters
----------
B_r : numeric or array_like
*Brightness* correlate :math:`B_r`.
B_rw : numeric or array_like
Ideal white *brightness* correlate :math:`B_{rw}`.
Returns
-------
numeric or ndarray
*Normalised achromatic Lightness* correlate :math:`L_n^\star`.
Examples
--------
>>> B_r = 62.626673467230766
>>> B_rw = 125.24353925846037
>>> normalised_achromatic_lightness_correlate(B_r, B_rw) # noqa # doctest: +ELLIPSIS
50.0039154...
"""
B_r = np.asarray(B_r)
B_rw = np.asarray(B_rw)
return 100 * (B_r / B_rw)
[docs]def hue_angle(p, t):
"""
Returns the *hue* angle :math:`h` in degrees.
Parameters
----------
p : numeric or array_like
Protanopic response :math:`p`.
t : numeric or array_like
Tritanopic response :math:`t`.
Returns
-------
numeric or ndarray
*Hue* angle :math:`h` in degrees.
Examples
--------
>>> p = -8.002142682085493e-05
>>> t = -1.7703650668990973e-05
>>> hue_angle(p, t) # doctest: +ELLIPSIS
257.5250300...
"""
p = np.asarray(p)
t = np.asarray(t)
h_L = np.degrees(np.arctan2(p, t)) % 360
return h_L
[docs]def chromatic_strength_function(theta):
"""
Defines the chromatic strength function :math:`E_s(\\theta)` used to
correct saturation scale as function of hue angle :math:`\\theta`.
Parameters
----------
theta : numeric or array_like
Hue angle :math:`\\theta`
Returns
-------
numeric or ndarray
Corrected saturation scale.
Examples
--------
>>> chromatic_strength_function(4.49462820973) # doctest: +ELLIPSIS
1.2267869...
"""
theta = np.asarray(theta)
E_s = 0.9394
E_s += -0.2478 * np.sin(1 * theta)
E_s += -0.0743 * np.sin(2 * theta)
E_s += +0.0666 * np.sin(3 * theta)
E_s += -0.0186 * np.sin(4 * theta)
E_s += -0.0055 * np.cos(1 * theta)
E_s += -0.0521 * np.cos(2 * theta)
E_s += -0.0573 * np.cos(3 * theta)
E_s += -0.0061 * np.cos(4 * theta)
return E_s
[docs]def saturation_components(h, bL_or, t, p):
"""
Returns the *saturation* components :math:`S_{RG}` and :math:`S_{YB}`.
Parameters
----------
h: numeric or array_like
Correlate of *hue* :math:`h` in degrees.
bL_or: numeric or array_like
Normalising chromatic adaptation exponential factor
:math:`\\beta_1(B_or)`.
t : numeric or array_like
Tritanopic response :math:`t`.
p : numeric or array_like
Protanopic response :math:`p`.
Returns
-------
numeric or ndarray
*Saturation* components :math:`S_{RG}` and :math:`S_{YB}`.
Examples
--------
>>> h = 257.52322689806243
>>> bL_or = 3.6810214956040888
>>> t = -1.7706764677181658e-05
>>> p = -8.0023561356363753e-05
>>> saturation_components(h, bL_or, t, p) # doctest: +ELLIPSIS
array([-0.0028852..., -0.0130396...])
"""
h = np.asarray(h)
bL_or = np.asarray(bL_or)
t = np.asarray(t)
p = np.asarray(p)
E_s = chromatic_strength_function(np.radians(h))
S_RG = (488.93 / bL_or) * E_s * t
S_YB = (488.93 / bL_or) * E_s * p
return tstack((S_RG, S_YB))
[docs]def saturation_correlate(S_RG, S_YB):
"""
Returns the correlate of *saturation* :math:`S`.
Parameters
----------
S_RG : numeric or array_like
*Saturation* component :math:`S_{RG}`.
S_YB : numeric or array_like
*Saturation* component :math:`S_{YB}`.
Returns
-------
numeric or ndarray
Correlate of *saturation* :math:`S`.
Examples
--------
>>> S_RG = -0.0028852716381965863
>>> S_YB = -0.013039632941332499
>>> saturation_correlate(S_RG, S_YB) # doctest: +ELLIPSIS
0.0133550...
"""
S_RG = np.asarray(S_RG)
S_YB = np.asarray(S_YB)
S = np.sqrt((S_RG ** 2) + (S_YB ** 2))
return S
[docs]def chroma_components(Lstar_P, S_RG, S_YB):
"""
Returns the *chroma* components :math:`C_{RG}` and :math:`C_{YB}`.
Parameters
----------
Lstar_P : numeric or array_like
*Achromatic Lightness* correlate :math:`L_p^\star`.
S_RG : numeric or array_like
*Saturation* component :math:`S_{RG}`.
S_YB : numeric or array_like
*Saturation* component :math:`S_{YB}`.
Returns
-------
ndarray
*Chroma* components :math:`C_{RG}` and :math:`C_{YB}`.
Examples
--------
>>> Lstar_P = 49.99988297570504
>>> S_RG = -0.0028852716381965863
>>> S_YB = -0.013039632941332499
>>> chroma_components(Lstar_P, S_RG, S_YB) # doctest: +ELLIPSIS
array([-0.00288527, -0.01303961])
"""
Lstar_P = np.asarray(Lstar_P)
S_RG = np.asarray(S_RG)
S_YB = np.asarray(S_YB)
C_RG = ((Lstar_P / 50) ** 0.7) * S_RG
C_YB = ((Lstar_P / 50) ** 0.7) * S_YB
return tstack((C_RG, C_YB))
[docs]def chroma_correlate(Lstar_P, S):
"""
Returns the correlate of *chroma* :math:`C`.
Parameters
----------
Lstar_P : numeric or array_like
*Achromatic Lightness* correlate :math:`L_p^\star`.
S : numeric or array_like
Correlate of *saturation* :math:`S`.
Returns
-------
numeric or ndarray
Correlate of *chroma* :math:`C`.
Examples
--------
>>> Lstar_P = 49.99988297570504
>>> S = 0.013355029751777615
>>> chroma_correlate(Lstar_P, S) # doctest: +ELLIPSIS
0.0133550...
"""
Lstar_P = np.asarray(Lstar_P)
S = np.asarray(S)
C = ((Lstar_P / 50) ** 0.7) * S
return C
[docs]def colourfulness_components(C_RG, C_YB, B_rw):
"""
Returns the *colourfulness* components :math:`M_{RG}` and :math:`M_{YB}`.
Parameters
----------
C_RG : numeric or array_like
*Chroma* component :math:`C_{RG}`.
C_YB : numeric or array_like
*Chroma* component :math:`C_{YB}`.
B_rw : numeric or array_like
Ideal white *brightness* correlate :math:`B_{rw}`.
Returns
-------
numeric or ndarray
*Colourfulness* components :math:`M_{RG}` and :math:`M_{YB}`.
Examples
--------
>>> C_RG = -0.0028852716381965863
>>> C_YB = -0.013039632941332499
>>> B_rw = 125.24353925846037
>>> colourfulness_components(C_RG, C_YB, B_rw) # doctest: +ELLIPSIS
(-0.0036136..., -0.0163312...)
"""
C_RG = np.asarray(C_RG)
C_YB = np.asarray(C_YB)
B_rw = np.asarray(B_rw)
M_RG = C_RG * B_rw / 100
M_YB = C_YB * B_rw / 100
return M_RG, M_YB
[docs]def colourfulness_correlate(C, B_rw):
"""
Returns the correlate of *colourfulness* :math:`M`.
Parameters
----------
C : numeric or array_like
Correlate of *chroma* :math:`C`.
B_rw : numeric or array_like
Ideal white *brightness* correlate :math:`B_{rw}`.
Returns
-------
numeric or ndarray
Correlate of *colourfulness* :math:`M`.
Examples
--------
>>> C = 0.013355007871688761
>>> B_rw = 125.24353925846037
>>> colourfulness_correlate(C, B_rw) # doctest: +ELLIPSIS
0.0167262...
"""
C = np.asarray(C)
B_rw = np.asarray(B_rw)
M = C * B_rw / 100
return M