据我所知,WKWebView 中加载的所有资源的缓存由
默认值:真
请记住,如果您请求相同的资源,webview 将不会从 Internet 加载内容,它会为您提供缓存资源中的内容。对于请求相同的资源,您可以使用一些 JavaScript 来获取内容。
看看下面的代码。加载 PDF 并点击保存按钮后。它将执行 JavaScript 代码,并在数据准备好通过 JavaScript 传递时
它会触发window.webkit.messageHandlers.myInterface.postMessage(base64)
让您的 ViewController 知道数据已准备好共享。
您可以通过以下方式进行验证
让它加载 PDF
关闭模拟器的网络(see)
点击保存按钮
您将获得 base64 格式的 pdf 数据。保存并分享它:)
import UIKit
import WebKit
class ViewController: UIViewController {
@IBOutlet weak var webView: WKWebView!
var activityIndicator: UIActivityIndicatorView?
override func viewDidLoad() {
super.viewDidLoad()
}
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
activityIndicator = UIActivityIndicatorView(activityIndicatorStyle: .gray)
webView.navigationDelegate = self
activityIndicator?.center = self.view.center
self.view.addSubview(activityIndicator!)
webView.configuration.userContentController.add(self, name: "myInterface")
webView.load(URLRequest(url: URL(string: "http://www.africau.edu/images/default/sample.pdf")!))
activityIndicator?.startAnimating()
}
@IBAction func saveAction(_ sender: Any) {
let s = """
var xhr = new XMLHttpRequest();
xhr.open('GET', "\(webView.url?.absoluteString ?? "")", true);
xhr.responseType = 'arraybuffer';
xhr.onload = function(e) {
if (this.status == 200) {
var uInt8Array = new Uint8Array(this.response);
var i = uInt8Array.length;
var binaryString = new Array(i);
while (i--){
binaryString[i] = String.fromCharCode(uInt8Array[i]);
}
var data = binaryString.join('');
var base64 = window.btoa(data);
window.webkit.messageHandlers.myInterface.postMessage(base64);
}
};
xhr.send();
"""
webView?.evaluateJavaScript(s, completionHandler: {(string,error) in
print(error ?? "no error")
})
}
}
extension ViewController: WKScriptMessageHandler{
func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) {
// print("Message received: \(message.name) with body: \(message.body)")
guard
var documentsURL = (FileManager.default.urls(for: .documentDirectory, in: .userDomainMask)).last,
let convertedData = Data.init(base64Encoded: message.body as! String)
else {
//handle error when getting documents URL
return
}
//name your file however you prefer
documentsURL.appendPathComponent("sample.pdf")
do {
try convertedData.write(to: documentsURL)
} catch {
//handle write error here
}
//if you want to get a quick output of where your
//file was saved from the simulator on your machine
//just print the documentsURL and go there in Finder
print(documentsURL)
let activityViewController = UIActivityViewController.init(activityItems: [documentsURL], applicationActivities: nil)
present(activityViewController, animated: true, completion: nil)
}
}
extension ViewController: WKNavigationDelegate{
func webView(_ webView: WKWebView, didFinish navigation: WKNavigation!) {
self.activityIndicator?.stopAnimating()
self.activityIndicator?.removeFromSuperview()
self.activityIndicator = nil
}
}
顺便说一句,您提供的 pdf 链接使用的是 HTTP 而不是 HTTPS。因此,出于测试目的,在您的 info.plist
中添加以下内容
<key>NSAppTransportSecurity</key>
<dict>
<key>NSAllowsArbitraryLoadsInWebContent</key>
<true/>
</dict>