KMURLCreatePDFWindowController.swift 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397
  1. //
  2. // KMURLCreatePDFWindowController.swift
  3. // PDF Reader Pro
  4. //
  5. // Created by Niehaoyu on 2024/10/9.
  6. //
  7. import Cocoa
  8. import KMComponentLibrary
  9. let kUrlToPDFFolderPath = (try? FileManager.default.url(for: .applicationSupportDirectory, in: .userDomainMask, appropriateFor: nil, create: true).appendingPathComponent(Bundle.main.bundleIdentifier ?? "").appendingPathComponent("WebPage"))?.path ?? ""
  10. @objc protocol KMURLCreatePDFWindowControllerDelegate: AnyObject {
  11. //选择打开文件
  12. @objc optional func urlCreateWCDidChooseFileToOpen(_ vc: KMURLCreatePDFWindowController, _ fileURL: URL)
  13. }
  14. class KMURLCreatePDFWindowController: NSWindowController {
  15. @IBOutlet var contendBox: NSBox!
  16. @IBOutlet var titleLabel: NSTextField!
  17. @IBOutlet var urlRadio: ComponentRadio!
  18. @IBOutlet var fileRadio: ComponentRadio!
  19. @IBOutlet var inputView: ComponentInput!
  20. @IBOutlet var fileInputBGView: NSView!
  21. @IBOutlet var fileInputView: ComponentInput!
  22. @IBOutlet var fileInputAddon: ComponentInputAddon!
  23. @IBOutlet var cancelButton: ComponentButton!
  24. @IBOutlet var openButton: ComponentButton!
  25. private var _isFromURL: Bool = true
  26. private var parentWindow: NSWindow?
  27. private var handler: ((String?) -> Void)!
  28. private var posterMaskView: KMBookletMaskView?
  29. var choosedFilePath: String?
  30. var filePath: String?
  31. weak open var delegate: KMURLCreatePDFWindowControllerDelegate?
  32. override func windowDidLoad() {
  33. super.windowDidLoad()
  34. // Implement this method to handle any initialization after your window controller's window has been loaded from its nib file.
  35. setUpProperty()
  36. reloadData()
  37. }
  38. var isFromURL: Bool {
  39. get {
  40. return _isFromURL
  41. }
  42. set {
  43. _isFromURL = newValue
  44. reloadData()
  45. }
  46. }
  47. func setUpProperty() {
  48. titleLabel.stringValue = NSLocalizedString("New From Web Page", comment: "")
  49. titleLabel.textColor = ComponentLibrary.shared.getComponentColorFromKey("colorText/1")
  50. titleLabel.font = ComponentLibrary.shared.getFontFromKey("mac/body-m-medium")
  51. urlRadio.properties = ComponentCheckBoxProperty(size: .s,
  52. state: .normal,
  53. text: KMLocalizedString("URL"),
  54. checkboxType: .normal)
  55. urlRadio.setTarget(self, action: #selector(typeRadioChanged(_:)))
  56. fileRadio.properties = ComponentCheckBoxProperty(size: .s,
  57. state: .normal,
  58. text: KMLocalizedString("File"),
  59. checkboxType: .normal)
  60. fileRadio.setTarget(self, action: #selector(typeRadioChanged(_:)))
  61. inputView.properties = ComponentInputProperty(size: .s,
  62. state: .normal,
  63. isError: false,
  64. isDisabled: false,
  65. placeholder: KMLocalizedString("https://pdfreaderpro.com"))
  66. inputView.delegate = self
  67. fileInputView.properties = ComponentInputProperty(size: .s,
  68. state: .normal,
  69. isError: false,
  70. isDisabled: false,
  71. placeholder: KMLocalizedString("Select Files(.html,.webarchive)"),
  72. creatable: false)
  73. fileInputView.properties.propertyInfo.cornerRadius_topRight = 0
  74. fileInputView.properties.propertyInfo.cornerRadius_bottomRight = 0
  75. fileInputAddon.properties = ComponentInputAddonProperty(size: .s,
  76. state: .normal,
  77. addOnBefore: false,
  78. onlyRead: false,
  79. addonType: .imageWithColor,
  80. iconImage: NSImage(named: "file_icon"))
  81. fileInputAddon.setTarget(self, action: #selector(chooseURLAction(_ :)))
  82. cancelButton.properties = ComponentButtonProperty(type: .default_tertiary,
  83. size: .s,
  84. state: .normal,
  85. buttonText: KMLocalizedString("Cancel"),
  86. keepPressState: false)
  87. cancelButton.setTarget(self, action: #selector(cancelButtonClicked(_ :)))
  88. cancelButton.keyEquivalent = KMKeyEquivalent.esc.string()
  89. openButton.properties = ComponentButtonProperty(type: .primary,
  90. size: .s,
  91. state: .normal,
  92. buttonText: KMLocalizedString("Open"),
  93. keepPressState: false)
  94. openButton.setTarget(self, action: #selector(openButtonClicked(_ :)))
  95. openButton.keyEquivalent = KMKeyEquivalent.enter // Enter key
  96. }
  97. func reloadData() {
  98. urlRadio.properties.checkboxType = isFromURL ? .selected : .normal
  99. fileRadio.properties.checkboxType = isFromURL ? .normal : .selected
  100. fileInputBGView.isHidden = isFromURL ? true : false
  101. inputView.isHidden = isFromURL ? false : true
  102. cancelButton.properties.state = .normal
  103. openButton.properties.state = .normal
  104. if let filePath = choosedFilePath {
  105. fileInputView.properties.text = filePath
  106. } else {
  107. fileInputView.properties.text = ""
  108. }
  109. if isFromURL {
  110. if inputView.properties.text.count > 0 {
  111. openButton.properties.isDisabled = false
  112. } else {
  113. openButton.properties.isDisabled = true
  114. }
  115. } else {
  116. if fileInputView.properties.text.count > 0 {
  117. openButton.properties.isDisabled = false
  118. } else {
  119. openButton.properties.isDisabled = true
  120. }
  121. }
  122. inputView.reloadData()
  123. fileInputView.reloadData()
  124. fileInputAddon.reloadData()
  125. urlRadio.reloadData()
  126. fileRadio.reloadData()
  127. cancelButton.reloadData()
  128. openButton.reloadData()
  129. fileInputView.reloadData()
  130. }
  131. //MARK: - Action
  132. @objc func typeRadioChanged(_ sender: NSView) {
  133. if sender == fileRadio {
  134. isFromURL = false
  135. } else if sender == self.urlRadio {
  136. isFromURL = true
  137. }
  138. }
  139. @objc func cancelButtonClicked(_ sender: NSView) {
  140. choosedFilePath = nil
  141. inputView.properties.text = ""
  142. hideWaitting()
  143. reloadData()
  144. KMURLToPDF.shareInstance.stopLoading()
  145. handler?(nil)
  146. parentWindow?.endSheet(self.window!)
  147. }
  148. @objc func openButtonClicked(_ sender: NSView) {
  149. convertInfoToPDF()
  150. }
  151. @objc func chooseURLAction(_ sender: NSView) {
  152. let openPanel = NSOpenPanel()
  153. openPanel.allowedFileTypes = ["html", "HTML"]
  154. openPanel.allowsMultipleSelection = false
  155. openPanel.beginSheetModal(for: self.window!) { [weak self] result in
  156. if result == NSApplication.ModalResponse.OK {
  157. if let url = openPanel.url {
  158. self?.choosedFilePath = url.path
  159. self?.reloadData()
  160. }
  161. }
  162. }
  163. }
  164. func convertInfoToPDF() {
  165. if !FileManager.default.fileExists(atPath: kUrlToPDFFolderPath) {
  166. try? FileManager.default.createDirectory(atPath: kUrlToPDFFolderPath, withIntermediateDirectories: false, attributes: nil)
  167. }
  168. var url: URL?
  169. if isFromURL {
  170. var urlString = inputView.properties.text
  171. var tUrl = URL(string: urlString)
  172. if tUrl?.scheme?.count ?? 0 < 1 {
  173. if urlString.hasSuffix(".com") == false {
  174. urlString = urlString + ".com"
  175. }
  176. tUrl = URL(string: "http://\(urlString)")
  177. }
  178. url = tUrl
  179. } else {
  180. if let choosedFilePath = choosedFilePath {
  181. url = URL(fileURLWithPath: choosedFilePath)
  182. }
  183. }
  184. if let url = url {
  185. inputView.properties.text = url.absoluteString
  186. inputView.reloadData()
  187. showWaitting()
  188. let outPutPath = kUrlToPDFFolderPath
  189. if isFromURL {
  190. KMURLToPDF.shareInstance.convertLinkString(string: url.absoluteString, toPath: outPutPath) {[weak self] url, error in
  191. self?.hideWaitting()
  192. if error == nil && url?.path.isEmpty == false {
  193. self?.choosedFilePath = nil
  194. self?.inputView.properties.text = ""
  195. self?.reloadData()
  196. self?.handler?(url?.path)
  197. if let window = self?.window {
  198. self?.parentWindow?.endSheet(window)
  199. }
  200. } else {
  201. if let window = self?.window {
  202. let alert = NSAlert()
  203. alert.alertStyle = .critical
  204. alert.messageText = NSLocalizedString("Conversion Failed", comment: "")
  205. if let description = error?.localizedDescription {
  206. alert.messageText = description
  207. }
  208. alert.beginSheetModal(for: window) { [weak self] result in
  209. }
  210. }
  211. }
  212. }
  213. } else {
  214. KMURLToPDF.shareInstance.convertHtmlFileToPDF(fromPath: url.path, toPath: outPutPath) {[weak self] url, error in
  215. self?.hideWaitting()
  216. if error == nil && url?.path.isEmpty == false {
  217. self?.choosedFilePath = nil
  218. self?.inputView.properties.text = ""
  219. self?.reloadData()
  220. self?.handler?(url?.path)
  221. if let window = self?.window {
  222. self?.parentWindow?.endSheet(window)
  223. }
  224. } else {
  225. if let window = self?.window {
  226. let alert = NSAlert()
  227. alert.alertStyle = .critical
  228. alert.messageText = NSLocalizedString("Conversion Failed", comment: "")
  229. alert.beginSheetModal(for: window) { [weak self] result in
  230. }
  231. }
  232. }
  233. }
  234. }
  235. }
  236. }
  237. //MARK: - WaitingView
  238. func showWaitting() {
  239. if posterMaskView == nil {
  240. posterMaskView = KMBookletMaskView(frame: NSMakeRect(0, 0, window?.frame.size.width ?? 0, window?.frame.size.height ?? 0))
  241. }
  242. contendBox.contentView?.addSubview(posterMaskView!, positioned: .below, relativeTo: self.cancelButton)
  243. }
  244. func hideWaitting() {
  245. posterMaskView?.removeFromSuperview()
  246. }
  247. //MARK: - public
  248. func own_beginSheetModal(for window: NSWindow?, completionHandler handler: ((String?) -> Void)?) {
  249. self.window?.makeFirstResponder(nil)
  250. if window != nil {
  251. parentWindow = window
  252. window!.beginSheet(self.window!) { ModalResponse in
  253. self.handler?(nil)
  254. }
  255. choosedFilePath = nil
  256. self.handler = handler
  257. hideWaitting()
  258. reloadData()
  259. }
  260. }
  261. override func becomeFirstResponder() -> Bool {
  262. return true
  263. }
  264. func formatFloat(_ f: Float) -> String {
  265. if f.truncatingRemainder(dividingBy: 1) == 0 {
  266. return String(format: "%.0f", f)
  267. } else if (f * 10).truncatingRemainder(dividingBy: 1) == 0 {
  268. return String(format: "%.1f", f)
  269. } else {
  270. return String(format: "%.2f", f)
  271. }
  272. }
  273. func isUrl(_ urlString: String?) -> Bool {
  274. guard let urlString = urlString else { return false }
  275. // 实现方法有问题,暂不使用
  276. var url: String
  277. if urlString.count > 4, urlString.prefix(4) == "www." {
  278. url = "http://\(urlString)"
  279. } else {
  280. url = urlString
  281. }
  282. let urlRegex = "\\bhttps?://[a-zA-Z0-9\\-.]+(?::(\\d+))?(?:(?:/[a-zA-Z0-9\\-._?,'+\\&%$=~*!():@\\\\]*)+)?"
  283. let urlTest = NSPredicate(format: "SELF MATCHES %@", urlRegex)
  284. return urlTest.evaluate(with: url)
  285. }
  286. func urlValueEncode(_ str: String) -> String? {
  287. let allowedCharacterSet = CharacterSet(charactersIn: "!*'();:@&=+$,?%#[]{}").inverted
  288. return str.addingPercentEncoding(withAllowedCharacters: allowedCharacterSet)
  289. }
  290. //MARK: - MouseEvent
  291. override func mouseDown(with event: NSEvent) {
  292. super.mouseDown(with: event)
  293. window?.makeFirstResponder(nil)
  294. }
  295. }
  296. //MARK: - ComponentInputDelegate
  297. extension KMURLCreatePDFWindowController: ComponentInputDelegate {
  298. func componentInputDidChanged(inputView: ComponentInput) {
  299. reloadData()
  300. }
  301. }