summaryrefslogtreecommitdiff
path: root/Sora/Data/ImageCacheManager.swift
blob: cd8b4e265fadea3baf2b243488798e686375db7f (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
import Combine
import SwiftUI

@MainActor
final class ImageCacheManager {
  // MARK: - Singleton
  static let shared = ImageCacheManager()

  // MARK: - Private Properties
  private let cache = URLCache(
    memoryCapacity: 100 * 1_024 * 1_024,  // 100 MB
    diskCapacity: 500 * 1_024 * 1_024,  // 500 MB
    directory: FileManager.default.urls(for: .cachesDirectory, in: .userDomainMask).first?
      .appendingPathComponent("SoraImageCache", isDirectory: true)
  )
  private var cancellables = Set<AnyCancellable>()
  private let downloadQueue = OperationQueue()
  private var preloadingURLs = Set<URL>()

  // MARK: - Initialisation
  private init() {
    downloadQueue.maxConcurrentOperationCount = 5
    downloadQueue.qualityOfService = .utility
  }

  // MARK: - Public Methods
  func preloadImages(_ urls: [URL]) {
    let newURLs = urls.filter { !preloadingURLs.contains($0) }

    for url in newURLs {
      preloadingURLs.insert(url)

      downloadQueue.addOperation {
        let cancellable = URLSession.shared.dataTaskPublisher(for: url)
          .map { CachedURLResponse(response: $0.response, data: $0.data) }
          .sink(
            receiveCompletion: { _ in
              DispatchQueue.main.async { [weak self] in
                self?.preloadingURLs.remove(url)
              }
            },
            receiveValue: { [weak self] cachedResponse in
              self?.cache.storeCachedResponse(cachedResponse, for: URLRequest(url: url))
            }
          )

        DispatchQueue.main.async { [weak self] in
          self?.cancellables.insert(cancellable)
        }
      }
    }
  }

  func clearCache() {
    cache.removeAllCachedResponses()
    cancellables.removeAll()
    preloadingURLs.removeAll()
  }

  func getCachedResponse(for url: URL) -> CachedURLResponse? {
    cache.cachedResponse(for: URLRequest(url: url))
  }
}