Почему мой CGImage не отображается правильно после setNeedsDisplay

Я рисую colorWheel в своем приложении. Я делаю это, создавая CGImage в отдельной функции и используя CGContextDrawImage в drawRect, чтобы вывести его на экран.

В начальной презентации все выглядит нормально, но после вызова setNeedsDisplay (или setNeedsDisplayInRect) изображение становится черным/искажается. Должно быть, я делаю что-то глупое, но не могу понять, что.

Крутое колесоКрутое колесо

Код DrawRect выглядит так:

override func drawRect(rect: CGRect) {

    if let context = UIGraphicsGetCurrentContext() {

        let wheelFrame = CGRectMake(0,0,circleRadius*2,circleRadius*2)
        CGContextSaveGState(context)

            //create clipping mask for circular wheel
        CGContextAddEllipseInRect(context, wheelFrame)
        CGContextClip(context)

            //draw the wheel
        if colorWheelImage != nil { CGContextDrawImage(context, CGRectMake(0,0,circleRadius*2,circleRadius*2), colorWheelImage) }
        CGContextRestoreGState(context)

            //draw a selector element
        self.drawSliderElement(context)
    }
}

Функция генерации CGimage:

func generateColorWheelImage() {

    let width       = Int(circleRadius*2)
    let height      = Int(circleRadius*2)

    var pixels = [PixelData]()

    for yIndex in 0..<width {
        for xIndex in 0..<height {
            pixels.append(colorAtPoint(CGPoint(x: xIndex, y: yIndex)))
        }
    }

    let rgbColorSpace   = CGColorSpaceCreateDeviceRGB()
    let bitmapInfo      = CGBitmapInfo.ByteOrderDefault

    let bitsPerComponent: Int   = 8
    let bitsPerPixel: Int       = 24

    let renderingIntent = CGColorRenderingIntent.RenderingIntentDefault

    assert(pixels.count == Int(width * height))

    var data = pixels

    let providerRef = CGDataProviderCreateWithData(nil, data, data.count * sizeof(PixelData), nil)

    colorWheelImage = CGImageCreate(width, height, bitsPerComponent, bitsPerPixel, width * Int(sizeof(PixelData)), rgbColorSpace, bitmapInfo, providerRef, nil, true,renderingIntent)

}

person Philip De Vries    schedule 12.06.2016    source источник


Ответы (1)


Наконец-то я нашел ответ (здесь: https://stackoverflow.com/a/10798750/5233176) на эту проблему, когда Я начал запускать код на iPad, а не на симуляторе, и получил BAD_ACCESS_ERROR, скорее увидев искаженное изображение.

Как поясняется в ответе: «[] CMSampleBuffer используется непосредственно для создания CGImageRef, поэтому [] CGImageRef станет недействительным после освобождения буфера».

Следовательно, проблема была с этой строкой:

    let providerRef = CGDataProviderCreateWithData(nil, data, data.count * sizeof(PixelData), nil)

И проблему можно решить, используя копию буфера, например:

    let providerData    = NSData(bytes: data, length: data.count * sizeof(PixelData))
    let provider        = CGDataProviderCreateWithCFData(providerData)
person Philip De Vries    schedule 18.08.2016