Quartz 2D编程指南之四:颜色与颜色空间

本文转载自:http://southpeak.github.io/2014/12/01/quartz2d-4/

不同的设备(显示器、打印机、扫描仪、摄像头)处理颜色的方式是不同的。每种设备都有其所能支持的颜色值范围。一种设备能支持的颜色可能在其它设备中无法支持。

为了有效的使用颜色及理解Quartz 2D中用于颜色及颜色空间的函数,我们需要熟悉在Color Management Overview文档中所使用的术语。该文档中讨论了色觉、颜色值、设备依赖及设备颜色空间、颜色匹配问题、再现意图(rendering intent)、颜色管理模块和ColorSync。

在本章中,我们将学习Quartz处理颜色和颜色空间,以及什么是alpha组件。本章同时也讨论如下问题:

创建颜色空间

创建和设置颜色

设置再现意图

颜色与颜色空间

Quartz中的颜色是用一组值来表示。而颜色空间用于解析这些颜色信息。例如,表4-1列出了在全亮度下蓝色值在不同颜色空间下的值。如果不知道颜色空间及颜色空间所能接受的值,我们没有办法知道一组值所表示的颜色。

Table 4-1 Color values in different color spaces

如果我们使用了错误的颜色空间,我们可能会获得完全不同的颜色,如图4-1所示。

Figure 4-1 Applying a BGR and an RGB color profile to the same image

颜色空间可以有不同数量的组件。表4-1中的颜色空间中其中三个只有三个组件,而CMYK有四个组件。值的范围与颜色空间有关。对大部分颜色空间来说,颜色值范围为[0.0, 1.0],1.0表示全亮度。例如,全亮度蓝色值在Quartz的RGB颜色空间中的值是(0, 0, 1.0)。在Quartz中,颜色值同样有一个alpha值来表示透明度。在表4-1中没有列出该值。

alpha值

alpha值是图形状态参数,Quartz用它来确定新的绘图对象如何与已存在的对象混合。在全强度下,新的绘图对象是不透明的。在0强度下,新的绘图对象是完全透明的。图4-2显示了5个大的方形,分别使用了alpha值为1.0, 0.75, 0.5, 0.1和0.0。随着大方形逐渐变得透明,底下的小的不透明的方形逐渐显现出来。

Figure 4-2 A comparison of large rectangles painted using various alpha values

我们可以将两个对象绘制到page上,而page可以在渲染前通过设置全局的graphics context来设置自己的透明度。图4-3显示了将全局的透明度设置为0.5和1.0的效果。

Figure 4-3 A comparison of global alpha values

在标准混合模式(图形状态的默认模式)下,Quartz使用下面的公式来混合源颜色和目标颜色的组件:

destination = (alphasource) + (1 - alpha)destination

其中源颜色是新绘制的颜色,目标颜色是背景颜色。该公式可用于新绘制的形状和图像。

对于对象透明度来说,alpha值为1.0时表示新对象是完全不透明的,值0.0表示新对象是完全透明的。0.0与1.0之间的值指定对象的透明度。我们可以为所有接受颜色的程序指定一个alpha值作为颜色值的最后一个组件。同样也可以使用CGContextSetAlpha函数来指定全局的alpha值。记住,如果同时设置以上两个值,Quartz将混合全局alpha值与对象的alpha值。

为了让page完全透明,我们可以调用CGContextClearRect函数来清除图形上下文(graphics context)的alpha通道。例如,我们可以在给图标创建一个透明遮罩或者使窗口的背景透明时,采用这种方法。

创建颜色空间

Quartz支持颜色管理系统使用的标准颜色空间,也支持通用的颜色空间、索引颜色空间和模式(pattern)颜色空间。设备颜色空间以一种简便的方法在不同设备间表示颜色。它用于在两种不同设备间的本地颜色空间转换颜色数据。设备依赖颜色空间的颜色在不同设备上显示时效果是一样的,它扩展了设备的能力。基于此,设备依赖颜色空间是显示颜色时最好的选择。

