[英]Authentication with WKWebView in Swift
在我的 iOS 應用程序中,我想使用 WKWebView 在應用程序中包裝外部 URL。 此 URL 需要基本身份驗證(它需要用戶和密碼憑據,如下面的屏幕截圖所示)。
經過一番調查,我嘗試使用didReceiveAuthenticationChallenge
方法來啟用自動登錄,所以我不明白它是如何工作的。
這是我的代碼。
import UIKit
import WebKit
class WebViewController: UIViewController, WKNavigationDelegate {
var webView: WKWebView?
private var request : NSURLRequest {
let baseUrl = "https://unimol.esse3.cineca.it/auth/Logon.do"
let URL = NSURL(string: baseUrl)!
return NSURLRequest(URL: URL)
}
/* Start the network activity indicator when the web view is loading */
func webView(webView: WKWebView,
didStartProvisionalNavigation navigation: WKNavigation){
UIApplication.sharedApplication().networkActivityIndicatorVisible = true
}
/* Stop the network activity indicator when the loading finishes */
func webView(webView: WKWebView,
didFinishNavigation navigation: WKNavigation){
UIApplication.sharedApplication().networkActivityIndicatorVisible = false
}
func webView(webView: WKWebView,
decidePolicyForNavigationResponse navigationResponse: WKNavigationResponse,
decisionHandler: ((WKNavigationResponsePolicy) -> Void)){
decisionHandler(.Allow)
}
func webView(webView: WKWebView, didReceiveAuthenticationChallenge challenge: NSURLAuthenticationChallenge, completionHandler: (NSURLSessionAuthChallengeDisposition, NSURLCredential?) -> Void) {
if challenge.protectionSpace.host == "https://unimol.esse3.cineca.it/auth/Logon.do" {
let user = "*******"
let password = "******"
let credential = NSURLCredential(user: user, password: password, persistence: NSURLCredentialPersistence.ForSession)
challenge.sender?.useCredential(credential, forAuthenticationChallenge: challenge)
}
}
override func viewDidLoad() {
/* Create our preferences on how the web page should be loaded */
let preferences = WKPreferences()
preferences.javaScriptEnabled = false
/* Create a configuration for our preferences */
let configuration = WKWebViewConfiguration()
configuration.preferences = preferences
/* Now instantiate the web view */
webView = WKWebView(frame: view.bounds, configuration: configuration)
if let theWebView = webView {
/* Load a web page into our web view */
let urlRequest = self.request
theWebView.loadRequest(urlRequest)
theWebView.navigationDelegate = self
view.addSubview(theWebView)
}
}
}
我正面臨這個例外:
Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: 'Completion handler passed to -[MyUnimol.WebViewController webView:didReceiveAuthenticationChallenge:completionHandler:] was not called'
如果我刪除了didReceiveAuthenticationChallenge
方法,我可以訪問該 URL,但顯然給了我錯誤的憑據。
任何人都可以解釋我做錯了什么嗎?
添加以下行:
completionHandler(NSURLSessionAuthChallengeDisposition.UseCredential, credential)
在didReceiveAuthenticationChallenge
結束時解決了這個問題。
我參加聚會遲到了,但這仍然對某人有用。
要在 WKWebview Swift 4 中支持身份驗證挑戰,請完成以下代碼
func webView(_ webView: WKWebView, didReceive challenge: URLAuthenticationChallenge, completionHandler: @escaping (URLSession.AuthChallengeDisposition, URLCredential?) -> Void) {
guard let hostname = webView.url?.host else {
return
}
let authenticationMethod = challenge.protectionSpace.authenticationMethod
if authenticationMethod == NSURLAuthenticationMethodDefault || authenticationMethod == NSURLAuthenticationMethodHTTPBasic || authenticationMethod == NSURLAuthenticationMethodHTTPDigest {
let av = UIAlertController(title: webView.title, message: String(format: "AUTH_CHALLENGE_REQUIRE_PASSWORD".localized, hostname), preferredStyle: .alert)
av.addTextField(configurationHandler: { (textField) in
textField.placeholder = "USER_ID".localized
})
av.addTextField(configurationHandler: { (textField) in
textField.placeholder = "PASSWORD".localized
textField.isSecureTextEntry = true
})
av.addAction(UIAlertAction(title: "BUTTON_OK".localized, style: .default, handler: { (action) in
guard let userId = av.textFields?.first?.text else{
return
}
guard let password = av.textFields?.last?.text else {
return
}
let credential = URLCredential(user: userId, password: password, persistence: .none)
completionHandler(.useCredential,credential)
}))
av.addAction(UIAlertAction(title: "BUTTON_CANCEL".localized, style: .cancel, handler: { _ in
completionHandler(.cancelAuthenticationChallenge, nil);
}))
self.parentViewController?.present(av, animated: true, completion: nil)
}else if authenticationMethod == NSURLAuthenticationMethodServerTrust{
// needs this handling on iOS 9
completionHandler(.performDefaultHandling, nil);
}else{
completionHandler(.cancelAuthenticationChallenge, nil);
}
}
以更簡單的方式:
雖然這個答案看起來多余但發布,因為這可能有助於其他天真的開發人員(如我)。
func webView(_ webView: WKWebView, didReceive challenge: URLAuthenticationChallenge, completionHandler: @escaping (URLSession.AuthChallengeDisposition, URLCredential?) -> Void) {
guard (webView.url?.host) != nil else {
return
}
let authenticationMethod = challenge.protectionSpace.authenticationMethod
if authenticationMethod == NSURLAuthenticationMethodDefault || authenticationMethod == NSURLAuthenticationMethodHTTPBasic || authenticationMethod == NSURLAuthenticationMethodHTTPDigest {
let credential = URLCredential(user: userName, password: password, persistence: .none)
completionHandler(.useCredential, credential)
} else if authenticationMethod == NSURLAuthenticationMethodServerTrust {
completionHandler(.performDefaultHandling, nil)
} else {
completionHandler(.cancelAuthenticationChallenge, nil)
}
}
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.