NCImageCache.swift 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. //
  2. // NCImageCache.swift
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 18/10/23.
  6. // Copyright © 2021 Marino Faggiana. All rights reserved.
  7. //
  8. // Author Marino Faggiana <marino.faggiana@nextcloud.com>
  9. //
  10. // This program is free software: you can redistribute it and/or modify
  11. // it under the terms of the GNU General Public License as published by
  12. // the Free Software Foundation, either version 3 of the License, or
  13. // (at your option) any later version.
  14. //
  15. // This program is distributed in the hope that it will be useful,
  16. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. // GNU General Public License for more details.
  19. //
  20. // You should have received a copy of the GNU General Public License
  21. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  22. //
  23. import UIKit
  24. import LRUCache
  25. import NextcloudKit
  26. import RealmSwift
  27. class NCImageCache: NSObject {
  28. public static let shared: NCImageCache = {
  29. let instance = NCImageCache()
  30. return instance
  31. }()
  32. // MARK: -
  33. private let limitCacheImagePreview: Int = 1000
  34. private var brandElementColor: UIColor?
  35. private var totalSize: Int64 = 0
  36. struct metadataInfo {
  37. var etag: String
  38. var date: NSDate
  39. var width: Int
  40. var height: Int
  41. }
  42. struct imageInfo {
  43. var image: UIImage?
  44. var size: CGSize?
  45. var date: Date
  46. }
  47. private typealias ThumbnailImagePreviewLRUCache = LRUCache<String, imageInfo>
  48. private typealias ThumbnailImageIconLRUCache = LRUCache<String, UIImage>
  49. private typealias ThumbnailSizePreviewLRUCache = LRUCache<String, CGSize?>
  50. private lazy var cacheImagePreview: ThumbnailImagePreviewLRUCache = {
  51. return ThumbnailImagePreviewLRUCache(countLimit: limitCacheImagePreview)
  52. }()
  53. private lazy var cacheImageIcon: ThumbnailImageIconLRUCache = {
  54. return ThumbnailImageIconLRUCache()
  55. }()
  56. private lazy var cacheSizePreview: ThumbnailSizePreviewLRUCache = {
  57. return ThumbnailSizePreviewLRUCache()
  58. }()
  59. private var metadatasInfo: [String: metadataInfo] = [:]
  60. private var metadatas: ThreadSafeArray<tableMetadata>?
  61. var createMediaCacheInProgress: Bool = false
  62. let showAllPredicateMediaString = "account == %@ AND serverUrl BEGINSWITH %@ AND (classFile == '\(NKCommon.TypeClassFile.image.rawValue)' OR classFile == '\(NKCommon.TypeClassFile.video.rawValue)') AND NOT (session CONTAINS[c] 'upload')"
  63. let showBothPredicateMediaString = "account == %@ AND serverUrl BEGINSWITH %@ AND (classFile == '\(NKCommon.TypeClassFile.image.rawValue)' OR classFile == '\(NKCommon.TypeClassFile.video.rawValue)') AND NOT (session CONTAINS[c] 'upload') AND NOT (livePhotoFile != '' AND classFile == '\(NKCommon.TypeClassFile.video.rawValue)')"
  64. let showOnlyPredicateMediaString = "account == %@ AND serverUrl BEGINSWITH %@ AND classFile == %@ AND NOT (session CONTAINS[c] 'upload') AND NOT (livePhotoFile != '' AND classFile == '\(NKCommon.TypeClassFile.video.rawValue)')"
  65. override private init() {}
  66. ///
  67. /// MEDIA CACHE
  68. ///
  69. func createMediaCache(account: String, withCacheSize: Bool) {
  70. if createMediaCacheInProgress {
  71. NextcloudKit.shared.nkCommonInstance.writeLog("[ERROR] ThumbnailLRUCache image process already in progress")
  72. return
  73. }
  74. createMediaCacheInProgress = true
  75. self.metadatasInfo.removeAll()
  76. self.metadatas = nil
  77. self.metadatas = getMediaMetadatas(account: account)
  78. let manager = FileManager.default
  79. let resourceKeys = Set<URLResourceKey>([.nameKey, .pathKey, .fileSizeKey, .creationDateKey])
  80. struct FileInfo {
  81. var path: URL
  82. var ocIdEtag: String
  83. var date: Date
  84. var fileSize: Int
  85. var width: Int
  86. var height: Int
  87. }
  88. var files: [FileInfo] = []
  89. let startDate = Date()
  90. if let metadatas = metadatas {
  91. metadatas.forEach { metadata in
  92. metadatasInfo[metadata.ocId] = metadataInfo(etag: metadata.etag, date: metadata.date, width: metadata.width, height: metadata.height)
  93. }
  94. }
  95. if let enumerator = manager.enumerator(at: URL(fileURLWithPath: NCUtilityFileSystem().directoryProviderStorage), includingPropertiesForKeys: [.isRegularFileKey], options: [.skipsHiddenFiles]) {
  96. for case let fileURL as URL in enumerator where fileURL.lastPathComponent.hasSuffix(NCGlobal.shared.storageExtPreview) {
  97. let fileName = fileURL.lastPathComponent
  98. let ocId = fileURL.deletingLastPathComponent().lastPathComponent
  99. guard let resourceValues = try? fileURL.resourceValues(forKeys: resourceKeys),
  100. let fileSize = resourceValues.fileSize,
  101. fileSize > 0 else { continue }
  102. let width = metadatasInfo[ocId]?.width ?? 0
  103. let height = metadatasInfo[ocId]?.height ?? 0
  104. if withCacheSize {
  105. if let date = metadatasInfo[ocId]?.date,
  106. let etag = metadatasInfo[ocId]?.etag,
  107. fileName == etag + NCGlobal.shared.storageExtPreview {
  108. files.append(FileInfo(path: fileURL, ocIdEtag: ocId + etag, date: date as Date, fileSize: fileSize, width: width, height: height))
  109. } else {
  110. let etag = fileName.replacingOccurrences(of: NCGlobal.shared.storageExtPreview, with: "")
  111. files.append(FileInfo(path: fileURL, ocIdEtag: ocId + etag, date: Date.distantPast, fileSize: fileSize, width: width, height: height))
  112. }
  113. } else if let date = metadatasInfo[ocId]?.date, let etag = metadatasInfo[ocId]?.etag, fileName == etag + NCGlobal.shared.storageExtPreview {
  114. files.append(FileInfo(path: fileURL, ocIdEtag: ocId + etag, date: date as Date, fileSize: fileSize, width: width, height: height))
  115. }
  116. }
  117. }
  118. files.sort(by: { $0.date > $1.date })
  119. if let firstDate = files.first?.date, let lastDate = files.last?.date {
  120. print("First date: \(firstDate)")
  121. print("Last date: \(lastDate)")
  122. }
  123. cacheImagePreview.removeAllValues()
  124. cacheSizePreview.removeAllValues()
  125. var counter: Int = 0
  126. for file in files {
  127. if !withCacheSize, counter > limitCacheImagePreview {
  128. break
  129. }
  130. autoreleasepool {
  131. if let image = UIImage(contentsOfFile: file.path.path) {
  132. if counter < limitCacheImagePreview {
  133. cacheImagePreview.setValue(imageInfo(image: image, size: image.size, date: file.date), forKey: file.ocIdEtag)
  134. totalSize = totalSize + Int64(file.fileSize)
  135. counter += 1
  136. }
  137. if file.width == 0, file.height == 0 {
  138. cacheSizePreview.setValue(image.size, forKey: file.ocIdEtag)
  139. }
  140. }
  141. }
  142. }
  143. let diffDate = Date().timeIntervalSinceReferenceDate - startDate.timeIntervalSinceReferenceDate
  144. NextcloudKit.shared.nkCommonInstance.writeLog("--------- ThumbnailLRUCache image process ---------")
  145. NextcloudKit.shared.nkCommonInstance.writeLog("Counter cache image: \(cacheImagePreview.count)")
  146. NextcloudKit.shared.nkCommonInstance.writeLog("Counter cache size: \(cacheSizePreview.count)")
  147. NextcloudKit.shared.nkCommonInstance.writeLog("Total size images process: " + NCUtilityFileSystem().transformedSize(totalSize))
  148. NextcloudKit.shared.nkCommonInstance.writeLog("Time process: \(diffDate)")
  149. NextcloudKit.shared.nkCommonInstance.writeLog("--------- ThumbnailLRUCache image process ---------")
  150. createMediaCacheInProgress = false
  151. NotificationCenter.default.postOnMainThread(name: NCGlobal.shared.notificationCenterCreateMediaCacheEnded)
  152. }
  153. func initialMetadatas() -> ThreadSafeArray<tableMetadata>? {
  154. defer { self.metadatas = nil }
  155. return self.metadatas
  156. }
  157. func getMediaMetadatas(account: String, predicate: NSPredicate? = nil) -> ThreadSafeArray<tableMetadata>? {
  158. guard let tableAccount = NCManageDatabase.shared.getAccount(predicate: NSPredicate(format: "account == %@", account)) else { return nil }
  159. let startServerUrl = NCUtilityFileSystem().getHomeServer(urlBase: tableAccount.urlBase, userId: tableAccount.userId) + tableAccount.mediaPath
  160. let predicateBoth = NSPredicate(format: showBothPredicateMediaString, account, startServerUrl)
  161. return NCManageDatabase.shared.getMediaMetadatas(predicate: predicate ?? predicateBoth)
  162. }
  163. ///
  164. /// PREVIEW CACHE
  165. ///
  166. func addPreviewImageCache(metadata: tableMetadata, image: UIImage) {
  167. cacheImagePreview.setValue(imageInfo(image: image, size: image.size, date: metadata.date as Date), forKey: metadata.ocId + metadata.etag)
  168. cacheSizePreview.setValue(image.size, forKey: metadata.ocId + metadata.etag)
  169. }
  170. func getPreviewImageCache(ocId: String, etag: String) -> UIImage? {
  171. if let cache = cacheImagePreview.value(forKey: ocId + etag) {
  172. return cache.image
  173. }
  174. return nil
  175. }
  176. ///
  177. /// SIZE CACHE
  178. ///
  179. func getPreviewSizeCache(ocId: String, etag: String) -> CGSize? {
  180. if let size = cacheSizePreview.value(forKey: ocId + etag) {
  181. return size
  182. } else {
  183. if let image = UIImage(contentsOfFile: NCUtilityFileSystem().getDirectoryProviderStoragePreviewOcId(ocId, etag: etag)) {
  184. return image.size
  185. }
  186. }
  187. return nil
  188. }
  189. ///
  190. /// ICON CACHE
  191. ///
  192. func setIconImageCache(ocId: String, etag: String, image: UIImage) {
  193. cacheImageIcon.setValue(image, forKey: ocId + etag)
  194. }
  195. func getIconImageCache(ocId: String, etag: String) -> UIImage? {
  196. return cacheImageIcon.value(forKey: ocId + etag)
  197. }
  198. // MARK: -
  199. struct images {
  200. static var file = UIImage()
  201. static var shared = UIImage()
  202. static var canShare = UIImage()
  203. static var shareByLink = UIImage()
  204. static var favorite = UIImage()
  205. static var comment = UIImage()
  206. static var livePhoto = UIImage()
  207. static var offlineFlag = UIImage()
  208. static var local = UIImage()
  209. static var folderEncrypted = UIImage()
  210. static var folderSharedWithMe = UIImage()
  211. static var folderPublic = UIImage()
  212. static var folderGroup = UIImage()
  213. static var folderExternal = UIImage()
  214. static var folderAutomaticUpload = UIImage()
  215. static var folder = UIImage()
  216. static var checkedYes = UIImage()
  217. static var checkedNo = UIImage()
  218. static var buttonMore = UIImage()
  219. static var buttonStop = UIImage()
  220. static var buttonMoreLock = UIImage()
  221. static var iconContacts = UIImage()
  222. static var iconTalk = UIImage()
  223. static var iconCalendar = UIImage()
  224. static var iconDeck = UIImage()
  225. static var iconMail = UIImage()
  226. static var iconConfirm = UIImage()
  227. static var iconPages = UIImage()
  228. static var iconFile = UIImage()
  229. }
  230. func createImagesCache() {
  231. let utility = NCUtility()
  232. images.file = utility.loadImage(named: "doc", colors: [NCBrandColor.shared.iconImageColor2])
  233. images.shared = utility.loadImage(named: "person.fill.badge.plus", colors: NCBrandColor.shared.iconImageMultiColors)
  234. images.canShare = utility.loadImage(named: "person.fill.badge.plus", colors: NCBrandColor.shared.iconImageMultiColors)
  235. images.shareByLink = utility.loadImage(named: "link", colors: [NCBrandColor.shared.iconImageColor])
  236. images.favorite = utility.loadImage(named: "star.fill", colors: [NCBrandColor.shared.yellowFavorite])
  237. images.livePhoto = utility.loadImage(named: "livephoto", colors: [NCBrandColor.shared.iconImageColor])
  238. images.offlineFlag = utility.loadImage(named: "arrow.down.circle.fill", colors: [.systemGreen])
  239. images.local = utility.loadImage(named: "checkmark.circle.fill", colors: [.systemGreen])
  240. images.checkedYes = utility.loadImage(named: "checkmark.circle.fill", colors: [NCBrandColor.shared.brandElement])
  241. images.checkedNo = utility.loadImage(named: "circle", colors: [NCBrandColor.shared.brandElement])
  242. images.buttonMore = utility.loadImage(named: "ellipsis", colors: [NCBrandColor.shared.iconImageColor])
  243. images.buttonStop = utility.loadImage(named: "stop.circle", colors: [NCBrandColor.shared.iconImageColor])
  244. images.buttonMoreLock = utility.loadImage(named: "lock.fill", colors: [NCBrandColor.shared.iconImageColor])
  245. createImagesBrandCache()
  246. }
  247. func createImagesBrandCache() {
  248. let brandElement = NCBrandColor.shared.brandElement
  249. guard brandElement != self.brandElementColor else { return }
  250. self.brandElementColor = brandElement
  251. let utility = NCUtility()
  252. images.folderEncrypted = UIImage(named: "folderEncrypted")!.image(color: brandElement)
  253. images.folderSharedWithMe = UIImage(named: "folder_shared_with_me")!.image(color: brandElement)
  254. images.folderPublic = UIImage(named: "folder_public")!.image(color: brandElement)
  255. images.folderGroup = UIImage(named: "folder_group")!.image(color: brandElement)
  256. images.folderExternal = UIImage(named: "folder_external")!.image(color: brandElement)
  257. images.folderAutomaticUpload = UIImage(named: "folderAutomaticUpload")!.image(color: brandElement)
  258. images.folder = UIImage(named: "folder")!.image(color: brandElement)
  259. images.iconContacts = utility.loadImage(named: "person.crop.rectangle.stack", colors: [NCBrandColor.shared.iconImageColor])
  260. images.iconTalk = UIImage(named: "talk-template")!.image(color: brandElement)
  261. images.iconCalendar = utility.loadImage(named: "calendar", colors: [NCBrandColor.shared.iconImageColor])
  262. images.iconDeck = utility.loadImage(named: "square.stack.fill", colors: [NCBrandColor.shared.iconImageColor])
  263. images.iconMail = utility.loadImage(named: "mail", colors: [NCBrandColor.shared.iconImageColor])
  264. images.iconConfirm = utility.loadImage(named: "arrow.right", colors: [NCBrandColor.shared.iconImageColor])
  265. images.iconPages = utility.loadImage(named: "doc.richtext", colors: [NCBrandColor.shared.iconImageColor])
  266. images.iconFile = utility.loadImage(named: "doc", colors: [NCBrandColor.shared.iconImageColor])
  267. NotificationCenter.default.postOnMainThread(name: NCGlobal.shared.notificationCenterChangeTheming)
  268. }
  269. }