AINewConfigWindowController.swift 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762
  1. //
  2. // AINewConfigWindowController.swift
  3. // PDF Reader Pro Edition
  4. //
  5. // Created by Niehaoyu on 2024/4/16.
  6. //
  7. import Cocoa
  8. protocol AIConfigWindowDelegate: AnyObject {
  9. func ai_InputViewDidChooseCurFile(aiConfigWindow: AINewConfigWindowController)
  10. }
  11. @objcMembers class AINewConfigWindowController: NSWindowController, NSWindowDelegate, AIInfoInputViewDelegate {
  12. static var currentWindowController: AINewConfigWindowController!
  13. @IBOutlet weak var contendBox: NSBox!
  14. var aiHeaderView: AIHeaderView!
  15. var aiChatView: AIChatView!
  16. var aiTypeItemView: AITypeItemChooseView!
  17. var aiInfoInputView: AIInfoInputView!
  18. var inputStringHeight: CGFloat = 40
  19. var didSetOriginFrame: Bool = false
  20. var eventLabel: String = "AITools_Start"
  21. weak var aiDelegate: AIConfigWindowDelegate?
  22. var chooseCurFileHandle: ((_ windowVC: AINewConfigWindowController) -> Void)?
  23. deinit {
  24. DistributedNotificationCenter.default.removeObserver(self)
  25. }
  26. @objc static func currentWC() -> AINewConfigWindowController {
  27. if currentWindowController != nil {
  28. return currentWindowController
  29. } else {
  30. let configWC: AINewConfigWindowController = AINewConfigWindowController.init(windowNibName: "AINewConfigWindowController")
  31. currentWindowController = configWC;
  32. return currentWindowController
  33. }
  34. }
  35. override func showWindow(_ sender: Any?) {
  36. super.showWindow(sender)
  37. self.window?.delegate = self
  38. self.aiInfoInputView.setUpTranslateUI()
  39. self.aiChatView.reloadData()
  40. self.aiTypeItemView.refreshUI()
  41. self.refreshViewUI()
  42. }
  43. override func windowDidLoad() {
  44. super.windowDidLoad()
  45. // Implement this method to handle any initialization after your window controller's window has been loaded from its nib file.
  46. self.window?.title = NSLocalizedString("AI Tools", comment: "")
  47. self.loadAIHeaderView()
  48. self.loadAIChatView()
  49. self.loadAIInputView()
  50. self.loadAITypeItemView()
  51. self.refreshViewUI()
  52. self.refreshViewColor()
  53. DistributedNotificationCenter.default().addObserver(self, selector: #selector(themeChange), name: NSNotification.Name(rawValue: "AppleInterfaceThemeChangedNotification"), object: nil)
  54. }
  55. func refreshViewColor() {
  56. if KMAppearance.isDarkMode() {
  57. self.contendBox.fillColor = NSColor(red: 33/255, green: 33/255, blue: 33/255, alpha: 1)
  58. } else {
  59. self.contendBox.fillColor = NSColor.white
  60. }
  61. self.aiHeaderView.refreshViewColor()
  62. self.aiInfoInputView.refreshUI()
  63. self.aiChatView.reloadData()
  64. self.aiTypeItemView.refreshViewColor()
  65. }
  66. func loadAIHeaderView() {
  67. self.aiHeaderView = AIHeaderView.createFromNib()
  68. self.aiHeaderView.frame = NSMakeRect(0, NSHeight(self.contendBox.frame)-58, NSWidth(self.contendBox.frame), 28)
  69. self.aiHeaderView.autoresizingMask = [.minXMargin, .maxXMargin, .width, .minYMargin]
  70. self.contendBox.addSubview(self.aiHeaderView)
  71. }
  72. func loadAIChatView() {
  73. self.aiChatView = AIChatView.createFromNib()
  74. self.aiChatView.wantsLayer = true
  75. self.aiChatView.layer?.backgroundColor = NSColor.clear.cgColor
  76. self.aiChatView.frame = NSMakeRect(0, 300, NSWidth(self.contendBox.frame), NSHeight(self.contendBox.frame)-400)
  77. self.aiChatView.autoresizingMask = [.width, .height]
  78. self.contendBox.addSubview(self.aiChatView)
  79. self.aiChatView.chooseConfigHandle = {[weak self] view, configType in
  80. DispatchQueue.main.async {
  81. self?.chooseAIFunctionWithType(configType)
  82. }
  83. }
  84. self.aiChatView.cancelAIHandle = {[weak self] view, chatInfoModel in
  85. DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 0.05) {
  86. chatInfoModel.chatInfoState = .stateCancel
  87. self?.aiChatView.reloadData()
  88. }
  89. }
  90. self.aiChatView.continueAITranslateHandle = {[weak self] view, chatInfoModel in
  91. DispatchQueue.main.async {
  92. chatInfoModel.chatInfoState = .stateLoading
  93. self?.aiChatView.reloadData()
  94. self?.continueAiTranslate(chatInfoModel)
  95. }
  96. }
  97. self.aiChatView.redoHandle = {[weak self] view, chatInfoModel in
  98. DispatchQueue.main.async {
  99. let newChatModel = AIChatInfoModel.init()
  100. newChatModel.aiConfigType = chatInfoModel.aiConfigType
  101. newChatModel.infoType = chatInfoModel.infoType
  102. newChatModel.filePath = chatInfoModel.filePath
  103. newChatModel.chatInfoState = .stateLoading
  104. newChatModel.uploadContent = chatInfoModel.uploadContent
  105. newChatModel.translateFromLanguage = chatInfoModel.translateFromLanguage
  106. newChatModel.translateToLanguage = chatInfoModel.translateToLanguage
  107. AIChatInfoManager.defaultManager.modelsArrM.append(newChatModel)
  108. self?.aiChatView.reloadData()
  109. if newChatModel.aiConfigType == .summarize {
  110. self?.aiSummarizeWithModel(newChatModel)
  111. } else if newChatModel.aiConfigType == .reWriting {
  112. self?.aiReWritingWithModel(newChatModel)
  113. } else if newChatModel.aiConfigType == .proofreading {
  114. self?.aiProofreadingWithModel(newChatModel)
  115. } else if newChatModel.aiConfigType == .translate {
  116. self?.startAiTranslateWithModel(newChatModel)
  117. } else if newChatModel.aiConfigType == .ponder {
  118. self?.aiPonderWithModel(newChatModel)
  119. }
  120. }
  121. }
  122. }
  123. func loadAIInputView() {
  124. self.aiInfoInputView = AIInfoInputView.createFromNib()
  125. self.aiInfoInputView.frame = NSMakeRect((NSWidth(self.contendBox.frame)-248)/2.0, 20, 248, 86)
  126. self.aiInfoInputView.autoresizingMask = [.minXMargin, .maxXMargin, .width, .maxYMargin]
  127. self.contendBox.addSubview(self.aiInfoInputView)
  128. self.aiInfoInputView.aiConfigType = .none
  129. self.aiInfoInputView.aiDelegate = self
  130. self.aiInfoInputView.reloadData()
  131. self.aiInfoInputView.startAIHandle = {[unowned self] view in
  132. if view.aiConfigType == .summarize {
  133. let chatModel = AIChatInfoModel.init()
  134. chatModel.aiConfigType = .summarize
  135. chatModel.infoType = .chatFileUpload
  136. chatModel.filePath = self.aiInfoInputView.filePath
  137. AIChatInfoManager.defaultManager.modelsArrM.append(chatModel)
  138. self.aiChatView.reloadData()
  139. self.aiSummarize()
  140. } else if view.aiConfigType == .reWriting {
  141. let chatModel = AIChatInfoModel.init()
  142. chatModel.aiConfigType = .reWriting
  143. chatModel.infoType = .chatStringUpload
  144. chatModel.uploadContent = self.aiInfoInputView.fileEmptyTextView.string
  145. AIChatInfoManager.defaultManager.modelsArrM.append(chatModel)
  146. self.aiChatView.reloadData()
  147. self.aiReWriting()
  148. } else if view.aiConfigType == .proofreading {
  149. let chatModel = AIChatInfoModel.init()
  150. chatModel.aiConfigType = .proofreading
  151. chatModel.infoType = .chatStringUpload
  152. chatModel.uploadContent = self.aiInfoInputView.fileEmptyTextView.string
  153. AIChatInfoManager.defaultManager.modelsArrM.append(chatModel)
  154. self.aiChatView.reloadData()
  155. self.aiProofreading()
  156. } else if view.aiConfigType == .translate {
  157. let chatModel = AIChatInfoModel.init()
  158. chatModel.aiConfigType = .translate
  159. if self.aiInfoInputView.filePath.isEmpty == false {
  160. //文件
  161. chatModel.filePath = self.aiInfoInputView.filePath
  162. chatModel.infoType = .chatFileUpload
  163. } else {
  164. //文字
  165. chatModel.uploadContent = self.aiInfoInputView.fileEmptyTextView.string
  166. chatModel.infoType = .chatStringUpload
  167. }
  168. chatModel.translateFromLanguage = self.aiInfoInputView.fromLanguage
  169. chatModel.translateToLanguage = self.aiInfoInputView.toLanguage
  170. AIChatInfoManager.defaultManager.modelsArrM.append(chatModel)
  171. self.aiChatView.reloadData()
  172. self.startAiTranslate()
  173. } else if view.aiConfigType == .ponder {
  174. let chatModel = AIChatInfoModel.init()
  175. chatModel.aiConfigType = .ponder
  176. chatModel.infoType = .chatStringUpload
  177. chatModel.uploadContent = self.aiInfoInputView.fileEmptyTextView.string
  178. AIChatInfoManager.defaultManager.modelsArrM.append(chatModel)
  179. self.aiChatView.reloadData()
  180. self.ponderReWriting()
  181. }
  182. self.refreshViewUI()
  183. self.aiTypeItemView.reloadData()
  184. }
  185. self.aiInfoInputView.inputFrameUpdateHandle = {[weak self] view, stringSize in
  186. if view.aiConfigType == .summarize {
  187. self?.inputStringHeight = max(stringSize.height, 40)
  188. } else if view.aiConfigType == .reWriting {
  189. self?.inputStringHeight = max(stringSize.height, 40)
  190. } else if view.aiConfigType == .proofreading {
  191. self?.inputStringHeight = max(stringSize.height, 40)
  192. } else if view.aiConfigType == .translate {
  193. if view.filePath.isEmpty {
  194. self?.inputStringHeight = max(stringSize.height, 40)
  195. } else {
  196. self?.inputStringHeight = 40
  197. }
  198. } else if view.aiConfigType == .ponder {
  199. self?.inputStringHeight = max(stringSize.height, 40)
  200. }
  201. self?.refreshViewUI()
  202. }
  203. self.aiInfoInputView.refreshUI()
  204. }
  205. func loadAITypeItemView() {
  206. self.aiTypeItemView = AITypeItemChooseView.createFromNib()
  207. self.aiTypeItemView.wantsLayer = true
  208. self.aiTypeItemView.frameCallback = {[weak self] in
  209. self?.refreshViewUI()
  210. }
  211. self.aiTypeItemView.layer?.backgroundColor = NSColor.clear.cgColor
  212. self.aiTypeItemView.frame = NSMakeRect((NSWidth(self.contendBox.frame)-246)/2, NSMaxY(self.aiInfoInputView.frame)+13, 246, 52)
  213. self.aiTypeItemView.autoresizingMask = [.minXMargin, .maxXMargin, .width, .maxYMargin]
  214. self.contendBox.addSubview(self.aiTypeItemView)
  215. self.aiTypeItemView.chooseTypeHandle = {[weak self] itemView, aiConfigtype in
  216. self?.chooseAIFunctionWithType(aiConfigtype)
  217. }
  218. self.aiTypeItemView.clearHandle = {[weak self] itemView in
  219. if AIChatInfoManager.defaultManager.modelsArrM.count < 1 {
  220. return
  221. }
  222. let alert = NSAlert()
  223. alert.alertStyle = .critical
  224. alert.informativeText = NSLocalizedString("All the AI content will be removed. Are you sure you want to clear the session?", comment: "")
  225. alert.messageText = NSLocalizedString("Clear session", comment: "")
  226. alert.addButton(withTitle: NSLocalizedString("Clear", comment: ""))
  227. alert.addButton(withTitle: NSLocalizedString("Cancel", comment: ""))
  228. let response = alert.runModal()
  229. if response.rawValue == 1000 {
  230. DispatchQueue.main.async {
  231. AIChatInfoManager.defaultManager.clearData()
  232. self?.aiChatView.reloadData()
  233. }
  234. }
  235. }
  236. }
  237. func chooseAIFunctionWithType(_ aiConfigType: AIConfigType) -> Void {
  238. if KMMemberInfo.shared.aiNewSubscription(aiConfigType) == false {
  239. return
  240. }
  241. if KMMemberInfo.shared.aiPoint == 0 &&
  242. (AIInfoManager.default().aiInfo.totalToken - AIInfoManager.default().aiInfo.usedTimes) < 1 {
  243. let alert = NSAlert()
  244. alert.alertStyle = .critical
  245. alert.informativeText = NSLocalizedString("No Value Present", comment: "")
  246. alert.addButton(withTitle: NSLocalizedString("OK", comment: ""))
  247. alert.runModal()
  248. return
  249. }
  250. if self.eventLabel.isEmpty {
  251. self.eventLabel = "AITools_Start"
  252. }
  253. if aiConfigType == .summarize {
  254. FMTrackEventManager.defaultManager.trackOnceEvent(event: "AITools", withProperties: [self.eventLabel:"AISum"])
  255. } else if aiConfigType == .reWriting {
  256. FMTrackEventManager.defaultManager.trackOnceEvent(event: "AITools", withProperties: [self.eventLabel:"AIRewrite"])
  257. } else if aiConfigType == .proofreading {
  258. FMTrackEventManager.defaultManager.trackOnceEvent(event: "AITools", withProperties: [self.eventLabel:"AIProofread"])
  259. } else if aiConfigType == .translate {
  260. FMTrackEventManager.defaultManager.trackOnceEvent(event: "AITools", withProperties: [self.eventLabel:"AITranslate"])
  261. } else if aiConfigType == .ponder {
  262. FMTrackEventManager.defaultManager.trackOnceEvent(event: "AITools", withProperties: [self.eventLabel:"AIDeepThink_Click"])
  263. }
  264. self.aiInfoInputView.aiConfigType = aiConfigType
  265. self.aiInfoInputView.reloadData()
  266. self.aiInfoInputView.aiFunctionTypeChanged()
  267. if self.aiInfoInputView.fileEmptyTextView.string.isEmpty == false {
  268. self.aiInfoInputView.updateCountLabelInfo()
  269. self.aiInfoInputView.refreshStringSize()
  270. }
  271. self.refreshViewUI()
  272. }
  273. func setCurrentPDFSelection(_ string: String) -> Void {
  274. self.aiInfoInputView.fileEmptyTextView.string = string
  275. }
  276. func refreshViewUI() {
  277. self.aiHeaderView.frame = NSMakeRect(0, NSHeight(self.contendBox.frame)-58, NSWidth(self.contendBox.frame), 28)
  278. if self.aiInfoInputView.aiConfigType == .none {
  279. self.aiInfoInputView.frame = NSMakeRect(13, 17, 240, 86)
  280. } else if self.aiInfoInputView.aiConfigType == .summarize {
  281. self.aiInfoInputView.frame = NSMakeRect(13, 17, 240, min(96+self.inputStringHeight, 500))
  282. } else if self.aiInfoInputView.aiConfigType == .reWriting {
  283. self.aiInfoInputView.frame = NSMakeRect(13, 17, 240, min(72+self.inputStringHeight, 500))
  284. } else if self.aiInfoInputView.aiConfigType == .proofreading {
  285. self.aiInfoInputView.frame = NSMakeRect(13, 17, 240, min(72+self.inputStringHeight, 500))
  286. } else if self.aiInfoInputView.aiConfigType == .translate {
  287. self.aiInfoInputView.frame = NSMakeRect(13, 17, 240, min(125+self.inputStringHeight, 500))
  288. } else if self.aiInfoInputView.aiConfigType == .ponder {
  289. self.aiInfoInputView.frame = NSMakeRect(13, 17, 240, min(72+self.inputStringHeight, 500))
  290. }
  291. self.aiTypeItemView.frame = NSMakeRect(13, NSMaxY(self.aiInfoInputView.frame)+8, 240, self.aiTypeItemView.viewHeight)
  292. self.aiChatView.frame = NSMakeRect(0, NSMaxY(self.aiTypeItemView.frame)+8, NSWidth(self.contendBox.frame), NSHeight(self.contendBox.frame) - NSMaxY(self.aiTypeItemView.frame) - 16 - 58)
  293. }
  294. //MARK: AI-Action
  295. func aiSummarize() -> Void {
  296. if FileManager.default.fileExists(atPath: self.aiInfoInputView.filePath) {
  297. let chatModel = AIChatInfoModel.init()
  298. chatModel.aiConfigType = .summarize
  299. chatModel.infoType = .chatStringResult
  300. chatModel.filePath = self.aiInfoInputView.filePath
  301. chatModel.chatInfoState = .stateLoading
  302. AIChatInfoManager.defaultManager.modelsArrM.append(chatModel)
  303. self.aiChatView.reloadData()
  304. self.aiSummarizeWithModel(chatModel)
  305. }
  306. }
  307. func aiSummarizeWithModel(_ chatModel: AIChatInfoModel) -> Void {
  308. if FileManager.default.fileExists(atPath: chatModel.filePath) {
  309. AIChatInfoManager.defaultManager.isAILoading = true
  310. //MARK: TestData
  311. // DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 3) {
  312. // DispatchQueue.main.async {
  313. // AIChatInfoManager.defaultManager.isAILoading = false
  314. // chatModel.chatResult = "AI Summary 返回的结果\nAI Summary 返回的结果\nAI Summary 返回的结果\nAI Summary 返回的结果\nAI Summary 返回的结果\nAI Summary 返回的结果\nAI Summary 返回的结果\nAI Summary 返回的结果"
  315. // chatModel.chatInfoState = .stateSuccess
  316. // self.aiChatView.reloadData()
  317. // self.aiTypeItemView.reloadData()
  318. // self.aiInfoInputView.reloadData()
  319. //
  320. // DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 4) {
  321. // DispatchQueue.main.async {
  322. // chatModel.chatInfoState = .stateFailed
  323. // chatModel.chatResult = "Unknown error"
  324. //
  325. // self.aiChatView.reloadData()
  326. // self.aiTypeItemView.reloadData()
  327. // self.aiInfoInputView.reloadData()
  328. // }
  329. // }
  330. // }
  331. // }
  332. //MARK: 正式数据
  333. KMAIRequestServerManager.defaultManager.aiAction(content: chatModel.filePath, state: .extractSummaryFile) { wrapper in
  334. DispatchQueue.main.async {
  335. AIChatInfoManager.defaultManager.isAILoading = false
  336. let resultStr = wrapper.content
  337. chatModel.chatResult = resultStr
  338. chatModel.chatInfoState = wrapper.success ? .stateSuccess : .stateFailed
  339. self.aiChatView.reloadData()
  340. self.aiTypeItemView.reloadData()
  341. self.aiInfoInputView.reloadData()
  342. }
  343. }
  344. } else {
  345. DispatchQueue.main.async {
  346. AIChatInfoManager.defaultManager.isAILoading = false
  347. let resultStr = NSLocalizedString("File Not Exist", comment: "")
  348. chatModel.chatResult = resultStr
  349. chatModel.chatInfoState = .stateFailed
  350. self.aiChatView.reloadData()
  351. self.aiTypeItemView.reloadData()
  352. self.aiInfoInputView.reloadData()
  353. }
  354. }
  355. }
  356. func aiReWriting() -> Void {
  357. //Loading(28+155)->Finish(实际大小)
  358. let chatModel = AIChatInfoModel.init()
  359. chatModel.aiConfigType = .reWriting
  360. chatModel.infoType = .chatStringResult
  361. chatModel.uploadContent = self.aiInfoInputView.fileEmptyTextView.string
  362. chatModel.chatInfoState = .stateLoading
  363. AIChatInfoManager.defaultManager.modelsArrM.append(chatModel)
  364. self.aiChatView.reloadData()
  365. self.aiReWritingWithModel(chatModel)
  366. }
  367. func ponderReWriting() -> Void {
  368. //Loading(28+155)->Finish(实际大小)
  369. let chatModel = AIChatInfoModel.init()
  370. chatModel.aiConfigType = .ponder
  371. chatModel.infoType = .chatStringResult
  372. chatModel.uploadContent = self.aiInfoInputView.fileEmptyTextView.string
  373. chatModel.chatInfoState = .stateLoading
  374. AIChatInfoManager.defaultManager.modelsArrM.append(chatModel)
  375. self.aiChatView.reloadData()
  376. self.aiPonderWithModel(chatModel)
  377. FMTrackEventManager.defaultManager.trackOnceEvent(event: "AITools", withProperties: ["AITools_Start":"AIDeepThink_Start"])
  378. }
  379. func aiReWritingWithModel(_ chatModel: AIChatInfoModel) -> Void {
  380. AIChatInfoManager.defaultManager.isAILoading = true
  381. //MARK: TestData
  382. // DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 3) {
  383. // DispatchQueue.main.async {
  384. // AIChatInfoManager.defaultManager.isAILoading = false
  385. // chatModel.chatInfoState = .stateSuccess
  386. // chatModel.chatResult = "AI Rewrite返回的结果AI Rewrite返回的结果AI Rewrite返回的结果AI Rewrite返回的结果AI Rewrite返回的结果AI Rewrite返回的结果AI Rewrite返回的结果AI Rewrite返回的结果AI Rewrite返回的结果AI Rewrite返回的结果AI Rewrite返回的结果AI Rewrite返回的结果AI Rewrite返回的结果"
  387. // self.aiChatView.reloadData()
  388. // self.aiTypeItemView.reloadData()
  389. // self.aiInfoInputView.reloadData()
  390. // }
  391. // }
  392. //MARK: 正式数据
  393. KMAIRequestServerManager.defaultManager.aiAction(content: chatModel.uploadContent,
  394. state: .rewrite) { wrapper in
  395. DispatchQueue.main.async {
  396. AIChatInfoManager.defaultManager.isAILoading = false
  397. let resultStr = wrapper.content
  398. chatModel.chatResult = resultStr
  399. chatModel.chatInfoState = wrapper.success ? .stateSuccess : .stateFailed
  400. self.aiChatView.reloadData()
  401. self.aiTypeItemView.reloadData()
  402. self.aiInfoInputView.reloadData()
  403. }
  404. }
  405. }
  406. func aiPonderWithModel(_ chatModel: AIChatInfoModel) -> Void {
  407. AIChatInfoManager.defaultManager.isAILoading = true
  408. //MARK: 正式数据
  409. KMAIRequestServerManager.defaultManager.aiAction(content: chatModel.uploadContent,
  410. state: .aiPonder) { wrapper in
  411. DispatchQueue.main.async {
  412. AIChatInfoManager.defaultManager.isAILoading = false
  413. let resultStr = wrapper.content
  414. chatModel.chatResult = resultStr
  415. chatModel.chatInfoState = wrapper.success ? .stateSuccess : .stateFailed
  416. self.aiChatView.reloadData()
  417. self.aiTypeItemView.reloadData()
  418. self.aiInfoInputView.reloadData()
  419. }
  420. }
  421. }
  422. func aiProofreading() -> Void {
  423. let chatModel = AIChatInfoModel.init()
  424. chatModel.aiConfigType = .proofreading
  425. chatModel.infoType = .chatStringResult
  426. chatModel.uploadContent = self.aiInfoInputView.fileEmptyTextView.string
  427. chatModel.chatInfoState = .stateLoading
  428. AIChatInfoManager.defaultManager.modelsArrM.append(chatModel)
  429. self.aiChatView.reloadData()
  430. self.aiProofreadingWithModel(chatModel)
  431. }
  432. func aiProofreadingWithModel(_ chatModel: AIChatInfoModel) -> Void {
  433. AIChatInfoManager.defaultManager.isAILoading = true
  434. //MARK: TestData
  435. // DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 3) {
  436. // DispatchQueue.main.async {
  437. // chatModel.chatInfoState = .stateSuccess
  438. // AIChatInfoManager.defaultManager.isAILoading = false
  439. // chatModel.chatResult = "AI Proofreading 返回的结果"
  440. // self.aiChatView.reloadData()
  441. // self.aiTypeItemView.reloadData()
  442. // self.aiInfoInputView.reloadData()
  443. //
  444. // }
  445. // }
  446. //MARK: 正式数据
  447. KMAIRequestServerManager.defaultManager.aiAction(content: chatModel.uploadContent,
  448. state: .correctTypos) { wrapper in
  449. DispatchQueue.main.async {
  450. AIChatInfoManager.defaultManager.isAILoading = false
  451. let resultStr = wrapper.content
  452. chatModel.chatResult = resultStr
  453. chatModel.chatInfoState = wrapper.success ? .stateSuccess : .stateFailed
  454. self.aiChatView.reloadData()
  455. self.aiTypeItemView.reloadData()
  456. self.aiInfoInputView.reloadData()
  457. }
  458. }
  459. }
  460. func startAiTranslate() -> Void {
  461. let chatModel = AIChatInfoModel.init()
  462. chatModel.aiConfigType = .translate
  463. if self.aiInfoInputView.filePath.isEmpty {
  464. //文字
  465. chatModel.infoType = .chatTranslateResult
  466. chatModel.uploadContent = self.aiInfoInputView.fileEmptyTextView.string
  467. chatModel.filePath = ""
  468. } else {
  469. //文件
  470. chatModel.infoType = .chatTranslateResult
  471. chatModel.filePath = self.aiInfoInputView.filePath
  472. chatModel.uploadContent = ""
  473. }
  474. chatModel.translateFromLanguage = self.aiInfoInputView.fromLanguage
  475. chatModel.translateToLanguage = self.aiInfoInputView.toLanguage
  476. chatModel.chatInfoState = .stateLoading
  477. AIChatInfoManager.defaultManager.modelsArrM.append(chatModel)
  478. self.aiChatView.reloadData()
  479. self.startAiTranslateWithModel(chatModel)
  480. }
  481. func startAiTranslateWithModel(_ chatModel: AIChatInfoModel) -> Void {
  482. //MARK: TestData
  483. // DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 3) {
  484. // DispatchQueue.main.async {
  485. // chatModel.chatInfoState = .stateInfoConfirm
  486. // chatModel.chatResult = "当前翻译需要的内容:"
  487. // self.aiChatView.reloadData()
  488. //
  489. // }
  490. // }
  491. //MARK: Data
  492. if chatModel.filePath.isEmpty == true {
  493. //文字
  494. AIChatInfoManager.defaultManager.isAILoading = true
  495. KMAIRequestServerManager.defaultManager.aiAction(content: chatModel.uploadContent,
  496. state: .textTranslate,
  497. from: chatModel.translateFromLanguage,
  498. to: chatModel.translateToLanguage) { wrapper in
  499. DispatchQueue.main.async {
  500. AIChatInfoManager.defaultManager.isAILoading = false
  501. let resultStr = wrapper.content
  502. chatModel.chatResult = resultStr
  503. chatModel.chatInfoState = wrapper.success ? .stateSuccess : .stateFailed
  504. self.aiChatView.reloadData()
  505. self.aiTypeItemView.reloadData()
  506. self.aiInfoInputView.reloadData()
  507. }
  508. };
  509. } else {
  510. //文件
  511. AIChatInfoManager.defaultManager.isAILoading = true
  512. KMAIRequestServerManager.defaultManager.aiAction(content: chatModel.filePath,
  513. state: .fileTranslate,
  514. from: chatModel.translateFromLanguage,
  515. to: chatModel.translateToLanguage) { wrapper in
  516. DispatchQueue.main.async {
  517. AIChatInfoManager.defaultManager.isAILoading = false
  518. var success = wrapper.success
  519. let resultStr = wrapper.content
  520. if success == false && resultStr == "501" {
  521. success = true
  522. }
  523. if success {
  524. let infoDict: NSDictionary = wrapper.result
  525. var credit: Int = 0
  526. if infoDict["credit"] != nil {
  527. credit = (infoDict["credit"] ?? "0") as! Int
  528. }
  529. var charCount: Int = 0
  530. if infoDict["charCount"] != nil {
  531. charCount = (infoDict["charCount"] ?? "0") as! Int
  532. if credit == 0 {
  533. if charCount < 10000 {
  534. credit = 2
  535. } else {
  536. credit = charCount/5000 + 1
  537. }
  538. }
  539. }
  540. chatModel.creditsValid = true
  541. if resultStr == "501" {
  542. chatModel.creditsValid = false
  543. }
  544. chatModel.costCredits = credit
  545. chatModel.totalChart = charCount
  546. chatModel.chatInfoState = .stateInfoConfirm
  547. chatModel.chatResult = resultStr
  548. self.aiChatView.reloadData()
  549. self.aiTypeItemView.reloadData()
  550. self.aiInfoInputView.reloadData()
  551. } else {
  552. chatModel.chatResult = resultStr
  553. chatModel.chatInfoState = .stateFailed
  554. self.aiChatView.reloadData()
  555. self.aiTypeItemView.reloadData()
  556. self.aiInfoInputView.reloadData()
  557. }
  558. }
  559. };
  560. }
  561. }
  562. func continueAiTranslate(_ chatModel: AIChatInfoModel) -> Void {
  563. //MARK: TestData
  564. // DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 3) {
  565. // DispatchQueue.main.async {
  566. // chatModel.chatInfoState = .stateSuccess
  567. // chatModel.chatResult = "/Users/kdanmobile/Desktop/Quick Start Guide v1.3.0.pdf"
  568. // self.aiChatView.reloadData()
  569. // self.aiTypeItemView.reloadData()
  570. // self.aiInfoInputView.reloadData()
  571. //
  572. // }
  573. // }
  574. KMAIRequestServerManager.defaultManager.aiTranslationFileTranslateHandle(fileKey: chatModel.chatResult,
  575. from: chatModel.translateFromLanguage,
  576. to: chatModel.translateToLanguage) { wrapper in
  577. DispatchQueue.main.async {
  578. AIChatInfoManager.defaultManager.isAILoading = false
  579. let resultStr = wrapper.content
  580. chatModel.chatResult = resultStr
  581. chatModel.chatInfoState = wrapper.success ? .stateSuccess : .stateFailed
  582. self.aiChatView.reloadData()
  583. self.aiTypeItemView.reloadData()
  584. self.aiInfoInputView.reloadData()
  585. }
  586. }
  587. }
  588. //MARK: AIInfoInputViewDelegate
  589. func ai_InputViewDidChooseCurFile(aiInputView: AIInfoInputView) {
  590. guard let callBack = self.chooseCurFileHandle else {
  591. return
  592. }
  593. callBack(self)
  594. }
  595. //MARK: NSWindowDelegate
  596. func windowDidBecomeMain(_ notification: Notification) {
  597. self.aiInfoInputView.reloadData()
  598. }
  599. func windowWillClose(_ notification: Notification) {
  600. AINewConfigWindowController.currentWindowController = nil
  601. }
  602. @objc func themeChange() {
  603. DispatchQueue.main.asyncAfter(deadline: DispatchTime.now() + 0.3) {
  604. self.refreshViewColor()
  605. }
  606. }
  607. }