2018-09-18 01:07:12 +00:00
|
|
|
//
|
|
|
|
// InAppBrowserWebViewController.swift
|
|
|
|
// flutter_inappbrowser
|
|
|
|
//
|
|
|
|
// Created by Lorenzo on 17/09/18.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Flutter
|
|
|
|
import UIKit
|
|
|
|
import WebKit
|
|
|
|
import Foundation
|
|
|
|
import AVFoundation
|
|
|
|
|
|
|
|
typealias OlderClosureType = @convention(c) (Any, Selector, UnsafeRawPointer, Bool, Bool, Any?) -> Void
|
|
|
|
typealias NewerClosureType = @convention(c) (Any, Selector, UnsafeRawPointer, Bool, Bool, Bool, Any?) -> Void
|
|
|
|
|
2018-10-09 23:52:27 +00:00
|
|
|
// the message needs to be concatenated with '' in order to have the same behavior like on Android
|
2018-10-13 19:12:32 +00:00
|
|
|
let consoleLogJS = """
|
2018-10-09 23:52:27 +00:00
|
|
|
(function() {
|
|
|
|
var oldLogs = {
|
|
|
|
'consoleLog': console.log,
|
|
|
|
'consoleDebug': console.debug,
|
|
|
|
'consoleError': console.error,
|
|
|
|
'consoleInfo': console.info,
|
|
|
|
'consoleWarn': console.warn
|
|
|
|
};
|
|
|
|
|
|
|
|
for (var k in oldLogs) {
|
|
|
|
(function(oldLog) {
|
|
|
|
console[oldLog.replace('console', '').toLowerCase()] = function() {
|
|
|
|
var message = '';
|
|
|
|
for (var i in arguments) {
|
|
|
|
if (message == '') {
|
|
|
|
message += arguments[i];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
message += ' ' + arguments[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
window.webkit.messageHandlers[oldLog].postMessage(message);
|
|
|
|
}
|
|
|
|
})(k);
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
"""
|
|
|
|
|
2018-10-13 19:12:32 +00:00
|
|
|
let resourceObserverJS = """
|
|
|
|
(function() {
|
|
|
|
var observer = new PerformanceObserver(function(list) {
|
|
|
|
list.getEntries().forEach(function(entry) {
|
|
|
|
window.webkit.messageHandlers['resourceLoaded'].postMessage(JSON.stringify(entry));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
observer.observe({entryTypes: ['resource', 'mark', 'measure']});
|
|
|
|
})();
|
|
|
|
"""
|
|
|
|
|
|
|
|
let JAVASCRIPT_BRIDGE_NAME = "flutter_inappbrowser"
|
|
|
|
|
|
|
|
let javaScriptBridgeJS = """
|
|
|
|
window.\(JAVASCRIPT_BRIDGE_NAME) = {};
|
|
|
|
window.\(JAVASCRIPT_BRIDGE_NAME).callHandler = function(handlerName, ...args) {
|
|
|
|
window.webkit.messageHandlers['callHandler'].postMessage( {'handlerName': handlerName, 'args': JSON.stringify(args)} );
|
|
|
|
}
|
|
|
|
"""
|
|
|
|
|
|
|
|
func currentTimeInMilliSeconds() -> Int {
|
|
|
|
let currentDate = Date()
|
|
|
|
let since1970 = currentDate.timeIntervalSince1970
|
|
|
|
return Int(since1970 * 1000)
|
|
|
|
}
|
|
|
|
|
|
|
|
func convertToDictionary(text: String) -> [String: Any]? {
|
|
|
|
if let data = text.data(using: .utf8) {
|
|
|
|
do {
|
|
|
|
return try JSONSerialization.jsonObject(with: data, options: []) as? [String: Any]
|
|
|
|
} catch {
|
|
|
|
print(error.localizedDescription)
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
}
|
2018-10-13 19:12:32 +00:00
|
|
|
return nil
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
2018-10-13 19:12:32 +00:00
|
|
|
|
|
|
|
//extension WKWebView{
|
|
|
|
//
|
|
|
|
// var keyboardDisplayRequiresUserAction: Bool? {
|
|
|
|
// get {
|
|
|
|
// return self.keyboardDisplayRequiresUserAction
|
|
|
|
// }
|
|
|
|
// set {
|
|
|
|
// self.setKeyboardRequiresUserInteraction(newValue ?? true)
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// func setKeyboardRequiresUserInteraction( _ value: Bool) {
|
|
|
|
//
|
|
|
|
// guard
|
|
|
|
// let WKContentViewClass: AnyClass = NSClassFromString("WKContentView") else {
|
|
|
|
// print("Cannot find the WKContentView class")
|
|
|
|
// return
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// let olderSelector: Selector = sel_getUid("_startAssistingNode:userIsInteracting:blurPreviousNode:userObject:")
|
|
|
|
// let newerSelector: Selector = sel_getUid("_startAssistingNode:userIsInteracting:blurPreviousNode:changingActivityState:userObject:")
|
|
|
|
//
|
|
|
|
// if let method = class_getInstanceMethod(WKContentViewClass, olderSelector) {
|
|
|
|
//
|
|
|
|
// let originalImp: IMP = method_getImplementation(method)
|
|
|
|
// let original: OlderClosureType = unsafeBitCast(originalImp, to: OlderClosureType.self)
|
|
|
|
// let block : @convention(block) (Any, UnsafeRawPointer, Bool, Bool, Any?) -> Void = { (me, arg0, arg1, arg2, arg3) in
|
|
|
|
// original(me, olderSelector, arg0, !value, arg2, arg3)
|
|
|
|
// }
|
|
|
|
// let imp: IMP = imp_implementationWithBlock(block)
|
|
|
|
// method_setImplementation(method, imp)
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// if let method = class_getInstanceMethod(WKContentViewClass, newerSelector) {
|
|
|
|
//
|
|
|
|
// let originalImp: IMP = method_getImplementation(method)
|
|
|
|
// let original: NewerClosureType = unsafeBitCast(originalImp, to: NewerClosureType.self)
|
|
|
|
// let block : @convention(block) (Any, UnsafeRawPointer, Bool, Bool, Bool, Any?) -> Void = { (me, arg0, arg1, arg2, arg3, arg4) in
|
|
|
|
// original(me, newerSelector, arg0, !value, arg2, arg3, arg4)
|
|
|
|
// }
|
|
|
|
// let imp: IMP = imp_implementationWithBlock(block)
|
|
|
|
// method_setImplementation(method, imp)
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
//}
|
|
|
|
|
2018-10-12 01:46:33 +00:00
|
|
|
class WKWebView_IBWrapper: WKWebView {
|
|
|
|
required convenience init?(coder: NSCoder) {
|
|
|
|
let config = WKWebViewConfiguration()
|
|
|
|
self.init(frame: .zero, configuration: config)
|
|
|
|
self.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-13 19:12:32 +00:00
|
|
|
class InAppBrowserWebViewController: UIViewController, WKUIDelegate, WKNavigationDelegate, UITextFieldDelegate, WKScriptMessageHandler {
|
2018-10-12 01:46:33 +00:00
|
|
|
@IBOutlet var webView: WKWebView_IBWrapper!
|
2018-09-18 01:07:12 +00:00
|
|
|
@IBOutlet var closeButton: UIButton!
|
|
|
|
@IBOutlet var reloadButton: UIBarButtonItem!
|
|
|
|
@IBOutlet var backButton: UIBarButtonItem!
|
|
|
|
@IBOutlet var forwardButton: UIBarButtonItem!
|
|
|
|
@IBOutlet var shareButton: UIBarButtonItem!
|
|
|
|
@IBOutlet var spinner: UIActivityIndicatorView!
|
|
|
|
@IBOutlet var toolbarTop: UIView!
|
|
|
|
@IBOutlet var toolbarBottom: UIToolbar!
|
|
|
|
@IBOutlet var urlField: UITextField!
|
|
|
|
|
|
|
|
weak var navigationDelegate: SwiftFlutterPlugin?
|
|
|
|
var currentURL: URL?
|
|
|
|
var tmpWindow: UIWindow?
|
|
|
|
var browserOptions: InAppBrowserOptions?
|
2018-09-20 00:48:24 +00:00
|
|
|
var initHeaders: [String: String]?
|
2018-09-23 16:35:32 +00:00
|
|
|
var isHidden = false
|
2018-09-30 19:52:56 +00:00
|
|
|
var uuid: String = ""
|
2018-10-13 19:12:32 +00:00
|
|
|
var WKNavigationMap: [String: [String: Any]] = [:]
|
|
|
|
var startPageTime = 0
|
2018-09-18 01:07:12 +00:00
|
|
|
|
|
|
|
required init(coder aDecoder: NSCoder) {
|
|
|
|
super.init(coder: aDecoder)!
|
|
|
|
}
|
|
|
|
|
|
|
|
override func viewWillAppear(_ animated: Bool) {
|
|
|
|
prepareWebView()
|
2018-09-26 00:56:56 +00:00
|
|
|
super.viewWillAppear(animated)
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
override func viewDidLoad() {
|
2018-10-12 01:46:33 +00:00
|
|
|
super.viewDidLoad()
|
2018-10-13 01:16:08 +00:00
|
|
|
|
2018-10-13 19:12:32 +00:00
|
|
|
//MyURLProtocol.wkWebViewDelegateMap[uuid] = self
|
2018-10-12 01:46:33 +00:00
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
webView.uiDelegate = self
|
|
|
|
webView.navigationDelegate = self
|
|
|
|
|
|
|
|
urlField.delegate = self
|
|
|
|
urlField.text = self.currentURL?.absoluteString
|
|
|
|
|
|
|
|
closeButton.addTarget(self, action: #selector(self.close), for: .touchUpInside)
|
|
|
|
|
|
|
|
forwardButton.target = self
|
|
|
|
forwardButton.action = #selector(self.goForward)
|
|
|
|
|
|
|
|
forwardButton.target = self
|
|
|
|
forwardButton.action = #selector(self.goForward)
|
|
|
|
|
|
|
|
backButton.target = self
|
|
|
|
backButton.action = #selector(self.goBack)
|
|
|
|
|
|
|
|
reloadButton.target = self
|
|
|
|
reloadButton.action = #selector(self.reload)
|
|
|
|
|
|
|
|
shareButton.target = self
|
|
|
|
shareButton.action = #selector(self.share)
|
|
|
|
|
|
|
|
spinner.hidesWhenStopped = true
|
|
|
|
spinner.isHidden = false
|
|
|
|
spinner.stopAnimating()
|
|
|
|
|
2018-09-20 00:48:24 +00:00
|
|
|
loadUrl(url: self.currentURL!, headers: self.initHeaders)
|
2018-10-13 19:12:32 +00:00
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Prevent crashes on closing windows
|
|
|
|
deinit {
|
|
|
|
webView?.uiDelegate = nil
|
|
|
|
}
|
|
|
|
|
|
|
|
override func viewWillDisappear (_ animated: Bool) {
|
|
|
|
super.viewWillDisappear(animated)
|
|
|
|
}
|
|
|
|
|
|
|
|
func prepareWebView() {
|
2018-09-26 00:56:56 +00:00
|
|
|
//UIApplication.shared.statusBarStyle = preferredStatusBarStyle
|
|
|
|
|
2018-09-19 02:10:00 +00:00
|
|
|
self.webView.configuration.userContentController = WKUserContentController()
|
|
|
|
self.webView.configuration.preferences = WKPreferences()
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
if (browserOptions?.hideUrlBar)! {
|
|
|
|
self.urlField.isHidden = true
|
|
|
|
self.urlField.isEnabled = false
|
|
|
|
}
|
|
|
|
|
|
|
|
if (browserOptions?.toolbarTop)! {
|
2018-09-21 01:41:04 +00:00
|
|
|
if browserOptions?.toolbarTopBackgroundColor != "" {
|
|
|
|
self.toolbarTop.backgroundColor = color(fromHexString: (browserOptions?.toolbarTopBackgroundColor)!)
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
self.toolbarTop.removeFromSuperview()
|
|
|
|
self.webView.bounds.size.height += self.toolbarTop.bounds.height
|
|
|
|
|
|
|
|
if #available(iOS 9.0, *) {
|
|
|
|
let topConstraint = webView.topAnchor.constraint(equalTo: self.view.topAnchor, constant: CGFloat(getStatusBarOffset()))
|
|
|
|
NSLayoutConstraint.activate([topConstraint])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (browserOptions?.toolbarBottom)! {
|
2018-09-21 01:41:04 +00:00
|
|
|
if browserOptions?.toolbarBottomBackgroundColor != "" {
|
|
|
|
self.toolbarBottom.backgroundColor = color(fromHexString: (browserOptions?.toolbarBottomBackgroundColor)!)
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
self.toolbarBottom.isTranslucent = (browserOptions?.toolbarBottomTranslucent)!
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
self.toolbarBottom.removeFromSuperview()
|
|
|
|
self.webView.bounds.size.height += self.toolbarBottom.bounds.height
|
|
|
|
|
|
|
|
if #available(iOS 9.0, *) {
|
|
|
|
let bottomConstraint = webView.bottomAnchor.constraint(equalTo: self.view.bottomAnchor)
|
|
|
|
NSLayoutConstraint.activate([bottomConstraint])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if browserOptions?.closeButtonCaption != "" {
|
|
|
|
closeButton.setTitle(browserOptions?.closeButtonCaption, for: .normal)
|
|
|
|
}
|
|
|
|
if browserOptions?.closeButtonColor != "" {
|
|
|
|
closeButton.tintColor = color(fromHexString: (browserOptions?.closeButtonColor)!)
|
|
|
|
}
|
|
|
|
|
|
|
|
self.modalPresentationStyle = UIModalPresentationStyle(rawValue: (browserOptions?.presentationStyle)!)!
|
|
|
|
self.modalTransitionStyle = UIModalTransitionStyle(rawValue: (browserOptions?.transitionStyle)!)!
|
|
|
|
|
|
|
|
// prevent webView from bouncing
|
|
|
|
if (browserOptions?.disallowOverScroll)! {
|
|
|
|
if self.webView.responds(to: #selector(getter: self.webView.scrollView)) {
|
|
|
|
self.webView.scrollView.bounces = false
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
for subview: UIView in self.webView.subviews {
|
|
|
|
if subview is UIScrollView {
|
|
|
|
(subview as! UIScrollView).bounces = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (browserOptions?.enableViewportScale)! {
|
|
|
|
let jscript = "var meta = document.createElement('meta'); meta.setAttribute('name', 'viewport'); meta.setAttribute('content', 'width=device-width'); document.getElementsByTagName('head')[0].appendChild(meta);"
|
|
|
|
let userScript = WKUserScript(source: jscript, injectionTime: .atDocumentEnd, forMainFrameOnly: true)
|
|
|
|
self.webView.configuration.userContentController.addUserScript(userScript)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Prevents long press on links that cause WKWebView exit
|
|
|
|
let jscriptWebkitTouchCallout = WKUserScript(source: "document.body.style.webkitTouchCallout='none';", injectionTime: .atDocumentEnd, forMainFrameOnly: true)
|
|
|
|
self.webView.configuration.userContentController.addUserScript(jscriptWebkitTouchCallout)
|
|
|
|
|
2018-10-13 19:12:32 +00:00
|
|
|
|
|
|
|
let consoleLogJSScript = WKUserScript(source: consoleLogJS, injectionTime: .atDocumentStart, forMainFrameOnly: false)
|
|
|
|
self.webView.configuration.userContentController.addUserScript(consoleLogJSScript)
|
2018-10-09 23:52:27 +00:00
|
|
|
self.webView.configuration.userContentController.add(self, name: "consoleLog")
|
|
|
|
self.webView.configuration.userContentController.add(self, name: "consoleDebug")
|
|
|
|
self.webView.configuration.userContentController.add(self, name: "consoleError")
|
|
|
|
self.webView.configuration.userContentController.add(self, name: "consoleInfo")
|
|
|
|
self.webView.configuration.userContentController.add(self, name: "consoleWarn")
|
2018-09-23 23:53:22 +00:00
|
|
|
|
2018-10-13 19:12:32 +00:00
|
|
|
let javaScriptBridgeJSScript = WKUserScript(source: javaScriptBridgeJS, injectionTime: .atDocumentStart, forMainFrameOnly: false)
|
|
|
|
self.webView.configuration.userContentController.addUserScript(javaScriptBridgeJSScript)
|
|
|
|
self.webView.configuration.userContentController.add(self, name: "callHandler")
|
|
|
|
|
|
|
|
if (browserOptions?.useOnLoadResource)! {
|
|
|
|
let resourceObserverJSScript = WKUserScript(source: resourceObserverJS, injectionTime: .atDocumentStart, forMainFrameOnly: false)
|
|
|
|
self.webView.configuration.userContentController.addUserScript(resourceObserverJSScript)
|
|
|
|
self.webView.configuration.userContentController.add(self, name: "resourceLoaded")
|
|
|
|
}
|
|
|
|
|
2018-09-23 23:53:22 +00:00
|
|
|
if #available(iOS 10.0, *) {
|
|
|
|
if (browserOptions?.mediaPlaybackRequiresUserGesture)! {
|
|
|
|
self.webView.configuration.mediaTypesRequiringUserActionForPlayback = .all
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
2018-09-23 23:53:22 +00:00
|
|
|
else {
|
|
|
|
self.webView.configuration.mediaTypesRequiringUserActionForPlayback = []
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Fallback on earlier versions
|
|
|
|
self.webView.configuration.mediaPlaybackRequiresUserAction = (browserOptions?.mediaPlaybackRequiresUserGesture)!
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
2018-09-23 23:53:22 +00:00
|
|
|
|
2018-09-19 02:10:00 +00:00
|
|
|
self.webView.configuration.allowsInlineMediaPlayback = (browserOptions?.allowsInlineMediaPlayback)!
|
2018-10-12 01:46:33 +00:00
|
|
|
|
|
|
|
//self.webView.keyboardDisplayRequiresUserAction = browserOptions?.keyboardDisplayRequiresUserAction
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
self.webView.configuration.suppressesIncrementalRendering = (browserOptions?.suppressesIncrementalRendering)!
|
2018-09-19 02:10:00 +00:00
|
|
|
self.webView.allowsBackForwardNavigationGestures = (browserOptions?.allowsBackForwardNavigationGestures)!
|
|
|
|
if #available(iOS 9.0, *) {
|
|
|
|
self.webView.allowsLinkPreview = (browserOptions?.allowsLinkPreview)!
|
|
|
|
} else {
|
|
|
|
// Fallback on earlier versions
|
|
|
|
}
|
|
|
|
if #available(iOS 10.0, *) {
|
|
|
|
self.webView.configuration.ignoresViewportScaleLimits = (browserOptions?.ignoresViewportScaleLimits)!
|
|
|
|
} else {
|
|
|
|
// Fallback on earlier versions
|
|
|
|
}
|
|
|
|
self.webView.configuration.allowsInlineMediaPlayback = (browserOptions?.allowsInlineMediaPlayback)!
|
|
|
|
if #available(iOS 9.0, *) {
|
|
|
|
self.webView.configuration.allowsPictureInPictureMediaPlayback = (browserOptions?.allowsPictureInPictureMediaPlayback)!
|
|
|
|
} else {
|
|
|
|
// Fallback on earlier versions
|
|
|
|
}
|
|
|
|
self.webView.configuration.preferences.javaScriptCanOpenWindowsAutomatically = (browserOptions?.javaScriptCanOpenWindowsAutomatically)!
|
|
|
|
self.webView.configuration.preferences.javaScriptEnabled = (browserOptions?.javaScriptEnabled)!
|
2018-09-20 00:48:24 +00:00
|
|
|
|
|
|
|
if ((browserOptions?.userAgent)! != "") {
|
|
|
|
if #available(iOS 9.0, *) {
|
|
|
|
self.webView.customUserAgent = (browserOptions?.userAgent)!
|
2018-10-13 01:16:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-20 00:48:24 +00:00
|
|
|
if (browserOptions?.clearCache)! {
|
|
|
|
clearCache()
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func loadUrl(url: URL, headers: [String: String]?) {
|
|
|
|
var request = URLRequest(url: url)
|
|
|
|
currentURL = url
|
|
|
|
updateUrlTextField(url: (currentURL?.absoluteString)!)
|
|
|
|
|
|
|
|
if headers != nil {
|
|
|
|
for (key, value) in headers! {
|
|
|
|
request.setValue(value, forHTTPHeaderField: key)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
webView.load(request)
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Load user requested url
|
|
|
|
func textFieldShouldReturn(_ textField: UITextField) -> Bool {
|
|
|
|
textField.resignFirstResponder()
|
|
|
|
if textField.text != nil && textField.text != "" {
|
|
|
|
let url = textField.text?.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed)
|
|
|
|
let request = URLRequest(url: URL(string: url!)!)
|
|
|
|
webView.load(request)
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
updateUrlTextField(url: (currentURL?.absoluteString)!)
|
|
|
|
}
|
|
|
|
//var list : WKBackForwardList = self.webView.backForwardList
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func setWebViewFrame(_ frame: CGRect) {
|
|
|
|
print("Setting the WebView's frame to \(NSStringFromCGRect(frame))")
|
|
|
|
webView.frame = frame
|
|
|
|
}
|
|
|
|
|
2018-09-20 00:48:24 +00:00
|
|
|
func clearCache() {
|
|
|
|
if #available(iOS 9.0, *) {
|
|
|
|
//let websiteDataTypes = NSSet(array: [WKWebsiteDataTypeDiskCache, WKWebsiteDataTypeMemoryCache])
|
|
|
|
let date = NSDate(timeIntervalSince1970: 0)
|
|
|
|
WKWebsiteDataStore.default().removeData(ofTypes: WKWebsiteDataStore.allWebsiteDataTypes(), modifiedSince: date as Date, completionHandler:{ })
|
|
|
|
} else {
|
|
|
|
var libraryPath = NSSearchPathForDirectoriesInDomains(FileManager.SearchPathDirectory.libraryDirectory, FileManager.SearchPathDomainMask.userDomainMask, false).first!
|
|
|
|
libraryPath += "/Cookies"
|
|
|
|
|
|
|
|
do {
|
|
|
|
try FileManager.default.removeItem(atPath: libraryPath)
|
|
|
|
} catch {
|
|
|
|
print("can't clear cache")
|
|
|
|
}
|
|
|
|
URLCache.shared.removeAllCachedResponses()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
@objc func reload () {
|
|
|
|
webView.reload()
|
|
|
|
}
|
|
|
|
|
|
|
|
@objc func share () {
|
|
|
|
let vc = UIActivityViewController(activityItems: [currentURL ?? ""], applicationActivities: [])
|
|
|
|
present(vc, animated: true, completion: nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
@objc func close() {
|
|
|
|
currentURL = nil
|
|
|
|
|
|
|
|
if (navigationDelegate != nil) {
|
2018-09-30 19:52:56 +00:00
|
|
|
navigationDelegate?.browserExit(uuid: self.uuid)
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
2018-09-19 02:10:00 +00:00
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
weak var weakSelf = self
|
|
|
|
|
|
|
|
// Run later to avoid the "took a long time" log message.
|
|
|
|
DispatchQueue.main.async(execute: {() -> Void in
|
|
|
|
if (weakSelf?.responds(to: #selector(getter: self.presentingViewController)))! {
|
|
|
|
weakSelf?.presentingViewController?.dismiss(animated: true, completion: {() -> Void in
|
|
|
|
self.tmpWindow?.windowLevel = 0.0
|
|
|
|
UIApplication.shared.delegate?.window??.makeKeyAndVisible()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
weakSelf?.parent?.dismiss(animated: true, completion: {() -> Void in
|
|
|
|
self.tmpWindow?.windowLevel = 0.0
|
|
|
|
UIApplication.shared.delegate?.window??.makeKeyAndVisible()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-09-20 00:48:24 +00:00
|
|
|
func canGoBack() -> Bool {
|
|
|
|
return webView.canGoBack
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
2018-09-19 02:10:00 +00:00
|
|
|
@objc func goBack() {
|
2018-09-20 00:48:24 +00:00
|
|
|
if canGoBack() {
|
2018-09-19 02:10:00 +00:00
|
|
|
webView.goBack()
|
|
|
|
updateUrlTextField(url: (webView?.url?.absoluteString)!)
|
|
|
|
}
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
2018-09-20 00:48:24 +00:00
|
|
|
func canGoForward() -> Bool {
|
|
|
|
return webView.canGoForward
|
|
|
|
}
|
|
|
|
|
2018-09-19 02:10:00 +00:00
|
|
|
@objc func goForward() {
|
2018-09-20 00:48:24 +00:00
|
|
|
if canGoForward() {
|
2018-09-19 02:10:00 +00:00
|
|
|
webView.goForward()
|
|
|
|
updateUrlTextField(url: (webView?.url?.absoluteString)!)
|
|
|
|
}
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func updateUrlTextField(url: String) {
|
|
|
|
urlField.text = url
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// On iOS 7 the status bar is part of the view's dimensions, therefore it's height has to be taken into account.
|
|
|
|
// The height of it could be hardcoded as 20 pixels, but that would assume that the upcoming releases of iOS won't
|
|
|
|
// change that value.
|
|
|
|
//
|
|
|
|
|
|
|
|
func getStatusBarOffset() -> Float {
|
|
|
|
let statusBarFrame: CGRect = UIApplication.shared.statusBarFrame
|
|
|
|
let statusBarOffset: Float = Float(min(statusBarFrame.size.width, statusBarFrame.size.height))
|
|
|
|
return statusBarOffset
|
|
|
|
}
|
|
|
|
|
|
|
|
// Helper function to convert hex color string to UIColor
|
|
|
|
// Assumes input like "#00FF00" (#RRGGBB).
|
|
|
|
// Taken from https://stackoverflow.com/questions/1560081/how-can-i-create-a-uicolor-from-a-hex-string
|
|
|
|
|
|
|
|
func color(fromHexString: String, alpha:CGFloat? = 1.0) -> UIColor {
|
|
|
|
|
|
|
|
// Convert hex string to an integer
|
|
|
|
let hexint = Int(self.intFromHexString(hexStr: fromHexString))
|
|
|
|
let red = CGFloat((hexint & 0xff0000) >> 16) / 255.0
|
|
|
|
let green = CGFloat((hexint & 0xff00) >> 8) / 255.0
|
|
|
|
let blue = CGFloat((hexint & 0xff) >> 0) / 255.0
|
|
|
|
let alpha = alpha!
|
|
|
|
|
|
|
|
// Create color object, specifying alpha as well
|
|
|
|
let color = UIColor(red: red, green: green, blue: blue, alpha: alpha)
|
|
|
|
return color
|
|
|
|
}
|
|
|
|
|
|
|
|
func intFromHexString(hexStr: String) -> UInt32 {
|
|
|
|
var hexInt: UInt32 = 0
|
|
|
|
// Create scanner
|
|
|
|
let scanner: Scanner = Scanner(string: hexStr)
|
|
|
|
// Tell scanner to skip the # character
|
|
|
|
scanner.charactersToBeSkipped = CharacterSet(charactersIn: "#")
|
|
|
|
// Scan hex value
|
|
|
|
scanner.scanHexInt32(&hexInt)
|
|
|
|
return hexInt
|
|
|
|
}
|
|
|
|
|
2018-09-19 02:10:00 +00:00
|
|
|
func webView(_ webView: WKWebView,
|
|
|
|
decidePolicyFor navigationAction: WKNavigationAction,
|
|
|
|
decisionHandler: @escaping (WKNavigationActionPolicy) -> Void) {
|
|
|
|
|
2018-10-13 19:12:32 +00:00
|
|
|
if let url = navigationAction.request.url {
|
|
|
|
|
|
|
|
if url.absoluteString != self.currentURL?.absoluteString && (browserOptions?.useOnLoadResource)! {
|
|
|
|
WKNavigationMap[url.absoluteString] = [
|
|
|
|
"startTime": currentTimeInMilliSeconds(),
|
|
|
|
"request": navigationAction.request
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
|
|
|
if navigationAction.navigationType == .linkActivated && (browserOptions?.useShouldOverrideUrlLoading)! {
|
|
|
|
navigationDelegate?.shouldOverrideUrlLoading(uuid: self.uuid, webView: webView, url: url)
|
|
|
|
decisionHandler(.cancel)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if navigationAction.navigationType == .linkActivated || navigationAction.navigationType == .backForward {
|
|
|
|
currentURL = url
|
|
|
|
updateUrlTextField(url: (url.absoluteString))
|
|
|
|
}
|
2018-09-23 19:38:31 +00:00
|
|
|
}
|
|
|
|
|
2018-09-19 02:10:00 +00:00
|
|
|
|
|
|
|
decisionHandler(.allow)
|
|
|
|
}
|
|
|
|
|
2018-10-12 01:46:33 +00:00
|
|
|
func webView(_ webView: WKWebView,
|
|
|
|
decidePolicyFor navigationResponse: WKNavigationResponse,
|
|
|
|
decisionHandler: @escaping (WKNavigationResponsePolicy) -> Void) {
|
2018-10-13 19:12:32 +00:00
|
|
|
|
|
|
|
if (browserOptions?.useOnLoadResource)! {
|
|
|
|
if let url = navigationResponse.response.url {
|
|
|
|
if WKNavigationMap[url.absoluteString] != nil {
|
|
|
|
let startResourceTime = (WKNavigationMap[url.absoluteString]!["startTime"] as! Int)
|
|
|
|
let startTime = startResourceTime - startPageTime;
|
|
|
|
let duration = currentTimeInMilliSeconds() - startResourceTime;
|
|
|
|
self.didReceiveResourceResponse(navigationResponse.response, fromRequest: WKNavigationMap[url.absoluteString]!["request"] as? URLRequest, withData: Data(), startTime: startTime, duration: duration)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-12 01:46:33 +00:00
|
|
|
decisionHandler(.allow)
|
|
|
|
}
|
|
|
|
|
2018-09-22 11:51:07 +00:00
|
|
|
// func webView(_ webView: WKWebView,
|
|
|
|
// decidePolicyFor navigationResponse: WKNavigationResponse,
|
|
|
|
// decisionHandler: @escaping (WKNavigationResponsePolicy) -> Void) {
|
|
|
|
// let mimeType = navigationResponse.response.mimeType
|
|
|
|
// if mimeType != nil && !mimeType!.starts(with: "text/") {
|
|
|
|
// download(url: webView.url)
|
|
|
|
// decisionHandler(.cancel)
|
|
|
|
// return
|
|
|
|
// }
|
|
|
|
// decisionHandler(.allow)
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// func download (url: URL?) {
|
|
|
|
// let filename = url?.lastPathComponent
|
|
|
|
//
|
|
|
|
// let destination: DownloadRequest.DownloadFileDestination = { _, _ in
|
|
|
|
// let documentsURL = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask)[0]
|
|
|
|
// let fileURL = documentsURL.appendingPathComponent(filename!)
|
|
|
|
//
|
|
|
|
// return (fileURL, [.removePreviousFile, .createIntermediateDirectories])
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// Alamofire.download((url?.absoluteString)!, to: destination).downloadProgress { progress in
|
|
|
|
// print("Download Progress: \(progress.fractionCompleted)")
|
|
|
|
// }.response { response in
|
|
|
|
// if response.error == nil, let path = response.destinationURL?.path {
|
|
|
|
// UIAlertView(title: nil, message: "File saved to " + path, delegate: nil, cancelButtonTitle: nil).show()
|
|
|
|
// }
|
|
|
|
// else {
|
|
|
|
// UIAlertView(title: nil, message: "Cannot save " + filename!, delegate: nil, cancelButtonTitle: nil).show()
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
func webView(_ webView: WKWebView, didStartProvisionalNavigation navigation: WKNavigation!) {
|
2018-10-13 19:12:32 +00:00
|
|
|
|
|
|
|
self.startPageTime = currentTimeInMilliSeconds()
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
// loading url, start spinner, update back/forward
|
|
|
|
backButton.isEnabled = webView.canGoBack
|
|
|
|
forwardButton.isEnabled = webView.canGoForward
|
2018-09-19 02:10:00 +00:00
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
if (browserOptions?.spinner)! {
|
|
|
|
spinner.startAnimating()
|
|
|
|
}
|
|
|
|
|
2018-09-30 19:52:56 +00:00
|
|
|
return (navigationDelegate?.onLoadStart(uuid: self.uuid, webView: webView))!
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func webView(_ webView: WKWebView, didFinish navigation: WKNavigation!) {
|
2018-10-13 19:12:32 +00:00
|
|
|
self.WKNavigationMap = [:]
|
2018-09-18 01:07:12 +00:00
|
|
|
// update url, stop spinner, update back/forward
|
|
|
|
currentURL = webView.url
|
|
|
|
updateUrlTextField(url: (currentURL?.absoluteString)!)
|
|
|
|
backButton.isEnabled = webView.canGoBack
|
|
|
|
forwardButton.isEnabled = webView.canGoForward
|
|
|
|
spinner.stopAnimating()
|
2018-09-30 19:52:56 +00:00
|
|
|
navigationDelegate?.onLoadStop(uuid: self.uuid, webView: webView)
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
2018-10-12 01:46:33 +00:00
|
|
|
func webView(_ view: WKWebView,
|
|
|
|
didFailProvisionalNavigation navigation: WKNavigation!,
|
|
|
|
withError error: Error) {
|
|
|
|
webView(view, didFail: navigation, withError: error)
|
|
|
|
}
|
2018-09-22 12:27:08 +00:00
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
func webView(_ webView: WKWebView, didFail navigation: WKNavigation!, withError error: Error) {
|
2018-09-22 12:27:08 +00:00
|
|
|
print("webView:didFailNavigationWithError - \(Int(error._code)): \(error.localizedDescription)")
|
2018-09-18 01:07:12 +00:00
|
|
|
backButton.isEnabled = webView.canGoBack
|
|
|
|
forwardButton.isEnabled = webView.canGoForward
|
|
|
|
spinner.stopAnimating()
|
2018-09-30 19:52:56 +00:00
|
|
|
navigationDelegate?.onLoadError(uuid: self.uuid, webView: webView, error: error)
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
2018-10-09 23:52:27 +00:00
|
|
|
|
2018-10-13 19:12:32 +00:00
|
|
|
func didReceiveResourceResponse(_ response: URLResponse, fromRequest request: URLRequest?, withData data: Data, startTime: Int, duration: Int) {
|
|
|
|
navigationDelegate?.onLoadResource(uuid: self.uuid, webView: webView, response: response, fromRequest: request, withData: data, startTime: startTime, duration: duration)
|
2018-10-12 01:46:33 +00:00
|
|
|
}
|
|
|
|
|
2018-10-09 23:52:27 +00:00
|
|
|
func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) {
|
|
|
|
if message.name.starts(with: "console") {
|
|
|
|
var messageLevel = "LOG"
|
|
|
|
switch (message.name) {
|
|
|
|
case "consoleLog":
|
|
|
|
messageLevel = "LOG"
|
|
|
|
break;
|
|
|
|
case "consoleDebug":
|
|
|
|
// on Android, console.debug is TIP
|
|
|
|
messageLevel = "TIP"
|
|
|
|
break;
|
|
|
|
case "consoleError":
|
|
|
|
messageLevel = "ERROR"
|
|
|
|
break;
|
|
|
|
case "consoleInfo":
|
|
|
|
// on Android, console.info is LOG
|
|
|
|
messageLevel = "LOG"
|
|
|
|
break;
|
|
|
|
case "consoleWarn":
|
|
|
|
messageLevel = "WARNING"
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
messageLevel = "LOG"
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
navigationDelegate?.onConsoleMessage(uuid: self.uuid, sourceURL: "", lineNumber: 1, message: message.body as! String, messageLevel: messageLevel)
|
|
|
|
}
|
2018-10-13 19:12:32 +00:00
|
|
|
else if message.name == "resourceLoaded" {
|
|
|
|
if let resource = convertToDictionary(text: message.body as! String) {
|
|
|
|
let url = URL(string: resource["name"] as! String)!
|
|
|
|
if !UIApplication.shared.canOpenURL(url) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
let startTime = Int(resource["startTime"] as! Double)
|
|
|
|
let duration = Int(resource["duration"] as! Double)
|
|
|
|
var urlRequest = URLRequest(url: url)
|
|
|
|
urlRequest.allHTTPHeaderFields = [:]
|
|
|
|
let config = URLSessionConfiguration.default
|
|
|
|
let session = URLSession(configuration: config)
|
|
|
|
let task = session.dataTask(with: urlRequest) { (data, response, error) in
|
|
|
|
if error != nil {
|
|
|
|
print(error)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
var withData = data
|
|
|
|
if withData == nil {
|
|
|
|
withData = Data()
|
|
|
|
}
|
|
|
|
self.didReceiveResourceResponse(response!, fromRequest: urlRequest, withData: withData!, startTime: startTime, duration: duration)
|
|
|
|
}
|
|
|
|
task.resume()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if message.name == "callHandler" {
|
|
|
|
let body = message.body as! [String: Any]
|
|
|
|
let handlerName = body["handlerName"] as! String
|
|
|
|
let args = body["args"] as! String
|
|
|
|
self.navigationDelegate?.onCallJsHandler(uuid: self.uuid, webView: webView, handlerName: handlerName, args: args)
|
|
|
|
}
|
2018-10-09 23:52:27 +00:00
|
|
|
}
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|