KMOutlineViewController.swift 46 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172
  1. //
  2. // KMOutlineViewController.swift
  3. // PDF Reader Pro
  4. //
  5. // Created by lxy on 2022/10/10.
  6. //
  7. import Cocoa
  8. import KMComponentLibrary
  9. extension KMNSearchKey.wholeWords {
  10. static let outline = "OutlineSearchWholeWordsKey"
  11. }
  12. extension KMNSearchKey.caseSensitive {
  13. static let outline = "OutlineSearchCaseSensitiveKey"
  14. }
  15. class KMOutlineViewController: KMNBotaBaseViewController {
  16. @IBOutlet var contendView: NSView!
  17. @IBOutlet weak var topView: NSView!
  18. @IBOutlet weak var titleLabel: NSTextField!
  19. @IBOutlet weak var lineView: NSView!
  20. @IBOutlet weak var addButton: NSButton!
  21. @IBOutlet weak var moreButton: NSButton!
  22. @IBOutlet var topSepline: NSView!
  23. @IBOutlet weak var emptyView: NSView!
  24. @IBOutlet weak var bigTipLabel: NSTextField!
  25. @IBOutlet weak var tipLabel: NSTextField!
  26. @IBOutlet weak var BOTAOutlineView: KMBOTAOutlineView!
  27. private weak var popover_: NSPopover?
  28. private lazy var addButton_: ComponentButton = {
  29. let view = ComponentButton()
  30. view.properties = ComponentButtonProperty(type: .text_gray, size: .xxs, state: .normal, isDisable: false, onlyIcon: true, keepPressState: false)
  31. return view
  32. }()
  33. private lazy var moreDropdown_: ComponentButton = {
  34. let view = ComponentButton()
  35. view.properties = ComponentButtonProperty(type: .text_gray, size: .xxs, state: .normal, isDisable: false, onlyIcon: true, icon: NSImage(named: "KMImageNameOutlineMore"))
  36. return view
  37. }()
  38. private var emptyView_: ComponentEmpty = {
  39. let view = ComponentEmpty()
  40. view.properties = ComponentEmptyProperty(emptyType: .noOutline, state: .normal, image: NSImage(named: "KMImageNameOutlineEmpty"), text: KMLocalizedString("No Outline"), subText: KMLocalizedString("Here is the description."))
  41. return view
  42. }()
  43. private var groupView_: ComponentGroup?
  44. private var menuGroupView_: ComponentGroup?
  45. private var outlineView_: KMOutlineView? {
  46. get {
  47. return BOTAOutlineView.outlineView
  48. }
  49. }
  50. var dragPDFOutline : KMBOTAOutlineItem!
  51. private var dragPDFOutlines_: [KMBOTAOutlineItem] = []
  52. var renameTextField : NSTextField!
  53. var renamePDFOutline : KMBOTAOutlineItem!
  54. let moreMenu = NSMenu()
  55. var isLocalEvent = false
  56. var model = KMNOutlineModel()
  57. var handdler = KMNOutlineHanddler()
  58. deinit {
  59. self.BOTAOutlineView.delegate = nil
  60. }
  61. override func viewWillDisappear() {
  62. super.viewWillDisappear()
  63. self.cancelSelect()
  64. }
  65. override func viewDidLoad() {
  66. super.viewDidLoad()
  67. handdler.delegate = self
  68. titleLabel.font = ComponentLibrary.shared.getFontFromKey("mac/body-m-bold")
  69. self.topView.wantsLayer = true
  70. addButton.image = nil
  71. topView.addSubview(searchButton)
  72. searchButton.km_add_size_constraint(size: NSMakeSize(24, 24))
  73. searchButton.km_add_centerY_constraint(constant: 1)
  74. searchButton.km_add_trailing_constraint(equalTo: addButton, attribute: .leading, constant: -4)
  75. searchButton.setTarget(self, action: #selector(_searchAction))
  76. addButton.addSubview(addButton_)
  77. addButton_.km_add_size_constraint(size: NSMakeSize(24, 24))
  78. addButton_.km_add_centerX_constraint()
  79. addButton_.km_add_centerY_constraint()
  80. addButton_.setTarget(self, action: #selector(addNewOutline))
  81. moreButton.image = nil
  82. moreButton.addSubview(moreDropdown_)
  83. moreDropdown_.km_add_size_constraint(size: NSMakeSize(24, 24))
  84. moreDropdown_.km_add_centerX_constraint()
  85. moreDropdown_.km_add_centerY_constraint()
  86. moreDropdown_.setTarget(self, action: #selector(_moreAction))
  87. if let data = headerSearchView {
  88. topView.addSubview(data)
  89. headerSearchView?.frame = topView.bounds
  90. headerSearchView?.autoresizingMask = [.width, .height]
  91. }
  92. hideHeaderSearch()
  93. headerSearchView?.itemClick = { [weak self] idx, params in
  94. if idx == 1 { // 显示搜索限制条件
  95. guard let button = params.first as? ComponentButton else {
  96. return
  97. }
  98. self?.showSearchGroupView(sender: button)
  99. } else if idx == 2 { // 关闭搜索
  100. self?.hideHeaderSearch()
  101. self?.reloadData()
  102. }
  103. }
  104. headerSearchView?.valueDidChange = { [weak self] sender, info in
  105. let value = info?[.newKey] as? String ?? ""
  106. self?.BOTAOutlineView.searchKey = value
  107. self?.reloadData()
  108. self?.BOTAOutlineView.outlineView.expandItem(nil, expandChildren: true)
  109. }
  110. emptyView.wantsLayer = true
  111. bigTipLabel.stringValue = ""
  112. tipLabel.stringValue = ""
  113. emptyView.addSubview(emptyView_)
  114. emptyView_.km_add_top_constraint(constant: 232)
  115. emptyView_.km_add_bottom_constraint()
  116. emptyView_.km_add_leading_constraint()
  117. emptyView_.km_add_trailing_constraint()
  118. self.BOTAOutlineView.delegate = self
  119. self.BOTAOutlineView.inputData = self.handdler.outlineRoot()
  120. self.BOTAOutlineView.outlineView.doubleAction = #selector(outlineViewDoubleAction)
  121. }
  122. override func updateUILanguage() {
  123. super.updateUILanguage()
  124. KMMainThreadExecute {
  125. self.titleLabel.stringValue = KMLocalizedString("Outline")
  126. }
  127. }
  128. override func updateUIThemeColor() {
  129. super.updateUIThemeColor()
  130. KMMainThreadExecute {
  131. self.contendView.wantsLayer = true
  132. let color = KMNColorTools.colorBg_layoutMiddle()
  133. self.contendView.layer?.backgroundColor = color.cgColor
  134. self.titleLabel.textColor = KMNColorTools.colorText_2()
  135. self.addButton_.properties.icon = NSImage(named: "KMBookmarkAdd")
  136. self.addButton_.reloadData()
  137. self.searchButton.properties.icon = NSImage(named: "KMImageNameOutlineSearch")
  138. self.searchButton.reloadData()
  139. let dividerColor = KMNColorTools.colorBorder_divider()
  140. self.topSepline.wantsLayer = true
  141. self.topSepline.layer?.backgroundColor = dividerColor.cgColor
  142. self.lineView.backgroundColor(dividerColor)
  143. }
  144. }
  145. override func showHeaderSearch() {
  146. super.showHeaderSearch()
  147. BOTAOutlineView.isSearchMode = true
  148. }
  149. override func hideHeaderSearch() {
  150. super.hideHeaderSearch()
  151. BOTAOutlineView.isSearchMode = false
  152. }
  153. func addRightMenu(view: NSView, event: NSEvent) {
  154. let point = event.locationInWindow
  155. let tempView = view
  156. var viewHeight: CGFloat = 0
  157. let items: [String] = ["Add Item", "Add Sub-Item", "Add A Higher Level","", "Delete","", "Edit", "Rename", "Change Destination","", "Promote", "Demote"]
  158. var menuItemArr: [ComponentMenuitemProperty] = []
  159. for value in items {
  160. if value.count == 0 {
  161. let property: ComponentMenuitemProperty = ComponentMenuitemProperty.divider()
  162. menuItemArr.append(property)
  163. viewHeight += 8
  164. } else {
  165. let properties_Menuitem: ComponentMenuitemProperty = ComponentMenuitemProperty(multipleSelect: false,
  166. itemSelected: false,
  167. isDisabled: false,
  168. keyEquivalent: nil,
  169. text: KMLocalizedString(value),
  170. identifier: value)
  171. menuItemArr.append(properties_Menuitem)
  172. viewHeight += 36
  173. }
  174. }
  175. if _isEmptySelection() {
  176. for data in menuItemArr {
  177. if data.text == KMLocalizedString("Add Item") {
  178. data.isDisabled = false
  179. } else {
  180. data.isDisabled = true
  181. }
  182. }
  183. } else if _isMutilSelection() {
  184. for data in menuItemArr {
  185. if data.text == KMLocalizedString("Delete") {
  186. data.isDisabled = false
  187. } else {
  188. data.isDisabled = true
  189. }
  190. }
  191. } else {
  192. let clickedRow = BOTAOutlineView.outlineView.clickedRow
  193. let outlineItem = BOTAOutlineView.outlineView.item(atRow: clickedRow) as? KMBOTAOutlineItem
  194. let idx = outlineItem?.outline.index ?? 0
  195. let canDemote = idx > 0
  196. let grandparentOutline = outlineItem?.outline.parent?.parent
  197. let canPromote = grandparentOutline != nil
  198. let canAddHigher = grandparentOutline != nil
  199. if BOTAOutlineView.isValidSearchMode() {
  200. for data in menuItemArr {
  201. if data.text == KMLocalizedString("Delete") || data.text == KMLocalizedString("Change Destination") {
  202. data.isDisabled = false
  203. } else if data.text == KMLocalizedString("Demote") {
  204. data.isDisabled = !canDemote
  205. } else if data.text == KMLocalizedString("Promote") {
  206. data.isDisabled = !canPromote
  207. } else {
  208. data.isDisabled = true
  209. }
  210. }
  211. } else {
  212. for data in menuItemArr {
  213. if data.text == KMLocalizedString("Add Sub-Item") || data.text == KMLocalizedString("Change Destination") {
  214. data.isDisabled = false
  215. } else if data.text == KMLocalizedString("Demote") {
  216. data.isDisabled = !canDemote
  217. } else if data.text == KMLocalizedString("Promote") {
  218. data.isDisabled = !canPromote
  219. } else if data.text == KMLocalizedString("Add A Higher Level") {
  220. data.isDisabled = !canAddHigher
  221. }
  222. }
  223. }
  224. }
  225. if menuGroupView_ == nil {
  226. menuGroupView_ = ComponentGroup.createFromNib(in: ComponentLibrary.shared.componentBundle())
  227. }
  228. if menuGroupView_ != nil {
  229. menuGroupView_?.clickedAutoHide = false
  230. menuGroupView_?.groupDelegate = self
  231. menuGroupView_?.frame = CGRectMake(0, 0, 180, viewHeight)
  232. menuGroupView_?.updateGroupInfo(menuItemArr)
  233. menuGroupView_?.showWithPoint(CGPoint(x: point.x, y: point.y - viewHeight), relativeTo: tempView)
  234. }
  235. }
  236. func reloadData() {
  237. self.BOTAOutlineView.reloadData(expandItemType: .none)
  238. }
  239. func editOutlineUI(editVC : KMOutlineEditViewController!) {
  240. if editVC.pageRadio.properties.checkboxType == .selected {
  241. let index = Int(editVC.pageInput.properties.text) ?? 0
  242. let pageIndex = max(0, index-1)
  243. if editVC.originalDestination?.pageIndex != pageIndex {
  244. let page = editVC.pdfView?.document.page(at: UInt(pageIndex))
  245. if page != nil {
  246. let destination = CPDFDestination.init(document: editVC.pdfView!.document, pageIndex: pageIndex)
  247. editVC.outline?.destination = destination
  248. } else {
  249. __NSBeep()
  250. }
  251. }
  252. } else if editVC.webRaido.properties.checkboxType == .selected {
  253. if editVC.originalURLString != editVC.webInput.properties.text {
  254. var urlString = editVC.webInput.properties.text
  255. let tLowerUrl = urlString.lowercased()
  256. if !tLowerUrl.hasPrefix("https://") && !tLowerUrl.hasPrefix("pf]://") && !urlString.hasPrefix("https://") &&
  257. urlString.lengthOfBytes(using: String.Encoding(rawValue: String.Encoding.utf16.rawValue)) > 0 {
  258. urlString = "http://\(urlString)"
  259. }
  260. let action = CPDFURLAction.init(url: urlString)
  261. editVC.outline?.action = action
  262. }
  263. } else if editVC.emailRadio.properties.checkboxType == .selected {
  264. var mailString = editVC.emailInput.properties.text
  265. let tLowerStr = mailString.lowercased()
  266. if !tLowerStr.hasPrefix("mailto:") {
  267. mailString = "mailto:\(mailString)"
  268. }
  269. if mailString != editVC.originalURLString {
  270. var action = CPDFURLAction.init(url: mailString)
  271. if action?.url == nil {
  272. action = CPDFURLAction.init(url: "mailto:")
  273. }
  274. editVC.outline?.action = action
  275. }
  276. }
  277. }
  278. // MARK: - Private Methods
  279. private func _showAlert(style: NSAlert.Style, message: String, info: String, buttons: [String]) -> NSApplication.ModalResponse {
  280. let alert = NSAlert()
  281. alert.alertStyle = style
  282. alert.messageText = message
  283. alert.informativeText = info
  284. for button in buttons {
  285. alert.addButton(withTitle: button)
  286. }
  287. return alert.runModal()
  288. }
  289. @objc private func _moreAction() {
  290. self.showGroupView()
  291. }
  292. private func _isEmptySelection() -> Bool {
  293. return BOTAOutlineView.outlineView.clickedRow == -1
  294. }
  295. private func _isMutilSelection() -> Bool {
  296. return BOTAOutlineView.outlineView.selectedRowIndexes.count > 1
  297. }
  298. // MARK: - Public Methods
  299. public func addOutline() {
  300. addItemAction()
  301. }
  302. public func removeAllOutline() {
  303. guard let data = self.BOTAOutlineView.data else { return }
  304. for item in data.children {
  305. item.toIndex = Int(item.outline.index)
  306. }
  307. self.deleteOutline(outlineItems: data.children)
  308. self.BOTAOutlineView.reloadData(expandItemType: .none)
  309. }
  310. //MARK: - GroupView
  311. func showGroupView() {
  312. var viewHeight: CGFloat = 8
  313. var menuItemArr: [ComponentMenuitemProperty] = []
  314. for i in ["Expand All", "Collapse All", "Remove All Outlines"] {
  315. let properties_Menuitem: ComponentMenuitemProperty = ComponentMenuitemProperty(multipleSelect: false,
  316. itemSelected: false,
  317. isDisabled: false,
  318. keyEquivalent: nil,
  319. text: KMLocalizedString(i))
  320. menuItemArr.append(properties_Menuitem)
  321. viewHeight += 36
  322. }
  323. if let data = menuItemArr.first {
  324. var canExpand = false
  325. for row in 0..<self.BOTAOutlineView.outlineView.numberOfRows {
  326. // 检查当前项目是否可以展开
  327. let item = self.BOTAOutlineView.outlineView.item(atRow: row)
  328. if self.BOTAOutlineView.outlineView.isExpandable(item) {
  329. if !self.BOTAOutlineView.outlineView.isItemExpanded(item) {
  330. canExpand = true
  331. break
  332. }
  333. }
  334. }
  335. data.isDisabled = !canExpand
  336. }
  337. if let data = menuItemArr.safe_element(for: 1) as? ComponentMenuitemProperty {
  338. var canCollapse = false
  339. for row in 0..<self.BOTAOutlineView.outlineView.numberOfRows {
  340. let item = self.BOTAOutlineView.outlineView.item(atRow: row)
  341. if self.BOTAOutlineView.outlineView.isExpandable(item) {
  342. if self.BOTAOutlineView.outlineView.isItemExpanded(item) {
  343. canCollapse = true
  344. break
  345. }
  346. }
  347. }
  348. data.isDisabled = !canCollapse
  349. }
  350. if let data = menuItemArr.last {
  351. if self.BOTAOutlineView.outlineView.item(atRow: 0) != nil {
  352. data.isDisabled = false
  353. } else {
  354. data.isDisabled = true
  355. }
  356. }
  357. if groupView_ == nil {
  358. groupView_ = ComponentGroup.createFromNib(in: ComponentLibrary.shared.componentBundle())
  359. }
  360. groupView_?.groupDelegate = self
  361. groupView_?.frame = CGRectMake(310, 0, 200, viewHeight)
  362. groupView_?.updateGroupInfo(menuItemArr)
  363. var point = moreDropdown_.convert(moreDropdown_.frame.origin, to: nil)
  364. point.y -= viewHeight
  365. groupView_?.showWithPoint(point, relativeTo: moreDropdown_)
  366. moreDropdown_.properties.state = .pressed
  367. moreDropdown_.reloadData()
  368. }
  369. func showSearchGroupView(sender: ComponentButton) {
  370. var viewHeight: CGFloat = 8
  371. var menuItemArr: [ComponentMenuitemProperty] = []
  372. let titles = ["Whole Words","Case Sensitive"]
  373. for i in titles {
  374. let menuI = ComponentMenuitemProperty(text: KMLocalizedString(i))
  375. menuItemArr.append(menuI)
  376. viewHeight += 36
  377. }
  378. if let info = menuItemArr.first {
  379. if KMDataManager.ud_bool(forKey: KMNSearchKey.wholeWords.outline) {
  380. info.righticon = NSImage(named: "KMNImageNameMenuSelect")
  381. }
  382. }
  383. if let info = menuItemArr.last {
  384. if KMDataManager.ud_bool(forKey: KMNSearchKey.caseSensitive.outline) {
  385. info.righticon = NSImage(named: "KMNImageNameMenuSelect")
  386. }
  387. }
  388. let groupView = ComponentGroup.createFromNib(in: ComponentLibrary.shared.componentBundle())
  389. searchGroupView = groupView
  390. groupView?.groupDelegate = self
  391. groupView?.frame = CGRectMake(310, 0, 200, viewHeight)
  392. groupView?.updateGroupInfo(menuItemArr)
  393. var point = sender.convert(sender.frame.origin, to: nil)
  394. point.y -= viewHeight
  395. groupView?.showWithPoint(point, relativeTo: sender)
  396. searchGroupTarget = sender
  397. }
  398. func removeGroupView() {
  399. if groupView_ != nil {
  400. groupView_?.removeFromSuperview()
  401. }
  402. moreDropdown_.properties.state = .normal
  403. moreDropdown_.reloadData()
  404. }
  405. func updateExtempViewState() {
  406. if BOTAOutlineView.isValidSearchMode() {
  407. if BOTAOutlineView.data?.searchChildren.isEmpty == true {
  408. self.emptyView.isHidden = false
  409. } else {
  410. self.emptyView.isHidden = true
  411. }
  412. } else {
  413. if(self.handdler.outlineRoot() == nil || self.handdler.outlineRoot()?.numberOfChildren == 0) { //无数据时的图
  414. self.emptyView.isHidden = false
  415. } else {
  416. self.emptyView.isHidden = true
  417. }
  418. }
  419. }
  420. }
  421. //MARK: - Menu 右键菜单
  422. extension KMOutlineViewController {
  423. @objc func outlineViewDoubleAction() {
  424. if(self.BOTAOutlineView.outlineView.clickedRow >= 0) {
  425. self.renameItemAction()
  426. }
  427. }
  428. @objc func addItemAction() {
  429. guard let outlineView = BOTAOutlineView.outlineView else {
  430. return
  431. }
  432. let selectRowIndexs = outlineView.selectedRowIndexes
  433. let dataCount = BOTAOutlineView.data?.children.count ?? 0
  434. var index: Int = 0
  435. var parent: KMBOTAOutlineItem?
  436. var outlineItem: KMBOTAOutlineItem?
  437. if selectRowIndexs.count == 0 {
  438. var lastOulineItem: KMBOTAOutlineItem?
  439. if dataCount == 0 {
  440. let item = KMBOTAOutlineItem()
  441. item.outline = self.handdler.document!.setNewOutlineRoot()
  442. item.parent = nil
  443. parent = item
  444. lastOulineItem = item
  445. } else {
  446. outlineItem = outlineView.item(atRow: outlineView.numberOfRows - 1) as? KMBOTAOutlineItem
  447. lastOulineItem = outlineItem
  448. while lastOulineItem?.parent != nil {
  449. lastOulineItem = lastOulineItem?.parent
  450. }
  451. parent = lastOulineItem
  452. }
  453. index = Int(lastOulineItem?.outline.numberOfChildren ?? 0)
  454. } else {
  455. outlineItem = outlineView.item(atRow: selectRowIndexs.last ?? 0) as? KMBOTAOutlineItem
  456. parent = outlineItem?.parent ?? KMBOTAOutlineItem()
  457. index = Int(((outlineItem?.outline.index) ?? 0) + 1)
  458. }
  459. self.addOutlineToIndex(index: index, parent: parent)
  460. }
  461. @objc func addChildItemAction() {
  462. let outlineView: KMOutlineView = self.BOTAOutlineView.outlineView
  463. let selectRowIndexs = outlineView.selectedRowIndexes
  464. if selectRowIndexs.count != 0 {
  465. let outlineItem: KMBOTAOutlineItem = outlineView.item(atRow: selectRowIndexs.last!) as! KMBOTAOutlineItem
  466. let index = outlineItem.outline.numberOfChildren
  467. self.addOutlineToIndex(index: NSInteger(index), parent: outlineItem)
  468. }
  469. }
  470. @objc func addHigherItemAction() {
  471. let outlineView: KMOutlineView = self.BOTAOutlineView.outlineView
  472. let selectRowIndexs = outlineView.selectedRowIndexes
  473. if selectRowIndexs.count != 0 {
  474. let outlineItem: KMBOTAOutlineItem = outlineView.item(atRow: selectRowIndexs.last!) as! KMBOTAOutlineItem
  475. var parent = outlineItem.parent
  476. let index = NSInteger(parent!.outline.index) + 1
  477. parent = parent?.parent
  478. if parent != nil {
  479. self.addOutlineToIndex(index: index, parent: parent!)
  480. }
  481. }
  482. }
  483. @objc func deleteItemAction() {
  484. let outlineView: KMOutlineView = self.BOTAOutlineView.outlineView
  485. let selectRowIndexs = outlineView.selectedRowIndexes
  486. if selectRowIndexs.count != 0 {
  487. var outlineItems: [KMBOTAOutlineItem] = []
  488. for index in selectRowIndexs {
  489. let outlineItem: KMBOTAOutlineItem = self.BOTAOutlineView.outlineView.item(atRow: index) as! KMBOTAOutlineItem
  490. outlineItem.toIndex = index
  491. outlineItem.parent = outlineItem.parent ?? KMBOTAOutlineItem()
  492. outlineItems.append(outlineItem)
  493. }
  494. self.deleteOutline(outlineItems: outlineItems)
  495. }
  496. }
  497. @objc func editItemAction() {
  498. let clickedRow = BOTAOutlineView.outlineView.clickedRow
  499. if clickedRow < 0 {
  500. NSSound.beep()
  501. return
  502. }
  503. if let rowView = self.BOTAOutlineView.outlineView.rowView(atRow: clickedRow, makeIfNecessary: true) {
  504. let item = self.BOTAOutlineView.outlineView.item(atRow: clickedRow) as? KMBOTAOutlineItem
  505. let vc = KMOutlineEditViewController.init(outline: item?.outline, document: self.handdler.pdfView)
  506. vc.pageCount = handdler.pageCount()
  507. vc.itemClick = { [weak self] idx, params in
  508. if idx == 1 {
  509. self?.popover_?.close()
  510. } else if idx == 2 {
  511. self?.popover_?.close()
  512. if let viewC = params.first as? KMOutlineEditViewController {
  513. let resp = self?._showAlert(style: .informational, message: KMLocalizedString("Are you sure you want to apply edits to this outline?"), info: "", buttons: [KMLocalizedString("Apply"), KMLocalizedString("Cancel")])
  514. if resp == .alertFirstButtonReturn {
  515. self?.editOutlineUI(editVC: viewC)
  516. }
  517. }
  518. }
  519. }
  520. let popover = NSPopover()
  521. popover_ = popover
  522. popover.delegate = self
  523. popover.contentViewController = vc
  524. popover.animates = true
  525. popover.behavior = .transient
  526. popover.setValue(true, forKey: "shouldHideAnchor")
  527. popover.show(relativeTo: rowView.bounds, of: rowView, preferredEdge: .minX)
  528. }
  529. }
  530. @objc func renameItemAction() {
  531. if self.BOTAOutlineView.outlineView.clickedRow >= 0 {
  532. self.renameOutlineWithRow(row: self.BOTAOutlineView.outlineView.clickedRow)
  533. } else {
  534. __NSBeep()
  535. }
  536. }
  537. @objc func changeItemAction() {
  538. guard let currentDest = handdler.currentDestination() else {
  539. NSSound.beep()
  540. return
  541. }
  542. guard let item = outlineView_?.clickedItem() as? KMBOTAOutlineItem else {
  543. NSSound.beep()
  544. return
  545. }
  546. let resp = _showAlert(style: .informational, message: KMLocalizedString("Are you sure you want to set the destination as the current location?"), info: "", buttons: [KMLocalizedString("Yes"), KMLocalizedString("No")])
  547. if resp == .alertFirstButtonReturn {
  548. handdler.changeLocation(outlineItem: item, destination: currentDest)
  549. }
  550. }
  551. @objc func promoteItemAction() {
  552. if self.BOTAOutlineView.outlineView.clickedRow >= 0 {
  553. let outlineItem: KMBOTAOutlineItem = self.BOTAOutlineView.outlineView.item(atRow: self.BOTAOutlineView.outlineView.clickedRow) as! KMBOTAOutlineItem
  554. var parent = outlineItem.parent
  555. let index = NSInteger(parent!.outline.index) + 1
  556. parent = parent?.parent
  557. if parent != nil {
  558. self.moveOutline(outlineItem: outlineItem, index: index, parent: parent)
  559. }
  560. }
  561. }
  562. @objc func demoteItemAction() {
  563. if self.BOTAOutlineView.outlineView.clickedRow >= 0 {
  564. let outlineItem: KMBOTAOutlineItem = self.BOTAOutlineView.outlineView.item(atRow: self.BOTAOutlineView.outlineView.clickedRow) as! KMBOTAOutlineItem
  565. let parent = outlineItem.parent
  566. let newParent = parent?.children[Int(outlineItem.outline.index) - 1]
  567. let index = newParent?.children.count
  568. if (index != nil) {
  569. self.moveOutline(outlineItem: outlineItem, index: NSInteger(index ?? 0), parent: newParent)
  570. }
  571. }
  572. }
  573. @objc private func expandAllComments(item: NSMenuItem) {
  574. self.BOTAOutlineView.expandAllComments(item: item)
  575. }
  576. @objc private func collapseAllComments(item: NSMenuItem) {
  577. self.BOTAOutlineView.collapseAllComments(item: item)
  578. }
  579. @objc private func removeAllOutlineItem(item: NSMenuItem) {
  580. let alter = NSAlert()
  581. alter.alertStyle = .informational
  582. alter.messageText = KMLocalizedString("This will permanently remove all outlines. Are you sure to continue?")
  583. alter.addButton(withTitle: KMLocalizedString("Yes"))
  584. alter.addButton(withTitle: KMLocalizedString("No"))
  585. let modlres = alter.runModal()
  586. if modlres == .alertFirstButtonReturn {
  587. self.removeAllOutline()
  588. }
  589. }
  590. @objc private func _searchAction() {
  591. searchButton.properties.state = .normal
  592. searchButton.reloadData()
  593. showHeaderSearch()
  594. }
  595. }
  596. //MARK: - Action
  597. extension KMOutlineViewController {
  598. @IBAction func addNewOutline(_ sender: Any) {
  599. self.addItemAction()
  600. }
  601. @IBAction func escButtonAction(_ sender: Any) {
  602. self.cancelSelect()
  603. }
  604. func cancelSelect() {
  605. self.BOTAOutlineView.cancelSelect()
  606. }
  607. func renameOutlineWithRow(row: NSInteger) {
  608. DispatchQueue.main.async {
  609. self.renamePDFOutline = self.BOTAOutlineView.outlineView.item(atRow: row) as? KMBOTAOutlineItem
  610. let cell : KMBOTAOutlineCellView = self.BOTAOutlineView.outlineView.view(atColumn: 0, row: row, makeIfNecessary: true) as! KMBOTAOutlineCellView
  611. self.renameTextField = cell.titleLabel
  612. self.renameTextField.delegate = self
  613. self.renameTextField.isEditable = true
  614. self.renameTextField.becomeFirstResponder()
  615. }
  616. }
  617. func addOutlineToIndex(index: Int, parent: KMBOTAOutlineItem?) {
  618. let pageIndex = self.handdler.currentPageIndex
  619. let label = self.fetchCurrentLabel(pageIndex: pageIndex)
  620. let destination = self.handdler.currentDestination()
  621. self.addOutlineToIndex(index: index, pageIndex: pageIndex, destination: destination, lable: label, parent: parent)
  622. }
  623. func addOutlineToIndex(index: Int, pageIndex: Int, destination: CPDFDestination?, lable: String, parent: KMBOTAOutlineItem?) {
  624. let outlineItem = KMBOTAOutlineItem()
  625. outlineItem.destination = destination
  626. outlineItem.label = lable
  627. outlineItem.parent = parent
  628. outlineItem.toIndex = index
  629. self.addOutline(outlineItems: [outlineItem])
  630. let tempOutlineView = self.BOTAOutlineView!
  631. var index = -1
  632. if tempOutlineView.outlineView.numberOfRows == 1 || tempOutlineView.data == nil {
  633. index = 0
  634. } else {
  635. index = tempOutlineView.outlineView.row(forItem: outlineItem)
  636. }
  637. tempOutlineView.selectIndex(index: index)
  638. //滑动到指定位置
  639. if(tempOutlineView.outlineView.selectedRow >= 0) {
  640. self.renameOutlineWithRow(row: tempOutlineView.outlineView.selectedRow)
  641. }
  642. let row = tempOutlineView.outlineView.row(forItem: outlineItem)
  643. if Thread.current.isMainThread {
  644. tempOutlineView.outlineView.scrollToVisible(tempOutlineView.outlineView.rect(ofRow: row))
  645. } else {
  646. DispatchQueue.main.async {
  647. tempOutlineView.outlineView.scrollToVisible(tempOutlineView.outlineView.rect(ofRow: row))
  648. }
  649. }
  650. }
  651. func updateOutlineSelection() {
  652. let currentPageIndex = self.handdler.currentPageIndex
  653. let numRows = self.BOTAOutlineView.outlineView.numberOfRows
  654. if numRows > 0 {
  655. for i in 0...numRows - 1 {
  656. let outlineItem: KMBOTAOutlineItem = self.BOTAOutlineView.outlineView.item(atRow: i) as! KMBOTAOutlineItem
  657. if (outlineItem.outline.destination == nil) {
  658. continue
  659. }
  660. if outlineItem.outline.destination.pageIndex == currentPageIndex {
  661. self.BOTAOutlineView.selectIndex(index: currentPageIndex)
  662. break
  663. }
  664. }
  665. }
  666. }
  667. func fetchCurrentLabel(pageIndex: Int) -> String {
  668. var label = "\(KMLocalizedString("Page"))\(pageIndex + 1)"
  669. let currentSelection = self.handdler.currentSelection()
  670. if currentSelection != nil && currentSelection?.selectionsByLine != nil {
  671. for data in currentSelection?.selectionsByLine ?? [] {
  672. label = data.string() ?? ""
  673. }
  674. }
  675. return label
  676. }
  677. }
  678. //MARK: - KMBOTAOutlineViewDelegate
  679. extension KMOutlineViewController: KMBOTAOutlineViewDelegate {
  680. func BOTAOutlineView(_ outlineView: KMBOTAOutlineView, rightDidMoseDown: KMBOTAOutlineItem, event: NSEvent) {
  681. let row = outlineView.outlineView.row(forItem: rightDidMoseDown)
  682. if outlineView.outlineView.rowView(atRow: row, makeIfNecessary: false) != nil {
  683. let rowView = outlineView.outlineView.rowView(atRow: row, makeIfNecessary: false)
  684. self.addRightMenu(view: rowView!, event: event)
  685. }
  686. }
  687. func BOTAOutlineView(_ outlineView: KMBOTAOutlineView, didReloadData: KMBOTAOutlineItem) {
  688. self.updateExtempViewState()
  689. }
  690. func BOTAOutlineView(_ outlineView: KMBOTAOutlineView, didSelectItem: [KMBOTAOutlineItem]) {
  691. if outlineView_?.selectedRowIndexes.count == 1 {
  692. isLocalEvent = true
  693. guard let item = outlineView_?.selectedItem() as? KMBOTAOutlineItem else {
  694. return
  695. }
  696. handdler.selectOutline(item.outline)
  697. }
  698. }
  699. func BOTAOutlineView(_ outlineView: KMBOTAOutlineView, writeItems items: [Any], to pasteboard: NSPasteboard) -> Bool {
  700. if outlineView.outlineView.selectedRow == -1 {
  701. return false
  702. }
  703. self.dragPDFOutline = items.first as? KMBOTAOutlineItem
  704. self.dragPDFOutlines_ = items as? [KMBOTAOutlineItem] ?? []
  705. let indexSet = [outlineView.outlineView.clickedRow]
  706. let indexSetData: Data = try!NSKeyedArchiver.archivedData(withRootObject: indexSet, requiringSecureCoding: true)
  707. pasteboard.declareTypes([NSPasteboard.PasteboardType(rawValue: "kKMPDFViewOutlineDragDataType")], owner: self)
  708. pasteboard.setData(indexSetData, forType: NSPasteboard.PasteboardType(rawValue: NSPasteboard.PasteboardType.RawValue("kKMPDFViewOutlineDragDataType")))
  709. return true
  710. }
  711. func BOTAOutlineView(_ outlineView: KMBOTAOutlineView, validateDrop info: NSDraggingInfo, proposedItem item: Any?, proposedChildIndex index: Int) -> NSDragOperation {
  712. var dragOperation = NSDragOperation.init(rawValue: 0)
  713. if index >= 0 {
  714. dragOperation = NSDragOperation.move
  715. }
  716. return dragOperation
  717. }
  718. func BOTAOutlineView(_ outlineView: KMBOTAOutlineView, acceptDrop info: NSDraggingInfo, item: Any?, childIndex index: Int) -> Bool {
  719. guard let dragOutlineItem = self.dragPDFOutline else { return false }
  720. let outlineItem: KMBOTAOutlineItem = (item ?? KMBOTAOutlineItem()) as! KMBOTAOutlineItem
  721. if index < 0 {
  722. return false
  723. }
  724. for dragOutlineItem in dragPDFOutlines_ {
  725. if outlineItem.parent == nil {
  726. var root = dragOutlineItem.parent
  727. while root?.parent?.children != nil {
  728. root = root?.parent!
  729. }
  730. if dragOutlineItem.parent!.isEqual(root) {
  731. if dragOutlineItem.outline.index > index {
  732. self.moveOutline(outlineItem: dragOutlineItem, index: index, parent: root)
  733. } else {
  734. self.moveOutline(outlineItem: dragOutlineItem, index: index - 1, parent: root)
  735. }
  736. } else {
  737. self.moveOutline(outlineItem: dragOutlineItem, index: index, parent: root)
  738. }
  739. } else {
  740. if dragOutlineItem.parent!.isEqual(item) {
  741. // if dragOutlineItem.outline.index != 0 {
  742. if dragOutlineItem.outline.index > index {
  743. self.moveOutline(outlineItem: dragOutlineItem, index: index, parent: item as? KMBOTAOutlineItem)
  744. } else {
  745. self.moveOutline(outlineItem: dragOutlineItem, index: index - 1, parent: item as? KMBOTAOutlineItem)
  746. }
  747. // } else {
  748. // return false
  749. // }
  750. } else {
  751. var tOutline = outlineItem
  752. var isContains = false
  753. while (tOutline.parent != nil) {
  754. if tOutline.outline.isEqual(dragOutlineItem.outline) {
  755. isContains = true
  756. break
  757. }
  758. tOutline = tOutline.parent!
  759. }
  760. if isContains == false {
  761. self.moveOutline(outlineItem: dragOutlineItem, index: index, parent: item as? KMBOTAOutlineItem)
  762. }
  763. }
  764. }
  765. }
  766. self.BOTAOutlineView.selectItem(outlineItem: dragOutlineItem)
  767. return true
  768. }
  769. }
  770. //MARK: - NSTextFieldDelegate
  771. extension KMOutlineViewController: NSTextFieldDelegate {
  772. func controlTextDidEndEditing(_ obj: Notification) {
  773. if (self.renameTextField.isEqual(obj.object)) {
  774. let textField : NSTextField = obj.object as! NSTextField
  775. self.renamePDFOutline(outlineItem: self.renamePDFOutline, label: textField.stringValue)
  776. }
  777. }
  778. }
  779. //MARK: - NSPopoverDelegate
  780. extension KMOutlineViewController: NSPopoverDelegate {
  781. func popoverWillClose(_ notification: Notification) {
  782. let popover : NSPopover = notification.object as! NSPopover
  783. if popover.contentViewController!.isKind(of: KMOutlineEditViewController.self) {
  784. }
  785. }
  786. func popoverDidClose(_ notification: Notification) {
  787. if popover_ == (notification.object as? NSPopover) {
  788. popover_ = nil
  789. }
  790. }
  791. }
  792. //MARK: - NSMenuItemValidation
  793. extension KMOutlineViewController: NSMenuDelegate, NSMenuItemValidation {
  794. func validateMenuItem(_ menuItem: NSMenuItem) -> Bool {
  795. let action = menuItem.action
  796. if (action == #selector(undo)) {
  797. return handdler.canUndo()
  798. }
  799. if (action == #selector(redo)) {
  800. return handdler.canRedo()
  801. }
  802. return true
  803. }
  804. }
  805. // MARK: - KMNOutlineHanddlerDelegate
  806. extension KMOutlineViewController: KMNOutlineHanddlerDelegate {
  807. func handdler(_ handdler: KMNOutlineHanddler, didAdd info: [String : Any]?) {
  808. let tempOutlineItems = info?["data"] as? [KMBOTAOutlineItem] ?? []
  809. let tempOutlineView = self.BOTAOutlineView
  810. if tempOutlineView?.data?.children.count == 0 || tempOutlineView?.data == nil {
  811. tempOutlineView?.inputData = self.handdler.outlineRoot()
  812. } else {
  813. if BOTAOutlineView.isValidSearchMode() {
  814. BOTAOutlineView.reloadSearchChildren(item: BOTAOutlineView.data)
  815. }
  816. tempOutlineView?.outlineView.reloadData()
  817. }
  818. //展开
  819. // DispatchQueue.main.async {
  820. for outlineItem in tempOutlineItems {
  821. var tempParent = outlineItem
  822. while tempParent.parent != nil {
  823. tempParent.isItemExpanded = true
  824. tempParent = tempParent.parent!
  825. tempOutlineView?.outlineView.expandItem(tempParent)
  826. }
  827. tempOutlineView?.outlineView.expandItem(tempParent.parent)
  828. }
  829. // }
  830. self.updateExtempViewState()
  831. }
  832. func handdler(_ handdler: KMNOutlineHanddler, didRemove info: [String : Any]?) {
  833. let tempOutlineItems = info?["data"] as? [KMBOTAOutlineItem] ?? []
  834. let tempOutlineView = self.BOTAOutlineView
  835. //展开
  836. for outlineItem in tempOutlineItems {
  837. outlineItem.parent?.isItemExpanded = true
  838. tempOutlineView?.outlineView.expandItem(outlineItem.parent)
  839. }
  840. if BOTAOutlineView.isValidSearchMode() {
  841. BOTAOutlineView.reloadSearchChildren(item: BOTAOutlineView.data)
  842. }
  843. tempOutlineView?.outlineView.reloadData()
  844. //删除需要取消选中
  845. tempOutlineView?.cancelSelect()
  846. //刷新nil数据
  847. self.updateExtempViewState()
  848. }
  849. func handdler(_ handdler: KMNOutlineHanddler, didRename outline: CPDFOutline?, info: [String : Any]?) {
  850. let outlineItem = info?["data"] as? KMBOTAOutlineItem
  851. let tempOutlineView = self.BOTAOutlineView
  852. tempOutlineView?.outlineView.reloadItem(outlineItem)
  853. }
  854. func handdler(_ handdler: KMNOutlineHanddler, didChangeLocation outline: CPDFOutline?, info: [String : Any]?) {
  855. let outlineItem = info?["data"] as? KMBOTAOutlineItem
  856. let tempOutlineView = self.BOTAOutlineView
  857. tempOutlineView?.outlineView.reloadItem(outlineItem)
  858. }
  859. func handdler(_ handdler: KMNOutlineHanddler, didMove outline: CPDFOutline?, info: [String : Any]?) {
  860. guard let outlineItem = info?["data"] as? KMBOTAOutlineItem else {
  861. return
  862. }
  863. let parent = info?["parent"] as? KMBOTAOutlineItem
  864. let tempOutlineView = self.BOTAOutlineView
  865. let index = info?["index"] as? Int ?? 0
  866. if BOTAOutlineView.isValidSearchMode() {
  867. BOTAOutlineView.reloadSearchChildren(item: BOTAOutlineView.data)
  868. }
  869. //显示数据刷新
  870. outlineItem.parent?.children.removeObject(outlineItem)
  871. parent?.children.insert(outlineItem, at: index)
  872. outlineItem.parent = parent
  873. tempOutlineView?.outlineView.reloadData()
  874. tempOutlineView?.cancelSelect()
  875. //展开
  876. outlineItem.isItemExpanded = true
  877. outlineItem.parent?.isItemExpanded = true
  878. tempOutlineView?.outlineView.expandItem(outlineItem)
  879. tempOutlineView?.outlineView.expandItem(outlineItem.parent)
  880. }
  881. }
  882. //MARK: - 快捷键
  883. extension KMOutlineViewController {
  884. @IBAction func delete(_ sender: Any) {
  885. self.deleteItemAction()
  886. }
  887. }
  888. //MARK: - undoRedo
  889. extension KMOutlineViewController {
  890. func moveOutline(outlineItem: KMBOTAOutlineItem, index: NSInteger, parent: KMBOTAOutlineItem!) {
  891. handdler.moveOutline(outlineItem: outlineItem, index: index, parent: parent)
  892. }
  893. func changeLocation(outlineItem: KMBOTAOutlineItem, destination: CPDFDestination) {
  894. handdler.changeLocation(outlineItem: outlineItem, destination: destination)
  895. }
  896. func renamePDFOutline(outlineItem: KMBOTAOutlineItem!, label: String) {
  897. let tempOutlineView = self.BOTAOutlineView!
  898. self.view.window?.makeFirstResponder(tempOutlineView.outlineView)
  899. self.renameTextField.isEditable = false
  900. if outlineItem.outline.label == label {
  901. return
  902. }
  903. handdler.renamePDFOutline(outlineItem: outlineItem, label: label)
  904. }
  905. func deleteOutline(outlineItems: [KMBOTAOutlineItem]) {
  906. NSApp.mainWindow?.makeFirstResponder(self.BOTAOutlineView)
  907. let tempOutlineView = self.BOTAOutlineView!
  908. handdler.deleteOutline(outlineItems: outlineItems)
  909. }
  910. func addOutline(outlineItems: [KMBOTAOutlineItem]) {
  911. NSApp.mainWindow?.makeFirstResponder(self.BOTAOutlineView)
  912. let tempOutlineView = self.BOTAOutlineView!
  913. //先取消选中
  914. tempOutlineView.cancelSelect()
  915. var tempOutlineItems: [KMBOTAOutlineItem] = outlineItems
  916. tempOutlineItems.sort(){$0.toIndex < $1.toIndex}
  917. handdler.addOutline(outlineItems: tempOutlineItems)
  918. }
  919. @IBAction func undo(_ sender: Any) {
  920. handdler.undo()
  921. }
  922. @IBAction func redo(_ sender: Any) {
  923. handdler.redo()
  924. }
  925. }
  926. //MARK: - ComponentDropdownDelegate
  927. extension KMOutlineViewController: ComponentDropdownDelegate {
  928. func componentDropdownDidShowMenuItem(dropdown: ComponentDropdown) {
  929. showGroupView()
  930. }
  931. }
  932. //MARK: - ComponentGroupDelegate
  933. extension KMOutlineViewController: ComponentGroupDelegate {
  934. func componentGroupDidDismiss(group: ComponentGroup?) {
  935. if group == groupView_ {
  936. removeGroupView()
  937. } else if group == menuGroupView_ {
  938. group?.removeFromSuperview()
  939. menuGroupView_ = nil
  940. } else if group == searchGroupView {
  941. searchGroupTarget?.properties.state = .normal
  942. searchGroupTarget?.reloadData()
  943. searchGroupTarget = nil
  944. }
  945. }
  946. func componentGroupDidSelect(group: ComponentGroup?, menuItemProperty: ComponentMenuitemProperty?) {
  947. if group == groupView_ {
  948. if let selItem = menuItemProperty {
  949. let index = group?.menuItemArr.firstIndex(of: selItem)
  950. if index == 0 {
  951. expandAllComments(item: NSMenuItem())
  952. } else if index == 1 {
  953. collapseAllComments(item: NSMenuItem())
  954. } else if index == 2 {
  955. removeAllOutlineItem(item: NSMenuItem())
  956. }
  957. }
  958. } else if group == menuGroupView_ {
  959. if let selItem = menuItemProperty {
  960. let index = group?.menuItemArr.firstIndex(of: selItem)
  961. if index == 0 {
  962. addItemAction()
  963. } else if index == 1 {
  964. addChildItemAction()
  965. } else if index == 2 {
  966. addHigherItemAction()
  967. } else if index == 4 {
  968. deleteItemAction()
  969. } else if index == 6 {
  970. group?.removeFromSuperview()
  971. editItemAction()
  972. } else if index == 7 {
  973. renameItemAction()
  974. } else if index == 8 {
  975. changeItemAction()
  976. } else if index == 10 {
  977. promoteItemAction()
  978. } else if index == 11 {
  979. demoteItemAction()
  980. }
  981. group?.removeFromSuperview()
  982. }
  983. } else if group == searchGroupView {
  984. guard let menuI = menuItemProperty else {
  985. return
  986. }
  987. let idx = group?.menuItemArr.firstIndex(of: menuI)
  988. if idx == 0 {
  989. let key = KMNSearchKey.wholeWords.outline
  990. let value = KMDataManager.ud_bool(forKey: key)
  991. KMDataManager.ud_set(!value, forKey: key)
  992. BOTAOutlineView.wholeWords = !value
  993. } else if idx == 1 {
  994. let key = KMNSearchKey.caseSensitive.outline
  995. let value = KMDataManager.ud_bool(forKey: key)
  996. KMDataManager.ud_set(!value, forKey: key)
  997. BOTAOutlineView.caseSensitive = !value
  998. }
  999. }
  1000. }
  1001. }