如果应用程序有精确的颜色表示需求,则应该总是使用设备依赖颜色空间。通用颜色空间(generic color space)是一种常用的设备依赖颜色空间。通用颜色空间通过操作系统为我们的应用程序提供最好的颜色空间。它能使在显示器上与在打印机上打印效果是一样的。

重要:IOS不支持设备依赖颜色空间或通用颜色空间。IOS应用程序必须使用设备颜色空间(device color space)。

创建设备依赖颜色空间

为了创建设备依赖颜色空间,我们需要给Quartz提供白色参考点,黑色参考点及特殊设备的gamma值。Quartz使用这些信息将源颜色空间的颜色值转化为输出设备颜色空间的颜色值。

Quartz支持设备依赖颜色空间,创建此空间的函数如下:

L*a*b是非线性转换,它属于Munsell颜色符号系统(该系统使用色度、值、饱和度来指定颜色)。 L组件表示亮度值,a组件表示绿色与红色之间的值,b组件表示蓝色与黄色之间的值。该颜色空间设计用于模拟人脑解码颜色。使用函数CGColorSpaceCreateLab来创建。

ICC颜色空间是由ICC(由国际色彩聪明,International Color Consortium)颜色配置而来的。ICC颜色配置了设备支持的颜色域,该颜色域与其它设备属性相符,所以该信息可被用于将一个设备的颜色空间精确地转换为另一个设备的颜色空间。大多数设备制造商都支持ICC配置。一些彩色显示器和打印机都内嵌了ICC信息,用于处理诸如TIFF的位图格式。使用函数CGColorSpaceCreateICCBased来创建。

标准化RGB是设备依赖的RGB颜色空间,它表示相对于白色参考点(设备可生成的最白的颜色)的颜色。 使用函数CGColorSpaceCreateCalibratedRGB来创建。

标准化灰度是设备依赖的灰度颜色空间,它表示相对于白色参考点(设备可生成的最白的颜色)的颜色。 使用函数CGColorSpaceCreateCalibratedGray来创建。

创建通用颜色空间

通用颜色空间的颜色与系统匹配。大部分情况下,结果是可接受的。就像名字所暗示的那样,每个“通用”颜色空间(generic gray, generic RGB, generic CMYK)都是一个指定的设备依赖颜色空间。

通过颜色空间非常容易使用;我们不需要提供任何参考点信息。我们使用函数CGColorSpaceCreateWithName来创建一个通用颜色空间,该函数可传入以下常量值:

kCGColorSpaceGenericGray:指定通用灰度颜色空间,该颜色空间是单色的,可以指定从0.0(纯黑)到1.0(纯白)范围内的颜色值。

kCGColorSpaceGenericRGB:指定通用RGB颜色空间,该颜色空间中的颜色值由三个组件(red, green, blue)组成,主要用于彩色显示器上的像素。RGB颜色空间中的每个组件的值范围是[0.0, 1.0]。

kCGColorSpaceGenericCMYK:指定通用CMYK颜色空间,该颜色空间的颜色值由四个组件(cyan, magenta, yellow, black),主要用于打印机。CMYK颜色空间的每个组件的值范围是[0.0, 1.0]。

创建设备颜色空间

设备颜色空间主要用于IOS应用程序,因为其它颜色空间无法在IOS上使用。大多数情况下,Mac OS X应用程序应使用通用颜色空间,而不使用设备颜色空间。但是有些Quartz程序希望图像使用设备颜色空间。例如,如果调用CGImageCreateWithMask函数来指定一个图像作为遮罩,图像必须在设备的灰度颜色空间(device gray color space)中定义。

我们可以使用以下函数来创建设备颜色空间:

CGColorSpaceCreateDeviceGray:创建设备依赖灰度颜色空间

CGColorSpaceCreateDeviceRGB:创建设备依赖RGB颜色空间

