mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-06-16 05:55:20 +00:00
185 lines
7.9 KiB
Swift
185 lines
7.9 KiB
Swift
import Foundation
|
|
import UIKit
|
|
import Display
|
|
import AsyncDisplayKit
|
|
import WebKit
|
|
import TelegramCore
|
|
import SwiftSignalKit
|
|
import TelegramPresentationData
|
|
import AccountContext
|
|
import ShareController
|
|
import UndoUI
|
|
|
|
private class WeakGameScriptMessageHandler: 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 GameControllerNode: ViewControllerTracingNode {
|
|
private var webView: WKWebView?
|
|
|
|
private let context: AccountContext
|
|
var presentationData: PresentationData
|
|
private let present: (ViewController, Any?) -> Void
|
|
private let message: EngineMessage?
|
|
|
|
init(context: AccountContext, presentationData: PresentationData, url: String, present: @escaping (ViewController, Any?) -> Void, message: EngineMessage?) {
|
|
self.context = context
|
|
self.presentationData = presentationData
|
|
self.present = present
|
|
self.message = message
|
|
|
|
super.init()
|
|
|
|
self.backgroundColor = .white
|
|
|
|
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(WeakGameScriptMessageHandler { [weak self] message in
|
|
if let strongSelf = self {
|
|
strongSelf.handleScriptMessage(message)
|
|
}
|
|
}, name: "performAction")
|
|
|
|
configuration.userContentController = userController
|
|
|
|
configuration.allowsInlineMediaPlayback = true
|
|
if #available(iOSApplicationExtension 10.0, iOS 10.0, *) {
|
|
configuration.mediaTypesRequiringUserActionForPlayback = []
|
|
} else {
|
|
configuration.mediaPlaybackRequiresUserAction = false
|
|
}
|
|
|
|
let webView = WKWebView(frame: CGRect(), configuration: configuration)
|
|
if #available(iOSApplicationExtension 9.0, iOS 9.0, *) {
|
|
webView.allowsLinkPreview = false
|
|
}
|
|
if #available(iOSApplicationExtension 11.0, iOS 11.0, *) {
|
|
webView.scrollView.contentInsetAdjustmentBehavior = .never
|
|
}
|
|
webView.interactiveTransitionGestureRecognizerTest = { point -> Bool in
|
|
return point.x > 30.0
|
|
}
|
|
|
|
self.view.addSubview(webView)
|
|
self.webView = webView
|
|
|
|
if let parsedUrl = URL(string: url) {
|
|
webView.load(URLRequest(url: parsedUrl))
|
|
}
|
|
}
|
|
|
|
func containerLayoutUpdated(_ layout: ContainerViewLayout, navigationBarHeight: CGFloat, transition: ContainedViewLayoutTransition) {
|
|
if let webView = self.webView {
|
|
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 shareData() -> (EnginePeer, String)? {
|
|
guard let message = self.message else {
|
|
return nil
|
|
}
|
|
var botPeer: EnginePeer?
|
|
var gameName: String?
|
|
for media in message.media {
|
|
if let game = media as? TelegramMediaGame {
|
|
inner: for attribute in message.attributes {
|
|
if let attribute = attribute as? InlineBotMessageAttribute, let peerId = attribute.peerId {
|
|
botPeer = message.peers[peerId].flatMap(EnginePeer.init)
|
|
break inner
|
|
}
|
|
}
|
|
if botPeer == nil {
|
|
botPeer = message.author
|
|
}
|
|
|
|
gameName = game.name
|
|
}
|
|
}
|
|
if let botPeer = botPeer, let gameName = gameName {
|
|
return (botPeer, gameName)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
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 == "share_game" || eventName == "share_score" {
|
|
if let (botPeer, gameName) = self.shareData(), let addressName = botPeer.addressName, !addressName.isEmpty, !gameName.isEmpty {
|
|
if eventName == "share_score" {
|
|
self.present(ShareController(context: self.context, subject: .fromExternal(1, { [weak self] peerIds, threadIds, requireStars, text, account, _ in
|
|
if let strongSelf = self, let message = strongSelf.message, let account = account as? ShareControllerAppAccountContext {
|
|
let signals = peerIds.map { TelegramEngine(account: account.context.account).messages.forwardGameWithScore(messageId: message.id, to: $0, threadId: threadIds[$0], as: nil) }
|
|
return .single(.preparing(false))
|
|
|> castError(ShareControllerError.self)
|
|
|> then(
|
|
combineLatest(signals)
|
|
|> castError(ShareControllerError.self)
|
|
|> mapToSignal { _ -> Signal<ShareControllerExternalStatus, ShareControllerError> in return .complete() }
|
|
)
|
|
|> then(.single(.done))
|
|
} else {
|
|
return .single(.done)
|
|
}
|
|
}), showInChat: nil, externalShare: false, immediateExternalShare: false), nil)
|
|
} else {
|
|
self.shareWithoutScore()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func shareWithoutScore() {
|
|
if let (botPeer, gameName) = self.shareData(), let addressName = botPeer.addressName, !addressName.isEmpty, !gameName.isEmpty {
|
|
let url = "https://t.me/\(addressName)?game=\(gameName)"
|
|
|
|
let context = self.context
|
|
let shareController = ShareController(context: context, subject: .url(url), showInChat: nil, externalShare: true)
|
|
shareController.actionCompleted = { [weak self] in
|
|
if let strongSelf = self {
|
|
let presentationData = context.sharedContext.currentPresentationData.with { $0 }
|
|
strongSelf.present(UndoOverlayController(presentationData: presentationData, content: .linkCopied(title: nil, text: presentationData.strings.Conversation_LinkCopied), elevatedLayout: false, animateInAsReplacement: false, action: { _ in return false }), nil)
|
|
}
|
|
}
|
|
self.present(shareController, nil)
|
|
}
|
|
}
|
|
}
|