如何比较日志中kCGColorSpaceModelRGB和UIExtendedSRGBColorSpace实例的2个UIColor对象?

时间:2016-12-19 15:26:31

标签: ios swift swift3 uicolor

现在我真的很困惑。 下面是变量如何实例化:

Utils.redColor = UIColor(red: CGFloat(red) / 255.0, green: CGFloat(green) / 255.0, blue: CGFloat(blue)/255.0, alpha: alpha) 

在这里我枚举一个属性文本的属性,如果它等于Utils.redColor,则跳过颜色:

    text.enumerateAttributes(in: NSRange(0..<text.length), options: []) { (attributes, range, _) -> Void in
                            for (attribute, object) in attributes {
                                if let textColor = object as? UIColor{
                                     NSLog("textColor = \(textColor) red = \(Utils.redColor!)") 
             if (!textColor.isEqual(Utils.redColor!)){
//I need to repaint any textColor other than red
             text.setAttributes(textAttributes , range: range)
            }
          }

所以,正如你在这段代码中看到的那样,textColor 也是一个UIColor对象,但是日志说:

textColor = kCGColorSpaceModelRGB 0.666667 0.172549 0.172549 1 red = UIExtendedSRGBColorSpace 0.666667 0.172549 0.172549 1

这是两种确切的颜色,但是是两个不同类的实例。这两个都是UIColor类的对象,这完全令人困惑!

这种比较从未触发,尽管它在Swift2中运行良好

我该如何修复它以及为什么会出现这个问题?

1 个答案:

答案 0 :(得分:17)

欢迎来到广泛色彩和色彩管理的野性和毛茸茸的世界。

你的两种颜色并不相等,每isEqual(或Swift ==,对于拥有它的ObjC类,它通过isEqual,因为它们有不同的颜色空间。 (它们不是不同的类; UIColor.description中的第一项是颜色空间的标识符,或者颜色空间没有名称的地方,颜色空间的模型 - 即,无论是基于RGB,基于CMYK,灰度等等。)

如果没有颜色空间将它们定义为颜色,则颜色的四个组件值没有可靠的含义,因此isEqual使用组件值和颜色空间来测试相等性。

除了颜色空间(跳过解决方案)

使用UIColor init(red:green:blue:alpha:)创建的颜色使用&#34;扩展sRGB&#34;色彩空间。此色彩空间旨在支持广泛的彩色显示(如iPhone 7中的P3彩色显示屏,iPad Pro 9.7和#34; iMac 2015年末,2016年末的MacBook Pro,以及可能接下来的其他任何内容),但应该是组件-value与其他设备上使用的sRGB色彩空间兼容。

例如,sRGB 1.0, 0.0, 0.0是&#34; red&#34;您可能最习惯......但如果您在P3颜色空间中使用RGB值1.0, 0.0, 0.0创建颜色,则会得到更多红色。如果您有一个应用程序,您需要同时支持sRGB和P3显示并直接使用颜色组件,这可能会让人感到困惑。因此,扩展sRGB空间允许相同的组件值表示相同的内容,但也允许使用0.0 - 1.0范围之外的值指定sRGB色域之外的颜色。例如,显示P3可以获得的最红色在扩展sRGB中表示为(大致)1.093, -0.227, -0.15

作为[初始值设定说明的文档,对于针对iOS 10或更高版本SDK链接的应用,init(red:green:blue:alpha:)会在扩展sRGB颜色空间中创建颜色,但对于较旧的应用(即使它们是&#39;)在iOS 10上运行它会在特定于设备的RGB空间中创建一种颜色(通常可以将其视为等同于sRGB)。

处理不同的色彩空间

因此,无论您的颜色替换代码或任何代码在您的属性字符串中创建颜色,都需要了解颜色空间。有几种可能的方法可以解决这个问题;选择最适合你的那个:

  1. 确保您的字符串创建代码和颜色替换代码都使用相同的设备无关颜色空间。 UIColor并没有为使用色彩空间提供大量实用工具,因此您可以use Display P3(在iOS 10及更高版本上),或者下拉到CGColor

    let sRGB = CGColorSpace(name: CGColorSpace.sRGB)!
    let cgDarkRed = CGColor(colorSpace: sRGB, components: [0.666667, 0.172549, 0.172549, 1])!
    let darkRed = UIColor(cgColor: cgDarkRed)
    
    // example creating attributed string...
    let attrString = NSAttributedString(string: "red", attributes: [NSForegroundColorAttributeName : darkRed])
    
    // example processing text...
    let redAttributes = [NSForegroundColorAttributeName: darkRed]
    text.enumerateAttributes(in: NSRange(0..<attrString.length)) { (attributes, range, stop) in
        for (_, textColor) in attributes where (textColor as? UIColor) != darkRed {
            text.setAttributes(redAttributes , range: range)
        }
    }
    
  2. 如果您无法控制输入颜色,请在比较前将它们转换为相同的颜色空间。这是UIColor扩展程序:

    extension UIColor {
        func isEqualWithConversion(_ color: UIColor) -> Bool {
            guard let space = self.cgColor.colorSpace
                else { return false }
            guard let converted = color.cgColor.converted(to: space, intent: .absoluteColorimetric, options: nil)
                else { return false }
            return self.cgColor == converted
        }
    }
    

    (然后,您可以在文字处理中使用此功能代替==isEqual。)

  3. 只需获取颜色的原始组件值并直接比较它们,基于您知道两者的颜色空间兼容的假设。有点脆弱,所以我推荐这个选项。