WebxdcViewController.swift 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  1. import UIKit
  2. import WebKit
  3. import DcCore
  4. class WebxdcViewController: WebViewViewController {
  5. enum WebxdcHandler: String {
  6. case log = "log"
  7. case setUpdateListener = "setUpdateListener"
  8. case sendStatusUpdate = "sendStatusUpdateHandler"
  9. }
  10. let INTERNALSCHEMA = "webxdc"
  11. var messageId: Int
  12. var dcContext: DcContext
  13. var webxdcUpdateObserver: NSObjectProtocol?
  14. var webxdcName: String = ""
  15. var sourceCodeUrl: String?
  16. private var shortcutManager: ShortcutManager?
  17. private lazy var moreButton: UIBarButtonItem = {
  18. let image: UIImage?
  19. if #available(iOS 13.0, *) {
  20. image = UIImage(systemName: "ellipsis.circle")
  21. } else {
  22. image = UIImage(named: "ic_more")
  23. }
  24. return UIBarButtonItem(image: image,
  25. style: .plain,
  26. target: self,
  27. action: #selector(moreButtonPressed))
  28. }()
  29. // Block just everything, except of webxdc urls
  30. let blockRules = """
  31. [
  32. {
  33. "trigger": {
  34. "url-filter": ".*"
  35. },
  36. "action": {
  37. "type": "block"
  38. }
  39. },
  40. {
  41. "trigger": {
  42. "url-filter": "webxdc://*"
  43. },
  44. "action": {
  45. "type": "ignore-previous-rules"
  46. }
  47. }
  48. ]
  49. """
  50. lazy var webxdcbridge: String = {
  51. let addr = dcContext.addr?
  52. .addingPercentEncoding(withAllowedCharacters: CharacterSet.urlQueryAllowed)
  53. let displayname = (dcContext.displayname ?? dcContext.addr)?
  54. .addingPercentEncoding(withAllowedCharacters: CharacterSet.urlQueryAllowed)
  55. let script = """
  56. window.webxdc = (() => {
  57. let setUpdateListenerPromise = null
  58. var log = (s)=>webkit.messageHandlers.log.postMessage(s);
  59. var update_listener = () => {};
  60. window.__webxdcUpdate = (updateString) => {
  61. try {
  62. var updates = JSON.parse(updateString);
  63. updates.forEach((update) => {
  64. update_listener(update);
  65. });
  66. } catch (e) {
  67. log("json error: "+ e.message)
  68. } finally {
  69. if (setUpdateListenerPromise) {
  70. setUpdateListenerPromise()
  71. setUpdateListenerPromise = null
  72. }
  73. }
  74. }
  75. return {
  76. selfAddr: decodeURI("\((addr ?? "unknown"))"),
  77. selfName: decodeURI("\((displayname ?? "unknown"))"),
  78. setUpdateListener: (cb, serial) => {
  79. update_listener = cb
  80. const promise = new Promise((res, _rej) => {
  81. setUpdateListenerPromise = res
  82. })
  83. webkit.messageHandlers.setUpdateListener.postMessage(typeof serial === "undefined" ? 0 : parseInt(serial));
  84. return promise
  85. },
  86. getAllUpdates: () => {
  87. console.error("deprecated 2022-02-20 all updates are returned through the callback set by setUpdateListener");
  88. return Promise.resolve([]);
  89. },
  90. sendUpdate: (payload, descr) => {
  91. // only one parameter is allowed, we we create a new parameter object here
  92. var parameter = {
  93. payload: payload,
  94. descr: descr
  95. };
  96. webkit.messageHandlers.sendStatusUpdateHandler.postMessage(parameter);
  97. },
  98. };
  99. })();
  100. """
  101. return script
  102. }()
  103. override var configuration: WKWebViewConfiguration {
  104. let config = WKWebViewConfiguration()
  105. let preferences = WKPreferences()
  106. let contentController = WKUserContentController()
  107. contentController.add(self, name: WebxdcHandler.sendStatusUpdate.rawValue)
  108. contentController.add(self, name: WebxdcHandler.setUpdateListener.rawValue)
  109. contentController.add(self, name: WebxdcHandler.log.rawValue)
  110. config.userContentController = contentController
  111. config.setURLSchemeHandler(self, forURLScheme: INTERNALSCHEMA)
  112. config.mediaTypesRequiringUserActionForPlayback = []
  113. config.allowsInlineMediaPlayback = true
  114. if #available(iOS 13.0, *) {
  115. preferences.isFraudulentWebsiteWarningEnabled = true
  116. }
  117. if #available(iOS 14.0, *) {
  118. config.defaultWebpagePreferences.allowsContentJavaScript = true
  119. } else {
  120. preferences.javaScriptEnabled = true
  121. }
  122. preferences.javaScriptCanOpenWindowsAutomatically = false
  123. config.preferences = preferences
  124. return config
  125. }
  126. init(dcContext: DcContext, messageId: Int) {
  127. self.dcContext = dcContext
  128. self.messageId = messageId
  129. self.shortcutManager = ShortcutManager(dcContext: dcContext, messageId: messageId)
  130. super.init()
  131. }
  132. required init?(coder: NSCoder) {
  133. fatalError("init(coder:) has not been implemented")
  134. }
  135. override func viewDidLoad() {
  136. super.viewDidLoad()
  137. let msg = dcContext.getMessage(id: messageId)
  138. let dict = msg.getWebxdcInfoDict()
  139. let document = dict["document"] as? String ?? ""
  140. webxdcName = dict["name"] as? String ?? "ErrName" // name should not be empty
  141. let chatName = dcContext.getChat(chatId: msg.chatId).name
  142. self.title = document.isEmpty ? "\(webxdcName) – \(chatName)" : "\(document) – \(chatName)"
  143. navigationItem.rightBarButtonItem = moreButton
  144. if let sourceCode = dict["source_code_url"] as? String,
  145. !sourceCode.isEmpty {
  146. sourceCodeUrl = sourceCode
  147. }
  148. }
  149. override func willMove(toParent parent: UIViewController?) {
  150. super.willMove(toParent: parent)
  151. let willBeRemoved = parent == nil
  152. navigationController?.interactivePopGestureRecognizer?.isEnabled = willBeRemoved
  153. if willBeRemoved {
  154. // remove observer
  155. let nc = NotificationCenter.default
  156. if let webxdcUpdateObserver = webxdcUpdateObserver {
  157. nc.removeObserver(webxdcUpdateObserver)
  158. }
  159. shortcutManager = nil
  160. } else {
  161. addObserver()
  162. }
  163. }
  164. private func addObserver() {
  165. let nc = NotificationCenter.default
  166. webxdcUpdateObserver = nc.addObserver(
  167. forName: dcNotificationWebxdcUpdate,
  168. object: nil,
  169. queue: OperationQueue.main
  170. ) { [weak self] notification in
  171. guard let self = self else { return }
  172. guard let ui = notification.userInfo,
  173. let messageId = ui["message_id"] as? Int else {
  174. logger.error("failed to handle dcNotificationWebxdcUpdate")
  175. return
  176. }
  177. if messageId == self.messageId {
  178. self.updateWebxdc()
  179. }
  180. }
  181. }
  182. override func webView(_ webView: WKWebView, decidePolicyFor navigationAction: WKNavigationAction, decisionHandler: @escaping (WKNavigationActionPolicy) -> Void) {
  183. // TODO: what about tel://
  184. if let url = navigationAction.request.url {
  185. if url.scheme == "mailto" {
  186. askToChatWith(url: url)
  187. decisionHandler(.cancel)
  188. return
  189. } else if url.scheme != INTERNALSCHEMA {
  190. logger.debug("cancel loading: \(url)")
  191. decisionHandler(.cancel)
  192. return
  193. }
  194. }
  195. logger.debug("loading: \(String(describing: navigationAction.request.url))")
  196. decisionHandler(.allow)
  197. }
  198. override func viewWillAppear(_ animated: Bool) {
  199. super.viewWillAppear(animated)
  200. loadRestrictedHtml()
  201. }
  202. override func viewDidDisappear(_ animated: Bool) {
  203. super.viewDidDisappear(animated)
  204. if #available(iOS 15.0, *) {
  205. webView.setAllMediaPlaybackSuspended(true)
  206. }
  207. }
  208. private func loadRestrictedHtml() {
  209. // TODO: compile only once
  210. WKContentRuleListStore.default().compileContentRuleList(
  211. forIdentifier: "WebxdcContentBlockingRules",
  212. encodedContentRuleList: blockRules) { (contentRuleList, error) in
  213. guard let contentRuleList = contentRuleList, error == nil else {
  214. return
  215. }
  216. let configuration = self.webView.configuration
  217. configuration.userContentController.add(contentRuleList)
  218. self.loadHtml()
  219. }
  220. }
  221. private func loadHtml() {
  222. DispatchQueue.global(qos: .userInitiated).async { [weak self] in
  223. guard let self = self else { return }
  224. let url = URL(string: "\(self.INTERNALSCHEMA)://acc\(self.dcContext.id)-msg\(self.messageId).localhost/index.html")
  225. let urlRequest = URLRequest(url: url!)
  226. DispatchQueue.main.async {
  227. self.webView.load(urlRequest)
  228. }
  229. }
  230. }
  231. var lastSerial: Int?
  232. private func updateWebxdc() {
  233. if let lastSerial = lastSerial {
  234. let statusUpdates = dcContext.getWebxdcStatusUpdates(msgId: messageId, lastKnownSerial: lastSerial)
  235. if let data: Data = statusUpdates.data(using: .utf8),
  236. let array = (try? JSONSerialization.jsonObject(with: data, options: [])) as? [Any],
  237. let first = array.first as? [String: Any],
  238. let maxSerial = first["max_serial"] as? Int {
  239. self.lastSerial = maxSerial
  240. }
  241. webView.evaluateJavaScript("window.__webxdcUpdate(atob(\"\(statusUpdates.toBase64())\"))", completionHandler: nil)
  242. }
  243. }
  244. @objc private func moreButtonPressed() {
  245. let alert = UIAlertController(title: webxdcName + " – " + String.localized("webxdc_app"),
  246. message: nil,
  247. preferredStyle: .safeActionSheet)
  248. let addToHomescreenAction = UIAlertAction(title: String.localized("add_to_home_screen"), style: .default, handler: addToHomeScreen(_:))
  249. alert.addAction(addToHomescreenAction)
  250. if sourceCodeUrl != nil {
  251. let sourceCodeAction = UIAlertAction(title: String.localized("source_code"), style: .default, handler: openUrl(_:))
  252. alert.addAction(sourceCodeAction)
  253. }
  254. let cancelAction = UIAlertAction(title: String.localized("cancel"), style: .cancel, handler: nil)
  255. alert.addAction(cancelAction)
  256. self.present(alert, animated: true, completion: nil)
  257. }
  258. private func addToHomeScreen(_ action: UIAlertAction) {
  259. shortcutManager?.showShortcutLandingPage()
  260. }
  261. private func openUrl(_ action: UIAlertAction) {
  262. if let sourceCodeUrl = sourceCodeUrl,
  263. let url = URL(string: sourceCodeUrl) {
  264. UIApplication.shared.open(url)
  265. }
  266. }
  267. private func askToChatWith(url: URL) {
  268. guard let appDelegate = UIApplication.shared.delegate as? AppDelegate,
  269. let emailAddress = parseEmailAddress(from: url) else {
  270. return
  271. }
  272. let alert = UIAlertController(title: String.localizedStringWithFormat(String.localized("ask_start_chat_with"), emailAddress),
  273. message: nil,
  274. preferredStyle: .safeActionSheet)
  275. alert.addAction(UIAlertAction(title: String.localized("start_chat"), style: .default, handler: { _ in
  276. RelayHelper.shared.askToChatWithMailto = false
  277. _ = appDelegate.application(UIApplication.shared, open: url)
  278. }))
  279. alert.addAction(UIAlertAction(title: String.localized("cancel"), style: .cancel, handler: nil))
  280. present(alert, animated: true, completion: nil)
  281. }
  282. private func parseEmailAddress(from url: URL) -> String? {
  283. if let urlComponents = URLComponents(url: url, resolvingAgainstBaseURL: false),
  284. !urlComponents.path.isEmpty {
  285. return RelayHelper.shared.splitString(urlComponents.path)[0]
  286. }
  287. return nil
  288. }
  289. }
  290. extension WebxdcViewController: WKScriptMessageHandler {
  291. func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) {
  292. let handler = WebxdcHandler(rawValue: message.name)
  293. switch handler {
  294. case .setUpdateListener:
  295. guard let lastKnownSerial = message.body as? Int else {
  296. logger.error("could not convert param \(message.body) to int")
  297. return
  298. }
  299. lastSerial = lastKnownSerial
  300. updateWebxdc()
  301. case .log:
  302. guard let msg = message.body as? String else {
  303. logger.error("could not convert param \(message.body) to string")
  304. return
  305. }
  306. logger.debug("webxdc log msg: "+msg)
  307. case .sendStatusUpdate:
  308. guard let dict = message.body as? [String: AnyObject],
  309. let payloadDict = dict["payload"] as? [String: AnyObject],
  310. let payloadJson = try? JSONSerialization.data(withJSONObject: payloadDict, options: []),
  311. let payloadString = String(data: payloadJson, encoding: .utf8),
  312. let description = dict["descr"] as? String else {
  313. logger.error("Failed to parse status update parameters \(message.body)")
  314. return
  315. }
  316. _ = dcContext.sendWebxdcStatusUpdate(msgId: messageId, payload: payloadString, description: description)
  317. default:
  318. logger.debug("another method was called")
  319. }
  320. }
  321. }
  322. extension WebxdcViewController: WKURLSchemeHandler {
  323. func webView(_ webView: WKWebView, start urlSchemeTask: WKURLSchemeTask) {
  324. if let url = urlSchemeTask.request.url, let scheme = url.scheme, scheme == INTERNALSCHEMA {
  325. let file = url.path
  326. let dcMsg = dcContext.getMessage(id: messageId)
  327. var data: Data
  328. if url.lastPathComponent == "webxdc.js" {
  329. data = Data(webxdcbridge.utf8)
  330. } else {
  331. data = dcMsg.getWebxdcBlob(filename: file)
  332. }
  333. let mimeType = DcUtils.getMimeTypeForPath(path: file)
  334. let response = URLResponse(url: url, mimeType: mimeType, expectedContentLength: data.count, textEncodingName: nil)
  335. urlSchemeTask.didReceive(response)
  336. urlSchemeTask.didReceive(data)
  337. urlSchemeTask.didFinish()
  338. } else {
  339. logger.debug("not loading \(String(describing: urlSchemeTask.request.url))")
  340. }
  341. }
  342. func webView(_ webView: WKWebView, stop urlSchemeTask: WKURLSchemeTask) {
  343. }
  344. }