2071 lines
93 KiB
Swift
Executable File
2071 lines
93 KiB
Swift
Executable File
//
|
|
// InAppWebView.swift
|
|
// flutter_inappbrowser
|
|
//
|
|
// Created by Lorenzo on 21/10/18.
|
|
//
|
|
|
|
import Flutter
|
|
import Foundation
|
|
import WebKit
|
|
|
|
func currentTimeInMilliSeconds() -> Int64 {
|
|
let currentDate = Date()
|
|
let since1970 = currentDate.timeIntervalSince1970
|
|
return Int64(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)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func JSONStringify(value: Any, prettyPrinted: Bool = false) -> String {
|
|
let options: JSONSerialization.WritingOptions = prettyPrinted ? .prettyPrinted : .init(rawValue: 0)
|
|
if JSONSerialization.isValidJSONObject(value) {
|
|
let data = try? JSONSerialization.data(withJSONObject: value, options: options)
|
|
if data != nil {
|
|
if let string = String(data: data!, encoding: .utf8) {
|
|
return string
|
|
}
|
|
}
|
|
}
|
|
return ""
|
|
}
|
|
|
|
let JAVASCRIPT_BRIDGE_NAME = "flutter_inappbrowser"
|
|
|
|
// the message needs to be concatenated with '' in order to have the same behavior like on Android
|
|
let consoleLogJS = """
|
|
(function(console) {
|
|
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);
|
|
}
|
|
})(window.console);
|
|
"""
|
|
|
|
let javaScriptBridgeJS = """
|
|
window.\(JAVASCRIPT_BRIDGE_NAME) = {};
|
|
window.\(JAVASCRIPT_BRIDGE_NAME).callHandler = function() {
|
|
var _callHandlerID = setTimeout(function(){});
|
|
window.webkit.messageHandlers['callHandler'].postMessage( {'handlerName': arguments[0], '_callHandlerID': _callHandlerID, 'args': JSON.stringify(Array.prototype.slice.call(arguments, 1))} );
|
|
return new Promise(function(resolve, reject) {
|
|
window.\(JAVASCRIPT_BRIDGE_NAME)[_callHandlerID] = resolve;
|
|
});
|
|
}
|
|
"""
|
|
|
|
let platformReadyJS = "window.dispatchEvent(new Event('flutterInAppBrowserPlatformReady'));";
|
|
|
|
let findTextHighlightJS = """
|
|
var wkwebview_SearchResultCount = 0;
|
|
var wkwebview_CurrentHighlight = 0;
|
|
var wkwebview_IsDoneCounting = false;
|
|
|
|
function wkwebview_FindAllAsyncForElement(element, keyword) {
|
|
if (element) {
|
|
if (element.nodeType == 3) {
|
|
// Text node
|
|
|
|
var elementTmp = element;
|
|
while (true) {
|
|
var value = elementTmp.nodeValue; // Search for keyword in text node
|
|
var idx = value.toLowerCase().indexOf(keyword);
|
|
|
|
if (idx < 0) break;
|
|
|
|
var span = document.createElement("span");
|
|
var text = document.createTextNode(value.substr(idx, keyword.length));
|
|
span.appendChild(text);
|
|
|
|
span.setAttribute(
|
|
"id",
|
|
"WKWEBVIEW_SEARCH_WORD_" + wkwebview_SearchResultCount
|
|
);
|
|
span.setAttribute("class", "wkwebview_Highlight");
|
|
var backgroundColor = wkwebview_SearchResultCount == 0 ? "#FF9732" : "#FFFF00";
|
|
span.setAttribute("style", "color: #000 !important; background: " + backgroundColor + " !important; padding: 0px !important; margin: 0px !important; border: 0px !important;");
|
|
|
|
text = document.createTextNode(value.substr(idx + keyword.length));
|
|
element.deleteData(idx, value.length - idx);
|
|
|
|
var next = element.nextSibling;
|
|
element.parentNode.insertBefore(span, next);
|
|
element.parentNode.insertBefore(text, next);
|
|
element = text;
|
|
|
|
wkwebview_SearchResultCount++;
|
|
elementTmp = document.createTextNode(
|
|
value.substr(idx + keyword.length)
|
|
);
|
|
|
|
window.webkit.messageHandlers["onFindResultReceived"].postMessage(
|
|
JSON.stringify({
|
|
activeMatchOrdinal: wkwebview_CurrentHighlight,
|
|
numberOfMatches: wkwebview_SearchResultCount,
|
|
isDoneCounting: wkwebview_IsDoneCounting
|
|
})
|
|
);
|
|
}
|
|
} else if (element.nodeType == 1) {
|
|
// Element node
|
|
if (
|
|
element.style.display != "none" &&
|
|
element.nodeName.toLowerCase() != "select"
|
|
) {
|
|
for (var i = element.childNodes.length - 1; i >= 0; i--) {
|
|
wkwebview_FindAllAsyncForElement(
|
|
element.childNodes[element.childNodes.length - 1 - i],
|
|
keyword
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// the main entry point to start the search
|
|
function wkwebview_FindAllAsync(keyword) {
|
|
wkwebview_ClearMatches();
|
|
wkwebview_FindAllAsyncForElement(document.body, keyword.toLowerCase());
|
|
wkwebview_IsDoneCounting = true;
|
|
window.webkit.messageHandlers["onFindResultReceived"].postMessage(
|
|
JSON.stringify({
|
|
activeMatchOrdinal: wkwebview_CurrentHighlight,
|
|
numberOfMatches: wkwebview_SearchResultCount,
|
|
isDoneCounting: wkwebview_IsDoneCounting
|
|
})
|
|
);
|
|
}
|
|
|
|
// helper function, recursively removes the highlights in elements and their childs
|
|
function wkwebview_ClearMatchesForElement(element) {
|
|
if (element) {
|
|
if (element.nodeType == 1) {
|
|
if (element.getAttribute("class") == "wkwebview_Highlight") {
|
|
var text = element.removeChild(element.firstChild);
|
|
element.parentNode.insertBefore(text, element);
|
|
element.parentNode.removeChild(element);
|
|
return true;
|
|
} else {
|
|
var normalize = false;
|
|
for (var i = element.childNodes.length - 1; i >= 0; i--) {
|
|
if (wkwebview_ClearMatchesForElement(element.childNodes[i])) {
|
|
normalize = true;
|
|
}
|
|
}
|
|
if (normalize) {
|
|
element.normalize();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// the main entry point to remove the highlights
|
|
function wkwebview_ClearMatches() {
|
|
wkwebview_SearchResultCount = 0;
|
|
wkwebview_CurrentHighlight = 0;
|
|
wkwebview_ClearMatchesForElement(document.body);
|
|
}
|
|
|
|
function wkwebview_FindNext(forward) {
|
|
if (wkwebview_SearchResultCount <= 0) return;
|
|
|
|
var idx = wkwebview_CurrentHighlight + (forward ? +1 : -1);
|
|
idx =
|
|
idx < 0
|
|
? wkwebview_SearchResultCount - 1
|
|
: idx >= wkwebview_SearchResultCount
|
|
? 0
|
|
: idx;
|
|
wkwebview_CurrentHighlight = idx;
|
|
|
|
var scrollTo = document.getElementById("WKWEBVIEW_SEARCH_WORD_" + idx);
|
|
if (scrollTo) {
|
|
var highlights = document.getElementsByClassName("wkwebview_Highlight");
|
|
for (var i = 0; i < highlights.length; i++) {
|
|
var span = highlights[i];
|
|
span.style.backgroundColor = "#FFFF00";
|
|
}
|
|
scrollTo.style.backgroundColor = "#FF9732";
|
|
|
|
scrollTo.scrollIntoView({
|
|
behavior: "auto",
|
|
block: "center"
|
|
});
|
|
|
|
window.webkit.messageHandlers["onFindResultReceived"].postMessage(
|
|
JSON.stringify({
|
|
activeMatchOrdinal: wkwebview_CurrentHighlight,
|
|
numberOfMatches: wkwebview_SearchResultCount,
|
|
isDoneCounting: wkwebview_IsDoneCounting
|
|
})
|
|
);
|
|
}
|
|
}
|
|
"""
|
|
|
|
let variableForOnLoadResourceJS = "window._flutter_inappbrowser_useOnLoadResource"
|
|
let enableVariableForOnLoadResourceJS = "\(variableForOnLoadResourceJS) = $PLACEHOLDER_VALUE;"
|
|
|
|
let resourceObserverJS = """
|
|
(function() {
|
|
var observer = new PerformanceObserver(function(list) {
|
|
list.getEntries().forEach(function(entry) {
|
|
if (window.\(variableForOnLoadResourceJS) == null || window.\(variableForOnLoadResourceJS) == true) {
|
|
window.\(JAVASCRIPT_BRIDGE_NAME).callHandler("onLoadResource", entry);
|
|
}
|
|
});
|
|
});
|
|
observer.observe({entryTypes: ['resource']});
|
|
})();
|
|
"""
|
|
|
|
let variableForShouldInterceptAjaxRequestJS = "window._flutter_inappbrowser_useShouldInterceptAjaxRequest"
|
|
let enableVariableForShouldInterceptAjaxRequestJS = "\(variableForShouldInterceptAjaxRequestJS) = $PLACEHOLDER_VALUE;"
|
|
|
|
let interceptAjaxRequestsJS = """
|
|
(function(ajax) {
|
|
var send = ajax.prototype.send;
|
|
var open = ajax.prototype.open;
|
|
var setRequestHeader = ajax.prototype.setRequestHeader;
|
|
ajax.prototype._flutter_inappbrowser_url = null;
|
|
ajax.prototype._flutter_inappbrowser_method = null;
|
|
ajax.prototype._flutter_inappbrowser_isAsync = null;
|
|
ajax.prototype._flutter_inappbrowser_user = null;
|
|
ajax.prototype._flutter_inappbrowser_password = null;
|
|
ajax.prototype._flutter_inappbrowser_password = null;
|
|
ajax.prototype._flutter_inappbrowser_already_onreadystatechange_wrapped = false;
|
|
ajax.prototype._flutter_inappbrowser_request_headers = {};
|
|
ajax.prototype.open = function(method, url, isAsync, user, password) {
|
|
isAsync = (isAsync != null) ? isAsync : true;
|
|
this._flutter_inappbrowser_url = url;
|
|
this._flutter_inappbrowser_method = method;
|
|
this._flutter_inappbrowser_isAsync = isAsync;
|
|
this._flutter_inappbrowser_user = user;
|
|
this._flutter_inappbrowser_password = password;
|
|
open.call(this, method, url, isAsync, user, password);
|
|
};
|
|
ajax.prototype.setRequestHeader = function(header, value) {
|
|
this._flutter_inappbrowser_request_headers[header] = value;
|
|
setRequestHeader.call(this, header, value);
|
|
};
|
|
function handleEvent(e) {
|
|
var self = this;
|
|
if (window.\(variableForShouldInterceptAjaxRequestJS) == null || window.\(variableForShouldInterceptAjaxRequestJS) == true) {
|
|
var headers = this.getAllResponseHeaders();
|
|
var responseHeaders = {};
|
|
if (headers != null) {
|
|
var arr = headers.trim().split(/[\\r\\n]+/);
|
|
arr.forEach(function (line) {
|
|
var parts = line.split(': ');
|
|
var header = parts.shift();
|
|
var value = parts.join(': ');
|
|
responseHeaders[header] = value;
|
|
});
|
|
}
|
|
var ajaxRequest = {
|
|
method: this._flutter_inappbrowser_method,
|
|
url: this._flutter_inappbrowser_url,
|
|
isAsync: this._flutter_inappbrowser_isAsync,
|
|
user: this._flutter_inappbrowser_user,
|
|
password: this._flutter_inappbrowser_password,
|
|
withCredentials: this.withCredentials,
|
|
headers: this._flutter_inappbrowser_request_headers,
|
|
readyState: this.readyState,
|
|
status: this.status,
|
|
responseURL: this.responseURL,
|
|
responseType: this.responseType,
|
|
responseText: this.responseText,
|
|
statusText: this.statusText,
|
|
responseHeaders, responseHeaders,
|
|
event: {
|
|
type: e.type,
|
|
loaded: e.loaded,
|
|
lengthComputable: e.lengthComputable
|
|
total: e.total
|
|
}
|
|
};
|
|
window.\(JAVASCRIPT_BRIDGE_NAME).callHandler('onAjaxProgress', ajaxRequest).then(function(result) {
|
|
if (result != null) {
|
|
switch (result) {
|
|
case 0:
|
|
self.abort();
|
|
return;
|
|
};
|
|
}
|
|
});
|
|
}
|
|
};
|
|
ajax.prototype.send = function(data) {
|
|
var self = this;
|
|
if (window.\(variableForShouldInterceptAjaxRequestJS) == null || window.\(variableForShouldInterceptAjaxRequestJS) == true) {
|
|
if (!this._flutter_inappbrowser_already_onreadystatechange_wrapped) {
|
|
this._flutter_inappbrowser_already_onreadystatechange_wrapped = true;
|
|
var onreadystatechange = this.onreadystatechange;
|
|
this.onreadystatechange = function() {
|
|
if (window.\(variableForShouldInterceptAjaxRequestJS) == null || window.\(variableForShouldInterceptAjaxRequestJS) == true) {
|
|
var headers = this.getAllResponseHeaders();
|
|
var responseHeaders = {};
|
|
if (headers != null) {
|
|
var arr = headers.trim().split(/[\\r\\n]+/);
|
|
arr.forEach(function (line) {
|
|
var parts = line.split(': ');
|
|
var header = parts.shift();
|
|
var value = parts.join(': ');
|
|
responseHeaders[header] = value;
|
|
});
|
|
}
|
|
var ajaxRequest = {
|
|
method: this._flutter_inappbrowser_method,
|
|
url: this._flutter_inappbrowser_url,
|
|
isAsync: this._flutter_inappbrowser_isAsync,
|
|
user: this._flutter_inappbrowser_user,
|
|
password: this._flutter_inappbrowser_password,
|
|
withCredentials: this.withCredentials,
|
|
headers: this._flutter_inappbrowser_request_headers,
|
|
readyState: this.readyState,
|
|
status: this.status,
|
|
responseURL: this.responseURL,
|
|
responseType: this.responseType,
|
|
responseText: this.responseText,
|
|
statusText: this.statusText,
|
|
responseHeaders: responseHeaders
|
|
};
|
|
window.\(JAVASCRIPT_BRIDGE_NAME).callHandler('onAjaxReadyStateChange', ajaxRequest).then(function(result) {
|
|
if (result != null) {
|
|
switch (result) {
|
|
case 0:
|
|
self.abort();
|
|
return;
|
|
};
|
|
}
|
|
if (onreadystatechange != null) {
|
|
onreadystatechange();
|
|
}
|
|
});
|
|
} else if (onreadystatechange != null) {
|
|
onreadystatechange();
|
|
}
|
|
};
|
|
}
|
|
this.addEventListener('loadstart', handleEvent);
|
|
this.addEventListener('load', handleEvent);
|
|
this.addEventListener('loadend', handleEvent);
|
|
this.addEventListener('progress', handleEvent);
|
|
this.addEventListener('error', handleEvent);
|
|
this.addEventListener('abort', handleEvent);
|
|
this.addEventListener('timeout', handleEvent);
|
|
var ajaxRequest = {
|
|
data: data,
|
|
method: this._flutter_inappbrowser_method,
|
|
url: this._flutter_inappbrowser_url,
|
|
isAsync: this._flutter_inappbrowser_isAsync,
|
|
user: this._flutter_inappbrowser_user,
|
|
password: this._flutter_inappbrowser_password,
|
|
withCredentials: this.withCredentials,
|
|
headers: this._flutter_inappbrowser_request_headers
|
|
};
|
|
window.\(JAVASCRIPT_BRIDGE_NAME).callHandler('shouldInterceptAjaxRequest', ajaxRequest).then(function(result) {
|
|
if (result != null) {
|
|
switch (result.action) {
|
|
case 0:
|
|
self.abort();
|
|
return;
|
|
};
|
|
data = result.data;
|
|
self.withCredentials = result.withCredentials;
|
|
for (var header in result.headers) {
|
|
var value = result.headers[header];
|
|
self.setRequestHeader(header, value);
|
|
};
|
|
if ((self._flutter_inappbrowser_method != result.method && result.method != null) || (self._flutter_inappbrowser_url != result.url && result.url != null)) {
|
|
self.abort();
|
|
self.open(result.method, result.url, result.isAsync, result.user, result.password);
|
|
return;
|
|
}
|
|
}
|
|
send.call(self, data);
|
|
});
|
|
} else {
|
|
send.call(this, data);
|
|
}
|
|
};
|
|
})(window.XMLHttpRequest);
|
|
"""
|
|
|
|
let variableForShouldInterceptFetchRequestsJS = "window._flutter_inappbrowser_useShouldInterceptFetchRequest"
|
|
let enableVariableForShouldInterceptFetchRequestsJS = "\(variableForShouldInterceptFetchRequestsJS) = $PLACEHOLDER_VALUE;"
|
|
|
|
let interceptFetchRequestsJS = """
|
|
(function(fetch) {
|
|
if (fetch == null) {
|
|
return;
|
|
}
|
|
function convertHeadersToJson(headers) {
|
|
var headersObj = {};
|
|
for (var header of headers.keys()) {
|
|
var value = headers.get(header);
|
|
headersObj[header] = value;
|
|
}
|
|
return headersObj;
|
|
}
|
|
function convertJsonToHeaders(headersJson) {
|
|
return new Headers(headersJson);
|
|
}
|
|
function convertBodyToArray(body) {
|
|
return new Response(body).arrayBuffer().then(function(arrayBuffer) {
|
|
var arr = Array.from(new Uint8Array(arrayBuffer));
|
|
return arr;
|
|
})
|
|
}
|
|
function convertArrayIntBodyToUint8Array(arrayIntBody) {
|
|
return new Uint8Array(arrayIntBody);
|
|
}
|
|
function convertCredentialsToJson(credentials) {
|
|
var credentialsObj = {};
|
|
if (window.FederatedCredential != null && credentials instanceof FederatedCredential) {
|
|
credentialsObj.type = credentials.type;
|
|
credentialsObj.id = credentials.id;
|
|
credentialsObj.name = credentials.name;
|
|
credentialsObj.protocol = credentials.protocol;
|
|
credentialsObj.provider = credentials.provider;
|
|
credentialsObj.iconURL = credentials.iconURL;
|
|
} else if (window.PasswordCredential != null && credentials instanceof PasswordCredential) {
|
|
credentialsObj.type = credentials.type;
|
|
credentialsObj.id = credentials.id;
|
|
credentialsObj.name = credentials.name;
|
|
credentialsObj.password = credentials.password;
|
|
credentialsObj.iconURL = credentials.iconURL;
|
|
} else {
|
|
credentialsObj.type = 'default';
|
|
credentialsObj.value = credentials;
|
|
}
|
|
}
|
|
function convertJsonToCredential(credentialsJson) {
|
|
var credentials;
|
|
if (window.FederatedCredential != null && credentialsJson.type === 'federated') {
|
|
credentials = new FederatedCredential({
|
|
id: credentialsJson.id,
|
|
name: credentialsJson.name,
|
|
protocol: credentialsJson.protocol,
|
|
provider: credentialsJson.provider,
|
|
iconURL: credentialsJson.iconURL
|
|
});
|
|
} else if (window.PasswordCredential != null && credentialsJson.type === 'password') {
|
|
credentials = new PasswordCredential({
|
|
id: credentialsJson.id,
|
|
name: credentialsJson.name,
|
|
password: credentialsJson.password,
|
|
iconURL: credentialsJson.iconURL
|
|
});
|
|
} else {
|
|
credentials = credentialsJson;
|
|
}
|
|
return credentials;
|
|
}
|
|
window.fetch = async function(resource, init) {
|
|
if (window.\(variableForShouldInterceptFetchRequestsJS) == null || window.\(variableForShouldInterceptFetchRequestsJS) == true) {
|
|
var fetchRequest = {
|
|
url: null,
|
|
method: null,
|
|
headers: null,
|
|
body: null,
|
|
mode: null,
|
|
credentials: null,
|
|
cache: null,
|
|
redirect: null,
|
|
referrer: null,
|
|
referrerPolicy: null,
|
|
integrity: null,
|
|
keepalive: null
|
|
};
|
|
if (resource instanceof Request) {
|
|
fetchRequest.url = resource.url;
|
|
fetchRequest.method = resource.method;
|
|
fetchRequest.headers = resource.headers;
|
|
fetchRequest.body = resource.body;
|
|
fetchRequest.mode = resource.mode;
|
|
fetchRequest.credentials = resource.credentials;
|
|
fetchRequest.cache = resource.cache;
|
|
fetchRequest.redirect = resource.redirect;
|
|
fetchRequest.referrer = resource.referrer;
|
|
fetchRequest.referrerPolicy = resource.referrerPolicy;
|
|
fetchRequest.integrity = resource.integrity;
|
|
fetchRequest.keepalive = resource.keepalive;
|
|
} else {
|
|
fetchRequest.url = resource;
|
|
if (init != null) {
|
|
fetchRequest.method = init.method;
|
|
fetchRequest.headers = init.headers;
|
|
fetchRequest.body = init.body;
|
|
fetchRequest.mode = init.mode;
|
|
fetchRequest.credentials = init.credentials;
|
|
fetchRequest.cache = init.cache;
|
|
fetchRequest.redirect = init.redirect;
|
|
fetchRequest.referrer = init.referrer;
|
|
fetchRequest.referrerPolicy = init.referrerPolicy;
|
|
fetchRequest.integrity = init.integrity;
|
|
fetchRequest.keepalive = init.keepalive;
|
|
}
|
|
}
|
|
if (fetchRequest.headers instanceof Headers) {
|
|
fetchRequest.headers = convertHeadersToJson(fetchRequest.headers);
|
|
}
|
|
fetchRequest.credentials = convertCredentialsToJson(fetchRequest.credentials);
|
|
return convertBodyToArray(fetchRequest.body).then(function(body) {
|
|
fetchRequest.body = body;
|
|
return window.\(JAVASCRIPT_BRIDGE_NAME).callHandler('shouldInterceptFetchRequest', fetchRequest).then(function(result) {
|
|
if (result != null) {
|
|
switch (result.action) {
|
|
case 0:
|
|
var controller = new AbortController();
|
|
if (init != null) {
|
|
init.signal = controller.signal;
|
|
} else {
|
|
init = {
|
|
signal: controller.signal
|
|
};
|
|
}
|
|
controller.abort();
|
|
break;
|
|
}
|
|
var resultResource = (result.url != null) ? result.url : resource;
|
|
var resultInit = init;
|
|
if (result.init != null) {
|
|
resultInit.method = result.method;
|
|
resultInit.headers = convertJsonToHeaders(result.headers);
|
|
resultInit.body = convertArrayIntBodyToUint8Array(result.body);
|
|
resultInit.mode = result.mode;
|
|
resultInit.credentials = convertJsonToCredential(result.credentials);
|
|
resultInit.cache = result.cache;
|
|
resultInit.redirect = result.redirect;
|
|
resultInit.referrer = result.referrer;
|
|
resultInit.referrerPolicy = result.referrerPolicy;
|
|
resultInit.integrity = result.integrity;
|
|
resultInit.keepalive = result.keepalive;
|
|
}
|
|
return fetch(resultResource, resultInit);
|
|
}
|
|
return fetch(resource, init);
|
|
});
|
|
});
|
|
} else {
|
|
return fetch(resource, init);
|
|
}
|
|
};
|
|
})(window.fetch);
|
|
"""
|
|
|
|
let interceptNavigationStateChangeJS = """
|
|
(function(window, document, history) {
|
|
history.pushState = (function(f) {
|
|
return function pushState(){
|
|
var ret = f.apply(this, arguments);
|
|
window.dispatchEvent(new Event('pushstate'));
|
|
window.dispatchEvent(new Event('_flutter_inappbrowser_locationchange'));
|
|
return ret;
|
|
};
|
|
})(history.pushState);
|
|
history.replaceState = ( function(f) {
|
|
return function replaceState(){
|
|
var ret = f.apply(this, arguments);
|
|
window.dispatchEvent(new Event('replacestate'));
|
|
window.dispatchEvent(new Event('_flutter_inappbrowser_locationchange'));
|
|
return ret;
|
|
};
|
|
})(history.replaceState);
|
|
window.addEventListener('popstate',function() {
|
|
window.dispatchEvent(new Event('_flutter_inappbrowser_locationchange'));
|
|
});
|
|
window.addEventListener('_flutter_inappbrowser_locationchange', function() {
|
|
window.webkit.messageHandlers["onNavigationStateChange"].postMessage(JSON.stringify({
|
|
url: document.location.href
|
|
}));
|
|
});
|
|
})(window, window.document, window.history);
|
|
"""
|
|
|
|
public class InAppWebView: WKWebView, UIScrollViewDelegate, WKUIDelegate, WKNavigationDelegate, WKScriptMessageHandler {
|
|
|
|
var IABController: InAppBrowserWebViewController?
|
|
var IAWController: FlutterWebViewController?
|
|
var options: InAppWebViewOptions?
|
|
var currentURL: URL?
|
|
var startPageTime: Int64 = 0
|
|
static var credentialsProposed: [URLCredential] = []
|
|
|
|
init(frame: CGRect, configuration: WKWebViewConfiguration, IABController: InAppBrowserWebViewController?, IAWController: FlutterWebViewController?) {
|
|
|
|
super.init(frame: frame, configuration: configuration)
|
|
self.IABController = IABController
|
|
self.IAWController = IAWController
|
|
uiDelegate = self
|
|
navigationDelegate = self
|
|
scrollView.delegate = self
|
|
}
|
|
|
|
required public init(coder aDecoder: NSCoder) {
|
|
super.init(coder: aDecoder)!
|
|
}
|
|
|
|
public func prepare() {
|
|
addObserver(self,
|
|
forKeyPath: #keyPath(WKWebView.estimatedProgress),
|
|
options: .new,
|
|
context: nil)
|
|
|
|
configuration.userContentController = WKUserContentController()
|
|
configuration.preferences = WKPreferences()
|
|
|
|
if (options?.transparentBackground)! {
|
|
isOpaque = false
|
|
backgroundColor = UIColor.clear
|
|
scrollView.backgroundColor = UIColor.clear
|
|
}
|
|
|
|
// prevent webView from bouncing
|
|
if (options?.disallowOverScroll)! {
|
|
if responds(to: #selector(getter: scrollView)) {
|
|
scrollView.bounces = false
|
|
}
|
|
else {
|
|
for subview: UIView in subviews {
|
|
if subview is UIScrollView {
|
|
(subview as! UIScrollView).bounces = false
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (options?.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)
|
|
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)
|
|
configuration.userContentController.addUserScript(jscriptWebkitTouchCallout)
|
|
|
|
let consoleLogJSScript = WKUserScript(source: consoleLogJS, injectionTime: .atDocumentStart, forMainFrameOnly: false)
|
|
configuration.userContentController.addUserScript(consoleLogJSScript)
|
|
configuration.userContentController.add(self, name: "consoleLog")
|
|
configuration.userContentController.add(self, name: "consoleDebug")
|
|
configuration.userContentController.add(self, name: "consoleError")
|
|
configuration.userContentController.add(self, name: "consoleInfo")
|
|
configuration.userContentController.add(self, name: "consoleWarn")
|
|
|
|
let javaScriptBridgeJSScript = WKUserScript(source: javaScriptBridgeJS, injectionTime: .atDocumentStart, forMainFrameOnly: false)
|
|
configuration.userContentController.addUserScript(javaScriptBridgeJSScript)
|
|
configuration.userContentController.add(self, name: "callHandler")
|
|
|
|
if (options?.useOnLoadResource)! {
|
|
let resourceObserverJSScript = WKUserScript(source: resourceObserverJS, injectionTime: .atDocumentStart, forMainFrameOnly: false)
|
|
configuration.userContentController.addUserScript(resourceObserverJSScript)
|
|
}
|
|
|
|
let findTextHighlightJSScript = WKUserScript(source: findTextHighlightJS, injectionTime: .atDocumentStart, forMainFrameOnly: false)
|
|
configuration.userContentController.addUserScript(findTextHighlightJSScript)
|
|
configuration.userContentController.add(self, name: "onFindResultReceived")
|
|
|
|
let interceptNavigationStateChangeJSScript = WKUserScript(source: interceptNavigationStateChangeJS, injectionTime: .atDocumentStart, forMainFrameOnly: false)
|
|
configuration.userContentController.addUserScript(interceptNavigationStateChangeJSScript)
|
|
configuration.userContentController.add(self, name: "onNavigationStateChange")
|
|
|
|
|
|
if (options?.useShouldInterceptAjaxRequest)! {
|
|
let interceptAjaxRequestsJSScript = WKUserScript(source: interceptAjaxRequestsJS, injectionTime: .atDocumentStart, forMainFrameOnly: false)
|
|
configuration.userContentController.addUserScript(interceptAjaxRequestsJSScript)
|
|
}
|
|
|
|
if (options?.useShouldInterceptFetchRequest)! {
|
|
let interceptFetchRequestsJSScript = WKUserScript(source: interceptFetchRequestsJS, injectionTime: .atDocumentStart, forMainFrameOnly: false)
|
|
configuration.userContentController.addUserScript(interceptFetchRequestsJSScript)
|
|
}
|
|
|
|
if #available(iOS 9.0, *) {
|
|
if ((options?.incognito)!) {
|
|
configuration.websiteDataStore = WKWebsiteDataStore.nonPersistent()
|
|
} else if ((options?.cacheEnabled)!) {
|
|
configuration.websiteDataStore = WKWebsiteDataStore.default()
|
|
}
|
|
}
|
|
|
|
if #available(iOS 11.0, *) {
|
|
if((options?.sharedCookiesEnabled)!) {
|
|
// More info to sending cookies with WKWebView
|
|
// https://stackoverflow.com/questions/26573137/can-i-set-the-cookies-to-be-used-by-a-wkwebview/26577303#26577303
|
|
// Set Cookies in iOS 11 and above, initialize websiteDataStore before setting cookies
|
|
// See also https://forums.developer.apple.com/thread/97194
|
|
// check if websiteDataStore has not been initialized before
|
|
if(!(options?.incognito)! && !(options?.cacheEnabled)!) {
|
|
configuration.websiteDataStore = WKWebsiteDataStore.nonPersistent()
|
|
}
|
|
for cookie in HTTPCookieStorage.shared.cookies ?? [] {
|
|
configuration.websiteDataStore.httpCookieStore.setCookie(cookie, completionHandler: nil)
|
|
}
|
|
}
|
|
}
|
|
|
|
configuration.suppressesIncrementalRendering = (options?.suppressesIncrementalRendering)!
|
|
allowsBackForwardNavigationGestures = (options?.allowsBackForwardNavigationGestures)!
|
|
if #available(iOS 9.0, *) {
|
|
allowsLinkPreview = (options?.allowsLinkPreview)!
|
|
configuration.allowsPictureInPictureMediaPlayback = (options?.allowsPictureInPictureMediaPlayback)!
|
|
if (options?.applicationNameForUserAgent != nil && (options?.applicationNameForUserAgent)! != "") {
|
|
configuration.applicationNameForUserAgent = (options?.applicationNameForUserAgent)!
|
|
}
|
|
if (options?.userAgent != nil && (options?.userAgent)! != "") {
|
|
customUserAgent = (options?.userAgent)!
|
|
}
|
|
}
|
|
|
|
configuration.preferences.javaScriptCanOpenWindowsAutomatically = (options?.javaScriptCanOpenWindowsAutomatically)!
|
|
configuration.preferences.javaScriptEnabled = (options?.javaScriptEnabled)!
|
|
configuration.preferences.minimumFontSize = CGFloat((options?.minimumFontSize)!)
|
|
configuration.selectionGranularity = WKSelectionGranularity.init(rawValue: (options?.selectionGranularity)!)!
|
|
|
|
if #available(iOS 10.0, *) {
|
|
configuration.ignoresViewportScaleLimits = (options?.ignoresViewportScaleLimits)!
|
|
|
|
var dataDetectorTypes = WKDataDetectorTypes.init(rawValue: 0)
|
|
for type in options?.dataDetectorTypes ?? [] {
|
|
let dataDetectorType = getDataDetectorType(type: type)
|
|
dataDetectorTypes = WKDataDetectorTypes(rawValue: dataDetectorTypes.rawValue | dataDetectorType.rawValue)
|
|
}
|
|
configuration.dataDetectorTypes = dataDetectorTypes
|
|
} else {
|
|
// Fallback on earlier versions
|
|
}
|
|
|
|
if #available(iOS 13.0, *) {
|
|
configuration.preferences.isFraudulentWebsiteWarningEnabled = (options?.isFraudulentWebsiteWarningEnabled)!
|
|
if options?.preferredContentMode != nil {
|
|
configuration.defaultWebpagePreferences.preferredContentMode = WKWebpagePreferences.ContentMode(rawValue: (options?.preferredContentMode)!)!
|
|
}
|
|
} else {
|
|
// Fallback on earlier versions
|
|
}
|
|
|
|
scrollView.showsVerticalScrollIndicator = (options?.verticalScrollBarEnabled)!
|
|
scrollView.showsHorizontalScrollIndicator = (options?.horizontalScrollBarEnabled)!
|
|
|
|
// options.debuggingEnabled is always enabled for iOS.
|
|
|
|
if (options?.clearCache)! {
|
|
clearCache()
|
|
}
|
|
}
|
|
|
|
@available(iOS 10.0, *)
|
|
public func getDataDetectorType(type: String) -> WKDataDetectorTypes {
|
|
switch type {
|
|
case "NONE":
|
|
return WKDataDetectorTypes.init(rawValue: 0)
|
|
case "PHONE_NUMBER":
|
|
return .phoneNumber
|
|
case "LINK":
|
|
return .link
|
|
case "ADDRESS":
|
|
return .address
|
|
case "CALENDAR_EVENT":
|
|
return .calendarEvent
|
|
case "TRACKING_NUMBER":
|
|
return .trackingNumber
|
|
case "FLIGHT_NUMBER":
|
|
return .flightNumber
|
|
case "LOOKUP_SUGGESTION":
|
|
return .lookupSuggestion
|
|
case "SPOTLIGHT_SUGGESTION":
|
|
return .spotlightSuggestion
|
|
case "ALL":
|
|
return .all
|
|
default:
|
|
return WKDataDetectorTypes.init(rawValue: 0)
|
|
}
|
|
}
|
|
|
|
public static func preWKWebViewConfiguration(options: InAppWebViewOptions?) -> WKWebViewConfiguration {
|
|
let configuration = WKWebViewConfiguration()
|
|
|
|
if #available(iOS 10.0, *) {
|
|
configuration.mediaTypesRequiringUserActionForPlayback = ((options?.mediaPlaybackRequiresUserGesture)!) ? .all : []
|
|
} else {
|
|
// Fallback on earlier versions
|
|
configuration.mediaPlaybackRequiresUserAction = (options?.mediaPlaybackRequiresUserGesture)!
|
|
}
|
|
|
|
configuration.allowsInlineMediaPlayback = (options?.allowsInlineMediaPlayback)!
|
|
|
|
if #available(iOS 11.0, *) {
|
|
if let schemes = options?.resourceCustomSchemes {
|
|
for scheme in schemes {
|
|
configuration.setURLSchemeHandler(CustomeSchemeHandler(), forURLScheme: scheme)
|
|
}
|
|
}
|
|
} else {
|
|
// Fallback on earlier versions
|
|
}
|
|
|
|
return configuration
|
|
}
|
|
|
|
override public func observeValue(forKeyPath keyPath: String?, of object: Any?,
|
|
change: [NSKeyValueChangeKey : Any]?, context: UnsafeMutableRawPointer?) {
|
|
if keyPath == #keyPath(WKWebView.estimatedProgress) {
|
|
let progress = Int(estimatedProgress * 100)
|
|
onProgressChanged(progress: progress)
|
|
}
|
|
}
|
|
|
|
public func goBackOrForward(steps: Int) {
|
|
if canGoBackOrForward(steps: steps) {
|
|
if (steps > 0) {
|
|
let index = steps - 1
|
|
go(to: self.backForwardList.forwardList[index])
|
|
}
|
|
else if (steps < 0){
|
|
let backListLength = self.backForwardList.backList.count
|
|
let index = backListLength + steps
|
|
go(to: self.backForwardList.backList[index])
|
|
}
|
|
}
|
|
}
|
|
|
|
public func canGoBackOrForward(steps: Int) -> Bool {
|
|
let currentIndex = self.backForwardList.backList.count
|
|
return (steps >= 0)
|
|
? steps <= self.backForwardList.forwardList.count
|
|
: currentIndex + steps >= 0
|
|
}
|
|
|
|
public func takeScreenshot (completionHandler: @escaping (_ screenshot: Data?) -> Void) {
|
|
if #available(iOS 11.0, *) {
|
|
takeSnapshot(with: nil, completionHandler: {(image, error) -> Void in
|
|
var imageData: Data? = nil
|
|
if let screenshot = image {
|
|
imageData = screenshot.pngData()!
|
|
}
|
|
completionHandler(imageData)
|
|
})
|
|
} else {
|
|
completionHandler(nil)
|
|
}
|
|
}
|
|
|
|
public func loadUrl(url: URL, headers: [String: String]?) {
|
|
var request = URLRequest(url: url)
|
|
currentURL = url
|
|
if headers != nil {
|
|
if let mutableRequest = (request as NSURLRequest).mutableCopy() as? NSMutableURLRequest {
|
|
for (key, value) in headers! {
|
|
mutableRequest.setValue(value, forHTTPHeaderField: key)
|
|
}
|
|
request = mutableRequest as URLRequest
|
|
}
|
|
}
|
|
load(request)
|
|
}
|
|
|
|
public func postUrl(url: URL, postData: Data, completionHandler: @escaping () -> Void) {
|
|
var request = URLRequest(url: url)
|
|
currentURL = url
|
|
request.httpMethod = "POST"
|
|
request.httpBody = postData
|
|
|
|
let task = URLSession.shared.dataTask(with: request) { (data : Data?, response : URLResponse?, error : Error?) in
|
|
var returnString = ""
|
|
if data != nil {
|
|
returnString = String(data: data!, encoding: .utf8) ?? ""
|
|
}
|
|
DispatchQueue.main.async(execute: {() -> Void in
|
|
self.loadHTMLString(returnString, baseURL: url)
|
|
completionHandler()
|
|
})
|
|
}
|
|
task.resume()
|
|
}
|
|
|
|
public func loadData(data: String, mimeType: String, encoding: String, baseUrl: String) {
|
|
let url = URL(string: baseUrl)!
|
|
currentURL = url
|
|
if #available(iOS 9.0, *) {
|
|
load(data.data(using: .utf8)!, mimeType: mimeType, characterEncodingName: encoding, baseURL: url)
|
|
} else {
|
|
loadHTMLString(data, baseURL: url)
|
|
}
|
|
}
|
|
|
|
public func loadFile(url: String, headers: [String: String]?) throws {
|
|
let key = SwiftFlutterPlugin.instance!.registrar!.lookupKey(forAsset: url)
|
|
let assetURL = Bundle.main.url(forResource: key, withExtension: nil)
|
|
if assetURL == nil {
|
|
throw NSError(domain: url + " asset file cannot be found!", code: 0)
|
|
}
|
|
loadUrl(url: assetURL!, headers: headers)
|
|
}
|
|
|
|
func setOptions(newOptions: InAppWebViewOptions, newOptionsMap: [String: Any]) {
|
|
|
|
if newOptionsMap["transparentBackground"] != nil && options?.transparentBackground != newOptions.transparentBackground {
|
|
if newOptions.transparentBackground {
|
|
isOpaque = false
|
|
backgroundColor = UIColor.clear
|
|
scrollView.backgroundColor = UIColor.clear
|
|
} else {
|
|
isOpaque = true
|
|
backgroundColor = nil
|
|
scrollView.backgroundColor = UIColor(red: 1, green: 1, blue: 1, alpha: 1)
|
|
}
|
|
}
|
|
|
|
if newOptionsMap["disallowOverScroll"] != nil && options?.disallowOverScroll != newOptions.disallowOverScroll {
|
|
if responds(to: #selector(getter: scrollView)) {
|
|
scrollView.bounces = !newOptions.disallowOverScroll
|
|
}
|
|
else {
|
|
for subview: UIView in subviews {
|
|
if subview is UIScrollView {
|
|
(subview as! UIScrollView).bounces = !newOptions.disallowOverScroll
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if #available(iOS 9.0, *) {
|
|
if (newOptionsMap["incognito"] != nil && options?.incognito != newOptions.incognito && newOptions.incognito) {
|
|
configuration.websiteDataStore = WKWebsiteDataStore.nonPersistent()
|
|
} else if (newOptionsMap["cacheEnabled"] != nil && options?.cacheEnabled != newOptions.cacheEnabled && newOptions.cacheEnabled) {
|
|
configuration.websiteDataStore = WKWebsiteDataStore.default()
|
|
}
|
|
}
|
|
|
|
if #available(iOS 11.0, *) {
|
|
if (newOptionsMap["sharedCookiesEnabled"] != nil && options?.sharedCookiesEnabled != newOptions.sharedCookiesEnabled && newOptions.sharedCookiesEnabled) {
|
|
if(!newOptions.incognito && !newOptions.cacheEnabled) {
|
|
configuration.websiteDataStore = WKWebsiteDataStore.nonPersistent()
|
|
}
|
|
for cookie in HTTPCookieStorage.shared.cookies ?? [] {
|
|
configuration.websiteDataStore.httpCookieStore.setCookie(cookie, completionHandler: nil)
|
|
}
|
|
}
|
|
}
|
|
|
|
if newOptionsMap["enableViewportScale"] != nil && options?.enableViewportScale != newOptions.enableViewportScale && newOptions.enableViewportScale {
|
|
let jscript = "var meta = document.createElement('meta'); meta.setAttribute('name', 'viewport'); meta.setAttribute('content', 'width=device-width'); document.getElementsByTagName('head')[0].appendChild(meta);"
|
|
evaluateJavaScript(jscript, completionHandler: nil)
|
|
}
|
|
|
|
if newOptionsMap["useOnLoadResource"] != nil && options?.useOnLoadResource != newOptions.useOnLoadResource && newOptions.useOnLoadResource {
|
|
let placeholderValue = newOptions.useOnLoadResource ? "true" : "false"
|
|
evaluateJavaScript(enableVariableForOnLoadResourceJS.replacingOccurrences(of: "$PLACEHOLDER_VALUE", with: placeholderValue), completionHandler: nil)
|
|
}
|
|
|
|
if newOptionsMap["useShouldInterceptAjaxRequest"] != nil && options?.useShouldInterceptAjaxRequest != newOptions.useShouldInterceptAjaxRequest && newOptions.useShouldInterceptAjaxRequest {
|
|
let placeholderValue = newOptions.useShouldInterceptAjaxRequest ? "true" : "false"
|
|
evaluateJavaScript(enableVariableForShouldInterceptAjaxRequestJS.replacingOccurrences(of: "$PLACEHOLDER_VALUE", with: placeholderValue), completionHandler: nil)
|
|
}
|
|
|
|
if newOptionsMap["useShouldInterceptFetchRequest"] != nil && options?.useShouldInterceptFetchRequest != newOptions.useShouldInterceptFetchRequest && newOptions.useShouldInterceptFetchRequest {
|
|
let placeholderValue = newOptions.useShouldInterceptFetchRequest ? "true" : "false"
|
|
evaluateJavaScript(enableVariableForShouldInterceptFetchRequestsJS.replacingOccurrences(of: "$PLACEHOLDER_VALUE", with: placeholderValue), completionHandler: nil)
|
|
}
|
|
|
|
if newOptionsMap["mediaPlaybackRequiresUserGesture"] != nil && options?.mediaPlaybackRequiresUserGesture != newOptions.mediaPlaybackRequiresUserGesture {
|
|
if #available(iOS 10.0, *) {
|
|
configuration.mediaTypesRequiringUserActionForPlayback = (newOptions.mediaPlaybackRequiresUserGesture) ? .all : []
|
|
} else {
|
|
// Fallback on earlier versions
|
|
configuration.mediaPlaybackRequiresUserAction = newOptions.mediaPlaybackRequiresUserGesture
|
|
}
|
|
}
|
|
|
|
if newOptionsMap["allowsInlineMediaPlayback"] != nil && options?.allowsInlineMediaPlayback != newOptions.allowsInlineMediaPlayback {
|
|
configuration.allowsInlineMediaPlayback = newOptions.allowsInlineMediaPlayback
|
|
}
|
|
|
|
if newOptionsMap["suppressesIncrementalRendering"] != nil && options?.suppressesIncrementalRendering != newOptions.suppressesIncrementalRendering {
|
|
configuration.suppressesIncrementalRendering = newOptions.suppressesIncrementalRendering
|
|
}
|
|
|
|
if newOptionsMap["allowsBackForwardNavigationGestures"] != nil && options?.allowsBackForwardNavigationGestures != newOptions.allowsBackForwardNavigationGestures {
|
|
allowsBackForwardNavigationGestures = newOptions.allowsBackForwardNavigationGestures
|
|
}
|
|
|
|
if newOptionsMap["allowsInlineMediaPlayback"] != nil && options?.allowsInlineMediaPlayback != newOptions.allowsInlineMediaPlayback {
|
|
configuration.allowsInlineMediaPlayback = newOptions.allowsInlineMediaPlayback
|
|
}
|
|
|
|
if newOptionsMap["javaScriptCanOpenWindowsAutomatically"] != nil && options?.javaScriptCanOpenWindowsAutomatically != newOptions.javaScriptCanOpenWindowsAutomatically {
|
|
configuration.preferences.javaScriptCanOpenWindowsAutomatically = newOptions.javaScriptCanOpenWindowsAutomatically
|
|
}
|
|
|
|
if newOptionsMap["javaScriptEnabled"] != nil && options?.javaScriptEnabled != newOptions.javaScriptEnabled {
|
|
configuration.preferences.javaScriptEnabled = newOptions.javaScriptEnabled
|
|
}
|
|
|
|
if newOptionsMap["minimumFontSize"] != nil && options?.minimumFontSize != newOptions.minimumFontSize {
|
|
configuration.preferences.minimumFontSize = CGFloat(newOptions.minimumFontSize)
|
|
}
|
|
|
|
if newOptionsMap["selectionGranularity"] != nil && options?.selectionGranularity != newOptions.selectionGranularity {
|
|
configuration.selectionGranularity = WKSelectionGranularity.init(rawValue: newOptions.selectionGranularity)!
|
|
}
|
|
|
|
if #available(iOS 10.0, *) {
|
|
if newOptionsMap["ignoresViewportScaleLimits"] != nil && options?.ignoresViewportScaleLimits != newOptions.ignoresViewportScaleLimits {
|
|
configuration.ignoresViewportScaleLimits = newOptions.ignoresViewportScaleLimits
|
|
}
|
|
|
|
if newOptionsMap["dataDetectorTypes"] != nil && options?.dataDetectorTypes != newOptions.dataDetectorTypes {
|
|
var dataDetectorTypes = WKDataDetectorTypes.init(rawValue: 0)
|
|
for type in newOptions.dataDetectorTypes {
|
|
let dataDetectorType = getDataDetectorType(type: type)
|
|
dataDetectorTypes = WKDataDetectorTypes(rawValue: dataDetectorTypes.rawValue | dataDetectorType.rawValue)
|
|
}
|
|
configuration.dataDetectorTypes = dataDetectorTypes
|
|
}
|
|
} else {
|
|
// Fallback on earlier versions
|
|
}
|
|
|
|
if #available(iOS 13.0, *) {
|
|
configuration.preferences.isFraudulentWebsiteWarningEnabled = (options?.isFraudulentWebsiteWarningEnabled)!
|
|
configuration.defaultWebpagePreferences.preferredContentMode = WKWebpagePreferences.ContentMode(rawValue: (options?.preferredContentMode)!)!
|
|
} else {
|
|
// Fallback on earlier versions
|
|
}
|
|
|
|
if newOptionsMap["verticalScrollBarEnabled"] != nil && options?.verticalScrollBarEnabled != newOptions.verticalScrollBarEnabled {
|
|
scrollView.showsVerticalScrollIndicator = newOptions.verticalScrollBarEnabled
|
|
}
|
|
if newOptionsMap["horizontalScrollBarEnabled"] != nil && options?.horizontalScrollBarEnabled != newOptions.horizontalScrollBarEnabled {
|
|
scrollView.showsHorizontalScrollIndicator = newOptions.horizontalScrollBarEnabled
|
|
}
|
|
|
|
if #available(iOS 9.0, *) {
|
|
if newOptionsMap["allowsLinkPreview"] != nil && options?.allowsLinkPreview != newOptions.allowsLinkPreview {
|
|
allowsLinkPreview = newOptions.allowsLinkPreview
|
|
}
|
|
if newOptionsMap["allowsPictureInPictureMediaPlayback"] != nil && options?.allowsPictureInPictureMediaPlayback != newOptions.allowsPictureInPictureMediaPlayback {
|
|
configuration.allowsPictureInPictureMediaPlayback = newOptions.allowsPictureInPictureMediaPlayback
|
|
}
|
|
if newOptionsMap["applicationNameForUserAgent"] != nil && options?.applicationNameForUserAgent != newOptions.applicationNameForUserAgent && newOptions.applicationNameForUserAgent != "" {
|
|
configuration.applicationNameForUserAgent = newOptions.applicationNameForUserAgent
|
|
}
|
|
if newOptionsMap["userAgent"] != nil && options?.userAgent != newOptions.userAgent && newOptions.userAgent != "" {
|
|
customUserAgent = newOptions.userAgent
|
|
}
|
|
}
|
|
|
|
|
|
|
|
if newOptionsMap["clearCache"] != nil && newOptions.clearCache {
|
|
clearCache()
|
|
}
|
|
|
|
if #available(iOS 11.0, *), newOptionsMap["contentBlockers"] != nil {
|
|
configuration.userContentController.removeAllContentRuleLists()
|
|
let contentBlockers = newOptions.contentBlockers
|
|
if contentBlockers.count > 0 {
|
|
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
|
|
}
|
|
self.configuration.userContentController.add(contentRuleList!)
|
|
}
|
|
} catch {
|
|
print(error.localizedDescription)
|
|
}
|
|
}
|
|
}
|
|
|
|
self.options = newOptions
|
|
}
|
|
|
|
func getOptions() -> [String: Any]? {
|
|
if (self.options == nil) {
|
|
return nil
|
|
}
|
|
return self.options!.getHashMap()
|
|
}
|
|
|
|
public 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()
|
|
}
|
|
}
|
|
|
|
public func injectDeferredObject(source: String, withWrapper jsWrapper: String, result: FlutterResult?) {
|
|
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?.count ?? 0) - 2))
|
|
let jsToInject = String(format: jsWrapper, sourceString!)
|
|
|
|
evaluateJavaScript(jsToInject, completionHandler: {(value, error) in
|
|
if result == nil {
|
|
return
|
|
}
|
|
|
|
if error != nil {
|
|
let userInfo = (error! as NSError).userInfo
|
|
self.onConsoleMessage(sourceURL: (userInfo["WKJavaScriptExceptionSourceURL"] as? URL)?.absoluteString ?? "", lineNumber: userInfo["WKJavaScriptExceptionLineNumber"] as! Int, message: userInfo["WKJavaScriptExceptionMessage"] as! String, messageLevel: 3)
|
|
}
|
|
|
|
if value == nil {
|
|
result!("")
|
|
return
|
|
}
|
|
|
|
do {
|
|
let data: Data = ("[" + String(describing: value!) + "]").data(using: String.Encoding.utf8, allowLossyConversion: false)!
|
|
let json: Array<Any> = try JSONSerialization.jsonObject(with: data, options: []) as! Array<Any>
|
|
if json[0] is String {
|
|
result!(json[0])
|
|
}
|
|
else {
|
|
result!(value)
|
|
}
|
|
} catch let error as NSError {
|
|
result!(FlutterError(code: "InAppBrowserFlutterPlugin", message: "Failed to load: \(error.localizedDescription)", details: error))
|
|
}
|
|
|
|
})
|
|
}
|
|
}
|
|
|
|
public func evaluateJavascript(source: String, result: FlutterResult?) {
|
|
let jsWrapper = "(function(){return JSON.stringify(eval(%@));})();"
|
|
injectDeferredObject(source: source, withWrapper: jsWrapper, result: result)
|
|
}
|
|
|
|
public func injectJavascriptFileFromUrl(urlFile: String) {
|
|
let jsWrapper = "(function(d) { var c = d.createElement('script'); c.src = %@; d.body.appendChild(c); })(document);"
|
|
injectDeferredObject(source: urlFile, withWrapper: jsWrapper, result: nil)
|
|
}
|
|
|
|
public func injectCSSCode(source: String) {
|
|
let jsWrapper = "(function(d) { var c = d.createElement('style'); c.innerHTML = %@; d.body.appendChild(c); })(document);"
|
|
injectDeferredObject(source: source, withWrapper: jsWrapper, result: nil)
|
|
}
|
|
|
|
public func injectCSSFileFromUrl(urlFile: String) {
|
|
let jsWrapper = "(function(d) { var c = d.createElement('link'); c.rel='stylesheet', c.type='text/css'; c.href = %@; d.body.appendChild(c); })(document);"
|
|
injectDeferredObject(source: urlFile, withWrapper: jsWrapper, result: nil)
|
|
}
|
|
|
|
public func getCopyBackForwardList() -> [String: Any] {
|
|
let currentList = backForwardList
|
|
let currentIndex = currentList.backList.count
|
|
var completeList = currentList.backList
|
|
if currentList.currentItem != nil {
|
|
completeList.append(currentList.currentItem!)
|
|
}
|
|
completeList.append(contentsOf: currentList.forwardList)
|
|
|
|
var history: [[String: String]] = []
|
|
|
|
for historyItem in completeList {
|
|
var historyItemMap: [String: String] = [:]
|
|
historyItemMap["originalUrl"] = historyItem.initialURL.absoluteString
|
|
historyItemMap["title"] = historyItem.title
|
|
historyItemMap["url"] = historyItem.url.absoluteString
|
|
history.append(historyItemMap)
|
|
}
|
|
|
|
var result: [String: Any] = [:]
|
|
result["history"] = history
|
|
result["currentIndex"] = currentIndex
|
|
|
|
return result;
|
|
}
|
|
|
|
public func webView(_ webView: WKWebView,
|
|
decidePolicyFor navigationAction: WKNavigationAction,
|
|
decisionHandler: @escaping (WKNavigationActionPolicy) -> Void) {
|
|
|
|
let app = UIApplication.shared
|
|
|
|
if let url = navigationAction.request.url {
|
|
// Handle target="_blank"
|
|
if navigationAction.targetFrame == nil && (options?.useOnTargetBlank)! {
|
|
onTargetBlank(url: url)
|
|
decisionHandler(.cancel)
|
|
return
|
|
}
|
|
|
|
if navigationAction.navigationType == .linkActivated && (options?.useShouldOverrideUrlLoading)! {
|
|
shouldOverrideUrlLoading(url: url)
|
|
decisionHandler(.cancel)
|
|
return
|
|
}
|
|
|
|
// Handle phone and email links
|
|
if url.scheme == "tel" || url.scheme == "mailto" {
|
|
if app.canOpenURL(url) {
|
|
if #available(iOS 10.0, *) {
|
|
app.open(url)
|
|
} else {
|
|
app.openURL(url)
|
|
}
|
|
}
|
|
decisionHandler(.cancel)
|
|
return
|
|
}
|
|
|
|
if navigationAction.navigationType == .linkActivated || navigationAction.navigationType == .backForward {
|
|
currentURL = url
|
|
if IABController != nil {
|
|
IABController!.updateUrlTextField(url: (currentURL?.absoluteString)!)
|
|
}
|
|
}
|
|
}
|
|
|
|
decisionHandler(.allow)
|
|
}
|
|
|
|
public func webView(_ webView: WKWebView,
|
|
decidePolicyFor navigationResponse: WKNavigationResponse,
|
|
decisionHandler: @escaping (WKNavigationResponsePolicy) -> Void) {
|
|
|
|
if (options?.useOnDownloadStart)! {
|
|
let mimeType = navigationResponse.response.mimeType
|
|
if let url = navigationResponse.response.url {
|
|
if mimeType != nil && !mimeType!.starts(with: "text/") {
|
|
onDownloadStart(url: url.absoluteString)
|
|
decisionHandler(.cancel)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
decisionHandler(.allow)
|
|
}
|
|
|
|
public func webView(_ webView: WKWebView, didStartProvisionalNavigation navigation: WKNavigation!) {
|
|
self.startPageTime = currentTimeInMilliSeconds()
|
|
onLoadStart(url: (currentURL?.absoluteString)!)
|
|
|
|
if IABController != nil {
|
|
// loading url, start spinner, update back/forward
|
|
IABController!.backButton.isEnabled = canGoBack
|
|
IABController!.forwardButton.isEnabled = canGoForward
|
|
|
|
if (IABController!.browserOptions?.spinner)! {
|
|
IABController!.spinner.startAnimating()
|
|
}
|
|
}
|
|
}
|
|
|
|
public func webView(_ webView: WKWebView, didFinish navigation: WKNavigation!) {
|
|
currentURL = url
|
|
InAppWebView.credentialsProposed = []
|
|
onLoadStop(url: (currentURL?.absoluteString)!)
|
|
evaluateJavaScript(platformReadyJS, completionHandler: nil)
|
|
|
|
if IABController != nil {
|
|
IABController!.updateUrlTextField(url: (currentURL?.absoluteString)!)
|
|
IABController!.backButton.isEnabled = canGoBack
|
|
IABController!.forwardButton.isEnabled = canGoForward
|
|
IABController!.spinner.stopAnimating()
|
|
}
|
|
}
|
|
|
|
public func webView(_ view: WKWebView,
|
|
didFailProvisionalNavigation navigation: WKNavigation!,
|
|
withError error: Error) {
|
|
webView(view, didFail: navigation, withError: error)
|
|
}
|
|
|
|
public func webView(_ webView: WKWebView, didFail navigation: WKNavigation!, withError error: Error) {
|
|
InAppWebView.credentialsProposed = []
|
|
|
|
onLoadError(url: (currentURL?.absoluteString)!, error: error)
|
|
|
|
if IABController != nil {
|
|
IABController!.backButton.isEnabled = canGoBack
|
|
IABController!.forwardButton.isEnabled = canGoForward
|
|
IABController!.spinner.stopAnimating()
|
|
}
|
|
}
|
|
|
|
public func webView(_ webView: WKWebView, didReceive challenge: URLAuthenticationChallenge, completionHandler: @escaping (URLSession.AuthChallengeDisposition, URLCredential?) -> Void) {
|
|
|
|
if challenge.protectionSpace.authenticationMethod == NSURLAuthenticationMethodHTTPBasic ||
|
|
challenge.protectionSpace.authenticationMethod == NSURLAuthenticationMethodDefault ||
|
|
challenge.protectionSpace.authenticationMethod == NSURLAuthenticationMethodHTTPDigest {
|
|
let host = challenge.protectionSpace.host
|
|
let prot = challenge.protectionSpace.protocol
|
|
let realm = challenge.protectionSpace.realm
|
|
let port = challenge.protectionSpace.port
|
|
onReceivedHttpAuthRequest(challenge: challenge, result: {(result) -> Void in
|
|
if result is FlutterError {
|
|
print((result as! FlutterError).message)
|
|
}
|
|
else if (result as? NSObject) == FlutterMethodNotImplemented {
|
|
completionHandler(.performDefaultHandling, nil)
|
|
}
|
|
else {
|
|
var response: [String: Any]
|
|
if let r = result {
|
|
response = r as! [String: Any]
|
|
var action = response["action"] as? Int
|
|
action = action != nil ? action : 0;
|
|
switch action {
|
|
case 0:
|
|
InAppWebView.credentialsProposed = []
|
|
completionHandler(.cancelAuthenticationChallenge, nil)
|
|
break
|
|
case 1:
|
|
let username = response["username"] as! String
|
|
let password = response["password"] as! String
|
|
let permanentPersistence = response["permanentPersistence"] as? Bool ?? false
|
|
let persistence = (permanentPersistence) ? URLCredential.Persistence.permanent : URLCredential.Persistence.forSession
|
|
let credential = URLCredential(user: username, password: password, persistence: persistence)
|
|
completionHandler(.useCredential, credential)
|
|
break
|
|
case 2:
|
|
if InAppWebView.credentialsProposed.count == 0 {
|
|
for (protectionSpace, credentials) in CredentialDatabase.credentialStore!.allCredentials {
|
|
if protectionSpace.host == host && protectionSpace.realm == realm &&
|
|
protectionSpace.protocol == prot && protectionSpace.port == port {
|
|
for credential in credentials {
|
|
InAppWebView.credentialsProposed.append(credential.value)
|
|
}
|
|
break
|
|
}
|
|
}
|
|
}
|
|
if InAppWebView.credentialsProposed.count == 0, let credential = challenge.proposedCredential {
|
|
InAppWebView.credentialsProposed.append(credential)
|
|
}
|
|
|
|
if let credential = InAppWebView.credentialsProposed.popLast() {
|
|
completionHandler(.useCredential, credential)
|
|
}
|
|
else {
|
|
completionHandler(.performDefaultHandling, nil)
|
|
}
|
|
break
|
|
default:
|
|
InAppWebView.credentialsProposed = []
|
|
completionHandler(.performDefaultHandling, nil)
|
|
}
|
|
return;
|
|
}
|
|
completionHandler(.performDefaultHandling, nil)
|
|
}
|
|
})
|
|
}
|
|
else if challenge.protectionSpace.authenticationMethod == NSURLAuthenticationMethodServerTrust {
|
|
|
|
guard let serverTrust = challenge.protectionSpace.serverTrust else {
|
|
completionHandler(.performDefaultHandling, nil)
|
|
return
|
|
}
|
|
|
|
onReceivedServerTrustAuthRequest(challenge: challenge, result: {(result) -> Void in
|
|
if result is FlutterError {
|
|
print((result as! FlutterError).message)
|
|
}
|
|
else if (result as? NSObject) == FlutterMethodNotImplemented {
|
|
completionHandler(.performDefaultHandling, nil)
|
|
}
|
|
else {
|
|
var response: [String: Any]
|
|
if let r = result {
|
|
response = r as! [String: Any]
|
|
var action = response["action"] as? Int
|
|
action = action != nil ? action : 0;
|
|
switch action {
|
|
case 0:
|
|
InAppWebView.credentialsProposed = []
|
|
completionHandler(.cancelAuthenticationChallenge, nil)
|
|
break
|
|
case 1:
|
|
let exceptions = SecTrustCopyExceptions(serverTrust)
|
|
SecTrustSetExceptions(serverTrust, exceptions)
|
|
let credential = URLCredential(trust: serverTrust)
|
|
completionHandler(.useCredential, credential)
|
|
break
|
|
default:
|
|
InAppWebView.credentialsProposed = []
|
|
completionHandler(.performDefaultHandling, nil)
|
|
}
|
|
return;
|
|
}
|
|
completionHandler(.performDefaultHandling, nil)
|
|
}
|
|
})
|
|
}
|
|
else if challenge.protectionSpace.authenticationMethod == NSURLAuthenticationMethodClientCertificate {
|
|
onReceivedClientCertRequest(challenge: challenge, result: {(result) -> Void in
|
|
if result is FlutterError {
|
|
print((result as! FlutterError).message)
|
|
}
|
|
else if (result as? NSObject) == FlutterMethodNotImplemented {
|
|
completionHandler(.performDefaultHandling, nil)
|
|
}
|
|
else {
|
|
var response: [String: Any]
|
|
if let r = result {
|
|
response = r as! [String: Any]
|
|
var action = response["action"] as? Int
|
|
action = action != nil ? action : 0;
|
|
switch action {
|
|
case 0:
|
|
completionHandler(.cancelAuthenticationChallenge, nil)
|
|
break
|
|
case 1:
|
|
let certificatePath = response["certificatePath"] as! String;
|
|
let certificatePassword = response["certificatePassword"] as? String ?? "";
|
|
|
|
let key = SwiftFlutterPlugin.instance!.registrar!.lookupKey(forAsset: certificatePath)
|
|
let path = Bundle.main.path(forResource: key, ofType: nil)!
|
|
let PKCS12Data = NSData(contentsOfFile: path)!
|
|
|
|
if let identityAndTrust: IdentityAndTrust = self.extractIdentity(PKCS12Data: PKCS12Data, password: certificatePassword) {
|
|
let urlCredential: URLCredential = URLCredential(
|
|
identity: identityAndTrust.identityRef,
|
|
certificates: identityAndTrust.certArray as? [AnyObject],
|
|
persistence: URLCredential.Persistence.forSession);
|
|
completionHandler(.useCredential, urlCredential)
|
|
} else {
|
|
completionHandler(.performDefaultHandling, nil)
|
|
}
|
|
break
|
|
case 2:
|
|
completionHandler(.cancelAuthenticationChallenge, nil)
|
|
break
|
|
default:
|
|
completionHandler(.performDefaultHandling, nil)
|
|
}
|
|
return;
|
|
}
|
|
completionHandler(.performDefaultHandling, nil)
|
|
}
|
|
})
|
|
}
|
|
else {
|
|
completionHandler(.performDefaultHandling, nil)
|
|
}
|
|
}
|
|
|
|
struct IdentityAndTrust {
|
|
|
|
var identityRef:SecIdentity
|
|
var trust:SecTrust
|
|
var certArray:AnyObject
|
|
}
|
|
|
|
func extractIdentity(PKCS12Data:NSData, password: String) -> IdentityAndTrust? {
|
|
var identityAndTrust:IdentityAndTrust?
|
|
var securityError:OSStatus = errSecSuccess
|
|
|
|
var importResult: CFArray? = nil
|
|
securityError = SecPKCS12Import(
|
|
PKCS12Data as NSData,
|
|
[kSecImportExportPassphrase as String: password] as NSDictionary,
|
|
&importResult
|
|
)
|
|
|
|
if securityError == errSecSuccess {
|
|
let certItems:CFArray = importResult! as CFArray;
|
|
let certItemsArray:Array = certItems as Array
|
|
let dict:AnyObject? = certItemsArray.first;
|
|
if let certEntry:Dictionary = dict as? Dictionary<String, AnyObject> {
|
|
// grab the identity
|
|
let identityPointer:AnyObject? = certEntry["identity"];
|
|
let secIdentityRef:SecIdentity = (identityPointer as! SecIdentity?)!;
|
|
// grab the trust
|
|
let trustPointer:AnyObject? = certEntry["trust"];
|
|
let trustRef:SecTrust = trustPointer as! SecTrust;
|
|
// grab the cert
|
|
let chainPointer:AnyObject? = certEntry["chain"];
|
|
identityAndTrust = IdentityAndTrust(identityRef: secIdentityRef, trust: trustRef, certArray: chainPointer!);
|
|
}
|
|
} else {
|
|
print("Security Error: " + securityError.description)
|
|
if #available(iOS 11.3, *) {
|
|
print(SecCopyErrorMessageString(securityError,nil))
|
|
}
|
|
}
|
|
return identityAndTrust;
|
|
}
|
|
|
|
|
|
func createAlertDialog(message: String?, responseMessage: String?, confirmButtonTitle: String?, completionHandler: @escaping () -> Void) {
|
|
let title = responseMessage != nil && !responseMessage!.isEmpty ? responseMessage : message
|
|
let okButton = confirmButtonTitle != nil && !confirmButtonTitle!.isEmpty ? confirmButtonTitle : NSLocalizedString("Ok", comment: "")
|
|
let alertController = UIAlertController(title: title, message: nil,
|
|
preferredStyle: UIAlertController.Style.alert);
|
|
|
|
alertController.addAction(UIAlertAction(title: okButton, style: UIAlertAction.Style.default) {
|
|
_ in completionHandler()}
|
|
);
|
|
|
|
let presentingViewController = ((self.IABController != nil) ? self.IABController! : self.window!.rootViewController!)
|
|
presentingViewController.present(alertController, animated: true, completion: {})
|
|
}
|
|
|
|
public func webView(_ webView: WKWebView, runJavaScriptAlertPanelWithMessage message: String,
|
|
initiatedByFrame frame: WKFrameInfo, completionHandler: @escaping () -> Void) {
|
|
|
|
onJsAlert(message: message, result: {(result) -> Void in
|
|
if result is FlutterError {
|
|
print((result as! FlutterError).message)
|
|
}
|
|
else if (result as? NSObject) == FlutterMethodNotImplemented {
|
|
self.createAlertDialog(message: message, responseMessage: nil, confirmButtonTitle: nil, completionHandler: completionHandler)
|
|
}
|
|
else {
|
|
let response: [String: Any]
|
|
var responseMessage: String?;
|
|
var confirmButtonTitle: String?;
|
|
|
|
if let r = result {
|
|
response = r as! [String: Any]
|
|
responseMessage = response["message"] as? String
|
|
confirmButtonTitle = response["confirmButtonTitle"] as? String
|
|
let handledByClient = response["handledByClient"] as? Bool
|
|
if handledByClient != nil, handledByClient! {
|
|
var action = response["action"] as? Int
|
|
action = action != nil ? action : 1;
|
|
switch action {
|
|
case 0:
|
|
completionHandler()
|
|
break
|
|
default:
|
|
completionHandler()
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
self.createAlertDialog(message: message, responseMessage: responseMessage, confirmButtonTitle: confirmButtonTitle, completionHandler: completionHandler)
|
|
}
|
|
})
|
|
}
|
|
|
|
func createConfirmDialog(message: String?, responseMessage: String?, confirmButtonTitle: String?, cancelButtonTitle: String?, completionHandler: @escaping (Bool) -> Void) {
|
|
let dialogMessage = responseMessage != nil && !responseMessage!.isEmpty ? responseMessage : message
|
|
let okButton = confirmButtonTitle != nil && !confirmButtonTitle!.isEmpty ? confirmButtonTitle : NSLocalizedString("Ok", comment: "")
|
|
let cancelButton = cancelButtonTitle != nil && !cancelButtonTitle!.isEmpty ? cancelButtonTitle : NSLocalizedString("Cancel", comment: "")
|
|
|
|
let alertController = UIAlertController(title: nil, message: dialogMessage, preferredStyle: .alert)
|
|
|
|
alertController.addAction(UIAlertAction(title: okButton, style: .default, handler: { (action) in
|
|
completionHandler(true)
|
|
}))
|
|
|
|
alertController.addAction(UIAlertAction(title: cancelButton, style: .cancel, handler: { (action) in
|
|
completionHandler(false)
|
|
}))
|
|
|
|
let presentingViewController = ((self.IABController != nil) ? self.IABController! : self.window!.rootViewController!)
|
|
presentingViewController.present(alertController, animated: true, completion: nil)
|
|
}
|
|
|
|
public func webView(_ webView: WKWebView, runJavaScriptConfirmPanelWithMessage message: String, initiatedByFrame frame: WKFrameInfo,
|
|
completionHandler: @escaping (Bool) -> Void) {
|
|
|
|
onJsConfirm(message: message, result: {(result) -> Void in
|
|
if result is FlutterError {
|
|
print((result as! FlutterError).message)
|
|
}
|
|
else if (result as? NSObject) == FlutterMethodNotImplemented {
|
|
self.createConfirmDialog(message: message, responseMessage: nil, confirmButtonTitle: nil, cancelButtonTitle: nil, completionHandler: completionHandler)
|
|
}
|
|
else {
|
|
let response: [String: Any]
|
|
var responseMessage: String?;
|
|
var confirmButtonTitle: String?;
|
|
var cancelButtonTitle: String?;
|
|
|
|
if let r = result {
|
|
response = r as! [String: Any]
|
|
responseMessage = response["message"] as? String
|
|
confirmButtonTitle = response["confirmButtonTitle"] as? String
|
|
cancelButtonTitle = response["cancelButtonTitle"] as? String
|
|
let handledByClient = response["handledByClient"] as? Bool
|
|
if handledByClient != nil, handledByClient! {
|
|
var action = response["action"] as? Int
|
|
action = action != nil ? action : 1;
|
|
switch action {
|
|
case 0:
|
|
completionHandler(true)
|
|
break
|
|
case 1:
|
|
completionHandler(false)
|
|
break
|
|
default:
|
|
completionHandler(false)
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
self.createConfirmDialog(message: message, responseMessage: responseMessage, confirmButtonTitle: confirmButtonTitle, cancelButtonTitle: cancelButtonTitle, completionHandler: completionHandler)
|
|
}
|
|
})
|
|
}
|
|
|
|
func createPromptDialog(message: String, defaultValue: String?, responseMessage: String?, confirmButtonTitle: String?, cancelButtonTitle: String?, value: String?, completionHandler: @escaping (String?) -> Void) {
|
|
let dialogMessage = responseMessage != nil && !responseMessage!.isEmpty ? responseMessage : message
|
|
let okButton = confirmButtonTitle != nil && !confirmButtonTitle!.isEmpty ? confirmButtonTitle : NSLocalizedString("Ok", comment: "")
|
|
let cancelButton = cancelButtonTitle != nil && !cancelButtonTitle!.isEmpty ? cancelButtonTitle : NSLocalizedString("Cancel", comment: "")
|
|
|
|
let alertController = UIAlertController(title: nil, message: dialogMessage, preferredStyle: .alert)
|
|
|
|
alertController.addTextField { (textField) in
|
|
textField.text = defaultValue
|
|
}
|
|
|
|
alertController.addAction(UIAlertAction(title: okButton, style: .default, handler: { (action) in
|
|
if let v = value {
|
|
completionHandler(v)
|
|
}
|
|
else if let text = alertController.textFields?.first?.text {
|
|
completionHandler(text)
|
|
} else {
|
|
completionHandler("")
|
|
}
|
|
}))
|
|
|
|
alertController.addAction(UIAlertAction(title: cancelButton, style: .cancel, handler: { (action) in
|
|
completionHandler(nil)
|
|
}))
|
|
|
|
let presentingViewController = ((self.IABController != nil) ? self.IABController! : self.window!.rootViewController!)
|
|
presentingViewController.present(alertController, animated: true, completion: nil)
|
|
}
|
|
|
|
public func webView(_ webView: WKWebView, runJavaScriptTextInputPanelWithPrompt message: String, defaultText defaultValue: String?, initiatedByFrame frame: WKFrameInfo,
|
|
completionHandler: @escaping (String?) -> Void) {
|
|
onJsPrompt(message: message, defaultValue: defaultValue, result: {(result) -> Void in
|
|
if result is FlutterError {
|
|
print((result as! FlutterError).message)
|
|
}
|
|
else if (result as? NSObject) == FlutterMethodNotImplemented {
|
|
self.createPromptDialog(message: message, defaultValue: defaultValue, responseMessage: nil, confirmButtonTitle: nil, cancelButtonTitle: nil, value: nil, completionHandler: completionHandler)
|
|
}
|
|
else {
|
|
let response: [String: Any]
|
|
var responseMessage: String?;
|
|
var confirmButtonTitle: String?;
|
|
var cancelButtonTitle: String?;
|
|
var value: String?;
|
|
|
|
if let r = result {
|
|
response = r as! [String: Any]
|
|
responseMessage = response["message"] as? String
|
|
confirmButtonTitle = response["confirmButtonTitle"] as? String
|
|
cancelButtonTitle = response["cancelButtonTitle"] as? String
|
|
let handledByClient = response["handledByClient"] as? Bool
|
|
value = response["value"] as? String;
|
|
if handledByClient != nil, handledByClient! {
|
|
var action = response["action"] as? Int
|
|
action = action != nil ? action : 1;
|
|
switch action {
|
|
case 0:
|
|
completionHandler(value)
|
|
break
|
|
case 1:
|
|
completionHandler(nil)
|
|
break
|
|
default:
|
|
completionHandler(nil)
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
self.createPromptDialog(message: message, defaultValue: defaultValue, responseMessage: responseMessage, confirmButtonTitle: confirmButtonTitle, cancelButtonTitle: cancelButtonTitle, value: value, completionHandler: completionHandler)
|
|
}
|
|
})
|
|
}
|
|
|
|
public func scrollViewDidScroll(_ scrollView: UIScrollView) {
|
|
if navigationDelegate != nil {
|
|
let x = Int(scrollView.contentOffset.x / scrollView.contentScaleFactor)
|
|
let y = Int(scrollView.contentOffset.y / scrollView.contentScaleFactor)
|
|
onScrollChanged(x: x, y: y)
|
|
}
|
|
setNeedsLayout()
|
|
}
|
|
|
|
public func onLoadStart(url: String) {
|
|
var arguments: [String: Any] = ["url": url]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onLoadStart", arguments: arguments)
|
|
}
|
|
}
|
|
|
|
public func onLoadStop(url: String) {
|
|
var arguments: [String: Any] = ["url": url]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onLoadStop", arguments: arguments)
|
|
}
|
|
}
|
|
|
|
public func onLoadError(url: String, error: Error) {
|
|
var arguments: [String: Any] = ["url": url, "code": error._code, "message": error.localizedDescription]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onLoadError", arguments: arguments)
|
|
}
|
|
}
|
|
|
|
public func onProgressChanged(progress: Int) {
|
|
var arguments: [String: Any] = ["progress": progress]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onProgressChanged", arguments: arguments)
|
|
}
|
|
}
|
|
|
|
public func onFindResultReceived(activeMatchOrdinal: Int, numberOfMatches: Int, isDoneCounting: Bool) {
|
|
var arguments: [String : Any] = [
|
|
"activeMatchOrdinal": activeMatchOrdinal,
|
|
"numberOfMatches": numberOfMatches,
|
|
"isDoneCounting": isDoneCounting
|
|
]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onFindResultReceived", arguments: arguments)
|
|
}
|
|
}
|
|
|
|
public func onNavigationStateChange(url: String) {
|
|
var arguments: [String : Any] = [
|
|
"url": url
|
|
]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onNavigationStateChange", arguments: arguments)
|
|
}
|
|
}
|
|
|
|
public func onScrollChanged(x: Int, y: Int) {
|
|
var arguments: [String: Any] = ["x": x, "y": y]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onScrollChanged", arguments: arguments)
|
|
}
|
|
}
|
|
|
|
public func onDownloadStart(url: String) {
|
|
var arguments: [String: Any] = ["url": url]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onDownloadStart", arguments: arguments)
|
|
}
|
|
}
|
|
|
|
public func onLoadResourceCustomScheme(scheme: String, url: String, result: FlutterResult?) {
|
|
var arguments: [String: Any] = ["scheme": scheme, "url": url]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onLoadResourceCustomScheme", arguments: arguments, result: result)
|
|
}
|
|
}
|
|
|
|
public func shouldOverrideUrlLoading(url: URL) {
|
|
var arguments: [String: Any] = ["url": url.absoluteString]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("shouldOverrideUrlLoading", arguments: arguments)
|
|
}
|
|
}
|
|
|
|
public func onTargetBlank(url: URL) {
|
|
var arguments: [String: Any] = ["url": url.absoluteString]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onTargetBlank", arguments: arguments)
|
|
}
|
|
}
|
|
|
|
public func onReceivedHttpAuthRequest(challenge: URLAuthenticationChallenge, result: FlutterResult?) {
|
|
var arguments: [String: Any?] = [
|
|
"host": challenge.protectionSpace.host,
|
|
"protocol": challenge.protectionSpace.protocol,
|
|
"realm": challenge.protectionSpace.realm,
|
|
"port": challenge.protectionSpace.port,
|
|
"previousFailureCount": challenge.previousFailureCount
|
|
]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onReceivedHttpAuthRequest", arguments: arguments, result: result)
|
|
}
|
|
}
|
|
|
|
public func onReceivedServerTrustAuthRequest(challenge: URLAuthenticationChallenge, result: FlutterResult?) {
|
|
var serverCertificateData: NSData?
|
|
let serverTrust = challenge.protectionSpace.serverTrust!
|
|
if let serverCertificate = SecTrustGetCertificateAtIndex(serverTrust, 0) {
|
|
let serverCertificateCFData = SecCertificateCopyData(serverCertificate)
|
|
let data = CFDataGetBytePtr(serverCertificateCFData)
|
|
let size = CFDataGetLength(serverCertificateCFData)
|
|
serverCertificateData = NSData(bytes: data, length: size)
|
|
}
|
|
|
|
var arguments: [String: Any?] = [
|
|
"host": challenge.protectionSpace.host,
|
|
"protocol": challenge.protectionSpace.protocol,
|
|
"realm": challenge.protectionSpace.realm,
|
|
"port": challenge.protectionSpace.port,
|
|
"previousFailureCount": challenge.previousFailureCount,
|
|
"serverCertificate": serverCertificateData,
|
|
"error": -1,
|
|
"message": "",
|
|
]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onReceivedServerTrustAuthRequest", arguments: arguments, result: result)
|
|
}
|
|
}
|
|
|
|
public func onReceivedClientCertRequest(challenge: URLAuthenticationChallenge, result: FlutterResult?) {
|
|
var arguments: [String: Any?] = [
|
|
"host": challenge.protectionSpace.host,
|
|
"protocol": challenge.protectionSpace.protocol,
|
|
"realm": challenge.protectionSpace.realm,
|
|
"port": challenge.protectionSpace.port
|
|
]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onReceivedClientCertRequest", arguments: arguments, result: result)
|
|
}
|
|
}
|
|
|
|
public func onJsAlert(message: String, result: FlutterResult?) {
|
|
var arguments: [String: Any] = ["message": message]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onJsAlert", arguments: arguments, result: result)
|
|
}
|
|
}
|
|
|
|
public func onJsConfirm(message: String, result: FlutterResult?) {
|
|
var arguments: [String: Any] = ["message": message]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onJsConfirm", arguments: arguments, result: result)
|
|
}
|
|
}
|
|
|
|
public func onJsPrompt(message: String, defaultValue: String?, result: FlutterResult?) {
|
|
var arguments: [String: Any] = ["message": message, "defaultValue": defaultValue as Any]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onJsPrompt", arguments: arguments, result: result)
|
|
}
|
|
}
|
|
|
|
public func onConsoleMessage(sourceURL: String, lineNumber: Int, message: String, messageLevel: Int) {
|
|
var arguments: [String: Any] = ["sourceURL": sourceURL, "lineNumber": lineNumber, "message": message, "messageLevel": messageLevel]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onConsoleMessage", arguments: arguments)
|
|
}
|
|
}
|
|
|
|
public func onCallJsHandler(handlerName: String, _callHandlerID: Int64, args: String) {
|
|
var arguments: [String: Any] = ["handlerName": handlerName, "args": args]
|
|
if IABController != nil {
|
|
arguments["uuid"] = IABController!.uuid
|
|
}
|
|
|
|
if let channel = getChannel() {
|
|
channel.invokeMethod("onCallJsHandler", arguments: arguments, result: {(result) -> Void in
|
|
if result is FlutterError {
|
|
print((result as! FlutterError).message)
|
|
}
|
|
else if (result as? NSObject) == FlutterMethodNotImplemented {}
|
|
else {
|
|
var json = "null"
|
|
if let r = result {
|
|
json = r as! String
|
|
}
|
|
self.evaluateJavaScript("window.\(JAVASCRIPT_BRIDGE_NAME)[\(_callHandlerID)](\(json)); delete window.\(JAVASCRIPT_BRIDGE_NAME)[\(_callHandlerID)];", completionHandler: nil)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
public func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) {
|
|
if message.name.starts(with: "console") {
|
|
var messageLevel = 1
|
|
switch (message.name) {
|
|
case "consoleLog":
|
|
messageLevel = 1
|
|
break;
|
|
case "consoleDebug":
|
|
// on Android, console.debug is TIP
|
|
messageLevel = 0
|
|
break;
|
|
case "consoleError":
|
|
messageLevel = 3
|
|
break;
|
|
case "consoleInfo":
|
|
// on Android, console.info is LOG
|
|
messageLevel = 1
|
|
break;
|
|
case "consoleWarn":
|
|
messageLevel = 2
|
|
break;
|
|
default:
|
|
messageLevel = 1
|
|
break;
|
|
}
|
|
onConsoleMessage(sourceURL: "", lineNumber: 1, message: message.body as! String, messageLevel: messageLevel)
|
|
}
|
|
else if message.name == "callHandler" {
|
|
let body = message.body as! [String: Any]
|
|
let handlerName = body["handlerName"] as! String
|
|
let _callHandlerID = body["_callHandlerID"] as! Int64
|
|
let args = body["args"] as! String
|
|
onCallJsHandler(handlerName: handlerName, _callHandlerID: _callHandlerID, args: args)
|
|
} else if message.name == "onFindResultReceived" {
|
|
if let resource = convertToDictionary(text: message.body as! String) {
|
|
let activeMatchOrdinal = resource["activeMatchOrdinal"] as! Int
|
|
let numberOfMatches = resource["numberOfMatches"] as! Int
|
|
let isDoneCounting = resource["isDoneCounting"] as! Bool
|
|
|
|
self.onFindResultReceived(activeMatchOrdinal: activeMatchOrdinal, numberOfMatches: numberOfMatches, isDoneCounting: isDoneCounting)
|
|
}
|
|
} else if message.name == "onNavigationStateChange" {
|
|
if let resource = convertToDictionary(text: message.body as! String) {
|
|
let url = resource["url"] as! String
|
|
|
|
self.onNavigationStateChange(url: url)
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
private func getChannel() -> FlutterMethodChannel? {
|
|
return (IABController != nil) ? SwiftFlutterPlugin.instance!.channel! : ((IAWController != nil) ? IAWController!.channel! : nil);
|
|
}
|
|
|
|
func findAllAsync(find: String?, completionHandler: ((Any?, Error?) -> Void)?) {
|
|
let startSearch = "wkwebview_FindAllAsync('\(find ?? "")');"
|
|
evaluateJavaScript(startSearch, completionHandler: completionHandler)
|
|
}
|
|
|
|
func findNext(forward: Bool, completionHandler: ((Any?, Error?) -> Void)?) {
|
|
evaluateJavaScript("wkwebview_FindNext(\(forward ? "true" : "false"));", completionHandler: completionHandler)
|
|
}
|
|
|
|
func clearMatches(completionHandler: ((Any?, Error?) -> Void)?) {
|
|
evaluateJavaScript("wkwebview_ClearMatches();", completionHandler: completionHandler)
|
|
}
|
|
|
|
public override func removeFromSuperview() {
|
|
configuration.userContentController.removeScriptMessageHandler(forName: "consoleLog")
|
|
configuration.userContentController.removeScriptMessageHandler(forName: "consoleDebug")
|
|
configuration.userContentController.removeScriptMessageHandler(forName: "consoleError")
|
|
configuration.userContentController.removeScriptMessageHandler(forName: "consoleInfo")
|
|
configuration.userContentController.removeScriptMessageHandler(forName: "consoleWarn")
|
|
configuration.userContentController.removeScriptMessageHandler(forName: "callHandler")
|
|
configuration.userContentController.removeScriptMessageHandler(forName: "onFindResultReceived")
|
|
configuration.userContentController.removeScriptMessageHandler(forName: "onNavigationStateChange")
|
|
configuration.userContentController.removeAllUserScripts()
|
|
removeObserver(self, forKeyPath: "estimatedProgress")
|
|
super.removeFromSuperview()
|
|
uiDelegate = nil
|
|
navigationDelegate = nil
|
|
scrollView.delegate = nil
|
|
IAWController?.channel?.setMethodCallHandler(nil)
|
|
IABController?.webView = nil
|
|
IAWController?.webView = nil
|
|
}
|
|
}
|