如何检测UIWebView中的单击文件链接?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了如何检测UIWebView中的单击文件链接?相关的知识,希望对你有一定的参考价值。

我在我的应用程序中使用UIWebView,我想检测用户何时单击指向文件(pdf,doc,docx ....)的链接而不是其他html页面。这将允许我下载文件并通过显示选项菜单打开它。

我试着使用以下功能:

webView(_ webView: UIWebView, shouldStartLoadWith request: URLRequest,  navigationType: UIWebViewNavigationType) -> Bool

在这个函数中,我获取URL,发送HEAD请求以获取响应的content-type,如果它不包含text/html,我下载它并显示选项菜单。

这不是一个完美的解决方案,因为我需要发出Sync请求才能获得content-type

以前有人遇到过这个问题吗?我该如何解决?

我试图在互联网上搜索,但找不到与此问题类似的任何内容

答案

试试这个

override func viewDidLoad() {
        super.viewDidLoad()
        self.myWebview.delegate = self;

        // Do any additional setup after loading the view, typically from a nib.
        let url = URL(string: "YOUR_URL_STRING")
        debugPrint(url!)
        myWebview.loadRequest(URLRequest(url: url!))
    }

    func webView(_ webView: UIWebView, shouldStartLoadWith request: URLRequest, navigationType: UIWebViewNavigationType) -> Bool {
        debugPrint("func myWebView has been called")
        debugPrint(request.url!)
        if navigationType == UIWebViewNavigationType.linkClicked {
            if (request.url!.host! == "stackoverflow.com"){
                return true
            } else {
                //UIApplication.sharedApplication().openURL(request.URL!)
                UIApplication.shared.open(request.url!)
                return false
            }
        }
        return true
    }

希望这会对你有所帮助。

另一答案

最后,我找到了完美的解决方案。

在shouldStartLoadWith中:

   var checkingProcesses: [URL:CheckingProcess] = [:]

   func webView(_ webView: UIWebView, shouldStartLoadWith request: URLRequest, navigationType: UIWebViewNavigationType) -> Bool {

    Logger.log("WebviewViewController", "shouldStartLoadWith", "with url: (String(describing: request.url?.absoluteString))")
    self.loadingViewLogic(hide: false)

    if let currentURL = request.url {

        if let process = self.checkingProcesses[currentURL]{

            Logger.log("WebviewViewController", "shouldStartLoadWith", "Known process: (String(describing: process))")

            if(process.status == CheckingStatus.finished){

                if(process.filePath != nil){
                    self.openFileByPresentingOptions(pathURL: process.filePath!)
                    return false

                }else if(process.errorMessage != nil){

                    //error
                    Util.showAlert(context: self, title: "Error getting URL type", message: process.errorMessage!, alertActions: nil)
                    process.clearResults()//In order to try the next time the user asks.
                    self.loadingViewLogic(hide: true)
                    return false

                }else{

                    //link - open it
                    return true
                }

            }else if(process.status == CheckingStatus.processing){

                Util.showAlert(context: self, title: "Processing...", message: "Please wait...", alertActions: nil)
                return false

            }else{//inactive

                self.checkingProcesses[currentURL]?.startCheck()
                return false
            }

        }else{// did not create the process yet

            Logger.log("WebviewViewController", "shouldStartLoadWith", "Creating new process for URL: (currentURL.absoluteString)")

            self.checkingProcesses[currentURL] = CheckingProcess(url: currentURL, webView: self.webView)
            self.checkingProcesses[currentURL]?.startCheck()
            return false
        }

    }else{

        Logger.log("WebviewViewController", "shouldStartLoadWith", "Couldn't get the currentURL")
        self.loadingViewLogic(hide: true)
        return false
    }

}

我创建了一个类“CheckingProcess.swift”:

import UIKit

public enum CheckingStatus {
    case inactive
    case processing
    case finished
}

class CheckingProcess: CustomStringConvertible {

    var status : CheckingStatus;
    var url: URL;
    var filePath: URL? = nil;
    var errorMessage: String? = nil;
    let webView: UIWebView?

    init(url: URL, webView: UIWebView) {

        Logger.log("CheckingProcess", "init", "with url: (url.absoluteString)")
        self.status = CheckingStatus.inactive;
        self.url = url;
        self.webView = webView
    }

    func startCheck(){

        Logger.log("CheckingProcess", "startCheck", "with url: (self.url.absoluteString), START")
        self.status = CheckingStatus.processing

        let destinationUrl = Util.generateLocalFileURL(remoteURL: self.url)

        DownloadManager.downloadFileAndSaveIfNotHTML(url: url, to: destinationUrl, completion: {(finalLocalURL, errorMessage) in

            Logger.log("CheckingProcess", "startCheck callback block", "url: (self.url.absoluteString), FinalLocalURL: (String(describing: finalLocalURL)), errorMessage: (String(describing: errorMessage))")
            self.filePath = finalLocalURL
            self.errorMessage = errorMessage

            self.status = CheckingStatus.finished

            self.webView?.loadRequest(NSURLRequest(url: self.url as URL) as URLRequest)
        })
    }

    func clearResults(){
        self.status = CheckingStatus.inactive
        self.filePath = nil
        self.errorMessage = nil
    }

    public var description: String {

        return "URL: (self.url), Status: (self.status), filePath: (String(describing: self.filePath)), errorMessage: (String(describing: self.errorMessage))"
    }

}

请享用!

以上是关于如何检测UIWebView中的单击文件链接?的主要内容,如果未能解决你的问题,请参考以下文章

如何检测 uiWebView 上 javascript 操作的启动?

使用“stringByEvaluatingJavaScriptFromString”单击 UIWebView 中的链接

当我单击 UIWebView 时,如何在 Safari 中打开链接?

XCode:如何在 UIWebView 中输入 javascript 以单击链接(但链接每次都不同)

UITableView:单击单元格并使用 UIWebView 打开单元格中的链接

摆脱 iOs UIWebView 中的链接单击“flash”指示器