CIQRCodeGenerator提供不一致的结果

时间:2015-12-27 16:44:22

标签: ios swift uiimageview uiimage core-image

我在使用CIQRCodeGenerator创建QR代码时遇到问题:当我首先生成代码时,它很清晰,但是当我再次运行该函数时,使用相同的输入,QR代码变得模糊:

初次运行(更清楚):

initial function run

第二次运行(更模糊):

second function run

首先在viewWillAppear中调用以下函数,然后在用户点击按钮后触发。

func generateQRCodeFromString(string: String) -> UIImage? {
    let data = string.dataUsingEncoding(NSISOLatin1StringEncoding)

    if let filter = CIFilter(name: "CIQRCodeGenerator") {
        filter.setValue(data, forKey: "inputMessage")
        filter.setValue("H", forKey: "inputCorrectionLevel")
        let transform = CGAffineTransformMakeScale(10, 10)

        if let output = filter.outputImage?.imageByApplyingTransform(transform) {
            return UIImage(CIImage: output)
        }
    }

    return nil
}

此处提供了说明问题的示例项目:http://jakeserver.com/Uploads/Apps/QR_Test.zip

在使用相同输入再次运行该功能后,UIImage是否会变得模糊?

编辑 - 添加了更多信息

override func viewDidLoad() {
    super.viewDidLoad()
    qrCode.image = generateQRCodeFromString("test", size: qrCode.frame.size);
}

override func viewWillLayoutSubviews() {
    qrCodeWidth.constant = self.view.frame.width * 0.8;
}
@IBAction func buttonTapped(sender: AnyObject) {
    qrCode.image = generateQRCodeFromString("test", size: qrCode.frame.size);
}

1 个答案:

答案 0 :(得分:1)

我不确定为什么运行之间的模糊性变化(可能是内部实现细节),但在Objective-C代码中我通过制作QR代码然后手动将图像写入更大尺寸的位图来解决这个问题。上下文。

我尝试将代码移植到Swift并提出了这个问题:

func generateQRCodeFromString(string: String, size: CGSize) -> UIImage? {
    guard let data = string.dataUsingEncoding(NSISOLatin1StringEncoding),
        let filter = CIFilter(name: "CIQRCodeGenerator") else { return nil }

    filter.setDefaults()
    filter.setValue(data, forKey: "inputMessage")
    filter.setValue("H", forKey: "inputCorrectionLevel")

    guard let image = filter.outputImage else { return nil }

    let extent = CGRectIntegral(image.extent)
    let scale = min(size.width / extent.width, size.height / extent.height);

    let (height, width) = (extent.height * scale, extent.width * scale)
    let colorSpace = CGColorSpaceCreateDeviceGray()
    let bitmapInfo = CGBitmapInfo(rawValue: CGImageAlphaInfo.None.rawValue)

    guard let bitmapContext = CGBitmapContextCreate(nil, Int(width), Int(height), 8, 0, colorSpace, bitmapInfo.rawValue) else { return nil }

    CGContextSetInterpolationQuality(bitmapContext, CGInterpolationQuality.None)
    CGContextScaleCTM(bitmapContext, CGFloat(scale), CGFloat(scale))
    CGContextDrawImage(bitmapContext, extent, CIContext().createCGImage(image, fromRect: extent))

    if let scaledImage = CGBitmapContextCreateImage(bitmapContext) {
        return UIImage(CGImage: scaledImage)
//        You might need to use this instead:
//        return UIImage(CGImage: <#T##CGImage#>, scale: <#T##CGFloat#>, orientation: <#T##UIImageOrientation#>)

    }

    return nil
}

这适用于您的用例吗?

顺便说一句,我不认为这导致了您的问题,但您没有解开datadataUsingEncoding(_:)返回NSData?而不是NSData)。