mirror of
https://github.com/mastodon/mastodon-ios.git
synced 2024-12-13 00:56:23 +01:00
0c224f47df
Co-authored-by: Marcus Kida <marcus.kida@bearologics.com> Co-authored-by: Jed Fox <git@jedfox.com>
161 lines
5.8 KiB
Swift
161 lines
5.8 KiB
Swift
//
|
|
// AttachmentViewModel+Load.swift
|
|
//
|
|
//
|
|
// Created by MainasuK on 2022/11/8.
|
|
//
|
|
|
|
import os.log
|
|
import UIKit
|
|
import AVKit
|
|
import UniformTypeIdentifiers
|
|
import Nuke
|
|
|
|
extension AttachmentViewModel {
|
|
|
|
@MainActor
|
|
func load(input: Input) async throws -> Output {
|
|
switch input {
|
|
case .image(let image):
|
|
guard let data = image.normalized()?.pngData() else {
|
|
throw AttachmentError.invalidAttachmentType
|
|
}
|
|
return .image(data, imageKind: .png)
|
|
case .url(let url):
|
|
do {
|
|
let output = try await AttachmentViewModel.load(url: url)
|
|
return output
|
|
} catch {
|
|
throw error
|
|
}
|
|
case .mastodonAssetUrl(let url, _):
|
|
return try await Self.loadMastodonAsset(url: url)
|
|
case .pickerResult(let pickerResult):
|
|
do {
|
|
let output = try await AttachmentViewModel.load(itemProvider: pickerResult.itemProvider)
|
|
return output
|
|
} catch {
|
|
throw error
|
|
}
|
|
case .itemProvider(let itemProvider):
|
|
do {
|
|
let output = try await AttachmentViewModel.load(itemProvider: itemProvider)
|
|
return output
|
|
} catch {
|
|
throw error
|
|
}
|
|
}
|
|
}
|
|
|
|
private static func loadMastodonAsset(url: URL) async throws -> Output {
|
|
guard !url.isFileURL else {
|
|
throw AttachmentError.invalidAttachmentType
|
|
}
|
|
let (imageData, _) = try await URLSession.shared.data(from: url)
|
|
return .image(imageData, imageKind: AssetType(imageData) == .png ? .png : .jpg)
|
|
}
|
|
|
|
private static func load(url: URL) async throws -> Output {
|
|
guard let uti = UTType(filenameExtension: url.pathExtension) else {
|
|
throw AttachmentError.invalidAttachmentType
|
|
}
|
|
|
|
if uti.conforms(to: .image) {
|
|
guard url.startAccessingSecurityScopedResource() else {
|
|
throw AttachmentError.invalidAttachmentType
|
|
}
|
|
defer { url.stopAccessingSecurityScopedResource() }
|
|
let imageData = try Data(contentsOf: url)
|
|
return .image(imageData, imageKind: AssetType(imageData) == .png ? .png : .jpg)
|
|
} else if uti.conforms(to: .movie) {
|
|
guard url.startAccessingSecurityScopedResource() else {
|
|
throw AttachmentError.invalidAttachmentType
|
|
}
|
|
defer { url.stopAccessingSecurityScopedResource() }
|
|
|
|
let fileName = UUID().uuidString
|
|
let tempDirectoryURL = FileManager.default.temporaryDirectory
|
|
let fileURL = tempDirectoryURL.appendingPathComponent(fileName).appendingPathExtension(url.pathExtension)
|
|
try FileManager.default.createDirectory(at: tempDirectoryURL, withIntermediateDirectories: true, attributes: nil)
|
|
try FileManager.default.copyItem(at: url, to: fileURL)
|
|
return .video(fileURL, mimeType: UTType.movie.preferredMIMEType ?? "video/mp4")
|
|
} else {
|
|
throw AttachmentError.invalidAttachmentType
|
|
}
|
|
}
|
|
|
|
private static func load(itemProvider: NSItemProvider) async throws -> Output {
|
|
if itemProvider.isImage() {
|
|
guard let result = try await itemProvider.loadImageData() else {
|
|
throw AttachmentError.invalidAttachmentType
|
|
}
|
|
let imageKind: Output.ImageKind = {
|
|
if let type = result.type {
|
|
if type == UTType.png {
|
|
return .png
|
|
}
|
|
if type == UTType.jpeg {
|
|
return .jpg
|
|
}
|
|
}
|
|
|
|
let imageData = result.data
|
|
let assetType = AssetType(imageData)
|
|
|
|
if assetType == .png {
|
|
return .png
|
|
}
|
|
if assetType == .jpeg {
|
|
return .jpg
|
|
}
|
|
|
|
assertionFailure("unknown image kind")
|
|
return .jpg
|
|
}()
|
|
return .image(result.data, imageKind: imageKind)
|
|
} else if itemProvider.isMovie() {
|
|
guard let result = try await itemProvider.loadVideoData() else {
|
|
throw AttachmentError.invalidAttachmentType
|
|
}
|
|
return .video(result.url, mimeType: "video/mp4")
|
|
} else {
|
|
assertionFailure()
|
|
throw AttachmentError.invalidAttachmentType
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
extension AttachmentViewModel {
|
|
static func createThumbnailForVideo(url: URL) -> UIImage? {
|
|
guard FileManager.default.fileExists(atPath: url.path) else { return nil }
|
|
let asset = AVURLAsset(url: url)
|
|
let assetImageGenerator = AVAssetImageGenerator(asset: asset)
|
|
assetImageGenerator.appliesPreferredTrackTransform = true // fix orientation
|
|
do {
|
|
let cgImage = try assetImageGenerator.copyCGImage(at: CMTimeMake(value: 0, timescale: 1), actualTime: nil)
|
|
let image = UIImage(cgImage: cgImage)
|
|
return image
|
|
} catch {
|
|
AttachmentViewModel.logger.log(level: .debug, "\((#file as NSString).lastPathComponent, privacy: .public)[\(#line, privacy: .public)], \(#function, privacy: .public): thumbnail generate fail: \(error.localizedDescription)")
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
|
|
extension NSItemProvider {
|
|
func isImage() -> Bool {
|
|
return hasRepresentationConforming(
|
|
toTypeIdentifier: UTType.image.identifier,
|
|
fileOptions: []
|
|
)
|
|
}
|
|
|
|
func isMovie() -> Bool {
|
|
return hasRepresentationConforming(
|
|
toTypeIdentifier: UTType.movie.identifier,
|
|
fileOptions: []
|
|
)
|
|
}
|
|
}
|