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

Vla*_*eev 7 uicolor ios swift swift3

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

Utils.redColor = UIColor(red: CGFloat(red) / 255.0, green: CGFloat(green) / 255.0, blue: CGFloat(blue)/255.0, alpha: alpha) 
Run Code Online (Sandbox Code Playgroud)

在这里,我枚举一个属性文本的属性,以跳过颜色,如果它等于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)
            }
          }
Run Code Online (Sandbox Code Playgroud)

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

textColor = kCGColorSpaceModelRGB 0.666667 0.172549 0.172549 1 red = UIExtendedSRGBColorSpace 0.666667 0.172549 0.172549 1
Run Code Online (Sandbox Code Playgroud)

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

虽然它在Swift2中运行良好,但这种比较从未触发

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

ric*_*ter 18

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

你的两种颜色是不相等的per isEqual(或Swift ==,它贯穿isEqual于拥有它的ObjC类),因为它们有不同的颜色空间.(它们不是不同的类;第一项UIColor.description是颜色空间的标识符,或者颜色空间没有名称的地方,颜色空间的模型 - 也就是说,它是否是基于RGB的,CMYK-基础,灰度等)

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

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

UIColor init(red:green:blue:alpha:)使用"Extended sRGB"颜色空间创建的颜色.此色彩空间旨在支持广泛的彩色显示(如iPhone 7中的P3彩色显示屏,iPad Pro 9.7",2015年末的iMac,2016年末的MacBook Pro,以及可能的其他任何产品),但兼容组件价值使用其他设备上使用的sRGB色彩空间.

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

作为[初始化程序注释的文档,对于与iOS 10或更高版本SDK链接的应用程序init(red:green:blue:alpha:),在Extended sRGB颜色空间中创建颜色,但对于较旧的应用程序(即使它们在iOS 10上运行),它会创建一个颜色特定于设备的RGB空间(通常可将其视为与sRGB等效).

处理不同的色彩空间

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

  1. 确保您的字符串创建代码和颜色替换代码都使用相同的设备无关颜色空间.UIColor没有提供很多用于处理色彩空间的工具,因此您可以使用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)
        }
    }
    
    Run Code Online (Sandbox Code Playgroud)
  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
        }
    }
    
    Run Code Online (Sandbox Code Playgroud)

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

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