首页 > 其他分享 >颜色空间的互相转换

颜色空间的互相转换

时间:2024-08-12 22:48:49浏览次数:17  
标签:颜色 const 互相 number param RGB 转换 255

前言

上一篇中,我们介绍了常见颜色空间的一些定义及表示,在这一章中,我们将大致了解各个颜色空间的互相转换

颜色转换算法

由于有些颜色空间可能并不能直接转换,或着过于繁杂,本文主要介绍由RGB向其它空间的转换,涉及到的代码也采用Ts进行演示讲解

在文章的最后面,会给出封装的转换算法(TS版),如对文章内容不感兴趣,可直接拖到文末查看获取方法

HEX

将一个RGB颜色转换为HEX模式,其实就是将十进制值转换为十六进制,没什么好说的,直接看代码理解即可

/**
 * RGB 转为 HEX
 * @param {number} r [0-255]红色通道值
 * @param {number} g [0-255]绿色通道值
 * @param {number} b [0-255]蓝色通道值
 * @param {boolean} ad 是否带 # ,默认带有
 * @return {HEX} 返回转换的 hex 值
 */
const rgbToHex = function (
 r: number,
 g: number,
 b: number,
 ad: boolean = true
): HEX {
  if (ad) return `#${((1 << 24) + (r << 16) + (g << 8) + b).toString(16).slice(1)}`;
  else return `${((1 << 24) + (r << 16) + (g << 8) + b).toString(16).slice(1)}`;
};

主要就是按顺序将RGB各分量值,通过左移运算转换为一个对应十六进制的十进制数,最后将其转换为十六进制字符串即可

CMYK

RGB只有三个通道(取值范围0-255),而CMYK有四个通道(取值范围0-100),故从RGBCMYK的转换主要两个过程:

  • RGBCMY

    • \(C = (255 - R) / 255\)
    • \(M = (255 - G) / 255\)
    • \(Y = (255 - B) / 255\)
  • CMYCMYK

    • \(K = \min (C, M, Y)\)
    • \(C = (C - K) / (1 - K)\)
    • \(M = (M - K) / (1 - K)\)
    • \(Y = (Y - K) / (1 - K)\)

具体的代码为:

/**
 * RGB 转为 CMYK
 * @param {number} r [0-255]红色通道值
 * @param {number} g [0-255]绿色通道值
 * @param {number} b [0-255]蓝色通道值
 * @return {CMYK} 返回转换的 cmyk 值
 */
const rgbToCmyk = function (
 r: number,
 g: number,
 b: number
): CMYK {
  let c = (255 - r) / 255;
  let m = (255 - g) / 255;
  let y = (255 - b) / 255;
  let k = Math.min(c, m, y);
  if (k === 1) {
    // 此时为纯黑色,其它分量均为零
    c = m = y = 0;
  } else {
    let kk = 1 - k;
    c = (c - k) / kk;
    m = (m - k) / kk;
    y = (y - k) / kk;
  }
  return {
    c: toFixed(c * 100, 0),
    m: toFixed(m * 100, 0),
    y: toFixed(y * 100, 0),
    k: toFixed(k * 100, 0)
  };
};

// toFixed为一个保留指定小数位的函数

该算法只是比较粗糙的转换,由于两个颜色空间色域并不一致,故转换过程中可能会存在一定程度的颜色偏差和失真

HSV

在开始转换之前,先分析一下各分量的值域,首先是RGB,值域为[0-255];接着是HSVH值域为[0-360]SV的值域是[0-100]

转换的第一步就是统一分量值域

  • 先将RGB的值域转换为[0-1],由此计算出来的SV值域也为[0-1](H不变),后续根据需要转变即可
  • 然后,根据一下公式步骤计算
  • hsv

对应的转换代码为:

/**
 * RGB 转为 HSV
 * @param {number} r [0-255]红色通道值
 * @param {number} g [0-255]绿色通道值
 * @param {number} b [0-255]蓝色通道值
 * @return {HSV} 返回转换的 hsv 值
 */
const rgbToHsv = function (
 r: number,
 g: number,
 b: number
): HSV {
  r = r / 255; // [0, 1]
  g = g / 255; // [0, 1]
  b = b / 255; // [0, 1]

  const max = Math.max(r, g, b);
  const min = Math.min(r, g, b);
  const d = max - min;

  const v = max;
  const s = max === 0 ? 0 : d / max;

  let h = 0;
  if (d !== 0) {
    switch (max) {
      case r:
        h = (g - b) / d + (g < b ? 6 : 0);
        break;
      case g:
        h = (b - r) / d + 2;
        break;
      case b:
        h = (r - g) / d + 4;
        break;
      default:
        break;
    }
    h = h / 6;
  }
  // 返回时再规整值域
  return {
    h: toFixed(h * 360),
    s: toFixed(s * 100),
    v: toFixed(v * 100)
  };
};

HSL

在转换前,同样需先确定好各分量的计算值域

