Представление PNG / JPEG из CIImage всегда возвращает ноль

В настоящее время я создаю приложение для редактирования фотографий.

Когда фотография выбирается пользователем, она автоматически конвертируется в черно-белую с помощью этого кода:

func blackWhiteImage(image: UIImage) -> Data {
    print("Starting black & white")

    let orgImg = CIImage(image: image)
    let bnwImg = orgImg?.applyingFilter("CIColorControls", withInputParameters: [kCIInputSaturationKey:0.0])

    let outputImage = UIImage(ciImage: bnwImg!)

    print("Black & white complete")

    return UIImagePNGRepresentation(outputImage)!
}

Проблема, с которой я столкнулся с этим кодом, заключается в том, что я продолжаю получать эту ошибку:

fatal error: unexpectedly found nil while unwrapping an Optional value

У меня был мой код в немного другой конфигурации, но он все равно ломается, когда попадает в раздел UIImagePNG/JPEGRepresentation(xx).

Есть ли способы получить данные PNG или JPEG из CIImage для использования в просмотре изображений / просто UIImage в целом?

Ни один из других методов не дает достаточно подробностей о том, какой код следует использовать.


person benfernandes    schedule 03.03.2017    source источник
comment
CIImage - это не изображение. Вы должны отрендерить его, чтобы получить изображение.   -  person matt    schedule 04.03.2017
comment
Дубликат представления stackoverflow.com/questions/40319229/ и многие другие   -  person matt    schedule 17.03.2017


Ответы (1)


Просто начните новый графический контекст и нарисуйте там свое изображение в оттенках серого. iOS 10 или новее вы можете использовать UIGraphicsImageRenderer, для синтаксиса более старой версии iOS проверьте историю изменений:

Xcode 11 • Swift 5.1

func blackWhiteImage(image: UIImage, isOpaque: Bool = false) -> Data? {
    guard let ciImage = CIImage(image: image)?.applyingFilter("CIColorControls", parameters: [kCIInputSaturationKey: 0]) else { return nil }
    let format = image.imageRendererFormat
    format.opaque = isOpaque
    return UIGraphicsImageRenderer(size: image.size, format: format).image { _ in
        UIImage(ciImage: ciImage).draw(in: CGRect(origin: .zero, size: image.size))
    }.pngData()
}

Вы также можете расширить UIImage, чтобы вернуть изображение в оттенках серого:

extension UIImage {
    var ciImage: CIImage? { CIImage(image: self) }
    func grayscale(isOpaque: Bool = false) -> UIImage? {
        guard let ciImage = ciImage?.applyingFilter("CIColorControls", parameters: [kCIInputSaturationKey: 0]) else { return nil }
        let format = imageRendererFormat
        format.opaque = isOpaque
        return UIGraphicsImageRenderer(size: size, format: format).image { _ in
             UIImage(ciImage: ciImage).draw(in: CGRect(origin: .zero, size: size))
        }
    }
}

let profilePicture = UIImage(data: try! Data(contentsOf: URL(string:"http://i.stack.imgur.com/Xs4RX.jpg")!))!

if let grayscale = profilePicture.grayscale(), let data = grayscale.pngData() {  // or Swift 4.1 or earlier -> let data = UIImagePNGRepresentation(grayscale)
    print(data.count)   //  689035
}
person Leo Dabus    schedule 04.03.2017