Swift: сбой кода по какой-то причине

Я новичок в Swift, и я прохожу курс онлайн, но, к сожалению, они не отвечают на мой вопрос. Я уверен, что ответ прост, но так как я новичок, я надеюсь, что люди здесь могут ответить мне.

import UIKit

public struct Pixel {
    public var value: UInt32 = 0

    public init() { }
    public var red: UInt8 {
        get {
            return UInt8(value & 0xFF)
        }
        set {
            value = UInt32(newValue) | (value & 0xFFFFFF00)
        }
    }

    public var green: UInt8 {
        get {
            return UInt8((value >> 8) & 0xFF)
        }
        set {
            value = (UInt32(newValue) << 8) | (value & 0xFFFF00FF)
        }
    }

    public var blue: UInt8 {
        get {
            return UInt8((value >> 16) & 0xFF)
        }
        set {
            value = (UInt32(newValue) << 16) | (value & 0xFF00FFFF)
        }
    }

    public var alpha: UInt8 {
        get {
            return UInt8((value >> 24) & 0xFF)
        }
        set {
            value = (UInt32(newValue) << 24) | (value & 0x00FFFFFF)
        }
    }
}

public class RGBAImage
{
    public var pixels: [Pixel]

    public var width: Int
    public var height: Int
    private var imageData: UnsafeMutablePointer<Pixel>

    public init?(image: UIImage) {
        guard let cgImage = image.cgImage else { return nil }

        // Redraw image for correct pixel format
        let colorSpace = CGColorSpaceCreateDeviceRGB()

        var bitmapInfo: UInt32 = CGBitmapInfo.byteOrder32Big.rawValue
        bitmapInfo |= CGImageAlphaInfo.premultipliedLast.rawValue & CGBitmapInfo.alphaInfoMask.rawValue

        width = Int(image.size.width)
        height = Int(image.size.height)
        let bytesPerRow = width * 4

        imageData = UnsafeMutablePointer<Pixel>.allocate(capacity: width * height)

        guard let imageContext = CGContext.init(data: imageData, width: width, height: height, bitsPerComponent: 8, bytesPerRow: bytesPerRow, space: colorSpace, bitmapInfo: bitmapInfo) else { return nil }
        imageContext.draw(cgImage, in: CGRect(origin: CGPoint.zero, size: image.size))

        let bufferPointer = UnsafeMutableBufferPointer<Pixel>(start: imageData, count: width * height)
        pixels = Array(bufferPointer) // <-- is this creating a copy of that bufferPointer contents or a reference?

        imageData.deinitialize(count: width * height) // <-- I tried commenting out this code, so not to free the memory but to no avail. I am not sure if the previous line copied the buffer or kept a reference to it
    }

    public func toUIImage() -> UIImage? {
        let colorSpace = CGColorSpaceCreateDeviceRGB()
        var bitmapInfo: UInt32 = CGBitmapInfo.byteOrder32Big.rawValue
        bitmapInfo |= CGImageAlphaInfo.premultipliedLast.rawValue & CGBitmapInfo.alphaInfoMask.rawValue

        let bytesPerRow = width * 4

        let imageDataReference = UnsafeMutablePointer<Pixel>(mutating: pixels)
        defer {
            imageDataReference.deinitialize(count: width * height)
        }
        let imageContext = CGContext(data: imageDataReference, width: width, height: height, bitsPerComponent: 8, bytesPerRow: bytesPerRow, space: colorSpace, bitmapInfo: bitmapInfo, releaseCallback: nil, releaseInfo: nil)

        guard let cgImage = imageContext!.makeImage() else {return nil}
        let image = UIImage(cgImage: cgImage)

        return image
    }
}

// Pixel visitor callback
public typealias PixelVisitor = (_ pixel: inout Pixel, _ args: AnyObject?) -> Void

public class ImageProcessor
{
    private var rgb_img: RGBAImage

    public init(img: UIImage) {
        rgb_img = RGBAImage(image: img)!
    }

    public func Pixels() -> [Pixel] {
        return rgb_img.pixels
    }

    public var width: Int {
        get {
            return rgb_img.width
        }
    }

    public var height: Int {
        get {
            return rgb_img.height
        }
    }

    public func visitPixels(_ callback: PixelVisitor, _ args: AnyObject?) {
        for y in 0..<self.rgb_img.height {
            let base = y * rgb_img.width
            for x in 0..<self.rgb_img.width {
                let index = base + x;
                var pixel = rgb_img.pixels[index]
                pixel.value += 0x202020 // <--- causing crash at index == 8?!
                //callback(&pixel, args)
            }
        }
    }

    public func toUIImage() -> UIImage? {
        return self.rgb_img.toUIImage()
    }
}

В комментарии к коду выше есть несколько вопросов ("<-"). Когда я запускаю посетителя, у меня происходит сбой при попытке изменить пиксель с индексом == 8.

let image = UIImage(named: "sample")
var filter_man = ImageProcessor(img: image!)
filter_man.visitPixels({ (p, args) in
    //p.red += 30
}, nil)

Что я делаю неправильно?

1 ответ

Самый простой способ избежать этого переполнения - это позволить swift сделать это

let res = pixel.value.addingReportingOverflow(0x202020)
pixel.value = res.partialValue

вместо вашей линии с крахом.

Если вы хотите обработать переполнение и в этом случае сделать пиксель белым / черным / красным / чем-то еще - вы можете проверить это с помощью свойства переполнения в res в моем примере.

Другие вопросы по тегам