HSLHSV比较类似,其中的H均表示色调(色相),值域为[0-360];接着规定SL的值域为[0-1],故RGB同样需转换为[0-1]的值

接着根据计算公式计算:

hsl

具体的代码如下:

/**
 * RGB 转为 HSL
 * @param {number} r [0-255]红色通道值
 * @param {number} g [0-255]绿色通道值
 * @param {number} b [0-255]蓝色通道值
 * @return {HSL} 返回转换的 hsl 值
 */
const rgbToHsl = function (
 r: number,
 g: number,
 b: number
): HSL {
  r = r / 255; // [0, 1]
  g = g / 255; // [0, 1]
  b = b / 255; // [0, 1]

  const max = Math.max(r, g, b);
  const min = Math.min(r, g, b);
  const d = max - min;

  const l = (max + min) / 2;
  const s = d === 0 ? 0 : l > 0.5 ? d / (2 - 2 * l) : d / (2 * l);

  let h = 0;
  if (d !== 0) {
    switch (max) {
      case r:
        h = (g - b) / d + (g < b ? 6 : 0);
        break;
      case g:
        h = (b - r) / d + 2;
        break;
      case b:
        h = (r - g) / d + 4;
        break;
      default:
        break;
    }
    h = h / 6;
  }
  return {
    h: toFixed(h * 360),
    s: toFixed(s * 100),
    l: toFixed(l * 100)
  };
};

LAB

LAB是一种色域极广的颜色模式,相较于RGB,其采用更加科学的颜色表示方法,它是基于人眼对颜色的感知来定义的,其主要有三个分量:L表示亮度,A表示绿色到红色的色差,B表示蓝色到黄色的色差

由于两个颜色空间的定义原理,RGB无法直接转换为LAB,需用XYZ作为一个中间层,即RGBXYZ,再转LAB

XYZ也是一个颜色空间,其全称为CIE 1931 XYZ色彩空间(也叫做CIE 1931色彩空间),由国际照明委员会(CIE)于1931年创立。

XYZ是为了解决更精确地定义色彩而提出来的, 其三个分量中, XY代表的是色度, 而Y既可以代表亮度也可以代表色度,单位为nit

在日常生活中,我们无法用RGB来精确定义颜色, 因为,不同的设备显示的RGB其实都是不一样的,不同的设备, 显示同一个RGB, 在人眼看出来可能是千差万别的, XYZ就是为了解决这样的问题,使不同设备颜色显示更精确

具体的转换步骤为:

  • RGB归一化,即值域转变为[0-1]
  • RGB值进行逆伽马校正
    • 具体就是将各分量值传入逆伽马校正的函数内求结果,可根据实际情况使用不同标准的逆伽马函数或者设备的校准曲线
  • 接着将校正后的RGB转换为XYZ空间值
  • 接着将XYZ进行归一计算,使其归一到参考白点,常用的是D65白点(0.950456, 1, 1.088754)
  • 然后将XYZ值再进行非线性变换
  • 最后计算转换为LAB

大致的数学公式如下:

rgb转lab

大致的代码如下:

/**
 * RGB 转为 XYZ
 * @param {number} r [0-255]红色通道值
 * @param {number} g [0-255]绿色通道值
 * @param {number} b [0-255]蓝色通道值
 * @return {XYZ} 返回转换的 xyz 值
 */
const rgbToXyz = function (
 r: number,
 g: number,
 b: number
): XYZ {
  // 归一化
  r = r / 255;
  g = g / 255;
  b = b / 255;

  r = gamma(r);
  g = gamma(g);
  b = gamma(b);

  let x = 0.4124564 * r + 0.3575761 * g + 0.1804375 * b;
  let y = 0.2126729 * r + 0.7151522 * g + 0.072175 * b;
  let z = 0.0193339 * r + 0.119192 * g + 0.9503041 * b;
  return {
    x,
    y,
    z
  };
};

const XN = 0.950456;
const YN = 1;
const ZN = 1.088754;

/**
 * XYZ 转 LAB
 * @param x
 * @param y
 * @param z
 */
const xyzToLab = function (
 x: number,
 y: number,
 z: number
): LAB {
  // 归一化
  x = x / XN;
  y = y / YN;
  z = z / ZN;

  x = f(x);
  y = f(y);
  z = f(z);

  let l = 116 * y - 16;
  let a = 500 * (x - y);
  let b = 200 * (y - z);
  return {
    l,
    a,
    b
  };
};

/**
 * RGB 转为 LAB
 * @param {number} r [0-255]红色通道值
 * @param {number} g [0-255]绿色通道值
 * @param {number} b [0-255]蓝色通道值
 * @return {LAB} 返回转换的 Lab 值
 */
const rgbToLab = function (
 r: number,
 g: number,
 b: number
): LAB {
  let xyz = rgbToXyz(r, g, b);
  let lab = xyzToLab(xyz.x, xyz.y, xyz.z);
  return {
    l: toFixed(lab.l),
    a: toFixed(lab.a),
    b: toFixed(lab.b)
  };
};

