KMBatesManager.swift 20 KB

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