CGColorSpaceCreateDeviceCMYK:创建设备依赖CMYK颜色空间

创建索引颜色空间和模式颜色空间

索引颜色空间包含一个有256个词目的颜色表,和词目映射到基础颜色空间。颜色表中每个词目指定一个基础颜色空间中的颜色值。使用CGColorSpaceCreateIndexed函数来创建。

模式颜色空间在绘制模式时使用。 使用CGColorSpaceCreatePattern函数来创建。

设置和创建颜色

Quartz提供了一套函数用于设置填充颜色、线框颜色、颜色空间和alpha值。每个颜色参数都是图形状态参数,这就意味着一旦设置了,设置将被保存并影响后续操作,直到被修改为止。

一个颜色必须有相关联的颜色空间。否则,Quartz不知道如何解析颜色值。进一步说,说是我们必须为绘制目标提供一个合适的颜色空间。如图4-4所示,左边是CMYK颜色空间中的蓝色填充色,右边是RGB颜色空间中的蓝色填充色。这两个颜色值在理论上是一样的,但只有在相同颜色空间下的相同颜色值显示出来才是一样的。

Figure 4-4 A CMYK fill color and an RGB fill color

我们可以使用CGContextSetFillColorSpace和CGContextSetStrokeColorSpace函数来设置填充和线框颜色空间,或者可以使用以下便利函数来设置设备颜色空间的颜色值。

Table 4-2 Color-setting functions

我们在填充及线框颜色空间中指定填充及线框颜色值。例如,在RGB颜色空间中,我们使用数组(1.0, 0.0, 0.0, 1.0)来表示红色。前三个值指定红色值为全强度,而绿色和蓝色为零强度。第四个值为alpha值,用于指定颜色的透明度。

如果需要在程序中重复使用颜色,最有效的方法是通过设置填充色和线框色来创建一个CGColor对象,然后将该对象传递给函数CGContextSetFillColorWithColorCGContextSetStrokeColorWithColor。我们可以按需要保持CGColor对象,并可以直接使用该对象来改进应用程序的显示。

我们可以调用CGColorCreate函数来创建CGColor对象,该函数需要两个参数:CGColorspace对象及颜色值数组。数组的最后一个值指定alpha值。

设置再现意图(Rending Intent)

“再现意图”用于指定如何将源颜色空间的颜色映射到图形上下文的目标颜色空间的颜色范围内。如果不显示指定再现意图,Quartz使用相对色度再现意图(relative colorimetric rendering intent)应用于所有绘制(不包含位图图像)。对于位图图像,Quartz默认使用感知(perceptual)再现意图。

我们可以调用CGContextSetRenderingIntent函数来设置再现意图,并传递图形上下文(graphics context)及下例常量作为参数:

kCGRenderingIntentDefault:使用默认的渲染意图。

kCGRenderingIntentAbsoluteColorimetric:绝对色度渲染意图。将输出设备颜色域外的颜色映射为输出设备域内与之最接近的颜色。这可以产生一个裁减效果,因为色域外的两个不同的颜色值可能被映射为色域内的同一个颜色值。当图形使用的颜色值同时包含在源色域及目标色域内时,这种方法是最好的。常用于logo或者使用专色(spot color)时。

kCGRenderingIntentRelativeColorimetric:相对色度渲染意图。转换所有的颜色(包括色域内的),以补偿图形上下文的白点与输出设备白点之间的色差。kCGRenderingIntentPerceptual:感知渲染意图。通过压缩图形上下文的色域来适应输出设备的色域,并保持源颜色空间的颜色之间的相对性。感知渲染意图适用于相片及其它复杂的高细度图片。

kCGRenderingIntentSaturation:饱和度渲染意图。把颜色转换到输出设备色域内时,保持颜色的相对饱和度。结果是包含亮度、饱和度颜色的图片。饱和度意图适用于生成低细度的图片,如描述性图表。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容