KMSearchReplaceWindowController.swift 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550
  1. //
  2. // KMSearchReplaceWindowController.swift
  3. // PDF Reader Pro
  4. //
  5. // Created by User-Tangchao on 2024/8/7.
  6. //
  7. import Cocoa
  8. import KMComponentLibrary
  9. class KMSearchReplaceWindowController_Window: NSWindow {
  10. override var canBecomeMain: Bool {
  11. return true
  12. }
  13. override var canBecomeKey: Bool {
  14. return true
  15. }
  16. }
  17. class KMSearchReplaceWindowController: KMNBaseWindowController {
  18. @IBOutlet weak var titleBarBox: NSBox!
  19. @IBOutlet weak var tabBox: NSBox!
  20. @IBOutlet weak var searchBox: NSBox!
  21. @IBOutlet weak var replaceBox: NSBox!
  22. var replaceCallback: (() -> Void)?
  23. var itemClick: KMCommonClickBlock?
  24. var closeCallback: (() -> Void)?
  25. private var _modalSession: NSApplication.ModalSession?
  26. var handdler: KMNSearchHanddler = KMNSearchHanddler() {
  27. didSet {
  28. if handdler.searchKey?.count != 0 && handdler.searchResults.count == 0{
  29. self.search(keyboard: handdler.searchKey ?? "")
  30. } else {
  31. self.reloadData()
  32. }
  33. }
  34. }
  35. private var currentSel: CPDFSelection?
  36. private lazy var titleBarView_: KMNSearchReplaceTitleBarView = {
  37. let view = KMNSearchReplaceTitleBarView()
  38. return view
  39. }()
  40. private lazy var searchItemView_: KMNSearchReplaceSearchItemView = {
  41. let view = KMNSearchReplaceSearchItemView()
  42. return view
  43. }()
  44. private lazy var replaceItemView_: KMNSearchReplacePopItemView = {
  45. let view = KMNSearchReplacePopItemView()
  46. return view
  47. }()
  48. var previousButton: ComponentButton {
  49. get {
  50. return searchItemView_.previousButton
  51. }
  52. }
  53. var nextButton: ComponentButton {
  54. get {
  55. return searchItemView_.nextButton
  56. }
  57. }
  58. var replaceAllButton: ComponentButton {
  59. get {
  60. return replaceItemView_.replaceAllButton
  61. }
  62. }
  63. var replaceButton: ComponentButton {
  64. get {
  65. return replaceItemView_.replaceButton
  66. }
  67. }
  68. private var searchGroupView_: ComponentGroup?
  69. convenience init(with pdfView: CPDFView?, type: KMNBotaSearchType) {
  70. self.init(windowNibName: "KMSearchReplaceWindowController")
  71. self.handdler.pdfView = pdfView
  72. handdler.type = type
  73. }
  74. override func windowDidLoad() {
  75. super.windowDidLoad()
  76. self.initDefaultValue()
  77. self.switchType(handdler.type)
  78. }
  79. @objc func didMoveNotification(_ noti: Notification) {
  80. let window = self.window
  81. let frame = window?.frame ?? .zero
  82. let supFrame = window?.parent?.frame ?? .zero
  83. KMPrint("frame: \(frame)")
  84. KMPrint("supFrame: \(supFrame)")
  85. }
  86. func initDefaultValue() {
  87. window?.isMovableByWindowBackground = true
  88. window?.contentView?.wantsLayer = true
  89. window?.contentView?.layer?.cornerRadius = ComponentLibrary.shared.getComponentValueFromKey("radius/m") as? CGFloat ?? 8
  90. window?.contentView?.layer?.masksToBounds = true
  91. window?.backgroundColor = .clear
  92. titleBarBox.boxType = .custom
  93. titleBarBox.borderWidth = 0
  94. titleBarBox.contentView = titleBarView_
  95. titleBarView_.itemClick = { [weak self] idx, _ in
  96. if idx == 1 {
  97. self?._closeAction(NSButton())
  98. } else if idx == 2 {
  99. self?._closeAction(NSButton())
  100. self?.itemClick?(1, self?.handdler)
  101. } else if idx == 3 {
  102. self?.switchType(.search, animate: true)
  103. self?.itemClick?(3, self?.handdler)
  104. self?.search(keyboard: self?.handdler.searchKey ?? "")
  105. } else if idx == 4 {
  106. self?.switchType(.replace, animate: true)
  107. self?.itemClick?(4, self?.handdler)
  108. self?.search(keyboard: self?.handdler.searchKey ?? "")
  109. }
  110. }
  111. searchBox.borderWidth = 0
  112. searchBox.contentView = searchItemView_
  113. searchItemView_.itemClick = { [unowned self] idx, _ in
  114. if idx == 1 { // Previous
  115. _previousAction(NSButton())
  116. } else if idx == 2 { // next
  117. _nextAction(NSButton())
  118. } else if idx == 3 {
  119. showSearchGroupView(sender: ComponentButton())
  120. }
  121. }
  122. searchItemView_.valueDidChange = { [unowned self] value, _ in
  123. if let data = value as? String {
  124. handdler.searchKey = data
  125. search(keyboard: data)
  126. updateButtonStatus()
  127. }
  128. }
  129. searchItemView_.inputDidEditBlock = { [unowned self] in
  130. updateButtonStatus()
  131. let value = searchItemView_.inputValue
  132. if value.isEmpty {
  133. } else {
  134. currentSel = nil
  135. }
  136. }
  137. searchItemView_.input.properties.showSuffix = false
  138. searchItemView_.input.reloadData()
  139. replaceBox.borderWidth = 0
  140. replaceBox.contentView = replaceItemView_
  141. replaceItemView_.itemClick = { [unowned self] idx, _ in
  142. if idx == 1 {
  143. _replaceAllAction(NSButton())
  144. } else if idx == 2 {
  145. _replaceAction(NSButton())
  146. }
  147. }
  148. replaceItemView_.valueDidChange = { [unowned self] value, _ in
  149. if let data = value as? String {
  150. handdler.replaceKey = data
  151. }
  152. }
  153. updateButtonStatus()
  154. if searchItemView_.inputValue.isEmpty {
  155. } else {
  156. self.currentSel = nil
  157. }
  158. }
  159. override func updateUILanguage() {
  160. super.updateUILanguage()
  161. }
  162. override func updateUIThemeColor() {
  163. super.updateUIThemeColor()
  164. KMMainThreadExecute {
  165. self.searchItemView_.input.properties.leftIcon = NSImage(named: "KMImageNameBotaSearch")
  166. self.searchItemView_.input.reloadData()
  167. self.replaceItemView_.input.properties.leftIcon = NSImage(named: "KMImagenameBotaSearchInputPrefiex")
  168. self.replaceItemView_.input.reloadData()
  169. self.updateViewColor()
  170. }
  171. }
  172. func updateButtonStatus() {
  173. let value = searchItemView_.inputValue
  174. if value.isEmpty {
  175. previousButton.properties.isDisabled = true
  176. previousButton.reloadData()
  177. nextButton.properties.isDisabled = true
  178. nextButton.reloadData()
  179. replaceButton.properties.isDisabled = true
  180. replaceButton.reloadData()
  181. replaceAllButton.properties.isDisabled = true
  182. replaceAllButton.reloadData()
  183. } else {
  184. replaceButton.properties.isDisabled = false
  185. replaceButton.reloadData()
  186. replaceAllButton.properties.isDisabled = false
  187. replaceAllButton.reloadData()
  188. previousButton.properties.isDisabled = (self.handdler.showIdx == 0) ? true : false
  189. previousButton.reloadData()
  190. nextButton.properties.isDisabled = ((self.handdler.showIdx == self.handdler.resultCount - 1) || self.handdler.resultCount == 0) ? true : false
  191. nextButton.reloadData()
  192. }
  193. }
  194. private func updateViewColor() {
  195. let isDark = KMAppearance.isDarkMode()
  196. if isDark {
  197. self.window?.contentView?.wantsLayer = true
  198. self.window?.contentView?.layer?.backgroundColor = ComponentLibrary.shared.getComponentColorFromKey("colorBg/popup").cgColor
  199. self.window?.contentView?.border(ComponentLibrary.shared.getComponentColorFromKey("colorBorder/popUp"), 1, 8)
  200. } else {
  201. self.window?.contentView?.wantsLayer = true
  202. self.window?.contentView?.layer?.backgroundColor = ComponentLibrary.shared.getComponentColorFromKey("colorBg/popup").cgColor
  203. self.window?.contentView?.border(ComponentLibrary.shared.getComponentColorFromKey("colorBorder/popUp"), 1, 8)
  204. }
  205. self.switchType(handdler.type)
  206. }
  207. func switchType(_ type: KMNBotaSearchType, animate: Bool = false) {
  208. if type == .replace {
  209. if KMMemberInfo.shared.isLogin == false {
  210. KMLoginWindowsController.shared.showWindow(nil)
  211. return
  212. }
  213. }
  214. handdler.type = type
  215. self.titleBarView_.type = type
  216. if type == .search { // 248 112
  217. self.replaceBox.isHidden = true
  218. var frame = self.window?.frame ?? .zero
  219. let height: CGFloat = 156
  220. let heightOffset = frame.size.height - height
  221. frame.origin.y += heightOffset
  222. frame.size.height = height
  223. self.window?.setFrame(frame, display: true, animate: animate)
  224. self.window?.minSize = frame.size
  225. self.window?.maxSize = frame.size
  226. } else if type == .replace { // 388 208
  227. DispatchQueue.main.async {
  228. self.replaceBox.isHidden = false
  229. }
  230. var frame = self.window?.frame ?? .zero
  231. let height:CGFloat = 252
  232. let heightOffset = frame.size.height-height
  233. frame.origin.y += heightOffset
  234. frame.size.height = height
  235. self.window?.setFrame(frame, display: true, animate: animate)
  236. self.window?.minSize = frame.size
  237. self.window?.maxSize = frame.size
  238. // 将事件回调出去
  239. self.replaceCallback?()
  240. }
  241. }
  242. func search(keyboard: String) {
  243. let isCase = KMDataManager.ud_bool(forKey: KMNSearchKey.caseSensitive.botaSearch)
  244. let isWholeWord = KMDataManager.ud_bool(forKey: KMNSearchKey.wholeWords.botaSearch)
  245. let isEditing = self.handdler.pdfView?.isEditing() ?? false
  246. self._beginLoading()
  247. self.handdler.search(keyword: keyboard, isCase: isCase, isWholeWord: isWholeWord, isEdit: isEditing, callback: { [weak self] datas in
  248. self?._endLoading()
  249. self?.reloadData()
  250. })
  251. }
  252. func reloadData() {
  253. let handdler = self.handdler
  254. searchItemView_.inputValue = handdler.searchKey ?? ""
  255. replaceItemView_.inputValue = handdler.replaceKey ?? ""
  256. if handdler.showIdx != 0 {
  257. let model = self.handdler.searchResults[handdler.showIdx]
  258. self.handdler.showSelection(model.selection)
  259. } else {
  260. let sels = handdler.searchResults
  261. if let sel = sels.first?.selection {
  262. handdler.showSelection(sel)
  263. } else {
  264. handdler.showSelection(CPDFSelection())
  265. }
  266. }
  267. self._showIndexTip()
  268. self.updateButtonStatus()
  269. }
  270. }
  271. //MARK: Actions
  272. extension KMSearchReplaceWindowController {
  273. @objc private func _closeAction(_ sender: NSButton) {
  274. self.window?.orderOut(nil)
  275. self.handdler.clearData()
  276. self.closeCallback?()
  277. }
  278. @objc public func _previousAction(_ sender: NSButton?) {
  279. let index = self.handdler.previous()
  280. if index < self.handdler.searchResults.count && index >= 0{
  281. let model = self.handdler.searchResults[index]
  282. self.handdler.showSelection(model.selection)
  283. self._showIndexTip()
  284. self.updateButtonStatus()
  285. }
  286. }
  287. @objc public func _nextAction(_ sender: NSButton?) {
  288. let index = self.handdler.next()
  289. if index < self.handdler.searchResults.count && index >= 0 {
  290. let model = self.handdler.searchResults[index]
  291. self.handdler.showSelection(model.selection)
  292. self._showIndexTip()
  293. self.updateButtonStatus()
  294. }
  295. }
  296. @objc private func _checkAction(_ sender: NSButton) {
  297. self.currentSel = nil
  298. }
  299. @objc private func _searchTabAction(_ sender: NSButton) {
  300. self.switchType(.search, animate: true)
  301. }
  302. @objc private func _replaceTabAction(_ sender: NSButton) {
  303. self.switchType(.replace, animate: true)
  304. }
  305. @objc private func _replaceAction(_ sender: NSButton) {
  306. let isEditing = self.handdler.pdfView?.isEditing() ?? false
  307. if isEditing == false {
  308. NSSound.beep()
  309. return
  310. }
  311. guard let selection = self.currentSel else { return }
  312. let searchS = self.searchItemView_.inputValue
  313. let replaceS = self.replaceItemView_.inputValue
  314. self.handdler.replace(searchS: searchS, replaceS: replaceS, sel: selection) { [weak self] newSel in
  315. self?.handdler.showSelection(newSel)
  316. }
  317. }
  318. @objc private func _replaceAllAction(_ sender: NSButton) {
  319. let isEditing = self.handdler.pdfView?.isEditing() ?? false
  320. if isEditing == false {
  321. NSSound.beep()
  322. return
  323. }
  324. let datas = self.handdler.pdfView?.document.findEditSelections() ?? []
  325. if datas.isEmpty {
  326. _showNoResultsAlert()
  327. return
  328. }
  329. let searchS = self.searchItemView_.inputValue
  330. let replaceS = self.replaceItemView_.inputValue
  331. self._beginLoading()
  332. DispatchQueue.global().async {
  333. self.handdler.pdfView?.document.replaceAllEditText(with: searchS, toReplace: replaceS)
  334. self.currentSel = nil
  335. DispatchQueue.main.async {
  336. self._endLoading()
  337. self.handdler.pdfView?.setHighlightedSelection(nil, animated: false)
  338. self.handdler.pdfView?.setNeedsDisplayForVisiblePages()
  339. }
  340. }
  341. }
  342. }
  343. //MARK: Alert
  344. extension KMSearchReplaceWindowController {
  345. private func _showNoResultsAlert() {
  346. _ = _showAlert(style: .critical, message: KMLocalizedString("No related content found, please change keyword."), info: "", buttons: [KMLocalizedString("OK", comment: "")])
  347. }
  348. private func _showAlert(style: NSAlert.Style, message: String, info: String, buttons: [String]) -> NSApplication.ModalResponse {
  349. let alert = NSAlert()
  350. alert.alertStyle = style
  351. alert.messageText = message
  352. alert.informativeText = info
  353. for button in buttons {
  354. alert.addButton(withTitle: button)
  355. }
  356. return alert.runModal()
  357. }
  358. private func _showIndexTip() {
  359. DispatchQueue.main.async {
  360. if self.handdler.resultCount == 0 {
  361. self.searchItemView_.input.properties.rightText = ""
  362. } else {
  363. self.searchItemView_.input.properties.rightText = "\(self.handdler.showIdx+1)/\(self.handdler.resultCount)"
  364. }
  365. self.searchItemView_.input.reloadData()
  366. }
  367. }
  368. }
  369. //MARK: ComponentGroupDelegate
  370. extension KMSearchReplaceWindowController: ComponentGroupDelegate {
  371. func showSearchGroupView(sender: ComponentButton) {
  372. var viewHeight: CGFloat = 8
  373. var menuItemArr: [ComponentMenuitemProperty] = []
  374. let titles = ["Search", "Find and Replace", "", "Whole Words", "Case Sensitive"]
  375. for i in titles {
  376. if i.isEmpty {
  377. let menuI = ComponentMenuitemProperty.divider()
  378. menuItemArr.append(menuI)
  379. viewHeight += 8
  380. } else {
  381. let menuI = ComponentMenuitemProperty(text: KMLocalizedString(i))
  382. menuItemArr.append(menuI)
  383. viewHeight += 36
  384. }
  385. }
  386. if handdler.type == .search {
  387. menuItemArr.first?.righticon = NSImage(named: "KMNImageNameMenuSelect")
  388. } else if handdler.type == .replace {
  389. let info = menuItemArr.safe_element(for: 1) as? ComponentMenuitemProperty
  390. info?.righticon = NSImage(named: "KMNImageNameMenuSelect")
  391. }
  392. if let info = menuItemArr.safe_element(for: 3) as? ComponentMenuitemProperty {
  393. if KMDataManager.ud_bool(forKey: KMNSearchKey.wholeWords.botaSearch) {
  394. info.righticon = NSImage(named: "KMNImageNameMenuSelect")
  395. }
  396. }
  397. if let info = menuItemArr.last {
  398. if KMDataManager.ud_bool(forKey: KMNSearchKey.caseSensitive.botaSearch) {
  399. info.righticon = NSImage(named: "KMNImageNameMenuSelect")
  400. }
  401. }
  402. let groupView = ComponentGroup.createFromNib(in: ComponentLibrary.shared.componentBundle())
  403. searchGroupView_ = groupView
  404. groupView?.groupDelegate = self
  405. groupView?.frame = CGRectMake(310, 0, 200, viewHeight)
  406. groupView?.updateGroupInfo(menuItemArr)
  407. let senderView = self.searchItemView_.input
  408. var point = senderView.convert(senderView.frame.origin, to: nil)
  409. point.y -= viewHeight
  410. point.y -= 30
  411. groupView?.showWithPoint(point, relativeTo: senderView)
  412. // searchGroupTarget = sender
  413. }
  414. func componentGroupDidSelect(group: ComponentGroup?, menuItemProperty: ComponentMenuitemProperty?) {
  415. if group == searchGroupView_ {
  416. guard let menuI = menuItemProperty else {
  417. return
  418. }
  419. let idx = group?.menuItemArr.firstIndex(of: menuI)
  420. if idx == 0 { // search
  421. switchType(.search)
  422. } else if idx == 1 { // replace
  423. switchType(.replace)
  424. } else if idx == 3 {
  425. let key = KMNSearchKey.wholeWords.botaSearch
  426. let value = KMDataManager.ud_bool(forKey: key)
  427. KMDataManager.ud_set(!value, forKey: key)
  428. currentSel = nil
  429. let data = searchItemView_.inputValue
  430. if data.isEmpty {
  431. search(keyboard: data)
  432. }
  433. } else if idx == 4 {
  434. let key = KMNSearchKey.caseSensitive.botaSearch
  435. let value = KMDataManager.ud_bool(forKey: key)
  436. KMDataManager.ud_set(!value, forKey: key)
  437. currentSel = nil
  438. let data = searchItemView_.inputValue
  439. if data.isEmpty {
  440. search(keyboard: data)
  441. }
  442. }
  443. }
  444. }
  445. }
  446. //MARK: Load
  447. extension KMSearchReplaceWindowController {
  448. private func _beginLoading() {
  449. self.window?.contentView?.beginLoading()
  450. }
  451. private func _endLoading() {
  452. self.window?.contentView?.endLoading()
  453. }
  454. }
  455. //MARK: Modal
  456. extension KMSearchReplaceWindowController {
  457. func startModal(_ sender: AnyObject?) {
  458. NSApp.stopModal()
  459. var modalCode: NSApplication.ModalResponse?
  460. if let _win = self.window {
  461. self._modalSession = NSApp.beginModalSession(for: _win)
  462. repeat {
  463. modalCode = NSApp.runModalSession(self._modalSession!)
  464. } while (modalCode == .continue)
  465. }
  466. }
  467. func endModal(_ sender: AnyObject?) {
  468. if let session = self._modalSession {
  469. NSApp.stopModal()
  470. NSApp.endModalSession(session)
  471. self.window?.orderOut(self)
  472. }
  473. if let winC = self.window?.kmCurrentWindowC, winC.isEqual(to: self) {
  474. self.window?.kmCurrentWindowC = nil
  475. }
  476. }
  477. }