/**
 * XYZ 转 LAB 的非线性变换
 * @param {number} t x, y, z的值
 */
function f(t: number): number {
  // Math.pow(29 / 6, 2) / 3 = 7.787037037037035
  // 16 / 116 = 0.13793103448275862
  if (t > 0.008856) {
    return Math.pow(t, 1 / 3);
  } else {
    return 7.787037037037035 * t + 0.13793103448275862;
  }
}

/**
 * gamma变换
 * @param t - r, g, b值
 */
function gamma(t: number) {
  return 
    t > 0.04045 ?
    Math.pow((t + 0.055) / 1.055, 2.4)
    : 
    t / 12.92;
}

上方给出的转换方法并不是绝对的,不同转换方法会有不同标准,具体根据自己需要选择

由于篇幅有限,这里只给出了RGB转换向其它的空间的,至于其它空间转换向RGB的并没给出,但具体的代码均封装为一个包(TS版),具体可按下列方法获取

封装的转换算法获取方法:在公众号:代码杂谈内回复颜色转换算法

标签:颜色,const,互相,number,param,RGB,转换,255
From: https://www.cnblogs.com/skmcj/p/18355885

相关文章

  • 预训练PDF数据格式转换
      大家好,我是herosunly。985院校硕士毕业,现担任算法研究员一职,热衷于大模型算法的研究与应用。曾担任百度千帆大模型比赛、BPAA算法大赛评委,编写微软OpenAI考试认证指导手册。曾获得阿里云天池比赛第一名,CCF比赛第二名,科大讯飞比赛第三名。授权多项发明专利。对机器学习和......
  • 数据类型的转换
    目录导言一、隐式类型转换1.整型隐式类型转换2.浮点型隐式类型转换3.字符型隐式类型转换4.布尔型隐式类型转换二、显式类型转换1.整型显式类型转换2.引用数据类型显式类型转换父类和子类之间的转换接口和实现类之间的转换三、类型转换的注意事项1.数据溢出和精度丢失2.强......
  • 词向量的转换理解以及与真实的关系
     实际上就是去除该矩阵的某一行。该矩阵实际上就是一个有N个词的300维向量,或者说该矩阵就是一个完整的词向量词汇表。而这个词汇表是通过交叉熵损失最小来构造的。即归根到底是“在特定语料库中(包含context中共现概率),以数学方法计算输入产生哪些输出”最复合“实际情况,即语料......
  • Android之集成Unity及互相调用
    Unity官方文档DemoUnity与原生交互之AndroidStudio篇——Unity导出Android工程,导入AndroidStudio打包APK全流程Unity3D与Android交互问题一Error:Unity.IL2CPP.Building.BuilderFailedException:Buildfailedwith0successfulnodesand0failedonesError:Inte......
  • 【办公软件学习】如何将Word格式转换为Markdown格式
    一键!将Word转换为Markdown参考链接1:https://zhuanlan.zhihu.com/p/30891168参考链接2:https://blog.csdn.net/qq15035899256/article/details/125547483参考链接3:https://word2md.com/方法一:Writage+Pandoc—双剑合璧!下载并安装Writage,下载地址:http://www.writage.c......
  • python datetime 时间格式转换
    1、字符串转时间datetime.strptimeimportdatetimedatetime.datetime.strptime('2020-08-1',"%Y-%m-%d")datetime.datetime.strptime('2020-08-123:30:59',"%Y-%m-%d%H:%M:%S")2、时间转字符串datetime.strftimeimportdatetimecurr......
  • 从字节到文件下载:揭秘 Spring 中 MultipartFile 的转换与浏览器端自动下载实现
    提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档文章目录前言一、下载功能核心Java代码二、代码解析1.设置Content-Disposition响应头2.设置Content-Type响应头3.写入文件内容到响应输出流4.文件写入到本地磁盘5.文件写入到Mnio中5.1主方法5.2实现......
  • 汇编语言语法练习与代码转换
     汇编语言语法练习与代码转换一、实验目的和要求1.掌握汇编语言源程序的格式;2.进一步学习汇编语言源程序的编辑、汇编、目标文件的链接和可执行文件的执行全过程;3.掌握编辑软件、TASM/MASM、TLINK和TD的使用方法、掌握汇编语言的语法规则。4.实验内容:(1)练习并掌握命......
  • Profibus DP(主站)转EtherNet/IP协议转换网关(通讯配置详解)
    作者的许多朋友均对如何实现ProfibusDP网络和EtherNet/IP网络的连接互通感到十分困扰,现在为大家统一作出解释。事实上,远创智控YC-DPM-EIP此款设备能够完美地解决这一问题。接下来,作者将会给各位全面且详尽地阐述该设备的功能、参数以及配置的方法。产品介绍本产品实现PROFIB......
  • 【C++高阶】:特殊类设计和四种类型转换
    ✨                      人生如梦,朝露夕花,宛若泡影   ......