2018-09-18 01:07:12 +00:00
|
|
|
//
|
|
|
|
// InAppBrowserWebViewController.swift
|
2019-11-29 15:59:18 +00:00
|
|
|
// flutter_inappwebview
|
2018-09-18 01:07:12 +00:00
|
|
|
//
|
|
|
|
// 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
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public class InAppWebView_IBWrapper: InAppWebView {
|
2019-05-30 21:49:51 +00:00
|
|
|
required init(coder: NSCoder) {
|
2018-10-12 01:46:33 +00:00
|
|
|
let config = WKWebViewConfiguration()
|
2020-05-21 01:34:39 +00:00
|
|
|
super.init(frame: .zero, configuration: config, IABController: nil, contextMenu: nil, channel: nil)
|
2018-10-12 01:46:33 +00:00
|
|
|
self.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public class InAppBrowserWebViewController: UIViewController, FlutterPlugin, UIScrollViewDelegate, WKUIDelegate, UITextFieldDelegate {
|
2018-10-15 23:27:58 +00:00
|
|
|
|
2019-06-07 00:13:56 +00:00
|
|
|
@IBOutlet var containerWebView: UIView!
|
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!
|
|
|
|
|
2018-10-17 23:38:36 +00:00
|
|
|
@IBOutlet var toolbarTop_BottomToWebViewTopConstraint: NSLayoutConstraint!
|
|
|
|
@IBOutlet var toolbarBottom_TopToWebViewBottomConstraint: NSLayoutConstraint!
|
2019-06-07 00:13:56 +00:00
|
|
|
@IBOutlet var containerWebView_BottomFullScreenConstraint: NSLayoutConstraint!
|
|
|
|
@IBOutlet var containerWebView_TopFullScreenConstraint: NSLayoutConstraint!
|
2018-10-17 23:38:36 +00:00
|
|
|
@IBOutlet var webView_BottomFullScreenConstraint: NSLayoutConstraint!
|
|
|
|
@IBOutlet var webView_TopFullScreenConstraint: NSLayoutConstraint!
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
var uuid: String = ""
|
2019-06-07 00:13:56 +00:00
|
|
|
var webView: InAppWebView!
|
2019-12-18 20:34:40 +00:00
|
|
|
var channel: FlutterMethodChannel?
|
2019-03-11 02:34:58 +00:00
|
|
|
var initURL: URL?
|
2020-05-21 01:34:39 +00:00
|
|
|
var contextMenu: [String: Any]?
|
2018-09-18 01:07:12 +00:00
|
|
|
var tmpWindow: UIWindow?
|
|
|
|
var browserOptions: InAppBrowserOptions?
|
2018-10-21 22:41:39 +00:00
|
|
|
var webViewOptions: InAppWebViewOptions?
|
2018-09-20 00:48:24 +00:00
|
|
|
var initHeaders: [String: String]?
|
2018-11-07 00:11:27 +00:00
|
|
|
var initData: String?
|
|
|
|
var initMimeType: String?
|
|
|
|
var initEncoding: String?
|
|
|
|
var initBaseUrl: String?
|
2018-09-23 16:35:32 +00:00
|
|
|
var isHidden = false
|
2018-10-17 23:38:36 +00:00
|
|
|
var viewPrepared = false
|
2019-12-18 20:34:40 +00:00
|
|
|
var previousStatusBarStyle = -1
|
2018-09-18 01:07:12 +00:00
|
|
|
|
|
|
|
required init(coder aDecoder: NSCoder) {
|
|
|
|
super.init(coder: aDecoder)!
|
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public static func register(with registrar: FlutterPluginRegistrar) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
public func prepareMethodChannel() {
|
|
|
|
channel = FlutterMethodChannel(name: "com.pichillilorenzo/flutter_inappbrowser_" + uuid, binaryMessenger: SwiftFlutterPlugin.instance!.registrar!.messenger())
|
|
|
|
SwiftFlutterPlugin.instance!.registrar!.addMethodCallDelegate(self, channel: channel!)
|
|
|
|
}
|
|
|
|
|
|
|
|
public func handle(_ call: FlutterMethodCall, result: @escaping FlutterResult) {
|
|
|
|
let arguments = call.arguments as? NSDictionary
|
|
|
|
|
|
|
|
switch call.method {
|
|
|
|
case "getUrl":
|
|
|
|
result(webView.url?.absoluteString)
|
|
|
|
break
|
|
|
|
case "getTitle":
|
|
|
|
result(webView.title)
|
|
|
|
break
|
|
|
|
case "getProgress":
|
|
|
|
let progress = Int(webView.estimatedProgress * 100)
|
|
|
|
result(progress)
|
|
|
|
break
|
|
|
|
case "loadUrl":
|
|
|
|
let url = arguments!["url"] as! String
|
|
|
|
let headers = arguments!["headers"] as? [String: String]
|
|
|
|
let absoluteUrl = URL(string: url)!.absoluteURL
|
|
|
|
webView.loadUrl(url: absoluteUrl, headers: headers)
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "loadData":
|
|
|
|
let data = arguments!["data"] as! String
|
|
|
|
let mimeType = arguments!["mimeType"] as! String
|
|
|
|
let encoding = arguments!["encoding"] as! String
|
|
|
|
let baseUrl = arguments!["baseUrl"] as! String
|
|
|
|
webView.loadData(data: data, mimeType: mimeType, encoding: encoding, baseUrl: baseUrl)
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "postUrl":
|
|
|
|
let url = arguments!["url"] as! String
|
|
|
|
let postData = arguments!["postData"] as! FlutterStandardTypedData
|
|
|
|
let absoluteUrl = URL(string: url)!.absoluteURL
|
|
|
|
webView.postUrl(url: absoluteUrl, postData: postData.data, completionHandler: { () -> Void in
|
|
|
|
result(true)
|
|
|
|
})
|
|
|
|
break
|
|
|
|
case "loadFile":
|
|
|
|
let url = arguments!["url"] as! String
|
|
|
|
let headers = arguments!["headers"] as? [String: String]
|
|
|
|
do {
|
|
|
|
try webView.loadFile(url: url, headers: headers)
|
|
|
|
result(true)
|
|
|
|
}
|
|
|
|
catch let error as NSError {
|
|
|
|
dump(error)
|
|
|
|
result(FlutterError(code: "InAppBrowserWebViewController", message: error.localizedDescription, details: nil))
|
|
|
|
}
|
|
|
|
break
|
|
|
|
case "close":
|
|
|
|
close()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "show":
|
|
|
|
show()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "hide":
|
|
|
|
hide()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "reload":
|
|
|
|
webView.reload()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "goBack":
|
|
|
|
webView.goBack()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "canGoBack":
|
|
|
|
result(webView.canGoBack)
|
|
|
|
break
|
|
|
|
case "goForward":
|
|
|
|
webView.goForward()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "canGoForward":
|
|
|
|
result(webView.canGoForward)
|
|
|
|
break
|
|
|
|
case "goBackOrForward":
|
|
|
|
let steps = arguments!["steps"] as! Int
|
|
|
|
webView.goBackOrForward(steps: steps)
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "canGoBackOrForward":
|
|
|
|
let steps = arguments!["steps"] as! Int
|
|
|
|
result(webView.canGoBackOrForward(steps: steps))
|
|
|
|
break
|
|
|
|
case "isLoading":
|
|
|
|
result(webView.isLoading == true)
|
|
|
|
break
|
|
|
|
case "stopLoading":
|
|
|
|
webView.stopLoading()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "isHidden":
|
|
|
|
result(isHidden == true)
|
|
|
|
break
|
|
|
|
case "evaluateJavascript":
|
|
|
|
let source = arguments!["source"] as! String
|
|
|
|
webView.evaluateJavascript(source: source, result: result)
|
|
|
|
break
|
|
|
|
case "injectJavascriptFileFromUrl":
|
|
|
|
let urlFile = arguments!["urlFile"] as! String
|
|
|
|
webView.injectJavascriptFileFromUrl(urlFile: urlFile)
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "injectCSSCode":
|
|
|
|
let source = arguments!["source"] as! String
|
|
|
|
webView.injectCSSCode(source: source)
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "injectCSSFileFromUrl":
|
|
|
|
let urlFile = arguments!["urlFile"] as! String
|
|
|
|
webView.injectCSSFileFromUrl(urlFile: urlFile)
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "takeScreenshot":
|
|
|
|
webView.takeScreenshot(completionHandler: { (screenshot) -> Void in
|
|
|
|
result(screenshot)
|
|
|
|
})
|
|
|
|
break
|
|
|
|
case "setOptions":
|
|
|
|
let inAppBrowserOptions = InAppBrowserOptions()
|
|
|
|
let inAppBrowserOptionsMap = arguments!["options"] as! [String: Any]
|
|
|
|
let _ = inAppBrowserOptions.parse(options: inAppBrowserOptionsMap)
|
|
|
|
self.setOptions(newOptions: inAppBrowserOptions, newOptionsMap: inAppBrowserOptionsMap)
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "getOptions":
|
|
|
|
result(getOptions())
|
|
|
|
break
|
|
|
|
case "getCopyBackForwardList":
|
|
|
|
result(webView.getCopyBackForwardList())
|
|
|
|
break
|
|
|
|
case "findAllAsync":
|
|
|
|
let find = arguments!["find"] as! String
|
|
|
|
webView.findAllAsync(find: find, completionHandler: {(value, error) in
|
|
|
|
if error != nil {
|
|
|
|
result(FlutterError(code: "InAppBrowserWebViewController", message: error?.localizedDescription, details: nil))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
result(true)
|
|
|
|
})
|
|
|
|
break
|
|
|
|
case "findNext":
|
|
|
|
let forward = arguments!["forward"] as! Bool
|
|
|
|
webView.findNext(forward: forward, completionHandler: {(value, error) in
|
|
|
|
if error != nil {
|
|
|
|
result(FlutterError(code: "InAppBrowserWebViewController", message: error?.localizedDescription, details: nil))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
result(true)
|
|
|
|
})
|
|
|
|
break
|
|
|
|
case "clearMatches":
|
|
|
|
webView.clearMatches(completionHandler: {(value, error) in
|
|
|
|
if error != nil {
|
|
|
|
result(FlutterError(code: "InAppBrowserWebViewController", message: error?.localizedDescription, details: nil))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
result(true)
|
|
|
|
})
|
|
|
|
break
|
|
|
|
case "clearCache":
|
|
|
|
webView.clearCache()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "scrollTo":
|
|
|
|
let x = arguments!["x"] as! Int
|
|
|
|
let y = arguments!["y"] as! Int
|
2020-06-13 01:50:19 +00:00
|
|
|
let animated = arguments!["animated"] as! Bool
|
|
|
|
webView.scrollTo(x: x, y: y, animated: animated)
|
2019-12-18 20:34:40 +00:00
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "scrollBy":
|
|
|
|
let x = arguments!["x"] as! Int
|
|
|
|
let y = arguments!["y"] as! Int
|
2020-06-13 01:50:19 +00:00
|
|
|
let animated = arguments!["animated"] as! Bool
|
|
|
|
webView.scrollTo(x: x, y: y, animated: animated)
|
2019-12-18 20:34:40 +00:00
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "pauseTimers":
|
|
|
|
webView.pauseTimers()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "resumeTimers":
|
|
|
|
webView.resumeTimers()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "printCurrentPage":
|
|
|
|
webView.printCurrentPage(printCompletionHandler: {(completed, error) in
|
|
|
|
if !completed, let _ = error {
|
|
|
|
result(false)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
result(true)
|
|
|
|
})
|
|
|
|
break
|
|
|
|
case "getContentHeight":
|
|
|
|
result(webView.getContentHeight())
|
|
|
|
break
|
|
|
|
case "reloadFromOrigin":
|
|
|
|
webView.reloadFromOrigin()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "getScale":
|
|
|
|
result(webView.getScale())
|
|
|
|
break
|
2020-05-09 02:36:07 +00:00
|
|
|
case "hasOnlySecureContent":
|
|
|
|
result(webView.hasOnlySecureContent)
|
|
|
|
break
|
2020-05-21 01:34:39 +00:00
|
|
|
case "getSelectedText":
|
|
|
|
if webView != nil {
|
|
|
|
webView!.getSelectedText { (value, error) in
|
|
|
|
if let err = error {
|
|
|
|
print(err.localizedDescription)
|
|
|
|
}
|
|
|
|
result(value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
result(nil)
|
|
|
|
}
|
|
|
|
break
|
|
|
|
case "getHitTestResult":
|
|
|
|
if webView != nil {
|
|
|
|
webView!.getHitTestResult { (value, error) in
|
|
|
|
if let err = error {
|
|
|
|
print(err.localizedDescription)
|
|
|
|
}
|
|
|
|
result(value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
result(nil)
|
|
|
|
}
|
|
|
|
break
|
2020-05-30 18:23:33 +00:00
|
|
|
case "clearFocus":
|
|
|
|
if webView != nil {
|
|
|
|
webView!.clearFocus()
|
|
|
|
result(true)
|
|
|
|
} else {
|
|
|
|
result(false)
|
|
|
|
}
|
|
|
|
|
|
|
|
break
|
|
|
|
case "setContextMenu":
|
|
|
|
if webView != nil {
|
|
|
|
let contextMenu = arguments!["contextMenu"] as? [String: Any]
|
|
|
|
webView!.contextMenu = contextMenu
|
|
|
|
result(true)
|
|
|
|
} else {
|
|
|
|
result(false)
|
|
|
|
}
|
|
|
|
break
|
2019-12-18 20:34:40 +00:00
|
|
|
default:
|
|
|
|
result(FlutterMethodNotImplemented)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public override func viewWillAppear(_ animated: Bool) {
|
2018-10-17 23:38:36 +00:00
|
|
|
if !viewPrepared {
|
2019-06-07 00:13:56 +00:00
|
|
|
let preWebviewConfiguration = InAppWebView.preWKWebViewConfiguration(options: webViewOptions)
|
2020-05-21 01:34:39 +00:00
|
|
|
self.webView = InAppWebView(frame: .zero, configuration: preWebviewConfiguration, IABController: self, contextMenu: contextMenu, channel: channel!)
|
2019-06-07 00:13:56 +00:00
|
|
|
self.containerWebView.addSubview(self.webView)
|
2018-10-17 23:38:36 +00:00
|
|
|
prepareConstraints()
|
|
|
|
prepareWebView()
|
2019-06-07 00:13:56 +00:00
|
|
|
|
2019-10-27 03:35:05 +00:00
|
|
|
if #available(iOS 11.0, *) {
|
2019-12-18 20:34:40 +00:00
|
|
|
if let contentBlockers = webView.options?.contentBlockers, contentBlockers.count > 0 {
|
2019-10-27 03:35:05 +00:00
|
|
|
do {
|
|
|
|
let jsonData = try JSONSerialization.data(withJSONObject: contentBlockers, options: [])
|
|
|
|
let blockRules = String(data: jsonData, encoding: String.Encoding.utf8)
|
|
|
|
WKContentRuleListStore.default().compileContentRuleList(
|
|
|
|
forIdentifier: "ContentBlockingRules",
|
|
|
|
encodedContentRuleList: blockRules) { (contentRuleList, error) in
|
|
|
|
|
|
|
|
if let error = error {
|
|
|
|
print(error.localizedDescription)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let configuration = self.webView!.configuration
|
|
|
|
configuration.userContentController.add(contentRuleList!)
|
|
|
|
|
|
|
|
self.initLoad(initURL: self.initURL, initData: self.initData, initMimeType: self.initMimeType, initEncoding: self.initEncoding, initBaseUrl: self.initBaseUrl, initHeaders: self.initHeaders)
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
self.onBrowserCreated()
|
2019-10-27 03:35:05 +00:00
|
|
|
}
|
|
|
|
return
|
|
|
|
} catch {
|
|
|
|
print(error.localizedDescription)
|
|
|
|
}
|
|
|
|
}
|
2019-06-07 00:13:56 +00:00
|
|
|
}
|
|
|
|
|
2019-10-27 03:35:05 +00:00
|
|
|
initLoad(initURL: initURL, initData: initData, initMimeType: initMimeType, initEncoding: initEncoding, initBaseUrl: initBaseUrl, initHeaders: initHeaders)
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
onBrowserCreated()
|
2018-10-17 23:38:36 +00:00
|
|
|
}
|
|
|
|
viewPrepared = true
|
2018-09-26 00:56:56 +00:00
|
|
|
super.viewWillAppear(animated)
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public func initLoad(initURL: URL?, initData: String?, initMimeType: String?, initEncoding: String?, initBaseUrl: String?, initHeaders: [String: String]?) {
|
2019-10-27 03:35:05 +00:00
|
|
|
if self.initData == nil {
|
|
|
|
loadUrl(url: self.initURL!, headers: self.initHeaders)
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
webView.loadData(data: initData!, mimeType: initMimeType!, encoding: initEncoding!, baseUrl: initBaseUrl!)
|
|
|
|
}
|
|
|
|
}
|
2018-10-17 23:38:36 +00:00
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public override func viewDidLoad() {
|
2018-10-12 01:46:33 +00:00
|
|
|
super.viewDidLoad()
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
urlField.delegate = self
|
2019-03-11 02:34:58 +00:00
|
|
|
urlField.text = self.initURL?.absoluteString
|
2018-09-18 01:07:12 +00:00
|
|
|
|
|
|
|
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()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Prevent crashes on closing windows
|
|
|
|
deinit {
|
2019-12-16 22:58:10 +00:00
|
|
|
print("InAppBrowserWebViewController - dealloc")
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public override func viewWillDisappear (_ animated: Bool) {
|
|
|
|
dispose()
|
2018-09-18 01:07:12 +00:00
|
|
|
super.viewWillDisappear(animated)
|
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public func prepareConstraints () {
|
|
|
|
containerWebView_BottomFullScreenConstraint = NSLayoutConstraint(item: self.containerWebView!, attribute: NSLayoutConstraint.Attribute.bottom, relatedBy: NSLayoutConstraint.Relation.equal, toItem: self.view, attribute: NSLayoutConstraint.Attribute.bottom, multiplier: 1, constant: 0)
|
|
|
|
containerWebView_TopFullScreenConstraint = NSLayoutConstraint(item: self.containerWebView!, attribute: NSLayoutConstraint.Attribute.top, relatedBy: NSLayoutConstraint.Relation.equal, toItem: self.view, attribute: NSLayoutConstraint.Attribute.top, multiplier: 1, constant: 0)
|
2019-06-07 00:13:56 +00:00
|
|
|
|
|
|
|
webView.translatesAutoresizingMaskIntoConstraints = false
|
2019-12-18 20:34:40 +00:00
|
|
|
let height = NSLayoutConstraint(item: self.webView!, attribute: .height, relatedBy: .equal, toItem: containerWebView, attribute: .height, multiplier: 1, constant: 0)
|
|
|
|
let width = NSLayoutConstraint(item: self.webView!, attribute: .width, relatedBy: .equal, toItem: containerWebView, attribute: .width, multiplier: 1, constant: 0)
|
|
|
|
let leftConstraint = NSLayoutConstraint(item: self.webView!, attribute: .leftMargin, relatedBy: .equal, toItem: containerWebView, attribute: .leftMargin, multiplier: 1, constant: 0)
|
|
|
|
let rightConstraint = NSLayoutConstraint(item: self.webView!, attribute: .rightMargin, relatedBy: .equal, toItem: containerWebView, attribute: .rightMargin, multiplier: 1, constant: 0)
|
|
|
|
let bottomContraint = NSLayoutConstraint(item: self.webView!, attribute: .bottomMargin, relatedBy: .equal, toItem: containerWebView, attribute: .bottomMargin, multiplier: 1, constant: 0)
|
2019-06-07 00:13:56 +00:00
|
|
|
containerWebView.addConstraints([height, width, leftConstraint, rightConstraint, bottomContraint])
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
webView_BottomFullScreenConstraint = NSLayoutConstraint(item: webView!, attribute: NSLayoutConstraint.Attribute.bottom, relatedBy: NSLayoutConstraint.Relation.equal, toItem: self.containerWebView, attribute: NSLayoutConstraint.Attribute.bottom, multiplier: 1, constant: 0)
|
|
|
|
webView_TopFullScreenConstraint = NSLayoutConstraint(item: webView!, attribute: NSLayoutConstraint.Attribute.top, relatedBy: NSLayoutConstraint.Relation.equal, toItem: self.containerWebView, attribute: NSLayoutConstraint.Attribute.top, multiplier: 1, constant: 0)
|
2018-10-17 23:38:36 +00:00
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public func prepareWebView() {
|
2019-03-11 02:34:58 +00:00
|
|
|
self.webView.options = webViewOptions
|
|
|
|
self.webView.prepare()
|
2018-09-19 02:10:00 +00:00
|
|
|
|
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 {
|
2018-10-17 23:38:36 +00:00
|
|
|
self.toolbarTop.isHidden = true
|
|
|
|
self.toolbarTop_BottomToWebViewTopConstraint.isActive = false
|
2019-06-07 00:13:56 +00:00
|
|
|
self.containerWebView_TopFullScreenConstraint.isActive = true
|
2018-10-17 23:38:36 +00:00
|
|
|
self.webView_TopFullScreenConstraint.isActive = true
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
2018-10-17 23:38:36 +00:00
|
|
|
self.toolbarBottom.isHidden = true
|
|
|
|
self.toolbarBottom_TopToWebViewBottomConstraint.isActive = false
|
2019-06-07 00:13:56 +00:00
|
|
|
self.containerWebView_BottomFullScreenConstraint.isActive = true
|
2018-10-17 23:38:36 +00:00
|
|
|
self.webView_BottomFullScreenConstraint.isActive = true
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if browserOptions?.closeButtonCaption != "" {
|
|
|
|
closeButton.setTitle(browserOptions?.closeButtonCaption, for: .normal)
|
|
|
|
}
|
|
|
|
if browserOptions?.closeButtonColor != "" {
|
|
|
|
closeButton.tintColor = color(fromHexString: (browserOptions?.closeButtonColor)!)
|
|
|
|
}
|
2019-12-18 20:42:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public func prepareBeforeViewWillAppear() {
|
2018-09-18 01:07:12 +00:00
|
|
|
self.modalPresentationStyle = UIModalPresentationStyle(rawValue: (browserOptions?.presentationStyle)!)!
|
|
|
|
self.modalTransitionStyle = UIModalTransitionStyle(rawValue: (browserOptions?.transitionStyle)!)!
|
2018-09-20 00:48:24 +00:00
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public func loadUrl(url: URL, headers: [String: String]?) {
|
2019-03-11 02:34:58 +00:00
|
|
|
webView.loadUrl(url: url, headers: headers)
|
|
|
|
updateUrlTextField(url: (webView.currentURL?.absoluteString)!)
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Load user requested url
|
2019-12-18 20:34:40 +00:00
|
|
|
public func textFieldShouldReturn(_ textField: UITextField) -> Bool {
|
2018-09-18 01:07:12 +00:00
|
|
|
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 {
|
2019-03-11 02:34:58 +00:00
|
|
|
updateUrlTextField(url: (webView.currentURL?.absoluteString)!)
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func setWebViewFrame(_ frame: CGRect) {
|
2019-09-23 02:29:13 +00:00
|
|
|
print("Setting the WebView's frame to \(NSCoder.string(for: frame))")
|
2018-09-18 01:07:12 +00:00
|
|
|
webView.frame = frame
|
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public func show() {
|
|
|
|
isHidden = false
|
|
|
|
view.isHidden = false
|
|
|
|
|
|
|
|
// Run later to avoid the "took a long time" log message.
|
|
|
|
DispatchQueue.main.async(execute: {() -> Void in
|
|
|
|
let baseWindowLevel = UIApplication.shared.keyWindow?.windowLevel
|
|
|
|
self.tmpWindow?.windowLevel = UIWindow.Level(baseWindowLevel!.rawValue + 1.0)
|
|
|
|
self.tmpWindow?.makeKeyAndVisible()
|
|
|
|
UIApplication.shared.delegate?.window??.makeKeyAndVisible()
|
|
|
|
self.tmpWindow?.rootViewController?.present(self, animated: true, completion: nil)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
public func hide() {
|
|
|
|
isHidden = true
|
|
|
|
|
|
|
|
// Run later to avoid the "took a long time" log message.
|
|
|
|
DispatchQueue.main.async(execute: {() -> Void in
|
|
|
|
self.presentingViewController?.dismiss(animated: true, completion: {() -> Void in
|
|
|
|
self.tmpWindow?.windowLevel = UIWindow.Level(rawValue: 0.0)
|
|
|
|
UIApplication.shared.delegate?.window??.makeKeyAndVisible()
|
|
|
|
if self.previousStatusBarStyle != -1 {
|
|
|
|
UIApplication.shared.statusBarStyle = UIStatusBarStyle(rawValue: self.previousStatusBarStyle)!
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
@objc public func reload () {
|
2018-09-18 01:07:12 +00:00
|
|
|
webView.reload()
|
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
@objc public func share () {
|
2019-03-11 02:34:58 +00:00
|
|
|
let vc = UIActivityViewController(activityItems: [webView.currentURL ?? ""], applicationActivities: [])
|
2018-09-18 01:07:12 +00:00
|
|
|
present(vc, animated: true, completion: nil)
|
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
@objc public func close() {
|
2018-09-18 01:07:12 +00:00
|
|
|
weak var weakSelf = self
|
|
|
|
|
2019-06-18 21:35:40 +00:00
|
|
|
if (weakSelf?.responds(to: #selector(getter: self.presentingViewController)))! {
|
|
|
|
weakSelf?.presentingViewController?.dismiss(animated: true, completion: {() -> Void in
|
2019-12-18 20:34:40 +00:00
|
|
|
|
2019-06-18 21:35:40 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
weakSelf?.parent?.dismiss(animated: true, completion: {() -> Void in
|
2019-12-18 20:34:40 +00:00
|
|
|
|
2019-06-18 21:35:40 +00:00
|
|
|
})
|
|
|
|
}
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
@objc public 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
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public func canGoBack() -> Bool {
|
2018-10-30 00:39:07 +00:00
|
|
|
return webView.canGoBack
|
2018-09-20 00:48:24 +00:00
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
@objc public 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
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public func canGoForward() -> Bool {
|
2018-10-30 00:39:07 +00:00
|
|
|
return webView.canGoForward
|
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
@objc public func goBackOrForward(steps: Int) {
|
2018-10-30 00:39:07 +00:00
|
|
|
webView.goBackOrForward(steps: steps)
|
|
|
|
updateUrlTextField(url: (webView?.url?.absoluteString)!)
|
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public func canGoBackOrForward(steps: Int) -> Bool {
|
2018-10-30 00:39:07 +00:00
|
|
|
return webView.canGoBackOrForward(steps: steps)
|
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public func updateUrlTextField(url: String) {
|
2018-09-18 01:07:12 +00:00
|
|
|
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-10-17 23:38:36 +00:00
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public func setOptions(newOptions: InAppBrowserOptions, newOptionsMap: [String: Any]) {
|
2018-10-17 23:38:36 +00:00
|
|
|
|
2018-10-21 22:41:39 +00:00
|
|
|
let newInAppWebViewOptions = InAppWebViewOptions()
|
2019-12-18 20:34:40 +00:00
|
|
|
let _ = newInAppWebViewOptions.parse(options: newOptionsMap)
|
2019-03-11 02:34:58 +00:00
|
|
|
self.webView.setOptions(newOptions: newInAppWebViewOptions, newOptionsMap: newOptionsMap)
|
2018-10-21 22:41:39 +00:00
|
|
|
|
2018-10-17 23:38:36 +00:00
|
|
|
if newOptionsMap["hidden"] != nil && browserOptions?.hidden != newOptions.hidden {
|
|
|
|
if newOptions.hidden {
|
2019-12-18 20:34:40 +00:00
|
|
|
hide()
|
2018-10-17 23:38:36 +00:00
|
|
|
}
|
|
|
|
else {
|
2019-12-18 20:34:40 +00:00
|
|
|
show()
|
2018-10-17 23:38:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if newOptionsMap["hideUrlBar"] != nil && browserOptions?.hideUrlBar != newOptions.hideUrlBar {
|
|
|
|
self.urlField.isHidden = newOptions.hideUrlBar
|
|
|
|
self.urlField.isEnabled = !newOptions.hideUrlBar
|
|
|
|
}
|
|
|
|
|
|
|
|
if newOptionsMap["toolbarTop"] != nil && browserOptions?.toolbarTop != newOptions.toolbarTop {
|
2019-06-07 00:13:56 +00:00
|
|
|
self.containerWebView_TopFullScreenConstraint.isActive = !newOptions.toolbarTop
|
2018-10-17 23:38:36 +00:00
|
|
|
self.webView_TopFullScreenConstraint.isActive = !newOptions.toolbarTop
|
|
|
|
self.toolbarTop.isHidden = !newOptions.toolbarTop
|
|
|
|
self.toolbarTop_BottomToWebViewTopConstraint.isActive = newOptions.toolbarTop
|
|
|
|
}
|
|
|
|
|
|
|
|
if newOptionsMap["toolbarTopBackgroundColor"] != nil && browserOptions?.toolbarTopBackgroundColor != newOptions.toolbarTopBackgroundColor && newOptions.toolbarTopBackgroundColor != "" {
|
|
|
|
self.toolbarTop.backgroundColor = color(fromHexString: newOptions.toolbarTopBackgroundColor)
|
|
|
|
}
|
|
|
|
|
|
|
|
if newOptionsMap["toolbarBottom"] != nil && browserOptions?.toolbarBottom != newOptions.toolbarBottom {
|
2019-06-07 00:13:56 +00:00
|
|
|
self.containerWebView_BottomFullScreenConstraint.isActive = !newOptions.toolbarBottom
|
2018-10-17 23:38:36 +00:00
|
|
|
self.webView_BottomFullScreenConstraint.isActive = !newOptions.toolbarBottom
|
|
|
|
self.toolbarBottom.isHidden = !newOptions.toolbarBottom
|
|
|
|
self.toolbarBottom_TopToWebViewBottomConstraint.isActive = newOptions.toolbarBottom
|
|
|
|
}
|
|
|
|
|
|
|
|
if newOptionsMap["toolbarBottomBackgroundColor"] != nil && browserOptions?.toolbarBottomBackgroundColor != newOptions.toolbarBottomBackgroundColor && newOptions.toolbarBottomBackgroundColor != "" {
|
|
|
|
self.toolbarBottom.backgroundColor = color(fromHexString: newOptions.toolbarBottomBackgroundColor)
|
|
|
|
}
|
|
|
|
|
|
|
|
if newOptionsMap["toolbarBottomTranslucent"] != nil && browserOptions?.toolbarBottomTranslucent != newOptions.toolbarBottomTranslucent {
|
|
|
|
self.toolbarBottom.isTranslucent = newOptions.toolbarBottomTranslucent
|
|
|
|
}
|
|
|
|
|
|
|
|
if newOptionsMap["closeButtonCaption"] != nil && browserOptions?.closeButtonCaption != newOptions.closeButtonCaption && newOptions.closeButtonCaption != "" {
|
|
|
|
closeButton.setTitle(newOptions.closeButtonCaption, for: .normal)
|
|
|
|
}
|
|
|
|
|
|
|
|
if newOptionsMap["closeButtonColor"] != nil && browserOptions?.closeButtonColor != newOptions.closeButtonColor && newOptions.closeButtonColor != "" {
|
|
|
|
closeButton.tintColor = color(fromHexString: newOptions.closeButtonColor)
|
|
|
|
}
|
|
|
|
|
|
|
|
if newOptionsMap["presentationStyle"] != nil && browserOptions?.presentationStyle != newOptions.presentationStyle {
|
|
|
|
self.modalPresentationStyle = UIModalPresentationStyle(rawValue: newOptions.presentationStyle)!
|
|
|
|
}
|
|
|
|
|
|
|
|
if newOptionsMap["transitionStyle"] != nil && browserOptions?.transitionStyle != newOptions.transitionStyle {
|
|
|
|
self.modalTransitionStyle = UIModalTransitionStyle(rawValue: newOptions.transitionStyle)!
|
|
|
|
}
|
|
|
|
|
|
|
|
self.browserOptions = newOptions
|
2018-10-21 22:41:39 +00:00
|
|
|
self.webViewOptions = newInAppWebViewOptions
|
2018-10-17 23:38:36 +00:00
|
|
|
}
|
|
|
|
|
2019-12-18 20:34:40 +00:00
|
|
|
public func getOptions() -> [String: Any?]? {
|
2020-05-29 17:56:03 +00:00
|
|
|
let webViewOptionsMap = self.webView.getOptions()
|
|
|
|
if (self.browserOptions == nil || webViewOptionsMap == nil) {
|
2018-10-21 22:41:39 +00:00
|
|
|
return nil
|
|
|
|
}
|
2020-05-29 17:56:03 +00:00
|
|
|
var optionsMap = self.browserOptions!.getRealOptions(obj: self)
|
|
|
|
optionsMap.merge(webViewOptionsMap!, uniquingKeysWith: { (current, _) in current })
|
2018-10-21 22:41:39 +00:00
|
|
|
return optionsMap
|
|
|
|
}
|
2019-12-18 20:34:40 +00:00
|
|
|
|
|
|
|
public func dispose() {
|
|
|
|
webView.dispose()
|
|
|
|
if previousStatusBarStyle != -1 {
|
|
|
|
UIApplication.shared.statusBarStyle = UIStatusBarStyle(rawValue: previousStatusBarStyle)!
|
|
|
|
}
|
|
|
|
transitioningDelegate = nil
|
|
|
|
urlField.delegate = nil
|
|
|
|
closeButton.removeTarget(self, action: #selector(self.close), for: .touchUpInside)
|
|
|
|
forwardButton.target = nil
|
|
|
|
forwardButton.target = nil
|
|
|
|
backButton.target = nil
|
|
|
|
reloadButton.target = nil
|
|
|
|
shareButton.target = nil
|
|
|
|
tmpWindow?.windowLevel = UIWindow.Level(rawValue: 0.0)
|
|
|
|
UIApplication.shared.delegate?.window??.makeKeyAndVisible()
|
|
|
|
onExit()
|
2020-05-21 01:34:39 +00:00
|
|
|
channel?.setMethodCallHandler(nil)
|
|
|
|
channel = nil
|
2019-12-18 20:34:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public func onBrowserCreated() {
|
|
|
|
channel!.invokeMethod("onBrowserCreated", arguments: [])
|
|
|
|
}
|
|
|
|
|
|
|
|
public func onExit() {
|
|
|
|
channel!.invokeMethod("onExit", arguments: [])
|
|
|
|
}
|
2018-09-18 01:07:12 +00:00
|
|
|
}
|