我正在构建一个基于CIFilters和MetalKit的实时照片编辑器.但是我遇到了在MTKView中显示宽色域图像的问题.

标准sRGB图像显示正常,但显示P3图像被清除.

我已经尝试将CIContext.render颜色空间设置为图像颜色空间,但仍然遇到问题.

以下是代码片段:

guard let inputimage = CIImage(mtlTexture: sourceTexture!) else { return }
                let outputimage = imageEditor.processImage(inputimage)
                print(colorSpace)
                context.render(outputimage,to: currentDrawable.texture,commandBuffer: commandBuffer,bounds: inputimage.extent,colorSpace: colorSpace)
                commandBuffer?.present(currentDrawable)

let pickedImage = info[UIImagePickerControllerOriginalImage] as! UIImage
        print(pickedImage.cgImage?.colorSpace)
        if let cspace = pickedImage.cgImage?.colorSpace {
            colorSpace = cspace
        }

我在Apple开发者论坛上发现了类似的问题,但没有任何答案:https://forums.developer.apple.com/thread/66166

解决方法

为了支持广泛的色域,您需要将MTKView的colorPixelFormat设置为 BGRA10_XR或 bgra10_XR_sRGB.我怀疑iOS上不支持macOS MTKViews的colorSpace属性,因为iOS中的色彩管理不活跃但是有针对性(阅读 Best practices for color management).

没有看到你的图像和他们的实际值,很难诊断,但我会解释我的发现和&实验.我建议你像我一样开始调试一种颜色.

例如,P3色彩空间中最红的点是什么?它可以通过UIColor定义,如下所示:

UIColor(displayP3Red: 1,green: 0,blue: 0,alpha: 1)

将UIButton添加到视图中,将背景设置为该颜色以进行调试.您可以在代码中获取组件以查看这些值在sRGB中的变化,

var fRed : CGFloat = 0
    var fGreen : CGFloat = 0
    var fBlue : CGFloat = 0
    var fAlpha : CGFloat = 0
    let c = UIColor(displayP3Red: 1,alpha: 1)
    c.getRed(&fRed,green: &fGreen,blue: &fBlue,alpha: &fAlpha)

或者您可以在macOS Color Sync Utility中使用计算器,

确保选择“扩展范围”,否则值将被钳制为0和1.

因此,正如您所看到的,您的P3(1,0)对应于扩展sRGB中的(1.0930,-0.2267,-0.1501).

现在,回到你的MTKView,

>如果将MTKView的colorPixelFormat设置为.BGRA10_XR,那么如果着色器的输出为,则获得最亮的红色,

(1.0930,-0.1501)
>如果将MTKView的colorPixelFormat设置为.bgra10_XR_sRGB,

(1.22486,-0.0420312,-0.0196301)

因为您必须编写线性RGB值,因为此纹理格式将为您应用伽马校正.应用反伽马时要小心,因为存在负值.我用这个功能,

let f = {(c: Float) -> Float in
    if fabs(c) <= 0.04045 {
        return c / 12.92
    }
    return sign(c) * powf((fabs(c) + 0.055) / 1.055,2.4)
}

最后一个缺失的部分是创建一个宽域UIImage.将颜色空间设置为CGColorSpace.displayP3并复制数据.但是有什么数据呢?这张照片中最亮的红色将是

(1,0)

或(65535,0)以16位整数形式.

我在我的代码中做的是使用.rgba16Unorm纹理来操作displayP3颜色空间中的图像,其中(1,0)将是P3中最亮的红色.这样,我可以直接将其内容复制到UIImage.然后,为了显示,我将颜色变换传递给着色器,以在显示之前从P3转换为扩展的sRGB(因此,不是饱和的颜色).我使用线性颜色,所以我的变换只是一个3×3矩阵.我将视图设置为.bgra10_XR_sRGB,因此伽玛将自动应用于我.

那个(列主要)矩阵是,

1.2249  -0.2247  0
-0.0420   1.0419  0
-0.0197  -0.0786  1.0979

你可以在这里阅读我是如何生成它的:Exploring the display-P3 color space

这是我使用UIButtons和MTKView构建的示例,在iPhoneX上进行屏幕捕获,

左侧的按钮是sRGB上最亮的红色,而右侧的按钮是使用displayP3颜色.在中心,我放置了一个MTKView,它输出如上所述的变换后的线性颜色.

同样的绿色实验,

现在,如果你在最近的iPhone或iPad上看到这个,你应该看到中间的方块和右边的按钮都有相同的亮色.如果您在无法显示它们的Mac上看到此内容,左侧按钮将显示相同的颜色.如果你在没有正确颜色管理的Windows机器或浏览器中看到这个,左边的按钮也可能看起来颜色不同,但这只是因为整个图像被解释为sRGB,显然这些像素具有不同的值……但外观不正确.

如果你想要更多的参考,请检查我在这里添加的testP3UIColor单元测试:ColorTests.swift,

我的函数初始化UIImage:Image.swift,

以及尝试转换的示例应用:SampleColorPalette

我没有尝试过CIImages,但我想同样的原则也适用.

