27
votes

There are several questions out there that ask about converting colors between colorspaces on Apple platforms. Unfortunately, the answers quite often involve NSColor or UIColor -- non portable Objective-C classes that cannot be interchangeably used on OS X and iOS.

So I'd like to ask a very specific thing that I'm sure there must be a good answer to out there. I simply cannot believe that Apple would not foresee the need for this.

How does one convert CGColor from one colorspace (for example, monochrome) to another one (for example, RGB) in a generic way, supporting all CGColorSpace types, while using solely the portable Core Graphics functions?


Some context. I need to multiply the value provided by an online service with a value stored in UIColor. Correct way to extract the RGB components prior to iOS5, which finally introduced the method -[UIColor getRed:green:blue:alpha:], is to use CGColorGetComponents(). I then multiply this color with the color fetched from the online service. This fails in case +[UIColor grayColor] was used to generate the UIColor. Meaning, I need to convert the color from the greyscale colorspace into RGB. In this case, it's easy. What about some other colorspace being provided? Or what if in a theoretical future scenario I just want to process a single pixel's color?

There is a suggestion somewhere that I paint a pixel into a bitmap context and then read this pixel. That's insane, and I hope it isn't the only way to do this. Obviously the drawing method can figure out how to perform the conversion; how can we leverage this without creating a bitmap context solely to draw a pixel in it?


Additional research:

  • This article on color conversion, aside from UIColor's undocumented/private method -styleString, more interestingly mentions also undocumented CGColorTransforms.
  • ColorCG.cpp from WebKit suggests there is a header named CoreGraphics/CGColorTransform.h. Unfortunately, this header does not exist, at least not on Mountain Lion. Why would Apple hide these APIs?
  • The only other sensible resource I found mentioning CGColorTransforms is FreeQuartz, a free/open source reimplementation of Core Graphics.

Filed a radar with Apple, #12141580, to open up and document CGColorTransform. I'm not holding my breath, though, so if there are other sensible suggestions, I'm all ears.

3
I've been doing this for years either using vImage (Accelerate) or bit map contexts. The demand for such utilities as you suggest is low - why would Apple invest the time and energy to do them? Given pixels in RGB format, you can convert to other spaces using easy-to-find C code online, process the data, then convert back into RGB.David H
@DavidH: Your argument is silly. Why does Apple provide -[NSString pathExtension], then? People happily use substrings in other languages. If the demand is so low, why did Apple add -[UIColor getRed:green:blue:alpha:]? You talk about energy - why would thousands of developers invest energy to reinvent the wheel a thousand times if Apple could? I understand your point of view, but I politely disagree with it in belief that something can be done without creating a context or NSImage/CGImage. Using vImage sounds interesting, care to post an example as an answer?Ivan Vučica
The proper way to vent your frustration is to enter an enhancement bug on bug reporter.apple.com. I have entered something like 300 or so over the past 8 years. Or, you could choose to make a nice open source project to do this and put it on github (I have several such projects), and attain fame and fortune through it!David H
@DavidH: Heh - I've filed only about 10 in the past three years, and only two got closed. Yes, I'm frustrated by Apple's lack of easily visible way to convert the colorspace :-) but I'm thankful for your attention to the question (despite you not offering an exact answer and your initial suggestion that I'm asking about existence of "low-demand" (i.e. unnecessary) features). Creating an open-source project is also not as interesting as discovering how Apple is doing this and expecting us to do this. Plus, I don't know anything about color profiles, so I'm hardly the right person. :-)Ivan Vučica

3 Answers

7
votes

Apple added this API in iOS 9 and macOS 10.11:

  • ObjC: CGColorCreateCopyByMatchingToColorSpace
  • Swift: CGColor.converted(to:intent:options:)
6
votes

Apple's official response to rdar #12141580 where I requested exposing CGColorTransform:

Thanks very much for your feedback.

Engineering has determined that the behavior will not be changed based on the following information:

Use ColorSync.

If you have questions regarding the resolution of this issue, please update your bug report with that information.

We are now closing this bug report.

Please be sure to regularly check new Apple releases for any updates that might affect this issue.

Whoever wrote that response obviously felt compelled to explain the decision in great detail. Especially a decision that recommends the use of a highly complex API such as ColorSync.

4
votes

The question explicitly says they want to avoid drawing on a bitmap context created solely for the purpose of converting colors, but i didn't realize and came up with this.

I'll leave this as reference for anyone which could stumble on this question and want to know how you would do the transformation by drawing on a temporary bitmap

CGColorRef source;
CGColorSpaceRef targetColorSpace;
int components = CGColorSpaceGetNumberOfComponents(targetColorSpace);
CGFloat temp-buffer[components];
CGContextRef temp-bmp = CGBitmapContextCreate(
    temp-buffer,
    1,1,8 * sizeof(CGFloat),components,
    targetColorSpace,
    kCGBitmapFloatComponents );
CGContextSetFillColorWithColor(temp-bmp,source);
CGContextFillRect(temp-bmp,CGRectMake(0,0,1,1));
CGColor target = CGColorCreate(targetColorSpace,temp-buffer);

/* then you cleanup and destroy created objects */

I didn't try this code, it could need some fixes.