金属多采样/锯齿边(iOS)

我正在尝试绘制一个将使用 Swift中的Metal进行动画处理的图形.我已成功绘制了一帧图形.从这张图片中可以看出,图形很简单.我无法弄清楚如何MultiSample绘图.一般来说,对Metal的引用似乎很少,特别是在Swift语法方面.

self.MetalLayer = cametallayer()
self.MetalLayer.device = self.device
self.MetalLayer.pixelFormat = .BGRA8Unorm
self.MetalLayer.framebufferOnly = true
self.MetalLayer.frame = self.view.frame
self.view.layer.addSublayer(self.MetalLayer)

self.renderer = SunRenderer(device: self.device,frame: self.view.frame)

let defaultLibrary = self.device.newDefaultLibrary()
let fragmentProgram = defaultLibrary!.newFunctionWithName("basic_fragment")
let vertexProgram = defaultLibrary!.newFunctionWithName("basic_vertex")

let pipelinestateDescriptor = MTLRenderPipelineDescriptor()
pipelinestateDescriptor.vertexFunction = vertexProgram
pipelinestateDescriptor.fragmentFunction = fragmentProgram
pipelinestateDescriptor.colorAttachments[0].pixelFormat = .BGRA8Unorm
pipelinestateDescriptor.colorAttachments[0].blendingEnabled = true

pipelinestateDescriptor.colorAttachments[0].rgbBlendOperation = MTLBlendOperation.Add
pipelinestateDescriptor.colorAttachments[0].alphaBlendOperation = MTLBlendOperation.Add
pipelinestateDescriptor.colorAttachments[0].sourceRGBBlendFactor = MTLBlendFactor.sourceAlpha
pipelinestateDescriptor.colorAttachments[0].sourceAlphaBlendFactor = MTLBlendFactor.sourceAlpha
pipelinestateDescriptor.colorAttachments[0].destinationRGBBlendFactor = MTLBlendFactor.OneMinusSourceAlpha
pipelinestateDescriptor.colorAttachments[0].destinationAlphaBlendFactor = MTLBlendFactor.OneMinusSourceAlpha

问题是,我如何平滑这些边缘?

更新:

所以我实现了一个MultiSample纹理,并将sampleCount设置为4.我没有注意到任何差异所以我怀疑我做错了什么.

最后:

所以最终确实出现了多次采样的工作原理.最初我有顶点用0 alpha包裹这些“光线”.这是一个使边缘更平滑的技巧.使用这些顶点,多重采样似乎没有改善边缘.当我恢复为每条光线有4个顶点时,多次采样改善了它们的边缘.

let defaultLibrary = self.device.newDefaultLibrary()
let fragmentProgram = defaultLibrary!.newFunctionWithName("basic_fragment")
let vertexProgram = defaultLibrary!.newFunctionWithName("basic_vertex")

let pipelinestateDescriptor = MTLRenderPipelineDescriptor()
pipelinestateDescriptor.vertexFunction = vertexProgram
pipelinestateDescriptor.fragmentFunction = fragmentProgram
pipelinestateDescriptor.colorAttachments[0].pixelFormat = .BGRA8Unorm
pipelinestateDescriptor.colorAttachments[0].blendingEnabled = true
pipelinestateDescriptor.sampleCount = 4

pipelinestateDescriptor.colorAttachments[0].rgbBlendOperation =    MTLBlendOperation.Add
pipelinestateDescriptor.colorAttachments[0].alphaBlendOperation = MTLBlendOperation.Add
pipelinestateDescriptor.colorAttachments[0].sourceRGBBlendFactor = MTLBlendFactor.sourceAlpha
pipelinestateDescriptor.colorAttachments[0].sourceAlphaBlendFactor = MTLBlendFactor.sourceAlpha

pipelinestateDescriptor.colorAttachments[0].destinationRGBBlendFactor = MTLBlendFactor.OneMinusSourceAlpha
pipelinestateDescriptor.colorAttachments[0].destinationAlphaBlendFactor = MTLBlendFactor.OneMinusSourceAlpha

let desc = MTLTextureDescriptor()
desc.textureType = MTLTextureType.Type2DMultisample
desc.width = Int(self.view.frame.width)
desc.height = Int(self.view.frame.height)
desc.sampleCount = 4
desc.pixelFormat = .BGRA8Unorm

self.sampletex = self.device.newTextureWithDescriptor(desc)


// When rendering
let renderPassDescriptor = MTLRenderPassDescriptor()
renderPassDescriptor.colorAttachments[0].texture = sampletex
renderPassDescriptor.colorAttachments[0].resolveTexture = drawable.texture
renderPassDescriptor.colorAttachments[0].loadAction = .Clear
renderPassDescriptor.colorAttachments[0].clearColor = MTLClearColor(red: 23/255.0,green: 26/255.0,blue: 31/255.0,alpha: 0.0)
renderPassDescriptor.colorAttachments[0].storeAction = .MultisampleResolve


let commandBuffer = commandQueue.commandBuffer()

let renderEncoder = commandBuffer.renderCommandEncoderWithDescriptor(renderPassDescriptor)
renderEncoder.setRenderPipelinestate(pipelinestate)

解决方法

这对MTKView来说简单得多(只需将sampleCount设置为视图和管道描述符上所需的MSAA样本数),但以下是滚动自己的步骤.

>创建渲染管道状态时,将渲染管道状态描述符的sampleCount设置为多重采样计数.>在启动时,每当图层调整大小时,通过创建纹理描述符(其textureType为MTLTextureType2DMultisample且其sampleCount为多重采样计数),创建尺寸等于图层可绘制尺寸的多重采样纹理.如果您使用的是深度和/或模板缓冲区,请在其描述符上设置这些属性.>渲染时,将MSAA纹理设置为渲染过程描述符的主要颜色附件的纹理,并将当前可绘制的纹理设置为resolveTexture.>将颜色附件的storeAction设置为MTLStoreActionMultisampleResolve,以便在传递结束时将MSAA纹理解析为渲染缓冲区.>像往常一样画画和呈现.

相关文章

UITabBarController 是 iOS 中用于管理和显示选项卡界面的一...
UITableView的重用机制避免了频繁创建和销毁单元格的开销,使...
Objective-C中,类的实例变量(instance variables)和属性(...
从内存管理的角度来看,block可以作为方法的传入参数是因为b...
WKWebView 是 iOS 开发中用于显示网页内容的组件,它是在 iO...
OC中常用的多线程编程技术: 1. NSThread NSThread是Objecti...