KMUserFeekbackWindowController.swift 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515
  1. //
  2. // KMUserFeekbackWindowController.swift
  3. // PDF Reader Pro Edition
  4. //
  5. // Created by tangchao on 2024/7/11.
  6. //
  7. import Cocoa
  8. @objcMembers class KMUserFeekbackWindowController: NSWindowController {
  9. static let shared = KMUserFeekbackWindowController(windowNibName: "KMUserFeekbackWindowController")
  10. @IBOutlet weak var emailBox: NSBox!
  11. @IBOutlet weak var typeBox: NSBox!
  12. @IBOutlet weak var despBox: NSBox!
  13. @IBOutlet weak var listHeaderBox: NSBox!
  14. @IBOutlet weak var listBox: NSBox!
  15. @IBOutlet weak var comfirmButton: NSButton!
  16. @IBOutlet weak var cancelButton: NSButton!
  17. private var emailItemView_: KMUserFbEmailItemView?
  18. private var typeItemView_: KMUserFbTypeItemView?
  19. private var despItemView_: KMUserFbDespItemView?
  20. private var listHeaderItemView_: KMUserFbListHeaderItemView?
  21. private var listItemView_: KMUserListItemView?
  22. typealias KMRequestServerComplete = (_ wrapper: ResultWrapper) -> Void
  23. private var filePaths_: [String] = []
  24. private var logPath_: String?
  25. private var popover_: NSPopover?
  26. var typeString: String = ""
  27. private var fileFormats_: [String] = ["jpg", "jpeg", "png", "gif", "bmp", "tiff", "tif", "psd", "svg", "pdf", "mp4", "mov", "avi", "mkv", "wmv", "flv", "mpg", "3gp", "doc", "docx", "xls" ,"xlsx", "ppt", "pptx", "txt", "rtf", "nfo"]
  28. private var fileFormatsString = "jpg, jpeg, png, gif, bmp, tiff, tif, psd, svg, pdf, mp4, mov, avi, mkv, wmv, flv, mpg, 3gp, doc, docx, xls,xlsx, ppt, pptx, txt, rtf, nfo"
  29. deinit {
  30. Swift.debugPrint("KMUserFeekbackWindowController deinit.")
  31. }
  32. override func windowDidLoad() {
  33. super.windowDidLoad()
  34. self._initDefaultValue()
  35. self._initSubViews()
  36. self.typeItemView_?.comBoBox.selectItem(at: 0)
  37. }
  38. private func _initDefaultValue() {
  39. if let btn = self.window?.standardWindowButton(.miniaturizeButton) {
  40. btn.isHidden = true
  41. }
  42. if let btn = self.window?.standardWindowButton(.zoomButton) {
  43. btn.isHidden = true
  44. }
  45. self.window?.title = NSLocalizedString("Feedback for PDF Reader Pro", comment: "")
  46. self.window?.backgroundColor = NSColor.white
  47. self.emailBox.borderWidth = 0
  48. self.typeBox.borderWidth = 0
  49. self.despBox.borderWidth = 0
  50. self.listHeaderBox.borderWidth = 0
  51. self.listBox.borderWidth = 0
  52. let email = VerificationManager.default()?.email ?? ""
  53. self.emailItemView_?.textfiled.stringValue = email
  54. self.emailItemView_?.textfiled.nextResponder = self.despItemView_?.textView
  55. self.cancelButton.title = NSLocalizedString("Cancel", comment: "")
  56. self.comfirmButton.title = NSLocalizedString("Submit", comment: "")
  57. self.cancelButton.target = self
  58. self.cancelButton.action = #selector(_cancelButtonAction)
  59. self.comfirmButton.target = self
  60. self.comfirmButton.action = #selector(_comfirmButtonAction)
  61. }
  62. private func _initSubViews() {
  63. self.emailItemView_ = KMUserFbEmailItemView.createFromNib()
  64. self.emailBox.contentView = self.emailItemView_
  65. self.typeItemView_ = KMUserFbTypeItemView.createFromNib()
  66. self.typeBox.contentView = self.typeItemView_
  67. self.despItemView_ = KMUserFbDespItemView.createFromNib()
  68. self.despBox.contentView = self.despItemView_
  69. self.listHeaderItemView_ = KMUserFbListHeaderItemView.createFromNib()
  70. self.listHeaderBox.contentView = self.listHeaderItemView_
  71. self.listItemView_ = KMUserListItemView.createFromNib()
  72. self.listBox.contentView = self.listItemView_
  73. self.listItemView_?.hiddenTip()
  74. self.listHeaderItemView_?.helpHoverCallback = { [weak self] action in
  75. if action.rawValue == 0 { // enter
  76. self?._showHelpPopover((self?.listHeaderItemView_?.helpButton)!)
  77. } else if action.rawValue == 2 { // exit
  78. self?._hiddenHelpPopover()
  79. }
  80. }
  81. self.listHeaderItemView_?.itemClick = { [weak self] idx in
  82. if idx == 1 { // 获取日志文件
  83. } else if idx == 3 { // 新增文件
  84. let panel = NSOpenPanel()
  85. // panel.message = self?.fileFormatsString
  86. panel.allowedFileTypes = self?.fileFormats_
  87. panel.allowsMultipleSelection = true
  88. panel.beginSheetModal(for: self!.window!) { resp in
  89. if resp == .cancel {
  90. return
  91. }
  92. for url in panel.urls {
  93. self?.filePaths_.append(url.path)
  94. }
  95. self?._updateListData()
  96. }
  97. }
  98. }
  99. self.listItemView_?.itemClick = { [weak self] idx, param in
  100. if let cnt = self?.filePaths_.count, idx < cnt {
  101. self?.filePaths_.remove(at: idx)
  102. self?._updateListData()
  103. }
  104. }
  105. self.listItemView_?.validateDropCallback = { [weak self] info, row, dropOperation in
  106. let datas = self?.filePaths_.count ?? 0
  107. if datas >= 10 {
  108. return NSDragOperation(rawValue: 0)
  109. }
  110. return .generic
  111. }
  112. self.listItemView_?.acceptDropCallback = { [weak self] info, row, dropOperation in
  113. let pborad = info.draggingPasteboard
  114. guard let _ = pborad.availableType(from: [.fileURL]) else {
  115. return false
  116. }
  117. guard let items = pborad.pasteboardItems else {
  118. return false
  119. }
  120. for item in items {
  121. let string = item.propertyList(forType: .fileURL) as? String ?? ""
  122. if let path = URL(string: string)?.path {
  123. let contains = self?._fileFormatIsContains(URL(string: string)!.pathExtension) ?? false
  124. if contains {
  125. self?.filePaths_.append(path)
  126. }
  127. }
  128. }
  129. self?._updateListData()
  130. return true
  131. }
  132. }
  133. override func showWindow(_ sender: Any?) {
  134. super.showWindow(sender)
  135. self.typeItemView_?.comBoBox.stringValue = self.typeString
  136. }
  137. // MARK: - Private Methods
  138. @objc private func _cancelButtonAction() {
  139. self.window?.orderOut(nil)
  140. }
  141. @objc private func _comfirmButtonAction() {
  142. if let data = self.listHeaderItemView_?.checkButton.state, data == .on {
  143. let fileaccess = AppSandboxFileAccess()
  144. // /Users/kdanmobile/Library/Logs/DiagnosticReports
  145. let url = FileManager.default.urls(for: .libraryDirectory, in: .userDomainMask).first
  146. let coms = url?.pathComponents ?? []
  147. var filePath: String = ""
  148. for (i, com) in coms.enumerated() {
  149. if i > 2 {
  150. break
  151. } else {
  152. filePath.append("/\(com)")
  153. }
  154. }
  155. let cmd = filePath.appending("/Library/Logs/DiagnosticReports/")
  156. // /Library/Logs/DiagnosticReports/
  157. // fileaccess?.accessFilePath(cmd, persistPermission: true, with: {
  158. //
  159. // })
  160. let logString = self._getLog()
  161. if logString.isEmpty == false {
  162. let bundleIdentifier = Bundle.main.bundleIdentifier ?? "PDF Reader Pro"
  163. if let url = FileManager.default.urls(for: .applicationSupportDirectory, in: .userDomainMask).first?.appendingPathComponent(bundleIdentifier) {
  164. if FileManager.default.fileExists(atPath: url.path) == false {
  165. try?FileManager.default.createDirectory(at: url, withIntermediateDirectories: false)
  166. }
  167. let fileUrl = url.appendingPathComponent("Log.txt")
  168. try?logString.write(to: fileUrl, atomically: true, encoding: .utf8)
  169. self.logPath_ = fileUrl.path
  170. }
  171. }
  172. } else {
  173. self.logPath_ = nil
  174. }
  175. self.window?.makeFirstResponder(nil)
  176. let state = self._isConnectionAvailable()
  177. if !state {
  178. self._showHud(msg: NSLocalizedString("Please make sure your internet connection is available.", comment: ""))
  179. return
  180. }
  181. guard let emailString = self.emailItemView_?.string(), emailString.isEmpty == false else {
  182. // 邮箱为空
  183. self.emailItemView_?.showTip(label: NSLocalizedString("请填写Email", comment: ""))
  184. return
  185. }
  186. if KMValidateEmail(email: emailString) == false {
  187. self.emailItemView_?.showTip(label: NSLocalizedString("Please enter valid email", comment: ""))
  188. return
  189. }
  190. guard let despString = self.despItemView_?.string(), despString.isEmpty == false else {
  191. // 描述信息为空
  192. self.despItemView_?.showTip()
  193. return
  194. }
  195. self.window?.contentView?.beginLoading()
  196. self.comfirmButton.isEnabled = false
  197. var filePaths: [String] = self.filePaths_
  198. if let data = self.listHeaderItemView_?.checkButton.state, data == .on {
  199. if let filePath = self.logPath_, filePath.isEmpty == false {
  200. filePaths.append(filePath)
  201. }
  202. }
  203. self._feekbackAction(filePaths: filePaths) { [weak self] wrapper in
  204. self?.window?.contentView?.endLoading()
  205. self?.comfirmButton.isEnabled = true
  206. if wrapper.success {
  207. self?._showHud(msg: NSLocalizedString("感谢您的反馈!\n客服将在1个工作日内反馈至您的邮箱", comment: ""))
  208. DispatchQueue.main.asyncAfter(deadline: .now() + 0.6) {
  209. self?.despItemView_?.textView.string = ""
  210. self?.despItemView_?.textView.placeholderLabel.isHidden = false;
  211. self?.filePaths_.removeAll()
  212. self?._updateListData()
  213. if let data = self?.logPath_, data.isEmpty == false {
  214. if FileManager.default.fileExists(atPath: data) {
  215. try?FileManager.default.removeItem(atPath: data)
  216. }
  217. }
  218. self?.logPath_ = nil
  219. self?.window?.orderOut(nil)
  220. }
  221. } else {
  222. self?._showHud(msg: wrapper.content)
  223. }
  224. }
  225. }
  226. private func _showHelpPopover(_ sender: NSButton) {
  227. if self.popover_ != nil {
  228. return
  229. }
  230. let vc = KMUserFbHelpPopController()
  231. vc.formatSting = self.fileFormatsString
  232. let popover = NSPopover()
  233. popover.contentViewController = vc
  234. popover.animates = true
  235. popover.behavior = .semitransient
  236. popover.delegate = self
  237. popover.setValue(true, forKey: "shouldHideAnchor")
  238. self.popover_ = popover
  239. popover.show(relativeTo: sender.bounds, of: sender, preferredEdge: .maxX)
  240. }
  241. private func _hiddenHelpPopover() {
  242. self.popover_?.close()
  243. }
  244. private func _updateListData() {
  245. self.listItemView_?.hiddenTip()
  246. var datas: [KMUserFbListModel] = []
  247. let maxSize: Double = 20 * 1024 * 1024
  248. var fileSize: Double = 0
  249. var filePaths: [String] = []
  250. var showFileSizeLimit = false
  251. var showFileCountLimit = false
  252. for (i, fileP) in self.filePaths_.enumerated() {
  253. let model = KMUserFbListModel()
  254. model.filePath = fileP
  255. let url = URL(fileURLWithPath: fileP)
  256. model.fileName = url.lastPathComponent
  257. let attri = try?FileManager.default.attributesOfItem(atPath: fileP)
  258. model.fileSize = attri?[FileAttributeKey.size] as? Double ?? 0
  259. // fileSize += model.fileSize
  260. fileSize = model.fileSize
  261. if fileSize >= maxSize {
  262. showFileSizeLimit = true
  263. continue
  264. }
  265. if i >= 10 {
  266. showFileCountLimit = true
  267. break
  268. }
  269. model.fileSizeString = self.fileSizeString(model.fileSize) ?? ""
  270. datas.append(model)
  271. filePaths.append(fileP)
  272. }
  273. self.filePaths_ = filePaths
  274. if showFileCountLimit {
  275. self.listItemView_?.showTip(tip: NSLocalizedString("添加失败:文件不能超过10", comment: ""))
  276. } else if showFileSizeLimit {
  277. self.listItemView_?.showTip(tip: NSLocalizedString("添加失败:附件大小不能超过20M", comment: ""))
  278. }
  279. if datas.count >= 10 {
  280. self.listHeaderItemView_?.addButton.isEnabled = false
  281. } else {
  282. self.listHeaderItemView_?.addButton.isEnabled = true
  283. }
  284. self.listItemView_?.datas = datas
  285. }
  286. private func _showHud(msg: String) {
  287. if let data = self.window?.contentView {
  288. // _ = CustomAlertView(message: msg, from: data, with: .black)
  289. CustomAlertView.alertView(message: msg, fromView: data, withStyle: .black)
  290. }
  291. }
  292. private func _getLog() -> String {
  293. let url = FileManager.default.urls(for: .libraryDirectory, in: .userDomainMask).first
  294. let coms = url?.pathComponents ?? []
  295. var filePath: String = ""
  296. for (i, com) in coms.enumerated() {
  297. if i > 2 {
  298. break
  299. } else {
  300. filePath.append("/\(com)")
  301. }
  302. }
  303. // ls /Library/Logs/DiagnosticReports/PDF*
  304. let cmd = "ls \(filePath)/Library/Logs/DiagnosticReports/PDF*"
  305. let filestrings = _runShellCommand(cmd)
  306. let files = filestrings.components(separatedBy: "\n")
  307. var results: [String] = []
  308. for string in files {
  309. if string.contains("PDF Reader Pro") {
  310. var tempS = string
  311. // tempS.replace(" ", with: "\\ ")
  312. // tempS.replacingCharacters(in: " ", with: "\\ ")
  313. // let ss = tempS.replacing(" ", with: "\\ ")
  314. let ss = tempS.replacingOccurrences(of: " ", with: "\\ ")
  315. results.append(ss)
  316. }
  317. }
  318. if let file = results.last {
  319. let cmd = "cat \(file)"
  320. // cat /Library/Logs/DiagnosticReports/PDF*
  321. let logs = _runShellCommand(cmd)
  322. return logs
  323. }
  324. return ""
  325. }
  326. private func _runShellCommand(_ command: String) -> String {
  327. let task = Process()
  328. let pipe = Pipe()
  329. task.standardOutput = pipe
  330. task.standardError = pipe
  331. task.arguments = ["-c", command]
  332. task.launchPath = "/bin/bash"
  333. do {
  334. try task.run()
  335. } catch {
  336. return "Error running command: \(error)"
  337. }
  338. let data = pipe.fileHandleForReading.readDataToEndOfFile()
  339. let output = String(data: data, encoding: .utf8) ?? ""
  340. return output
  341. }
  342. func test() {
  343. var convertString = "TARGET_BUNDLE_IDENTIFIER=\"com.brother.pdfreaderprofree.mac\"\n"
  344. convertString = convertString.appendingFormat("SANDBOX_ROOT=$(osascript -e 'tell application \"System Events\" to get the path of the home folder of (system info)')\n")
  345. convertString = convertString.appendingFormat("CONTAINER_LOG_PATH=$(osascript <<EOF\n")
  346. convertString.append("tell application \"System Events\"\n")
  347. // convertString.append("set isRun to running\n")
  348. convertString.append("set container_path to POSIX path of (the home folder as text) & \"Library:Containers:\" & \"$TARGET_BUNDLE_IDENTIFIER\"\n")
  349. convertString.append("set log_path to container_path & \":Logs:\" & \"Console.app-History\"\n")
  350. convertString.append("return log_path\n")
  351. convertString.append("end tell\n")
  352. convertString.append("EOF)\n")
  353. convertString.append("echo \"日志文件路径: $CONTAINER_LOG_PATH\"\n")
  354. var dic: NSDictionary?
  355. let docScript = NSAppleScript(source: convertString)
  356. docScript?.executeAndReturnError(&dic)
  357. }
  358. func fileSizeString(_ fSize: Double) -> String {
  359. let fileSize = fSize / 1024
  360. let size = fileSize >= 1024 ? (fileSize < 1048576 ? fileSize/1024 : fileSize/1048576.0) : fileSize
  361. let unit = fileSize >= 1024 ? (fileSize < 1048576 ? "M" : "G") : "K"
  362. return String(format: "%0.1f %@", size, unit)
  363. }
  364. private func _feekbackAction(filePaths: [String], complete: @escaping KMRequestServerComplete) {
  365. let urlString = kVerificationServer + "/api/feedback/feedback"
  366. var fileDatas: [Data] = []
  367. for filePath in filePaths {
  368. if let fileData = FileManager.default.contents(atPath: filePath) {
  369. fileDatas.append(fileData)
  370. }
  371. }
  372. let uuid = GetHardwareUUID()
  373. let params: [String: Any] = ["email": self.emailItemView_?.string() ?? "",
  374. "title": self.typeItemView_?.string() ?? "",
  375. "content": self.despItemView_?.string() ?? "",
  376. "unique_sn" : uuid!,
  377. "files[]" : fileDatas]
  378. KMAIRequestServer.requestServer.uploadFile(urlString: urlString, params: params) { formData in
  379. for i in 0 ..< fileDatas.count {
  380. let path = filePaths[i]
  381. let fileURL = URL(fileURLWithPath: path)
  382. try? formData.appendPart(withFileURL: fileURL, name: "files[]", fileName: fileURL.lastPathComponent, mimeType: "application/octet-stream")
  383. }
  384. } requestSerializer: { requestSerializer in
  385. requestSerializer.setValue("Apifox/1.0.0 (https://www.apifox.cn)", forHTTPHeaderField: "User-Agent")
  386. } completion: { task, responseObject, error in
  387. if responseObject != nil {
  388. let data: NSDictionary = responseObject as? NSDictionary ?? [:]
  389. // var code: String = responseObject!["code"] as? String ?? "06005"
  390. let code = data["code"] as? Int ?? 06005
  391. if code == 06005 {
  392. // let tempCode: Int = responseObject!["code"] as? Int ?? 0
  393. // if tempCode == 501 {
  394. // code = "501"
  395. // }
  396. }
  397. let message: String = data["message"] as? String ?? "unknown error"
  398. let error = NSError(domain: message, code: code)
  399. if code == 200 {
  400. let wrapper = ResultWrapper(success: true, resultData: data)
  401. // wrapper.content = data["fileKey"] as! String
  402. complete(wrapper)
  403. } else {
  404. let wrapper = ResultWrapper(success: false, resultData: data)
  405. if code == 501 {
  406. wrapper.content = "501"
  407. } else {
  408. wrapper.content = message
  409. }
  410. complete(wrapper)
  411. }
  412. } else {
  413. complete(ResultWrapper(success: false, content: "unknown error"))
  414. }
  415. }
  416. }
  417. private func _isConnectionAvailable() -> Bool {
  418. if Reachability.forInternetConnection().currentReachabilityStatus().rawValue == 0 {
  419. return false
  420. }
  421. return true
  422. }
  423. private func _fileFormatIsContains(_ exn: String) -> Bool {
  424. let _exn = exn.lowercased()
  425. return self.fileFormats_.contains(_exn)
  426. }
  427. }
  428. extension KMUserFeekbackWindowController: NSPopoverDelegate {
  429. func popoverWillClose(_ notification: Notification) {
  430. if let data = self.popover_?.isEqual(to: notification.object), data {
  431. self.popover_ = nil
  432. }
  433. }
  434. }