KMBatesManager.swift 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545
  1. //
  2. // KMBatesManager.swift
  3. // PDF Reader Pro
  4. //
  5. // Created by tangchao on 2022/12/28.
  6. //
  7. import Cocoa
  8. let KMBatesDatasUpdateNotiName = NSNotification.Name("KMBatesDatasUpdateNotiName")
  9. class KMBatesManager: NSObject {
  10. let kBatesFolderPath = NSSearchPathForDirectoriesInDomains(FileManager.SearchPathDirectory.applicationSupportDirectory, FileManager.SearchPathDomainMask.userDomainMask, true).last?.stringByAppendingPathComponent(Bundle.main.bundleIdentifier!).stringByAppendingPathComponent("bates")
  11. let kBatesPlistPath = NSSearchPathForDirectoriesInDomains(FileManager.SearchPathDirectory.applicationSupportDirectory, FileManager.SearchPathDomainMask.userDomainMask, true).last?.stringByAppendingPathComponent(Bundle.main.bundleIdentifier!).stringByAppendingPathComponent("bates").stringByAppendingPathComponent("bates.plist")
  12. static let defaultManager = KMBatesManager()
  13. var datas: Array<KMBatesModel> = []
  14. var defaultModel: KMBatesModel = KMBatesModel()
  15. override init() {
  16. super.init()
  17. #if DEBUG
  18. print("kBatesPlistPath = \(kBatesPlistPath ?? "")")
  19. #endif
  20. if (FileManager.default.fileExists(atPath: kBatesPlistPath!)) {
  21. let dataDict = NSDictionary(contentsOfFile: kBatesPlistPath!)
  22. if (dataDict == nil) {
  23. return
  24. }
  25. for keyIndex in 0 ..< (dataDict?.allKeys.count)! {
  26. let key: String = dataDict?.allKeys[keyIndex] as! String
  27. let modelDict: NSDictionary = dataDict?.object(forKey: key) as! NSDictionary
  28. let model = parseDictionary(dict: modelDict)
  29. model.tag = key
  30. self.datas.append(model)
  31. }
  32. /// 根据id进行排序(升序)
  33. self.datas.sort(){$0.tag > $1.tag}
  34. }
  35. defaultModel.name = self.fetchBatesAvailableName()
  36. }
  37. //MARK: - Get
  38. func fetchBatesAvailableName() -> String {
  39. var availableIndex = 1
  40. for item in datas {
  41. if item.name.hasPrefix("Bates-") {
  42. if let index = Int(item.name.dropFirst("Bates-".count)), index >= availableIndex {
  43. availableIndex = index + 1
  44. }
  45. }
  46. }
  47. return "Bates-\(availableIndex)"
  48. }
  49. //MARK: - 增删改查
  50. func addTemplate(_ model: KMBatesModel) -> Bool {
  51. if (!FileManager.default.fileExists(atPath: kBatesFolderPath!)) {
  52. let create: ()? = try?FileManager.default.createDirectory(atPath: kBatesFolderPath!, withIntermediateDirectories: false)
  53. if (create == nil) {
  54. return false
  55. }
  56. }
  57. if (!FileManager.default.fileExists(atPath: kBatesPlistPath!)) {
  58. let create = try?FileManager.default.createFile(atPath: kBatesPlistPath!, contents: nil)
  59. if (create == nil) {
  60. return false
  61. }
  62. }
  63. let dict = NSDictionary(contentsOfFile: kBatesPlistPath!)
  64. var newDict:NSMutableDictionary!
  65. if (dict != nil) {
  66. newDict = NSMutableDictionary(dictionary: dict!)
  67. } else {
  68. newDict = NSMutableDictionary()
  69. }
  70. let modelDict = self.parseModel(model: model)
  71. let tag = model.tag
  72. newDict.addEntries(from: [tag : modelDict])
  73. let result = newDict.write(toFile: kBatesPlistPath!, atomically: true)
  74. if (result) {
  75. if (self.datas.count < 1) {
  76. self.datas.append(model)
  77. } else {
  78. self.datas.insert(model, at: 0)
  79. }
  80. }
  81. NotificationCenter.default.post(name: KMBatesDatasUpdateNotiName, object: nil)
  82. return result
  83. }
  84. func deleteTemplate(_ model: KMBatesModel) -> Bool {
  85. if (model.tag.isEmpty) {
  86. return false
  87. }
  88. if (!FileManager.default.fileExists(atPath: kBatesPlistPath!)) {
  89. return false
  90. }
  91. let key: String = model.tag
  92. let dictionary = NSDictionary(contentsOfFile: kBatesPlistPath!)
  93. var newDictionary: NSMutableDictionary!
  94. if (dictionary != nil) {
  95. newDictionary = NSMutableDictionary(dictionary: dictionary!)
  96. } else {
  97. newDictionary = NSMutableDictionary()
  98. }
  99. newDictionary.removeObject(forKey: key)
  100. let result = newDictionary.write(toFile: kBatesPlistPath!, atomically: true)
  101. if (result) {
  102. if (self.datas.contains(model)) {
  103. self.datas.removeObject(model)
  104. }
  105. }
  106. NotificationCenter.default.post(name: KMBatesDatasUpdateNotiName, object: nil)
  107. return result
  108. }
  109. func deleteAllTemplate() -> Bool {
  110. if (!FileManager.default.fileExists(atPath: kBatesPlistPath!)) {
  111. return false
  112. }
  113. let dictionary = NSDictionary(contentsOfFile: kBatesPlistPath!)
  114. var newDictionary: NSMutableDictionary!
  115. if (dictionary != nil) {
  116. newDictionary = NSMutableDictionary(dictionary: dictionary!)
  117. } else {
  118. newDictionary = NSMutableDictionary()
  119. }
  120. newDictionary.removeAllObjects()
  121. let result = newDictionary.write(toFile: kBatesPlistPath!, atomically: true)
  122. if (result) {
  123. self.datas.removeAll()
  124. }
  125. NotificationCenter.default.post(name: KMBatesDatasUpdateNotiName, object: nil)
  126. return result
  127. }
  128. func updateTemplate(_ model: KMBatesModel) -> Bool {
  129. if (!FileManager.default.fileExists(atPath: kBatesFolderPath!)) {
  130. let create = try?FileManager.default.createDirectory(atPath: kBatesFolderPath!, withIntermediateDirectories: false)
  131. if (create == nil) {
  132. return false
  133. }
  134. }
  135. if (!FileManager.default.fileExists(atPath: kBatesPlistPath!)) {
  136. let create = try?FileManager.default.createFile(atPath: kBatesPlistPath!, contents: nil)
  137. if (create == nil) {
  138. return false
  139. }
  140. }
  141. var flagModel: KMBatesModel!
  142. for model_ in self.datas {
  143. if (model_.tag == model.tag) {
  144. flagModel = model_
  145. break
  146. }
  147. }
  148. if (flagModel == nil) {
  149. return false
  150. }
  151. let dict = NSDictionary(contentsOfFile: kBatesPlistPath!)
  152. var newDict:NSMutableDictionary!
  153. if (dict != nil) {
  154. newDict = NSMutableDictionary(dictionary: dict!)
  155. } else {
  156. newDict = NSMutableDictionary()
  157. }
  158. let modelDict = self.parseModel(model: model)
  159. newDict.setObject(modelDict, forKey: flagModel.tag as NSCopying)
  160. let result = newDict.write(toFile: kBatesPlistPath!, atomically: true)
  161. if (result) {
  162. if let index = self.datas.firstIndex(of: flagModel) {
  163. self.datas[index] = model
  164. }
  165. }
  166. NotificationCenter.default.post(name: KMBatesDatasUpdateNotiName, object: nil)
  167. return result
  168. }
  169. func updateModel(_ model: KMBatesModel, with dict: NSDictionary) {
  170. /// 字体相关
  171. model.fontName = dict["fontName"] as! String
  172. model.fontsize = dict["fontsize"] as! CGFloat
  173. if let value = dict.object(forKey: "color") {
  174. model.color = NSColor.km_init(hex: value as! String)
  175. }
  176. /// 页边距
  177. model.leftMargin = dict["leftMargin"] as! Int
  178. model.rightMargin = dict["rightMargin"] as! Int
  179. model.bottomMargin = dict["bottomMargin"] as! Int
  180. model.topMargin = dict["topMargin"] as! Int
  181. /// 内容
  182. model.topLeftString = dict["topLeftString"] as! String
  183. model.topCenterString = dict["topCenterString"] as! String
  184. model.topRightString = dict["topRightString"] as! String
  185. model.bottomLeftString = dict["bottomLeftString"] as! String
  186. model.bottomCenterString = dict["bottomCenterString"] as! String
  187. model.bottomRightString = dict["bottomRightString"] as! String
  188. model.prefixString = dict["prefixString"] as! String
  189. model.suffixString = dict["suffixString"] as! String
  190. model.digits = dict["digits"] as! Int
  191. model.startString = dict["startString"] as! String
  192. model.tag = dict["tag"] as! String
  193. if let value = dict.object(forKey: "name") {
  194. model.name = value as! String
  195. }
  196. NotificationCenter.default.post(name: KMBatesDatasUpdateNotiName, object: nil)
  197. }
  198. func updateCPDFBates(_ pdfBates: CPDFBates, withModel model: KMBatesModel, _ totalPDFCount: Int) {
  199. let fontSize = model.fontsize
  200. let fontName: String = model.fontName
  201. let font = NSFont.boldSystemFont(ofSize:fontSize)
  202. let style = NSMutableParagraphStyle()
  203. style.alignment = .center
  204. style.lineBreakMode = .byCharWrapping
  205. let size: NSSize = "text".boundingRect(with: NSSize(width: 1000, height: 1000), options: NSString.DrawingOptions(rawValue: 3), attributes: [NSAttributedString.Key.font : font, NSAttributedString.Key.paragraphStyle : style]).size
  206. pdfBates.margin = NSEdgeInsetsMake(max(CGFloat(model.topMargin)-size.height, 0), CGFloat(model.leftMargin), max(CGFloat(model.bottomMargin)-size.height, 0), CGFloat(model.rightMargin))
  207. let strings = KMBatesManager.parseModel(model: model, UInt(totalPDFCount))
  208. var count: Int = 0
  209. let color = model.color
  210. for text in strings {
  211. pdfBates.setText(text, at: UInt(count))
  212. pdfBates.setTextColor(color, at: UInt(count))
  213. pdfBates.setFontSize(fontSize, at: UInt(count))
  214. pdfBates.setFontName(fontName, at: UInt(count))
  215. count += 1
  216. }
  217. }
  218. //MARK: - Parse
  219. func parseModel(model: KMBatesModel) -> Dictionary<String, Any> {
  220. var dict: [String : Any] = [:]
  221. /// 字体相关
  222. dict["fontName"] = model.fontName
  223. dict["fontsize"] = model.fontsize
  224. dict["color"] = model.color.toHex()
  225. /// 页边距
  226. dict["leftMargin"] = model.leftMargin
  227. dict["rightMargin"] = model.rightMargin
  228. dict["bottomMargin"] = model.bottomMargin
  229. dict["topMargin"] = model.topMargin
  230. /// 内容
  231. dict["topLeftString"] = model.topLeftString
  232. dict["topCenterString"] = model.topCenterString
  233. dict["topRightString"] = model.topRightString
  234. dict["bottomLeftString"] = model.bottomLeftString
  235. dict["bottomCenterString"] = model.bottomCenterString
  236. dict["bottomRightString"] = model.bottomRightString
  237. dict["prefixString"] = model.prefixString
  238. dict["suffixString"] = model.suffixString
  239. dict["digits"] = model.digits
  240. dict["startString"] = model.startString
  241. dict["tag"] = model.tag
  242. dict["name"] = model.name
  243. return dict
  244. }
  245. private func parseDictionary(dict: NSDictionary) -> KMBatesModel {
  246. let model = KMBatesModel()
  247. /// 字体相关
  248. model.fontName = dict["fontName"] as! String
  249. model.fontsize = dict["fontsize"] as! CGFloat
  250. if let value = dict.object(forKey: "color") {
  251. model.color = NSColor.km_init(hex: value as! String)
  252. }
  253. /// 页边距
  254. model.leftMargin = dict["leftMargin"] as! Int
  255. model.rightMargin = dict["rightMargin"] as! Int
  256. model.bottomMargin = dict["bottomMargin"] as! Int
  257. model.topMargin = dict["topMargin"] as! Int
  258. /// 内容
  259. model.topLeftString = dict["topLeftString"] as! String
  260. model.topCenterString = dict["topCenterString"] as! String
  261. model.topRightString = dict["topRightString"] as! String
  262. model.bottomLeftString = dict["bottomLeftString"] as! String
  263. model.bottomCenterString = dict["bottomCenterString"] as! String
  264. model.bottomRightString = dict["bottomRightString"] as! String
  265. model.prefixString = dict["prefixString"] as! String
  266. model.suffixString = dict["suffixString"] as! String
  267. model.digits = dict["digits"] as! Int
  268. model.startString = dict["startString"] as! String
  269. if let value = dict["tag"] {
  270. model.tag = value as! String
  271. }
  272. if let value = dict["name"] {
  273. model.name = value as! String
  274. }
  275. return model
  276. }
  277. //MARK: - Compare
  278. class func compareIsChangedModel(_ model: KMBatesModel, withDict dict: NSDictionary) -> Bool {
  279. if let value = dict["fontName"] {
  280. if model.fontName != (value as! String) {
  281. return true
  282. }
  283. }
  284. if let value = dict["fontsize"] {
  285. if model.fontsize != (value as! CGFloat) {
  286. return true
  287. }
  288. }
  289. if let value = dict["color"] {
  290. if model.color.toHex() != (value as! String) {
  291. return true
  292. }
  293. }
  294. if let value = dict["leftMargin"] {
  295. if model.leftMargin != (value as! Int) {
  296. return true
  297. }
  298. }
  299. if let value = dict["rightMargin"] {
  300. if model.rightMargin != (value as! Int) {
  301. return true
  302. }
  303. }
  304. if let value = dict["bottomMargin"] {
  305. if model.bottomMargin != (value as! Int) {
  306. return true
  307. }
  308. }
  309. if let value = dict["topMargin"] {
  310. if model.topMargin != (value as! Int) {
  311. return true
  312. }
  313. }
  314. if let value = dict["topLeftString"] {
  315. if model.topLeftString != (value as! String) {
  316. return true
  317. }
  318. }
  319. if let value = dict["topCenterString"] {
  320. if model.topCenterString != (value as! String) {
  321. return true
  322. }
  323. }
  324. if let value = dict["topRightString"] {
  325. if model.topRightString != (value as! String) {
  326. return true
  327. }
  328. }
  329. if let value = dict["bottomLeftString"] {
  330. if model.bottomLeftString != (value as! String) {
  331. return true
  332. }
  333. }
  334. if let value = dict["bottomCenterString"] {
  335. if model.bottomCenterString != (value as! String) {
  336. return true
  337. }
  338. }
  339. if let value = dict["bottomRightString"] {
  340. if model.bottomRightString != (value as! String) {
  341. return true
  342. }
  343. }
  344. if let value = dict["prefixString"] {
  345. if model.prefixString != (value as! String) {
  346. return true
  347. }
  348. }
  349. if let value = dict["suffixString"] {
  350. if model.suffixString != (value as! String) {
  351. return true
  352. }
  353. }
  354. if let value = dict["digits"] {
  355. if model.digits != (value as! Int) {
  356. return true
  357. }
  358. }
  359. if let value = dict["startString"] {
  360. if model.startString != (value as! String) {
  361. return true
  362. }
  363. }
  364. return false
  365. }
  366. }
  367. //Class
  368. extension KMBatesManager {
  369. class func parseModel(model: KMBatesModel, _ pageCount: UInt) -> [String] {
  370. var topLeftString: String = ""
  371. if (!model.topLeftString.isEmpty) {
  372. var string = KMBatesManager.parsePageFormat(formatString: model.topLeftString, startPage: model.startString, pageCount: "\(pageCount)")
  373. string = KMBatesManager.parseDateFormat(formatString: string)
  374. topLeftString = string
  375. }
  376. var topCenterString: String = ""
  377. if (!model.topCenterString.isEmpty) {
  378. var string = KMBatesManager.parsePageFormat(formatString: model.topCenterString, startPage: model.startString, pageCount: "\(pageCount)")
  379. string = KMBatesManager.parseDateFormat(formatString: string)
  380. topCenterString = string
  381. }
  382. var topRightString: String = ""
  383. if (!model.topRightString.isEmpty) {
  384. var string = KMBatesManager.parsePageFormat(formatString: model.topRightString, startPage: model.startString, pageCount: "\(pageCount)")
  385. string = KMBatesManager.parseDateFormat(formatString: string)
  386. topRightString = string
  387. }
  388. var bottomLeftString: String = ""
  389. if (!model.bottomLeftString.isEmpty) {
  390. var string = KMBatesManager.parsePageFormat(formatString: model.bottomLeftString, startPage: model.startString, pageCount: "\(pageCount)")
  391. string = KMBatesManager.parseDateFormat(formatString: string)
  392. bottomLeftString = string
  393. }
  394. var bottomCenterString: String = ""
  395. if (!model.bottomCenterString.isEmpty) {
  396. var string = KMBatesManager.parsePageFormat(formatString: model.bottomCenterString, startPage: model.startString, pageCount: "\(pageCount)")
  397. string = KMBatesManager.parseDateFormat(formatString: string)
  398. bottomCenterString = string
  399. }
  400. var bottomRightString: String = ""
  401. if (!model.bottomRightString.isEmpty) {
  402. var string = KMBatesManager.parsePageFormat(formatString: model.bottomRightString, startPage: model.startString, pageCount: "\(pageCount)")
  403. string = KMBatesManager.parseDateFormat(formatString: string)
  404. bottomRightString = string
  405. }
  406. return [topLeftString, topCenterString, topRightString, bottomLeftString, bottomCenterString, bottomRightString]
  407. }
  408. class func parsePageFormat(formatString: String, startPage: String, pageCount: String) -> String {
  409. var result = formatString
  410. for pageFormat in self.getPageFormats() {
  411. let string = "<<\(pageFormat)>>"
  412. if (result.contains(string)) {
  413. var tempString = ""
  414. if (string == "<<1>>") {
  415. tempString.append("<<\(startPage)>>")
  416. } else if (string == "<<1 of n>>") {
  417. tempString.append("<<\(startPage)>>")
  418. tempString.append(" of \(pageCount)")
  419. } else if (string == "<<1/n>>") {
  420. tempString.append("<<\(startPage)>>")
  421. tempString.append("/\(pageCount)")
  422. } else if (string == "<<Page 1>>") {
  423. tempString.append("Page \(startPage)")
  424. } else if (string == "<<Page 1 of n>>") {
  425. tempString.append("Page \(startPage)")
  426. tempString.append("of \(pageCount)")
  427. }
  428. result = result.replacingOccurrences(of: string, with: tempString)
  429. }
  430. }
  431. return result
  432. }
  433. class func parseDateFormat(formatString: String) -> String {
  434. var result: String = formatString
  435. for dateFormat in self.getDateFormats() {
  436. if (result.contains(dateFormat)) {
  437. let formatString: String = dateFormat.replacingOccurrences(of: "m", with: "M")
  438. let replace = "<<\(dateFormat)>>"
  439. let date = Date()
  440. let dateFormatter = DateFormatter()
  441. dateFormatter.dateFormat = formatString
  442. let dateString = dateFormatter.string(from: date)
  443. result = result.replacingOccurrences(of: replace, with: dateString)
  444. }
  445. }
  446. return result
  447. }
  448. class func getPageFormats() -> [String] {
  449. return ["1",
  450. "1 of n",
  451. "1/n",
  452. "Page 1",
  453. "Page 1 of n"]
  454. }
  455. @objc class func getDateFormats() -> [String] {
  456. return ["m/d", "m/d/yy", "m/d/yyyy",
  457. "mm/dd/yy", "mm/dd/yyyy",
  458. "d/m/yy", "d/m/yyyy",
  459. "dd/mm/yy", "dd/mm/yyyy",
  460. "mm/yy", "mm/yyyy",
  461. "m.d.yy", "m.d.yyyy",
  462. "mm.dd.yy", "mm.dd.yyyy", "mm.yy", "mm.yyyy",
  463. "d.m.yy", "d.m.yyyy",
  464. "dd.mm.yy", "dd.mm.yyyy",
  465. "yy-mm-dd",
  466. "yyyy-mm-dd"]
  467. }
  468. }