2018-09-14 00:21:51 +00:00
|
|
|
/*
|
|
|
|
Licensed to the Apache Software Foundation (ASF) under one
|
|
|
|
or more contributor license agreements. See the NOTICE file
|
|
|
|
distributed with this work for additional information
|
|
|
|
regarding copyright ownership. The ASF licenses this file
|
|
|
|
to you under the Apache License, Version 2.0 (the
|
|
|
|
"License"); you may not use this file except in compliance
|
|
|
|
with the License. You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing,
|
|
|
|
software distributed under the License is distributed on an
|
|
|
|
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
|
|
KIND, either express or implied. See the License for the
|
|
|
|
specific language governing permissions and limitations
|
|
|
|
under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import Flutter
|
|
|
|
import UIKit
|
|
|
|
import WebKit
|
|
|
|
import Foundation
|
|
|
|
import AVFoundation
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
let WEBVIEW_STORYBOARD = "WebView"
|
|
|
|
let WEBVIEW_STORYBOARD_CONTROLLER_ID = "viewController"
|
2018-09-14 00:21:51 +00:00
|
|
|
|
|
|
|
public class SwiftFlutterPlugin: NSObject, FlutterPlugin {
|
2018-09-18 01:07:12 +00:00
|
|
|
var webViewController: InAppBrowserWebViewController?
|
2018-09-14 00:21:51 +00:00
|
|
|
|
|
|
|
var tmpWindow: UIWindow?
|
|
|
|
var channel: FlutterMethodChannel
|
|
|
|
private var previousStatusBarStyle = -1
|
|
|
|
|
|
|
|
public init(with registrar: FlutterPluginRegistrar) {
|
|
|
|
channel = FlutterMethodChannel(name: "com.pichillilorenzo/flutter_inappbrowser", binaryMessenger: registrar.messenger())
|
|
|
|
}
|
|
|
|
|
|
|
|
public static func register(with registrar: FlutterPluginRegistrar) {
|
|
|
|
let channel = FlutterMethodChannel(name: "com.pichillilorenzo/flutter_inappbrowser", binaryMessenger: registrar.messenger())
|
|
|
|
let instance = SwiftFlutterPlugin(with: registrar)
|
|
|
|
registrar.addMethodCallDelegate(instance, channel: channel)
|
|
|
|
}
|
|
|
|
|
|
|
|
public func handle(_ call: FlutterMethodCall, result: @escaping FlutterResult) {
|
|
|
|
let arguments = call.arguments as? NSDictionary
|
|
|
|
switch call.method {
|
2018-09-18 01:07:12 +00:00
|
|
|
case "open":
|
|
|
|
self.open(arguments: arguments!, result: result)
|
|
|
|
break
|
|
|
|
case "close":
|
|
|
|
self.close()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "show":
|
|
|
|
self.show()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "hide":
|
|
|
|
self.hide()
|
|
|
|
result(true)
|
|
|
|
break
|
2018-09-19 02:10:00 +00:00
|
|
|
case "reload":
|
|
|
|
self.webViewController?.reload()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "goBack":
|
|
|
|
self.webViewController?.goBack()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "goForward":
|
|
|
|
self.webViewController?.goForward()
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "isLoading":
|
|
|
|
result(self.webViewController?.webView.isLoading == true)
|
|
|
|
break
|
|
|
|
case "stopLoading":
|
|
|
|
self.webViewController?.webView.stopLoading()
|
|
|
|
result(true)
|
|
|
|
break
|
2018-09-18 01:07:12 +00:00
|
|
|
case "injectScriptCode":
|
|
|
|
self.injectScriptCode(arguments: arguments!)
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "injectScriptFile":
|
|
|
|
self.injectScriptFile(arguments: arguments!)
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "injectStyleCode":
|
|
|
|
self.injectStyleCode(arguments: arguments!)
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
case "injectStyleFile":
|
|
|
|
self.injectStyleFile(arguments: arguments!)
|
|
|
|
result(true)
|
|
|
|
break
|
|
|
|
default:
|
|
|
|
break
|
2018-09-14 00:21:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func close() {
|
2018-09-18 01:07:12 +00:00
|
|
|
if webViewController == nil {
|
2018-09-14 00:21:51 +00:00
|
|
|
print("IAB.close() called but it was already closed.")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// Things are cleaned up in browserExit.
|
2018-09-18 01:07:12 +00:00
|
|
|
webViewController?.close()
|
2018-09-14 00:21:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func isSystemUrl(_ url: URL) -> Bool {
|
|
|
|
if (url.host == "itunes.apple.com") {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
public func open(arguments: NSDictionary, result: @escaping FlutterResult) {
|
|
|
|
let url: String? = (arguments["url"] as? String)!
|
|
|
|
var target: String? = (arguments["target"] as? String)!
|
2018-09-18 01:07:12 +00:00
|
|
|
target = target != nil ? target : "_self"
|
|
|
|
let options = (arguments["options"] as? [String: Any])!
|
|
|
|
|
2018-09-14 00:21:51 +00:00
|
|
|
if url != nil {
|
2018-09-18 01:07:12 +00:00
|
|
|
let absoluteUrl = URL(string: url!)?.absoluteURL
|
2018-09-14 00:21:51 +00:00
|
|
|
|
|
|
|
if isSystemUrl(absoluteUrl!) {
|
2018-09-18 01:07:12 +00:00
|
|
|
target = "_system"
|
2018-09-14 00:21:51 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
if (target == "_self" || target == "_target") {
|
|
|
|
openIn(inAppBrowser: absoluteUrl!, withOptions: options)
|
2018-09-14 00:21:51 +00:00
|
|
|
}
|
2018-09-18 01:07:12 +00:00
|
|
|
else if (target == "_system") {
|
2018-09-14 00:21:51 +00:00
|
|
|
open(inSystem: absoluteUrl!)
|
|
|
|
}
|
|
|
|
else {
|
2018-09-18 01:07:12 +00:00
|
|
|
// anything else
|
2018-09-14 00:21:51 +00:00
|
|
|
openIn(inAppBrowser: absoluteUrl!, withOptions: options)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
result(false)
|
|
|
|
}
|
|
|
|
result(true)
|
|
|
|
}
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
func openIn(inAppBrowser url: URL, withOptions options: [String: Any]) {
|
|
|
|
|
|
|
|
let browserOptions = InAppBrowserOptions()
|
|
|
|
browserOptions.parse(options: options)
|
2018-09-14 00:21:51 +00:00
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
if browserOptions.clearCache {
|
2018-09-14 00:21:51 +00:00
|
|
|
let _: HTTPCookie?
|
|
|
|
let storage = HTTPCookieStorage.shared
|
|
|
|
for cookie in storage.cookies! {
|
|
|
|
if !(cookie.domain.isEqual(".^filecookies^") ) {
|
|
|
|
storage.deleteCookie(cookie)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
if browserOptions.clearSessionCache {
|
2018-09-14 00:21:51 +00:00
|
|
|
let storage = HTTPCookieStorage.shared
|
|
|
|
for cookie in storage.cookies! {
|
|
|
|
if !(cookie.domain.isEqual(".^filecookies^") && cookie.isSessionOnly) {
|
|
|
|
storage.deleteCookie(cookie)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
if webViewController == nil {
|
2018-09-14 00:21:51 +00:00
|
|
|
|
|
|
|
if !(self.tmpWindow != nil) {
|
|
|
|
let frame: CGRect = UIScreen.main.bounds
|
|
|
|
self.tmpWindow = UIWindow(frame: frame)
|
|
|
|
}
|
2018-09-18 01:07:12 +00:00
|
|
|
|
|
|
|
let storyboard = UIStoryboard(name: WEBVIEW_STORYBOARD, bundle: nil)
|
|
|
|
let vc = storyboard.instantiateViewController(withIdentifier: WEBVIEW_STORYBOARD_CONTROLLER_ID)
|
|
|
|
webViewController = vc as? InAppBrowserWebViewController
|
|
|
|
webViewController?.browserOptions = browserOptions
|
|
|
|
webViewController?.tmpWindow = tmpWindow
|
|
|
|
webViewController?.currentURL = url
|
|
|
|
webViewController?.navigationDelegate = self
|
2018-09-14 00:21:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if !browserOptions.hidden {
|
|
|
|
show()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func show() {
|
2018-09-18 01:07:12 +00:00
|
|
|
if webViewController == nil {
|
2018-09-14 00:21:51 +00:00
|
|
|
print("Tried to show IAB after it was closed.")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if previousStatusBarStyle != -1 {
|
|
|
|
print("Tried to show IAB while already shown")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
weak var weakSelf: SwiftFlutterPlugin? = self
|
|
|
|
|
|
|
|
// Run later to avoid the "took a long time" log message.
|
|
|
|
DispatchQueue.main.async(execute: {() -> Void in
|
2018-09-18 01:07:12 +00:00
|
|
|
if weakSelf?.webViewController != nil {
|
2018-09-14 00:21:51 +00:00
|
|
|
if !(self.tmpWindow != nil) {
|
|
|
|
let frame: CGRect = UIScreen.main.bounds
|
|
|
|
self.tmpWindow = UIWindow(frame: frame)
|
|
|
|
}
|
|
|
|
let tmpController = UIViewController()
|
|
|
|
let baseWindowLevel = UIApplication.shared.keyWindow?.windowLevel
|
|
|
|
self.tmpWindow?.rootViewController = tmpController
|
|
|
|
self.tmpWindow?.windowLevel = UIWindowLevel(baseWindowLevel! + 1)
|
|
|
|
self.tmpWindow?.makeKeyAndVisible()
|
2018-09-18 01:07:12 +00:00
|
|
|
|
|
|
|
tmpController.present(self.webViewController!, animated: true, completion: nil)
|
2018-09-14 00:21:51 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
public func hide() {
|
2018-09-18 01:07:12 +00:00
|
|
|
if webViewController == nil {
|
2018-09-14 00:21:51 +00:00
|
|
|
print("Tried to hide IAB after it was closed.")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if previousStatusBarStyle == -1 {
|
|
|
|
print("Tried to hide IAB while already hidden")
|
|
|
|
return
|
|
|
|
}
|
2018-09-18 01:07:12 +00:00
|
|
|
|
2018-09-14 00:21:51 +00:00
|
|
|
previousStatusBarStyle = UIApplication.shared.statusBarStyle.rawValue
|
|
|
|
// Run later to avoid the "took a long time" log message.
|
|
|
|
DispatchQueue.main.async(execute: {() -> Void in
|
2018-09-18 01:07:12 +00:00
|
|
|
if self.webViewController != nil {
|
2018-09-14 00:21:51 +00:00
|
|
|
self.previousStatusBarStyle = -1
|
2018-09-18 01:07:12 +00:00
|
|
|
self.webViewController?.presentingViewController?.dismiss(animated: true, completion: {() -> Void in
|
2018-09-14 00:21:51 +00:00
|
|
|
self.tmpWindow?.windowLevel = 0.0
|
|
|
|
UIApplication.shared.delegate?.window??.makeKeyAndVisible()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func open(inSystem url: URL) {
|
|
|
|
if UIApplication.shared.openURL(url) == false {
|
|
|
|
NotificationCenter.default.post(Notification(name: Notification.Name(rawValue: "CDVPluginHandleOpenURLNotification"), object: url))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is a helper method for the inject{Script|Style}{Code|File} API calls, which
|
|
|
|
// provides a consistent method for injecting JavaScript code into the document.
|
|
|
|
//
|
|
|
|
// If a wrapper string is supplied, then the source string will be JSON-encoded (adding
|
|
|
|
// quotes) and wrapped using string formatting. (The wrapper string should have a single
|
|
|
|
// '%@' marker).
|
|
|
|
//
|
|
|
|
// If no wrapper is supplied, then the source string is executed directly.
|
|
|
|
func injectDeferredObject(_ source: String, withWrapper jsWrapper: String) {
|
|
|
|
if jsWrapper != nil {
|
|
|
|
let jsonData: Data? = try? JSONSerialization.data(withJSONObject: [source], options: [])
|
|
|
|
let sourceArrayString = String(data: jsonData!, encoding: String.Encoding.utf8)
|
|
|
|
if sourceArrayString != nil {
|
|
|
|
let sourceString: String? = (sourceArrayString! as NSString).substring(with: NSRange(location: 1, length: (sourceArrayString?.characters.count ?? 0) - 2))
|
|
|
|
let jsToInject = String(format: jsWrapper, sourceString!)
|
2018-09-18 01:07:12 +00:00
|
|
|
webViewController?.webView?.evaluateJavaScript(jsToInject)
|
2018-09-14 00:21:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2018-09-18 01:07:12 +00:00
|
|
|
webViewController?.webView?.evaluateJavaScript(source)
|
2018-09-14 00:21:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func injectScriptCode(arguments: NSDictionary) {
|
|
|
|
let jsWrapper = "(function(){JSON.stringify([eval(%@)])})()"
|
|
|
|
injectDeferredObject(arguments["source"] as! String, withWrapper: jsWrapper)
|
|
|
|
}
|
|
|
|
|
|
|
|
public func injectScriptFile(arguments: NSDictionary) {
|
|
|
|
let jsWrapper = "(function(d) { var c = d.createElement('script'); c.src = %@; d.body.appendChild(c); })(document)"
|
|
|
|
injectDeferredObject(arguments["urlFile"] as! String, withWrapper: jsWrapper)
|
|
|
|
}
|
|
|
|
|
|
|
|
public func injectStyleCode(arguments: NSDictionary) {
|
|
|
|
let jsWrapper = "(function(d) { var c = d.createElement('style'); c.innerHTML = %@; d.body.appendChild(c); })(document)"
|
|
|
|
injectDeferredObject(arguments["source"] as! String, withWrapper: jsWrapper)
|
|
|
|
}
|
|
|
|
|
|
|
|
public func injectStyleFile(arguments: NSDictionary) {
|
|
|
|
let jsWrapper = "(function(d) { var c = d.createElement('link'); c.rel='stylesheet', c.type='text/css'; c.href = %@; d.body.appendChild(c); })(document)"
|
|
|
|
injectDeferredObject(arguments["urlFile"] as! String, withWrapper: jsWrapper)
|
|
|
|
}
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
func webViewDidStartLoad(_ webView: WKWebView) {
|
|
|
|
let url: String = webViewController!.currentURL!.absoluteString
|
2018-09-14 00:21:51 +00:00
|
|
|
channel.invokeMethod("loadstart", arguments: ["type": "loadstart", "url": url])
|
|
|
|
}
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
func webViewDidFinishLoad(_ webView: WKWebView) {
|
|
|
|
let url: String = webViewController!.currentURL!.absoluteString
|
2018-09-14 00:21:51 +00:00
|
|
|
channel.invokeMethod("loadstop", arguments: ["type": "loadstop", "url": url])
|
|
|
|
}
|
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
func webView(_ webView: WKWebView, didFailLoadWithError error: Error) {
|
|
|
|
let url: String = webViewController!.currentURL!.absoluteString
|
2018-09-14 00:21:51 +00:00
|
|
|
let arguments = ["type": "loaderror", "url": url, "code": error._code, "message": error.localizedDescription] as [String : Any]
|
|
|
|
channel.invokeMethod("loaderror", arguments: arguments)
|
|
|
|
}
|
|
|
|
|
|
|
|
func browserExit() {
|
|
|
|
|
|
|
|
channel.invokeMethod("exit", arguments: ["type": "exit"])
|
2018-09-18 01:07:12 +00:00
|
|
|
|
2018-09-14 00:21:51 +00:00
|
|
|
// Set navigationDelegate to nil to ensure no callbacks are received from it.
|
2018-09-18 01:07:12 +00:00
|
|
|
webViewController?.navigationDelegate = nil
|
2018-09-14 00:21:51 +00:00
|
|
|
// Don't recycle the ViewController since it may be consuming a lot of memory.
|
|
|
|
// Also - this is required for the PDF/User-Agent bug work-around.
|
2018-09-18 01:07:12 +00:00
|
|
|
webViewController = nil
|
2018-09-14 00:21:51 +00:00
|
|
|
|
2018-09-18 01:07:12 +00:00
|
|
|
if previousStatusBarStyle != -1 {
|
|
|
|
UIApplication.shared.statusBarStyle = UIStatusBarStyle(rawValue: previousStatusBarStyle)!
|
|
|
|
}
|
2018-09-14 00:21:51 +00:00
|
|
|
|
|
|
|
previousStatusBarStyle = -1
|
|
|
|
// this value was reset before reapplying it. caused statusbar to stay black on ios7
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|