Implementación de CIEDE2000 en Swift
| Número de visitas | 617 |
|---|---|
| Número de archivos consultados | 406 + 392 |
Esta página presenta una implementación de referencia de la fórmula de diferencia de color CIEDE2000 en Swift. Si desea garantizar una compatibilidad perfecta (hasta el décimo decimal) con algunas implementaciones de terceros, es posible que tenga que modificar los comentarios del código fuente. Para facilitarlo, el siguiente enlace automatiza esta operación.
La función ΔE2000 en Swift
Consideremos la más común y académica (Sharma, 2005) de las dos formulaciones.
// Esta función escrita en Swift es de dominio público y no
// está afiliada a la CIE (Comisión Internacional de Iluminación).
import Foundation
// La implementación clásica de CIEDE2000 que recibe dos colores L*a*b* y devuelve su diferencia.
// El componente "L" varía de 0 a 100. "a" y "b" normalmente se proyectan entre -128 y 127.
func ciede_2000(l_1: Double, a_1: Double, b_1: Double, l_2: Double, a_2: Double, b_2: Double) -> Double {
// Ejecuta el cálculo de diferencia de color CIEDE2000 en Swift.
// k_l, k_c y k_h son factores paramétricos a ajustar según
// sus necesidades de visualización (texturas, fondos).
let k_l = 1.0, k_c = 1.0, k_h = 1.0;
var n = (sqrt(a_1 * a_1 + b_1 * b_1) + sqrt(a_2 * a_2 + b_2 * b_2)) * 0.5;
n = n * n * n * n * n * n * n;
// Un factor que implica la croma elevada a la potencia 7,
// diseñado para tener más en cuenta la influencia de la croma.
n = 1.0 + 0.5 * (1.0 - sqrt(n / (n + 6103515625.0)));
// Aplicación del factor de corrección de la croma para compensar su no linealidad.
let c_1 = sqrt(a_1 * a_1 * n * n + b_1 * b_1);
let c_2 = sqrt(a_2 * a_2 * n * n + b_2 * b_2);
// La función atan2 es preferible a atan porque calcula el ángulo de un
// punto (x, y) en todos los cuadrantes, teniendo en cuenta el signo de x e y.
var h_1 = atan2(b_1, a_1 * n), h_2 = atan2(b_2, a_2 * n);
if h_1 < 0.0 { h_1 += 2.0 * .pi; }
if h_2 < 0.0 { h_2 += 2.0 * .pi; }
n = abs(h_2 - h_1);
// Evita que la rama dependa del RoundingMode del lenguaje de programación.
if .pi - 1E-14 < n && n < .pi + 1E-14 { n = .pi; }
// Cuando los ángulos de tono se encuentran en diferentes cuadrantes,
// la media aritmética simple puede dar un ángulo incorrecto, las
// líneas siguientes aplican la corrección angular necesaria.
var h_m = (h_1 + h_2) * 0.5, h_d = (h_2 - h_1) * 0.5;
if .pi < n {
h_d += .pi;
// 📜 La formulación de Sharma no utiliza la línea siguiente, sino la que le sigue.
// Nota: estas dos variantes sólo difieren en ±0,0003 en la diferencia de color final.
h_m += .pi;
// if h_m < .pi { h_m += .pi; } else { h_m -= .pi; }
}
let p = 36.0 * h_m - 55.0 * .pi;
n = (c_1 + c_2) * 0.5;
n = n * n * n * n * n * n * n;
// El término de corrección de la rotación del tono mejora la precisión
// del algoritmo, especialmente cuando la comparación se refiere a azules.
let r_t = -2.0 * sqrt(n / (n + 6103515625.0))
* sin(.pi / 3.0 * exp(p * p / (-25.0 * .pi * .pi)));
n = (l_1 + l_2) * 0.5;
n = (n - 50.0) * (n - 50.0);
// Luminosidad .
let l = (l_2 - l_1) / (k_l * (1.0 + 0.015 * n / sqrt(20.0 + n)));
// Estos coeficientes se utilizan para modular la influencia de
// los componentes armónicos en el cálculo de la diferencia de tono.
let t = 1.0 + 0.24 * sin(2.0 * h_m + .pi * 0.5)
+ 0.32 * sin(3.0 * h_m + 8.0 * .pi / 15.0)
- 0.17 * sin(h_m + .pi / 3.0)
- 0.20 * sin(4.0 * h_m + 3.0 * .pi / 20.0);
n = c_1 + c_2;
// Tono.
let h = 2.0 * sqrt(c_1 * c_2) * sin(h_d) / (k_h * (1.0 + 0.0075 * n * t));
// Croma.
let c = (c_2 - c_1) / (k_c * (1.0 + 0.0225 * n));
// Devolver la raíz cuadrada garantiza que dE00 representa una distancia
// geométrica (que va de 0 a aproximadamente 185) en el espacio CIELAB.
return sqrt(l * l + h * h + c * c + c * h * r_t);
}
// Proyecto GitHub : https://github.com/michel-leonard/ciede2000-color-matching
// Pruebas en línea : https://michel-leonard.github.io/ciede2000-color-matching
// L1 = 44.4 a1 = 30.3 b1 = 3.3
// L2 = 43.8 a2 = 24.5 b2 = -2.4
// CIE ΔE00 = 4.4951265489 (Bruce Lindbloom, Netflix’s VMAF, ...)
// CIE ΔE00 = 4.4951440582 (Gaurav Sharma, OpenJDK, ...)
// Desviación entre implementaciones ≈ 1.8e-5
// Consulte los comentarios del código fuente para pasar de una de estas variantes de implementación ΔE*00 a la otra.Parámetros k_l, k_c y k_h
Los parámetros k_l, k_c y k_h de la fórmula CIEDE2000 son factores de ponderación que se aplican a los componentes de luminosidad (ΔL*), croma (ΔC*) y tono (ΔH*), respectivamente. En el código fuente, se definen como constantes cuyo valor por defecto es 1, lo que se corresponde con las condiciones de observación estándar establecidas por la Comisión Internacional de Iluminación (CIE). En la práctica, puede ser necesario ajustar estos coeficientes para reflejar condiciones específicas: por ejemplo, k_l = 2 se utiliza a veces para dar más peso a las diferencias de brillo (algo habitual en la industria textil), mientras que k_c o k_h pueden reducirse para aumentar la tolerancia a las variaciones de saturación o tono. En resumen, estos coeficientes suelen oscilar entre 0,5 y 2, siendo 1 el valor más habitual.
Precisión y fiabilidad del código fuente
La diferencia entre la formulación académica de Sharma y la simplificada de Lindbloom no supera ±0,0003 en el ΔE2000 final. La implementación que aquí se presenta es de 64 bits y garantiza una precisión de más de 10 decimales; por lo tanto, la elección de una formulación en lugar de otra es un detalle técnico. En la parte superior de esta página, puedes elegir entre las dos formulaciones; la que se muestra actualmente es la formulación simplificada.
¿Cómo determinar si una implementación determinada de CIEDE2000 es académica o simplificada?
- Evalúa
ciede_2000(27.5, 24.6, 9.2, 15.7, 56.6, -18.5)con precisión - Si el resultado es
19.999944, entonces es del tipo académico (como Sharma, OpenJDK, etc.) - Si el resultado es
20.000002, entonces es del tipo simplificado (como Lindbloom, Netflix VMAF, etc.)
¿Cómo se convierten los colores RGB a L*a*b*?
Vaya a la página AWK, C, Dart, Java, JavaScript, Kotlin, Lua, PHP, Python, Ruby o Rust, donde ya está implementado dicho conversor (que utiliza el iluminante D65), además de la función de comparación de colores.
Rangos de valores en CIELAB e interpretación del ΔE2000
En el espacio de color CIELAB, el componente L* representa la luminosidad y suele variar de 0 (negro) a 100 (blanco). Los componentes a* y b* definen los ejes de color: a* va del verde al rojo, y b* del azul al amarillo. En la práctica, los valores de a* y b* casi siempre se limitan a un intervalo comprendido entre -128 y +127, aunque la norma no establece un límite oficial para estos dos componentes.
| Color 1 | Color 2 | Valor de ΔE2000 |
|---|---|---|
| 1 | ||
| 2 | ||
| 3 |
| Color 1 | Color 2 | Valor de ΔE2000 |
|---|---|---|
| 5 | ||
| 10 | ||
| 15 |
ΔE2000 (CIEDE2000) cuantifica la diferencia perceptible entre dos colores: 0 significa que son idénticos, y valores más altos (hasta 185 y más) indican una diferencia más notable. Por ejemplo, un ΔE2000 alrededor de 5 indica colores cercanos, mientras que alrededor de 15 indica colores claramente distintos. Cuando el valor ΔE2000 supera 40, los colores comparados ya casi no tienen nada en común, y no podemos extraer de ellos información precisa.
Ejemplo de uso en Swift
// Compute the Delta E (CIEDE2000) color difference between two L*a*b* colors in Swift
let (l1, a1, b1) = (30.0, 28.0, -3.7)
let (l2, a2, b2) = (32.5, 21.8, 3.6)
let delta_e = ciede_2000(l_1: l1, a_1: a1, b_1: b1, l_2: l2, a_2: a2, b_2: b2)
print(delta_e)
// .................................................. This shows a ΔE2000 of 5.9485905742
// As explained in the comments, compliance with Gaurav Sharma would display 5.9485740608Los resultados de las pruebas
Nuestro programa de pruebas, escrito en C99, incluye 250 pruebas estáticas precisas. Los resultados muestran que esta función de CIEDE2000 en Swift es interoperable con los otros 41 lenguajes de programación.
CIEDE2000 Verification Summary :
First Verified Line : 5,50.66,-34,47.8,1.8,2,41.336597284307338
Duration : 41.23 s
Successes : 10000000
Errors : 0
Average Delta E : 62.9542
Average Deviation : 6.6143187649192467e-15
Maximum Deviation : 2.2737367544323206e-13Archivos para descargar
Siéntase libre de utilizar estos archivos puestos a disposición por Michel, incluso con fines comerciales.
| Archivo | Tamaño | Número de clics |
|---|---|---|
| ciede-2000.swift | 4 KB | 102 |
| ciede-2000-driver.swift | 6 KB | 112 |
| ciede-2000-random.swift | 6 KB | 87 |
| test-swift.yml | 2 KB | 58 |
| vs-pretty.yml | 4 KB | 47 |
| reference-dataset.txt | 4 KB | 392 |
| Haga clic en swift.zip para descargar todos estos archivos en un archivo. | ||
Comunidad
¿Qué opina de este código fuente o de CIEDE2000? Su opinión es importante para nosotros. El libro de visitas ya contiene 9 mensajes - incluyendo 1 en español. Eche un vistazo y comparta su opinión.