// // NCViewerRichdocument.swift // Nextcloud // // Created by Marino Faggiana on 06/09/18. // Copyright © 2018 Marino Faggiana. All rights reserved. // // Author Marino Faggiana // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . // import Foundation import WebKit import NCCommunication class NCViewerRichdocument: WKWebView, WKNavigationDelegate, WKScriptMessageHandler, NCSelectDelegate { let appDelegate = UIApplication.shared.delegate as! AppDelegate @objc var metadata: tableMetadata! var documentInteractionController: UIDocumentInteractionController! var view: UIView! var viewController: UIViewController! override init(frame: CGRect, configuration: WKWebViewConfiguration) { super.init(frame: frame, configuration: configuration) let contentController = configuration.userContentController contentController.add(self, name: "RichDocumentsMobileInterface") autoresizingMask = [.flexibleWidth, .flexibleHeight] navigationDelegate = self NotificationCenter.default.addObserver(self, selector: #selector(self.grabFocus), name: NSNotification.Name(rawValue: k_notificationCenter_richdocumentGrabFocus), object: nil) } required init?(coder: NSCoder) { super.init(coder: coder) } override func layoutSubviews() { super.layoutSubviews() self.frame = CGRect(x: 0, y: 0, width: self.frame.width, height: self.frame.height) } @objc func viewRichDocumentAt(_ link: String, metadata: tableMetadata, view: UIView, viewController: UIViewController) { self.metadata = metadata self.view = view self.viewController = viewController NotificationCenter.default.addObserver(self, selector: #selector(keyboardDidShow), name: UIResponder.keyboardDidShowNotification, object: nil) NotificationCenter.default.addObserver(self, selector: #selector(keyboardWillHide), name: UIResponder.keyboardWillHideNotification, object: nil) var request = URLRequest(url: URL(string: link)!) request.addValue("true", forHTTPHeaderField: "OCS-APIRequest") let language = NSLocale.preferredLanguages[0] as String request.addValue(language, forHTTPHeaderField: "Accept-Language") let userAgent : String = CCUtility.getUserAgent() customUserAgent = userAgent load(request) self.view.addSubview(self) } @objc func keyboardDidShow(notification: Notification) { guard let info = notification.userInfo else { return } guard let frameInfo = info[UIResponder.keyboardFrameEndUserInfoKey] as? NSValue else { return } let keyboardFrame = frameInfo.cgRectValue //print("keyboardFrame: \(keyboardFrame)") frame.size.height = view.frame.height - keyboardFrame.size.height } @objc func keyboardWillHide(notification: Notification) { frame = view.frame } //MARK: - public func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) { if (message.name == "RichDocumentsMobileInterface") { if message.body as? String == "close" { appDelegate.activeDetail.viewUnload() appDelegate.activeMain.readFileReloadFolder() } if message.body as? String == "insertGraphic" { let storyboard = UIStoryboard(name: "NCSelect", bundle: nil) let navigationController = storyboard.instantiateInitialViewController() as! UINavigationController let viewController = navigationController.topViewController as! NCSelect viewController.delegate = self viewController.hideButtonCreateFolder = true viewController.selectFile = true viewController.includeDirectoryE2EEncryption = false viewController.includeImages = true viewController.type = "" viewController.layoutViewSelect = k_layout_view_richdocument navigationController.modalPresentationStyle = UIModalPresentationStyle.fullScreen self.viewController.present(navigationController, animated: true, completion: nil) } if message.body as? String == "share" { NCMainCommon.sharedInstance.openShare(ViewController: viewController, metadata: metadata, indexPage: 2) } if let param = message.body as? Dictionary { if param["MessageName"] as? String == "downloadAs" { if let values = param["Values"] as? Dictionary { guard let type = values["Type"] as? String else { return } guard let urlString = values["URL"] as? String else { return } guard let url = URL(string: urlString) else { return } guard let components = URLComponents(url: url, resolvingAgainstBaseURL: false) else { return } let filename = (components.path as NSString).lastPathComponent let fileNameLocalPath = CCUtility.getDirectoryUserData() + "/" + filename if type == "print" { NCUtility.sharedInstance.startActivityIndicator(view: self, bottom: 0) } NCCommunication.shared.download(serverUrlFileName: urlString, fileNameLocalPath: fileNameLocalPath, requestHandler: { (_) in }, progressHandler: { (_) in }, completionHandler: { (account, etag, date, lenght, error, errorCode, errorDescription) in if errorCode == 0 && account == self.metadata.account { if type == "print" { NCUtility.sharedInstance.stopActivityIndicator() let pic = UIPrintInteractionController.shared let printInfo = UIPrintInfo.printInfo() printInfo.outputType = UIPrintInfo.OutputType.general printInfo.orientation = UIPrintInfo.Orientation.portrait printInfo.jobName = "Document" pic.printInfo = printInfo pic.printingItem = URL(fileURLWithPath: fileNameLocalPath) pic.present(from: CGRect.zero, in: self, animated: true, completionHandler: { (pci, completed, error) in // end. }) } else { self.documentInteractionController = UIDocumentInteractionController() self.documentInteractionController.url = URL(fileURLWithPath: fileNameLocalPath) self.documentInteractionController.presentOptionsMenu(from: self.appDelegate.window.rootViewController!.view.bounds, in: self.appDelegate.window.rootViewController!.view, animated: true) } } else { NCContentPresenter.shared.messageNotification("_error_", description: errorDescription, delay: TimeInterval(k_dismissAfterSecond), type: NCContentPresenter.messageType.error, errorCode: errorCode) } }) } } else if param["MessageName"] as? String == "fileRename" { if let values = param["Values"] as? Dictionary { guard let newName = values["NewName"] as? String else { return } metadata.fileName = newName metadata.fileNameView = newName } } } if message.body as? String == "documentLoaded" { print("documentLoaded") } if message.body as? String == "paste" { self.paste(self) } } } //MARK: - @objc func grabFocus() { let functionJS = "OCA.RichDocuments.documentsMain.postGrabFocus()" evaluateJavaScript(functionJS) { (result, error) in } } //MARK: - func dismissSelect(serverUrl: String?, metadata: tableMetadata?, type: String, buttonType: String, overwrite: Bool) { if serverUrl != nil && metadata != nil { let path = CCUtility.returnFileNamePath(fromFileName: metadata!.fileName, serverUrl: serverUrl!, activeUrl: appDelegate.activeUrl)! NCCommunication.shared.createAssetRichdocuments(path: path) { (account, url, errorCode, errorDescription) in if errorCode == 0 && account == self.appDelegate.activeAccount { let functionJS = "OCA.RichDocuments.documentsMain.postAsset('\(metadata!.fileNameView)', '\(url!)')" self.evaluateJavaScript(functionJS, completionHandler: { (result, error) in }) } else if errorCode != 0 { NCContentPresenter.shared.messageNotification("_error_", description: errorDescription, delay: TimeInterval(k_dismissAfterSecond), type: NCContentPresenter.messageType.error, errorCode: Int(k_CCErrorInternalError)) } else { print("[LOG] It has been changed user during networking process, error.") } } } } func select(_ metadata: tableMetadata!, serverUrl: String!) { let path = CCUtility.returnFileNamePath(fromFileName: metadata!.fileName, serverUrl: serverUrl!, activeUrl: appDelegate.activeUrl)! NCCommunication.shared.createAssetRichdocuments(path: path) { (account, url, errorCode, errorDescription) in if errorCode == 0 && account == self.appDelegate.activeAccount { let functionJS = "OCA.RichDocuments.documentsMain.postAsset('\(metadata.fileNameView)', '\(url!)')" self.evaluateJavaScript(functionJS, completionHandler: { (result, error) in }) } else if errorCode != 0 { NCContentPresenter.shared.messageNotification("_error_", description: errorDescription, delay: TimeInterval(k_dismissAfterSecond), type: NCContentPresenter.messageType.error, errorCode: Int(k_CCErrorInternalError)) } else { print("[LOG] It has been changed user during networking process, error.") } } } //MARK: - public func webView(_ webView: WKWebView, didReceive challenge: URLAuthenticationChallenge, completionHandler: @escaping (URLSession.AuthChallengeDisposition, URLCredential?) -> Void) { if let serverTrust = challenge.protectionSpace.serverTrust { completionHandler(Foundation.URLSession.AuthChallengeDisposition.useCredential, URLCredential(trust: serverTrust)) } else { completionHandler(URLSession.AuthChallengeDisposition.useCredential, nil); } } public func webView(_ webView: WKWebView, didStartProvisionalNavigation navigation: WKNavigation!) { print("didStartProvisionalNavigation"); } public func webView(_ webView: WKWebView, didReceiveServerRedirectForProvisionalNavigation navigation: WKNavigation!) { print("didReceiveServerRedirectForProvisionalNavigation"); } public func webView(_ webView: WKWebView, didFinish navigation: WKNavigation!) { NCUtility.sharedInstance.stopActivityIndicator() } }