1
0
mirror of https://github.com/mastodon/mastodon-ios.git synced 2024-12-12 08:36:02 +01:00
mastodon-app-ufficiale-ipho.../MastodonSDK/Sources/MastodonCore/Service/PhotoLibraryService.swift
Marcus Kida 4ea600403b
Move all UI*FeedbackGenerators to FeedbackGenerator and disable them for now (IOS-247) (#1267)
* Move all UI*FeedbackGenerators to FeedbackGenerator and disable them for now (IOS-247)

* Fix copyright header

* Remove empty private constructor
2024-04-09 16:41:47 +02:00

165 lines
5.0 KiB
Swift

//
// PhotoLibraryService.swift
// Mastodon
//
// Created by MainasuK Cirno on 2021-4-29.
//
import UIKit
import Combine
import Photos
import Alamofire
import AlamofireImage
public final class PhotoLibraryService: NSObject {
}
extension PhotoLibraryService {
public enum PhotoLibraryError: Error {
case noPermission
case badPayload
}
public enum ImageSource {
case url(URL)
case image(UIImage)
}
}
extension PhotoLibraryService {
public func save(imageSource source: ImageSource) -> AnyPublisher<Void, Error> {
let feedbackGenerator = FeedbackGenerator.shared
let imageDataPublisher: AnyPublisher<Data, Error> = {
switch source {
case .url(let url):
return PhotoLibraryService.fetchImageData(url: url)
case .image(let image):
return PhotoLibraryService.fetchImageData(image: image)
}
}()
return imageDataPublisher
.flatMap { data in
PhotoLibraryService.save(imageData: data)
}
.handleEvents(receiveSubscription: { _ in
feedbackGenerator.generate(.impact(.light))
}, receiveCompletion: { completion in
switch completion {
case .failure:
feedbackGenerator.generate(.notification(.error))
case .finished:
feedbackGenerator.generate(.notification(.success))
}
})
.eraseToAnyPublisher()
}
}
extension PhotoLibraryService {
public func copy(imageSource source: ImageSource) -> AnyPublisher<Void, Error> {
let feedbackGenerator = FeedbackGenerator.shared
let imageDataPublisher: AnyPublisher<Data, Error> = {
switch source {
case .url(let url):
return PhotoLibraryService.fetchImageData(url: url)
case .image(let image):
return PhotoLibraryService.fetchImageData(image: image)
}
}()
return imageDataPublisher
.flatMap { data in
PhotoLibraryService.copy(imageData: data)
}
.handleEvents(receiveSubscription: { _ in
feedbackGenerator.generate(.impact(.light))
}, receiveCompletion: { completion in
switch completion {
case .failure:
feedbackGenerator.generate(.notification(.error))
case .finished:
feedbackGenerator.generate(.notification(.success))
}
})
.eraseToAnyPublisher()
}
}
extension PhotoLibraryService {
static func fetchImageData(url: URL) -> AnyPublisher<Data, Error> {
AF.request(url).publishData()
.tryMap { response in
switch response.result {
case .success(let data):
return data
case .failure(let error):
throw error
}
}
.eraseToAnyPublisher()
}
static func fetchImageData(image: UIImage) -> AnyPublisher<Data, Error> {
return Future<Data, Error> { promise in
DispatchQueue.global().async {
let imageData = image.pngData()
DispatchQueue.main.async {
if let imageData = imageData {
promise(.success(imageData))
} else {
promise(.failure(PhotoLibraryError.badPayload))
}
}
}
}
.eraseToAnyPublisher()
}
static func save(imageData: Data) -> AnyPublisher<Void, Error> {
guard PHPhotoLibrary.authorizationStatus(for: .addOnly) != .denied else {
return Fail(error: PhotoLibraryError.noPermission).eraseToAnyPublisher()
}
return Future<Void, Error> { promise in
PHPhotoLibrary.shared().performChanges {
PHAssetCreationRequest.forAsset().addResource(with: .photo, data: imageData, options: nil)
} completionHandler: { isSuccess, error in
if let error = error {
promise(.failure(error))
} else {
promise(.success(Void()))
}
}
}
.eraseToAnyPublisher()
}
static func copy(imageData: Data) -> AnyPublisher<Void, Error> {
Future<Void, Error> { promise in
DispatchQueue.global().async {
let image = UIImage(data: imageData, scale: UIScreen.main.scale)
DispatchQueue.main.async {
if let image = image {
UIPasteboard.general.image = image
promise(.success(Void()))
} else {
promise(.failure(PhotoLibraryError.badPayload))
}
}
}
}
.eraseToAnyPublisher()
}
}