You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
session-ios/Session/Utilities/QRCode.swift

121 lines
4.6 KiB
Swift

// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
import UIKit
enum QRCode {
/// Generates a QRCode for the give string
///
/// **Note:** If the `hasBackground` value is true then the QRCode will be black and white and
/// the `withRenderingMode(.alwaysTemplate)` won't work correctly on some iOS versions (eg. iOS 16)
static func generate(for string: String, hasBackground: Bool) -> UIImage {
let data = string.data(using: .utf8)
var qrCodeAsCIImage: CIImage
let filter1 = CIFilter(name: "CIQRCodeGenerator")! // stringlint:disable
filter1.setValue(data, forKey: "inputMessage")
qrCodeAsCIImage = filter1.outputImage!
guard !hasBackground else {
let filter2 = CIFilter(name: "CIFalseColor")! // stringlint:disable
filter2.setValue(qrCodeAsCIImage, forKey: "inputImage")
filter2.setValue(CIColor(color: .black), forKey: "inputColor0")
filter2.setValue(CIColor(color: .white), forKey: "inputColor1")
qrCodeAsCIImage = filter2.outputImage!
let scaledQRCodeAsCIImage = qrCodeAsCIImage.transformed(by: CGAffineTransform(scaleX: 6.4, y: 6.4))
return UIImage(ciImage: scaledQRCodeAsCIImage)
}
let filter2 = CIFilter(name: "CIColorInvert")! // stringlint:disable
filter2.setValue(qrCodeAsCIImage, forKey: "inputImage")
qrCodeAsCIImage = filter2.outputImage!
let filter3 = CIFilter(name: "CIMaskToAlpha")! // stringlint:disable
filter3.setValue(qrCodeAsCIImage, forKey: "inputImage")
qrCodeAsCIImage = filter3.outputImage!
let scaledQRCodeAsCIImage = qrCodeAsCIImage.transformed(by: CGAffineTransform(scaleX: 6.4, y: 6.4))
// Note: It looks like some internal method was changed in iOS 16.0 where images
// generated from a CIImage don't have the same color information as normal images
// as a result tinting using the `alwaysTemplate` rendering mode won't work - to
// work around this we convert the image to data and then back into an image
let imageData: Data = UIImage(ciImage: scaledQRCodeAsCIImage).pngData()!
return UIImage(data: imageData)!
}
}
import SwiftUI
import SessionUIKit
struct QRCodeView: View {
let string: String
let hasBackground: Bool
let logo: String?
let themeStyle: UIUserInterfaceStyle
var backgroundThemeColor: ThemeValue {
switch themeStyle {
case .light:
return .backgroundSecondary
default:
return .textPrimary
}
}
var qrCodeThemeColor: ThemeValue {
switch themeStyle {
case .light:
return .textPrimary
default:
return .backgroundPrimary
}
}
static private var cornerRadius: CGFloat = 10
static private var logoSize: CGFloat = 66
var body: some View {
ZStack(alignment: .center) {
ZStack(alignment: .center) {
RoundedRectangle(cornerRadius: Self.cornerRadius)
.fill(themeColor: backgroundThemeColor)
Image(uiImage: QRCode.generate(for: string, hasBackground: hasBackground))
.resizable()
.renderingMode(.template)
.foregroundColor(themeColor: qrCodeThemeColor)
.scaledToFit()
.frame(
maxWidth: .infinity,
maxHeight: .infinity
)
.padding(.vertical, Values.smallSpacing)
if let logo = logo {
ZStack(alignment: .center) {
Rectangle()
.fill(themeColor: backgroundThemeColor)
Image(logo)
.resizable()
.renderingMode(.template)
.foregroundColor(themeColor: qrCodeThemeColor)
.scaledToFit()
.frame(
maxWidth: .infinity,
maxHeight: .infinity
)
.padding(.all, 4)
}
.frame(
width: Self.logoSize,
height: Self.logoSize
)
}
}
.frame(
maxWidth: 400,
maxHeight: 400
)
}
.frame(maxWidth: .infinity)
}
}