import Foundation import UIKit import Display import AsyncDisplayKit import WebKit import TelegramPresentationData private class WeakPaymentScriptMessageHandler: NSObject, WKScriptMessageHandler { private let f: (WKScriptMessage) -> () init(_ f: @escaping (WKScriptMessage) -> ()) { self.f = f super.init() } func userContentController(_ controller: WKUserContentController, didReceive scriptMessage: WKScriptMessage) { self.f(scriptMessage) } } final class BotCheckoutWebInteractionControllerNode: ViewControllerTracingNode, WKNavigationDelegate { private var presentationData: PresentationData private let intent: BotCheckoutWebInteractionControllerIntent private var webView: WKWebView? init(presentationData: PresentationData, url: String, intent: BotCheckoutWebInteractionControllerIntent) { self.presentationData = presentationData self.intent = intent super.init() self.backgroundColor = .white let webView: WKWebView switch intent { case .addPaymentMethod: let js = "var TelegramWebviewProxyProto = function() {}; " + "TelegramWebviewProxyProto.prototype.postEvent = function(eventName, eventData) { " + "window.webkit.messageHandlers.performAction.postMessage({'eventName': eventName, 'eventData': eventData}); " + "}; " + "var TelegramWebviewProxy = new TelegramWebviewProxyProto();" let configuration = WKWebViewConfiguration() let userController = WKUserContentController() let userScript = WKUserScript(source: js, injectionTime: .atDocumentStart, forMainFrameOnly: false) userController.addUserScript(userScript) userController.add(WeakPaymentScriptMessageHandler { [weak self] message in if let strongSelf = self { strongSelf.handleScriptMessage(message) } }, name: "performAction") configuration.userContentController = userController webView = WKWebView(frame: CGRect(), configuration: configuration) if #available(iOSApplicationExtension 9.0, iOS 9.0, *) { webView.allowsLinkPreview = false } case .externalVerification: webView = WKWebView() if #available(iOSApplicationExtension 9.0, iOS 9.0, *) { webView.allowsLinkPreview = false } webView.navigationDelegate = self } if #available(iOSApplicationExtension 11.0, iOS 11.0, *) { webView.scrollView.contentInsetAdjustmentBehavior = .never } self.webView = webView self.view.addSubview(webView) if let parsedUrl = URL(string: url) { webView.load(URLRequest(url: parsedUrl)) } } func containerLayoutUpdated(_ layout: ContainerViewLayout, navigationBarHeight: CGFloat, transition: ContainedViewLayoutTransition) { self.webView?.frame = CGRect(origin: CGPoint(x: 0.0, y: navigationBarHeight), size: CGSize(width: layout.size.width, height: max(1.0, layout.size.height - navigationBarHeight))) } func animateIn() { self.layer.animatePosition(from: CGPoint(x: self.layer.position.x, y: self.layer.position.y + self.layer.bounds.size.height), to: self.layer.position, duration: 0.5, timingFunction: kCAMediaTimingFunctionSpring) } func animateOut(completion: (() -> Void)? = nil) { self.layer.animatePosition(from: self.layer.position, to: CGPoint(x: self.layer.position.x, y: self.layer.position.y + self.layer.bounds.size.height), duration: 0.2, timingFunction: CAMediaTimingFunctionName.easeInEaseOut.rawValue, removeOnCompletion: false, completion: { _ in completion?() }) } private func handleScriptMessage(_ message: WKScriptMessage) { guard let body = message.body as? [String: Any] else { return } guard let eventName = body["eventName"] as? String else { return } if eventName == "payment_form_submit" { guard let eventString = body["eventData"] as? String else { return } guard let eventData = eventString.data(using: .utf8) else { return } guard let dict = (try? JSONSerialization.jsonObject(with: eventData, options: [])) as? [String: Any] else { return } guard let title = dict["title"] as? String else { return } guard let credentials = dict["credentials"] else { return } guard let credentialsData = try? JSONSerialization.data(withJSONObject: credentials, options: []) else { return } guard let credentialsString = String(data: credentialsData, encoding: .utf8) else { return } if case let .addPaymentMethod(_, completion) = self.intent { completion(BotCheckoutPaymentWebToken(title: title, data: credentialsString, saveOnServer: false)) } } } func webView(_ webView: WKWebView, decidePolicyFor navigationAction: WKNavigationAction, decisionHandler: @escaping (WKNavigationActionPolicy) -> Void) { if case let .externalVerification(completion) = self.intent, let host = navigationAction.request.url?.host { if host == "t.me" || host == "telegram.me" { decisionHandler(.cancel) completion(true) } else { decisionHandler(.allow) } } else { decisionHandler(.allow) } } }