// 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)
    }
}