Захват window.print() из WKWebView

Я пытаюсь захватить вызов JavaScript window.print() из WKWebView с моим приложением, чтобы фактически показать диалог печати и позволить им печатать с помощью кнопки, представленной на странице (вместо кнопки внутри моего приложения). Кто-нибудь знает лучший способ сделать это?

В настоящее время для меня, щелкнув ссылку, которая вызывает window.print(), ничего не происходит, кроме как с помощью метода делегата definePolicyForNavigationAction, и я не смог найти в нем ничего подходящего.

2 ответа

Решение

Разобрался сразу после публикации (очевидно)... Надеюсь, это поможет кому-то еще.

При создании вашего веб-просмотра...

let configuration = WKWebViewConfiguration()
let script = WKUserScript(source: "window.print = function() { window.webkit.messageHandlers.print.postMessage('print') }", injectionTime: WKUserScriptInjectionTime.AtDocumentEnd, forMainFrameOnly: true)
configuration.userContentController.addUserScript(script)
configuration.userContentController.addScriptMessageHandler(self, name: "print")
self.webView = WKWebView(frame: CGRectZero, configuration: configuration)

Затем просто примите протокол WKScriptMessageHandler в вашем контроллере представления и добавьте необходимый метод...

func userContentController(userContentController: WKUserContentController, didReceiveScriptMessage message: WKScriptMessage) {
    if message.name == "print" {
        printCurrentPage()
    } else {
        println("Some other message sent from web page...")
    }
}

func printCurrentPage() {
    let printController = UIPrintInteractionController.sharedPrintController()
    let printFormatter = self.webView.viewPrintFormatter()
    printController?.printFormatter = printFormatter

    let completionHandler: UIPrintInteractionCompletionHandler = { (printController, completed, error) in
        if !completed {
            if let e = error? {
                println("[PRINT] Failed: \(e.domain) (\(e.code))")
            } else {
                println("[PRINT] Canceled")
            }
        }
    }

    if let controller = printController? {
        if UIDevice.currentDevice().userInterfaceIdiom == .Pad {
            controller.presentFromBarButtonItem(someBarButtonItem, animated: true, completionHandler: completionHandler)
        } else {
            controller.presentAnimated(true, completionHandler: completionHandler)
        }
    }
}

Есть недокументированный делегат для перехвата window.print()

class MyApp: WKUIDelegate {
  func makeWebview() {
      ...
      self.webview.UIDelegate = self
  }
  func _webView(webView: WKWebView!, printFrame: WKFrameInfo) {
      println("JS: window.print()")
      printCurrentPage()
  }     
}
Другие вопросы по тегам