"""
Blackbody - Planckian Radiator
==============================
Define objects to compute the spectral radiance of a planckian radiator
and its spectral distribution.
References
----------
- :cite:`CIETC1-482004i` : CIE TC 1-48. (2004). APPENDIX E. INFORMATION ON
THE USE OF PLANCK'S EQUATION FOR STANDARD AIR. In CIE 015:2004 Colorimetry,
3rd Edition (pp. 77-82). ISBN:978-3-901906-33-6
- :cite:`Wikipedia2003f` : Wikipedia. (2003). Rayleigh-Jeans law. Retrieved
February 12, 2022, from https://en.wikipedia.org/wiki/Rayleigh-Jeans_law
"""
from __future__ import annotations
import typing
import numpy as np
from colour.colorimetry import (
SPECTRAL_SHAPE_DEFAULT,
SpectralDistribution,
SpectralShape,
)
from colour.constants import CONSTANT_BOLTZMANN, CONSTANT_LIGHT_SPEED
if typing.TYPE_CHECKING:
from colour.hints import ArrayLike, NDArrayFloat
from colour.utilities import as_float, as_float_array
from colour.utilities.common import attest
__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__ = [
"CONSTANT_C1",
"CONSTANT_C2",
"CONSTANT_N",
"planck_law",
"blackbody_spectral_radiance",
"sd_blackbody",
"rayleigh_jeans_law",
"sd_rayleigh_jeans",
]
# 2 * math.pi * CONSTANT_PLANCK * CONSTANT_LIGHT_SPEED ** 2
CONSTANT_C1: float = 3.741771e-16
# CONSTANT_PLANCK * CONSTANT_LIGHT_SPEED / CONSTANT_BOLTZMANN
CONSTANT_C2: float = 1.4388e-2
CONSTANT_N: float = 1
[docs]
def planck_law(
wavelength: ArrayLike,
temperature: ArrayLike,
c1: float = CONSTANT_C1,
c2: float = CONSTANT_C2,
n: float = CONSTANT_N,
) -> NDArrayFloat:
"""
Compute the spectral radiance of a blackbody as a function of
wavelength at specified thermodynamic temperature :math:`T[K]` in a
medium with index of refraction :math:`n`.
Parameters
----------
wavelength
Wavelength in meters.
temperature
Temperature :math:`T[K]` in kelvin degrees.
c1
The official value of :math:`c1` is provided by the Committee on
Data for Science and Technology (CODATA) and is
:math:`c1=3.741771 \\times 10^{-16}\\ \\mathrm{W/m^2}` *(Mohr and
Taylor, 2000)*.
c2
Since :math:`T` is measured on the International Temperature
Scale, the value of :math:`c2` used in colorimetry should follow
that adopted in the current International Temperature Scale
(ITS-90) *(Preston-Thomas, 1990; Mielenz et al., 1991)*, namely
:math:`c2=1.4388 \\times 10^{-2}\\ \\mathrm{m \\cdot K}`.
n
Medium index of refraction. For dry air at 15°C and 101 325 Pa,
containing 0.03 percent by volume of carbon dioxide, it is
approximately 1.00028 throughout the visible region although
*CIE 15:2004* recommends using :math:`n=1`.
Returns
-------
:class:`numpy.ndarray`
Radiance in *watts per steradian per square metre*
(:math:`\\mathrm{W \\cdot sr^{-1} \\cdot m^{-2}}`).
Warnings
--------
The :func:`colour.colorimetry.planck_law` definition behaviour with
n-dimensional arrays is unusual: The ``wavelength`` and
``temperature`` parameters are first raveled using
:func:`numpy.ravel`. Then, they are *broadcasted* together by
transposing the ``temperature`` parameter. Finally, and for
convenience, the return value is squeezed using :func:`numpy.squeeze`.
Notes
-----
- The following implementation is expressed in terms of wavelength.
- The SI unit of radiance is *watts per steradian per square
metre* (:math:`\\mathrm{W \\cdot sr^{-1} \\cdot m^{-2}}`).
References
----------
:cite:`CIETC1-482004i`
Examples
--------
>>> planck_law(500 * 1e-9, 5500) # doctest: +ELLIPSIS
np.float64(20472701909806.5...)
>>> planck_law(500 * 1e-9, [5000, 5500, 6000]) # doctest: +ELLIPSIS
array([1.2106064...e+13, 2.0472701...e+13, 3.1754431...e+13])
"""
l = as_float_array(wavelength) # noqa: E741
t = as_float_array(temperature)
attest(np.all(l > 0), "Wavelengths must be positive real numbers!")
l = np.ravel(l)[..., None] # noqa: E741
t = np.ravel(t)[None, ...]
d = 1 / np.expm1(c2 / (n * l * t))
p = ((c1 * n**-2 * l**-5) / np.pi) * d
return as_float(np.squeeze(p))
blackbody_spectral_radiance = planck_law
[docs]
def sd_blackbody(
temperature: float,
shape: SpectralShape = SPECTRAL_SHAPE_DEFAULT,
c1: float = CONSTANT_C1,
c2: float = CONSTANT_C2,
n: float = CONSTANT_N,
) -> SpectralDistribution:
"""
Generate the spectral distribution of the planckian radiator for the
specified temperature :math:`T[K]` with values in *watts per steradian
per square metre per nanometre* (:math:`W/sr/m^2/nm`).
Parameters
----------
temperature
Temperature :math:`T[K]` in kelvins.
shape
Spectral shape used to create the spectral distribution of the
planckian radiator.
c1
The official value of :math:`c_1` is provided by the Committee on
Data for Science and Technology (CODATA) and is
:math:`c_1=3.741771 \\times 10^{16}\\ W/m^2` *(Mohr and Taylor,
2000)*.
c2
Since :math:`T` is measured on the International Temperature
Scale, the value of :math:`c_2` used in colorimetry should follow
that adopted in the current International Temperature Scale
(ITS-90) *(Preston-Thomas, 1990; Mielenz et al., 1991)*, namely
:math:`c_2=1.4388 \\times 10^{-2}\\ m \\cdot K`.
n
Medium index of refraction. For dry air at 15°C and 101 325 Pa,
containing 0.03 percent by volume of carbon dioxide, it is
approximately 1.00028 throughout the visible region although
*CIE 15:2004* recommends using :math:`n=1`.
Returns
-------
:class:`colour.SpectralDistribution`
Blackbody spectral distribution with values in *watts per
steradian per square metre per nanometre* (:math:`W/sr/m^2/nm`).
Examples
--------
>>> from colour.utilities import numpy_print_options
>>> with numpy_print_options(suppress=True):
... sd_blackbody(5000, shape=SpectralShape(400, 700, 20))
... # doctest: +ELLIPSIS
SpectralDistribution([[ 400. , 8742.5713329...],
[ 420. , 9651.6810212...],
[ 440. , 10447.3423137...],
[ 460. , 11121.8597759...],
[ 480. , 11673.7121534...],
[ 500. , 12106.0645344...],
[ 520. , 12425.4166118...],
[ 540. , 12640.4550541...],
[ 560. , 12761.1284859...],
[ 580. , 12797.9345572...],
[ 600. , 12761.3938171...],
[ 620. , 12661.6795247...],
[ 640. , 12508.3723863...],
[ 660. , 12310.3119640...],
[ 680. , 12075.5205176...],
[ 700. , 11811.1793602...]],
SpragueInterpolator,
{},
Extrapolator,
{'method': 'Constant', 'left': None, 'right': None})
"""
return SpectralDistribution(
planck_law(shape.wavelengths * 1e-9, temperature, c1, c2, n) * 1e-9,
shape.wavelengths,
name=f"{temperature}K Blackbody",
)
[docs]
def rayleigh_jeans_law(wavelength: ArrayLike, temperature: ArrayLike) -> NDArrayFloat:
"""
Approximate the spectral radiance of a blackbody as a function of
wavelength at specified thermodynamic temperature :math:`T[K]` according
to the *Rayleigh-Jeans* law.
Parameters
----------
wavelength
Wavelength in meters.
temperature
Temperature :math:`T[K]` in kelvin degrees.
Returns
-------
:class:`numpy.ndarray`
Radiance in *watts per steradian per square metre*
(:math:`W/sr/m^2`).
Warnings
--------
The :func:`colour.colorimetry.rayleigh_jeans_law` definition behaviour
with n-dimensional arrays is unusual: The ``wavelength`` and
``temperature`` parameters are first raveled using :func:`numpy.ravel`.
Then, they are *broadcasted* together by transposing the
``temperature`` parameter. Finally, and for convenience, the return
value is squeezed using :func:`numpy.squeeze`.
Notes
-----
- The *Rayleigh-Jeans* law agrees with experimental results at large
wavelengths (low frequencies) but strongly disagrees at short
wavelengths (high frequencies). This inconsistency between
observations and the predictions of classical physics is commonly
known as the *ultraviolet catastrophe*.
- The following implementation is expressed in terms of wavelength.
- The SI unit of radiance is *watts per steradian per square metre*
(:math:`W/sr/m^2`).
References
----------
:cite:`Wikipedia2003f`
Examples
--------
>>> rayleigh_jeans_law(500 * 1e-9, 5500) # doctest: +ELLIPSIS
np.float64(728478884562351.5...)
>>> rayleigh_jeans_law(500 * 1e-9, [5000, 5500, 6000])
... # doctest: +ELLIPSIS
array([6.6225353...e+14, 7.2847888...e+14, 7.9470423...e+14])
"""
l = as_float_array(wavelength) # noqa: E741
t = as_float_array(temperature)
l = np.ravel(l)[..., None] # noqa: E741
t = np.ravel(t)[None, ...]
c = CONSTANT_LIGHT_SPEED
k_B = CONSTANT_BOLTZMANN
B = (2 * c * k_B * t) / (l**4)
return as_float(np.squeeze(B))
[docs]
def sd_rayleigh_jeans(
temperature: float,
shape: SpectralShape = SPECTRAL_SHAPE_DEFAULT,
) -> SpectralDistribution:
"""
Generate the spectral distribution of the planckian radiator for the
specified temperature :math:`T[K]` with values in *watts per steradian
per square metre per nanometre* (:math:`W/sr/m^2/nm`) according to the
*Rayleigh-Jeans* law.
Parameters
----------
temperature
Temperature :math:`T[K]` in kelvins.
shape
Spectral shape used to create the spectral distribution of the
planckian radiator.
Returns
-------
:class:`colour.SpectralDistribution`
Blackbody spectral distribution with values in *watts per steradian
per square metre per nanometre* (:math:`W/sr/m^2/nm`).
Notes
-----
- The *Rayleigh-Jeans* law agrees with experimental results at large
wavelengths (low frequencies) but strongly disagrees at short
wavelengths (high frequencies). This inconsistency between
observations and the predictions of classical physics is commonly
known as the *ultraviolet catastrophe*.
Examples
--------
>>> from colour.utilities import numpy_print_options
>>> with numpy_print_options(suppress=True):
... sd_rayleigh_jeans(5000, shape=SpectralShape(400, 700, 20))
... # doctest: +ELLIPSIS
SpectralDistribution([[ 400. , 1616829.9106941...],
[ 420. , 1330169.9688456...],
[ 440. , 1104316.5840408...],
[ 460. , 924427.7490112...],
[ 480. , 779721.2146480...],
[ 500. , 662253.5314203...],
[ 520. , 566097.0941823...],
[ 540. , 486776.1157138...],
[ 560. , 420874.0917050...],
[ 580. , 365756.7299433...],
[ 600. , 319373.8095198...],
[ 620. , 280115.7588306...],
[ 640. , 246708.6655722...],
[ 660. , 218136.6091932...],
[ 680. , 193583.6389284...],
[ 700. , 172390.0279623...]],
SpragueInterpolator,
{},
Extrapolator,
{'method': 'Constant', 'left': None, 'right': None})
"""
return SpectralDistribution(
rayleigh_jeans_law(shape.wavelengths * 1e-9, temperature) * 1e-9,
shape.wavelengths,
name=f"{temperature}K Rayleigh-Jeans",
)