| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145 |
- //
- // LivePhotoSource.swift
- // Kingfisher
- //
- // Created by onevcat on 2024/10/01.
- //
- // Copyright (c) 2024 Wei Wang <onevcat@gmail.com>
- //
- // Permission is hereby granted, free of charge, to any person obtaining a copy
- // of this software and associated documentation files (the "Software"), to deal
- // in the Software without restriction, including without limitation the rights
- // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- // copies of the Software, and to permit persons to whom the Software is
- // furnished to do so, subject to the following conditions:
- //
- // The above copyright notice and this permission notice shall be included in
- // all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- // THE SOFTWARE.
- import Foundation
- /// A resource type representing a component of a Live Photo, which consists of a still image and a video.
- ///
- /// ``LivePhotoResource`` encapsulates the necessary information to download and cache a single components of a Live
- /// Photo: it is either a still image (typically in HEIC format) or a video (typically in MOV format). Multiple
- /// ``LivePhotoResource`` values (typically two, one for the image and one for the video) can form a ``LivePhotoSource``,
- /// which is expected by Kingfisher in its live photo loading high level APIs.
- ///
- /// The Live Photo data can be retrieved by `PHAssetResourceManager.requestData` method and uploaded to your server.
- /// You should not modify the metadata or other information of the data, otherwise, it is possible that the
- /// `PHLivePhoto` class cannot read and recognize it anymore. For more information, please refer to Apple's
- /// documentation of Photos framework.
- public struct LivePhotoResource: Sendable {
-
- /// The file type of a ``LivePhotoResource``.
- public enum FileType: Sendable, Equatable {
- /// File type HEIC. Usually it represents the still image in a Live Photo.
- case heic
- /// File type MOV. Usually it represents the video in a Live Photo.
- case mov
- /// Other file types with the file extension.
- case other(String)
-
- var fileExtension: String {
- switch self {
- case .heic: return "heic"
- case .mov: return "mov"
- case .other(let ext): return ext
- }
- }
- }
-
- public let dataSource: Source
- public let referenceFileType: FileType
-
- var cacheKey: String { dataSource.cacheKey }
- var downloadURL: URL? { dataSource.url }
-
- public init(downloadURL: URL, cacheKey: String? = nil, fileType: FileType? = nil) {
- let resource = KF.ImageResource(downloadURL: downloadURL, cacheKey: cacheKey)
- dataSource = .network(resource)
- referenceFileType = fileType ?? resource.guessedFileType
- }
-
- public init(resource: any Resource, fileType: FileType? = nil) {
- self.dataSource = .network(resource)
- referenceFileType = fileType ?? resource.guessedFileType
- }
- }
- extension LivePhotoResource.FileType {
- func determinedFileExtension(_ data: Data) -> String? {
- switch self {
- case .mov: return "mov"
- case .heic: return "heic"
- case .other(let ext):
- if !ext.isEmpty {
- return ext
- }
- return Self.guessedFileExtension(from: data)
- }
- }
-
- static let fytpChunk: [UInt8] = [0x66, 0x74, 0x79, 0x70]
- static let heicChunk: [UInt8] = [0x68, 0x65, 0x69, 0x63]
- static let qtChunk: [UInt8] = [0x71, 0x74, 0x20, 0x20] // quicktime
-
- static func guessedFileExtension(from data: Data) -> String? {
-
- guard data.count >= 12 else { return nil }
-
- var buffer = [UInt8](repeating: 0, count: 12)
- data.copyBytes(to: &buffer, count: 12)
-
- guard Array(buffer[4..<8]) == fytpChunk else {
- return nil
- }
-
- let fileTypeChunk = Array(buffer[8..<12])
- if fileTypeChunk == heicChunk {
- return "heic"
- }
- if fileTypeChunk == qtChunk {
- return "mov"
- }
- return nil
- }
- }
- extension Resource {
- var guessedFileType: LivePhotoResource.FileType {
- let pathExtension = downloadURL.pathExtension.lowercased()
- return switch pathExtension {
- case "mov": .mov
- case "heic": .heic
- default: .other(pathExtension)
- }
- }
- }
- public struct LivePhotoSource: Sendable {
-
- public let resources: [LivePhotoResource]
-
- public init(resources: [any Resource]) {
- let livePhotoResources = resources.map { LivePhotoResource(resource: $0) }
- self.init(livePhotoResources)
- }
-
- public init(urls: [URL]) {
- let resources = urls.map { KF.ImageResource(downloadURL: $0) }
- self.init(resources: resources)
- }
-
- public init(_ resources: [LivePhotoResource]) {
- self.resources = resources
- }
- }
|