在我的Swift iOS应用程序中,我想从远程服务器下载一些动态HTML页面,将它们保存在文档目录中,并从文档目录中显示这些页面。
我用它来加载页面:
var appWebView:WKWebView?
...
appWebView!.loadRequest(NSURLRequest(URL: NSURL(fileURLWithPath: htmlPath)))
一切都在模拟器上运行,但当我转移到真正的手机时,它只是显示一个空白页面。然后我使用Safari连接到应用程序,发现它抱怨“无法加载资源”。
然后我尝试首先阅读htmlPath
页面的内容,然后使用
appWebView!.loadHTMLString()
加载页面。它在HTML页面很简单时有效。但是,如果HTML引用其他内容,即文档目录中的JavaScript文件(使用类似<script src="file:////var/mobile/Containers/Data/Application/762035C9-2BF2-4CDD-B5B1-574A0E2B0728/Documents/xxxxx.js">
的绝对路径),则无法加载。
有谁知道为什么会这样,以及如何解决这个问题?
更多信息:
答案 0 :(得分:32)
这是我用来加载我的项目中的本地文件(iOS 10,Swift 3)的简化版本。根据Raghuram和Fox5150在评论中的要求,我在iOS 10.3.1和iPhone 7+上再次测试后,更新了我的代码(7.5.2017)。
更新19.04.2018:添加了一个新功能,用HTML,CSS,JS文件下载.zip,将其解压缩到/ Documents /(Alamofire + Zip)中,然后将这些文件加载到web视图。您也可以在GitHub sample project中找到它。再次,随意叉和星! :)
更新08.02.2018:最后添加了GitHub sample project,其中还包含本地JavaScript文件。随意分叉&amp;星! :)
ViewController.swift
import UIKit
import WebKit
class ViewController: UIViewController, WKNavigationDelegate {
override func viewDidLoad() {
super.viewDidLoad()
let webView = WKWebView()
let htmlPath = Bundle.main.path(forResource: "index", ofType: "html")
let htmlUrl = URL(fileURLWithPath: htmlPath!, isDirectory: false)
webView.loadFileURL(htmlUrl, allowingReadAccessTo: htmlUrl)
webView.navigationDelegate = self
view = webView
}
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
}
}
ViewController.swift
import UIKit
import WebKit
class ViewController: UIViewController, WKNavigationDelegate {
override func viewDidLoad() {
super.viewDidLoad()
let webView = WKWebView()
let htmlPath = Bundle.main.path(forResource: "index", ofType: "html")
let folderPath = Bundle.main.bundlePath
let baseUrl = URL(fileURLWithPath: folderPath, isDirectory: true)
do {
let htmlString = try NSString(contentsOfFile: htmlPath!, encoding: String.Encoding.utf8.rawValue)
webView.loadHTMLString(htmlString as String, baseURL: baseUrl)
} catch {
// catch error
}
webView.navigationDelegate = self
view = webView
}
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
}
}
需要注意:
css/styles.css
因为iOS会压缩你的文件结构,而style.css与index.html处于同一级别,所以写<link rel="stylesheet" type="text/css" href="styles.css">
而不是鉴于这两个版本,这里的问题是来自项目的html / css文件:
幅/ index.html中
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>Offline WebKit</title>
<link rel="stylesheet" type="text/css" href="styles.css">
</head>
<body>
<h1 id="webkit-h1">Offline WebKit!</h1>
</body>
</html>
网/ CSS / styles.css的
#webkit-h1 {
font-size: 80px;
color: lightblue;
}
如果有人想要一个GitHub示例项目,请在评论部分告诉我,我会上传它。
答案 1 :(得分:7)
此方法允许WKWebView正确读取链接的CSS / JS文件的目录和子目录的层次结构。您 NOT 需要更改HTML,CSS或JS代码。
针对Xcode 9.3进行了更新
将本地Web文件的文件夹导入项目的任何位置。请确保你:
☑️根据需要复制项目
☑️创建文件夹引用 (不是“创建群组”)
☑️添加到目标
使用WKWebView转到View Controller,并将以下代码添加到matvec(J) += value(p,q) * x(I)
方法中:
viewDidLoad
let url = Bundle.main.url(forResource: "index", withExtension: "html", subdirectory: "website")!
webView.loadFileURL(url, allowingReadAccessTo: url)
let request = URLRequest(url: url)
webView.load(request)
- 要加载的文件的名称(不含index
扩展名).html
- 您的网络文件夹的名称(website
应位于此目录的根目录)整体代码应如下所示:
index.html
如果您对此方法或代码有任何疑问,我会尽力回答。 :)
答案 2 :(得分:2)
这个解决方案帮助了我:
[configuration.preferences setValue:@YES forKey:@"allowFileAccessFromFileURLs"];
答案 3 :(得分:1)
这很有效(Swift 3,Xcode 8):
import UIKit
import WebKit
class ViewController: UIViewController, WKNavigationDelegate {
var webView: WKWebView!
override func loadView() {
webView = WKWebView()
webView.navigationDelegate = self
view = webView
}
override func viewDidLoad() {
super.viewDidLoad()
if let url = Bundle.main.url(forResource: "file", withExtension: "txt")
{
do
{
let contents = try String(contentsOfFile: url.path)
webView.loadHTMLString(contents, baseURL: url.deletingLastPathComponent())
}
catch
{
print("Could not load the HTML string.")
}
}
}
}
答案 4 :(得分:1)
以这种方式构造URL允许我使用WKWebView从文档目录中加载资源:
guard let docDir = try? FileManager.default.url(for: .documentDirectory, in: .userDomainMask, appropriateFor: nil, create: false) else {
return
}
let resourceURL = docDir.appendingPathComponent("/Path/To/Your/Resource")
self.wkWebView.loadFileURL(resourceURL, allowingReadAccessTo: docDir)
答案 5 :(得分:0)
这可以很好地与文件URL或远程URL配合使用,无论文件是在分发包中还是在文档中:
if url.isFileURL {
webView.loadFileURL(url, allowingReadAccessTo: url)
} else {
let request = URLRequest(url: url)
webView.load(request)
}
答案 6 :(得分:0)
文件必须在文档目录中。
我实现了以下操作来检索文档:
let documentDirUrl = try! FileManager.default.url(for: .documentDirectory, in: .userDomainMask, appropriateFor: nil, create: false)
let fileNameWithExtension = "IMG_0002.PNG"
let indexFileUrl = documentDirUrl.appendingPathComponent(fileNameWithExtension)
if FileManager.default.fileExists(atPath: indexFileUrl.path) {
webView.loadFileURL(indexFileUrl, allowingReadAccessTo: documentDirUrl)
}
答案 7 :(得分:-1)
检查该票证:iOS: How to load local files (not in the bundle) in a WKWebView?
var nsurl = URL(fileURLWithPath: URL(fileURLWithPath: URL(fileURLWithPath: documentsDirectory()).appendingPathComponent(user_appli).absoluteString).appendingPathComponent("index.html").absoluteString) //locally
var readAccessToURL: URL? = nsurl.deletingLastPathComponent?.deletingLastPathComponent
if let anURL = readAccessToURL {
webView?.loadFileURL(nsurl, allowingReadAccessTo: anURL)
}