我希望这些信息有所帮助.我花了很长时间才弄清楚如何正确显示颜色,因为我在Metal SDK文档中找不到任何对displayP3支持的明确引用.

ios – 显示宽色域P3色彩空间的MTKView的更多相关文章

  1. ios – 显示宽色域P3色彩空间的MTKView

    但外观不正确.如果你想要更多的参考,请检查我在这里添加的testP3UIColor单元测试:ColorTests.swift,我的函数初始化UIImage:Image.swift,以及尝试转换的示例应用:SampleColorPalette我没有尝试过CIImages,但我想同样的原则也适用.我希望这些信息有所帮助.我花了很长时间才弄清楚如何正确显示颜色,因为我在MetalSDK文档中找不到任何对displayP3支持的明确引用.

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

    现在我真的很困惑.下面是变量如何实例化:在这里,我枚举一个属性文本的属性,以跳过颜色,如果它等于Utils.redColor:所以,正如你在这段代码中看到的,textColor也是一个UIColor对象,但是日志说:这是两种确切的颜色,但是是两个不同类的实例.这两个都是UIColor类的对象,这完全令人困惑!虽然它在Swift2中运行良好,但这种比较从未触发我该如何解决它以及为什么会出现这个问题?

随机推荐

  1. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  2. iOS – genstrings:无法连接到输出目录en.lproj

    使用我桌面上的项目文件夹,我启动终端输入:cd然后将我的项目文件夹拖到终端,它给了我路径.然后我将这行代码粘贴到终端中找.-name*.m|xargsgenstrings-oen.lproj我在终端中收到此错误消息:genstrings:无法连接到输出目录en.lproj它多次打印这行,然后说我的项目是一个目录的路径?没有.strings文件.对我做错了什么的想法?

  3. iOS 7 UIButtonBarItem图像没有色调

    如何确保按钮图标采用全局色调?解决方法只是想将其转换为根注释,以便为“回答”复选标记提供更好的上下文,并提供更好的格式.我能想出这个!

  4. ios – 在自定义相机层的AVFoundation中自动对焦和自动曝光

    为AVFoundation定制图层相机创建精确的自动对焦和曝光的最佳方法是什么?

  5. ios – Xcode找不到Alamofire,错误:没有这样的模块’Alamofire’

    我正在尝试按照github(https://github.com/Alamofire/Alamofire#cocoapods)指令将Alamofire包含在我的Swift项目中.我创建了一个新项目,导航到项目目录并运行此命令sudogeminstallcocoapods.然后我面临以下错误:搜索后我设法通过运行此命令安装cocoapodssudogeminstall-n/usr/local/bin

  6. ios – 在没有iPhone6s或更新的情况下测试ARKit

    我在决定下载Xcode9之前.我想玩新的框架–ARKit.我知道要用ARKit运行app我需要一个带有A9芯片或更新版本的设备.不幸的是我有一个较旧的.我的问题是已经下载了新Xcode的人.在我的情况下有可能运行ARKit应用程序吗?那个或其他任何模拟器?任何想法或我将不得不购买新设备?解决方法任何iOS11设备都可以使用ARKit,但是具有高质量AR体验的全球跟踪功能需要使用A9或更高版本处理器的设备.使用iOS11测试版更新您的设备是必要的.

  7. 将iOS应用移植到Android

    我们制作了一个具有2000个目标c类的退出大型iOS应用程序.我想知道有一个最佳实践指南将其移植到Android?此外,由于我们的应用程序大量使用UINavigation和UIView控制器,我想知道在Android上有类似的模型和实现.谢谢到目前为止,guenter解决方法老实说,我认为你正在计划的只是制作难以维护的糟糕代码.我意识到这听起来像很多工作,但从长远来看它会更容易,我只是将应用程序的概念“移植”到android并从头开始编写.

  8. ios – 在Swift中覆盖Objective C类方法

    我是Swift的初学者,我正在尝试在Swift项目中使用JSONModel.我想从JSONModel覆盖方法keyMapper,但我没有找到如何覆盖模型类中的Objective-C类方法.该方法的签名是:我怎样才能做到这一点?解决方法您可以像覆盖实例方法一样执行此操作,但使用class关键字除外:

  9. ios – 在WKWebView中获取链接URL

    我想在WKWebView中获取tapped链接的url.链接采用自定义格式,可触发应用中的某些操作.例如HTTP://我的网站/帮助#深层链接对讲.我这样使用KVO:这在第一次点击链接时效果很好.但是,如果我连续两次点击相同的链接,它将不报告链接点击.是否有解决方法来解决这个问题,以便我可以检测每个点击并获取链接?任何关于这个的指针都会很棒!解决方法像这样更改addobserver在observeValue函数中,您可以获得两个值

  10. ios – 在Swift的UIView中找到UILabel

    我正在尝试在我的UIViewControllers的超级视图中找到我的UILabels.这是我的代码:这是在Objective-C中推荐的方式,但是在Swift中我只得到UIViews和CALayer.我肯定在提供给这个方法的视图中有UILabel.我错过了什么?我的UIViewController中的调用:解决方法使用函数式编程概念可以更轻松地实现这一目标.

返回
顶部