forked from mirrors/linphone-iphone
170 lines
5.5 KiB
Swift
170 lines
5.5 KiB
Swift
/*
|
|
* Copyright (c) 2010-2023 Belledonne Communications SARL.
|
|
*
|
|
* This file is part of linphone-iphone
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
import SwiftUI
|
|
import PhotosUI
|
|
|
|
struct PhotoPicker: UIViewControllerRepresentable {
|
|
typealias UIViewControllerType = PHPickerViewController
|
|
|
|
let filter: PHPickerFilter?
|
|
var limit: Int = 0
|
|
let onComplete: ([PHPickerResult]) -> Void
|
|
|
|
func makeUIViewController(context: Context) -> PHPickerViewController {
|
|
|
|
var configuration = PHPickerConfiguration()
|
|
if filter != nil {
|
|
configuration.filter = filter
|
|
}
|
|
configuration.selectionLimit = limit
|
|
|
|
let controller = PHPickerViewController(configuration: configuration)
|
|
|
|
controller.delegate = context.coordinator
|
|
return controller
|
|
}
|
|
|
|
static func convertToUIImageArray(fromResults results: [PHPickerResult], onComplete: @escaping ([UIImage]?, Error?) -> Void) {
|
|
var images = [UIImage]()
|
|
|
|
let dispatchGroup = DispatchGroup()
|
|
for result in results {
|
|
dispatchGroup.enter()
|
|
let itemProvider = result.itemProvider
|
|
if itemProvider.canLoadObject(ofClass: UIImage.self) {
|
|
itemProvider.loadObject(ofClass: UIImage.self) { (imageOrNil, errorOrNil) in
|
|
if let error = errorOrNil {
|
|
onComplete(nil, error)
|
|
}
|
|
if let image = imageOrNil as? UIImage {
|
|
images.append(image)
|
|
}
|
|
dispatchGroup.leave()
|
|
}
|
|
}
|
|
}
|
|
dispatchGroup.notify(queue: .main) {
|
|
onComplete(images, nil)
|
|
}
|
|
}
|
|
|
|
static func convertToAttachmentArray(fromResults results: [PHPickerResult], onComplete: @escaping ([Attachment]?, Error?) -> Void) {
|
|
var medias = [Attachment]()
|
|
|
|
let dispatchGroup = DispatchGroup()
|
|
for result in results {
|
|
dispatchGroup.enter()
|
|
let itemProvider = result.itemProvider
|
|
if itemProvider.hasItemConformingToTypeIdentifier(UTType.image.identifier) {
|
|
itemProvider.loadFileRepresentation(forTypeIdentifier: UTType.image.identifier) { urlFile, error in
|
|
if urlFile != nil {
|
|
do {
|
|
let dataResult = try Data(contentsOf: urlFile!)
|
|
let urlImage = self.saveMedia(name: urlFile!.lastPathComponent, data: dataResult, type: .image)
|
|
if urlImage != nil {
|
|
let attachment = Attachment(id: UUID().uuidString, url: urlImage!, type: .image)
|
|
medias.append(attachment)
|
|
}
|
|
} catch {
|
|
|
|
}
|
|
}
|
|
|
|
dispatchGroup.leave()
|
|
}
|
|
} else if itemProvider.hasItemConformingToTypeIdentifier(UTType.movie.identifier) {
|
|
itemProvider.loadFileRepresentation(forTypeIdentifier: UTType.movie.identifier) { urlFile, error in
|
|
if urlFile != nil {
|
|
do {
|
|
let dataResult = try Data(contentsOf: urlFile!)
|
|
let urlImage = self.saveMedia(name: urlFile!.lastPathComponent, data: dataResult, type: .video)
|
|
let urlThumbnail = getURLThumbnail(name: urlFile!.lastPathComponent)
|
|
|
|
if urlImage != nil {
|
|
let attachment = Attachment(id: UUID().uuidString, thumbnail: urlThumbnail, full: urlImage!, type: .video)
|
|
medias.append(attachment)
|
|
}
|
|
} catch {
|
|
|
|
}
|
|
}
|
|
dispatchGroup.leave()
|
|
}
|
|
}
|
|
}
|
|
|
|
dispatchGroup.notify(queue: .main) {
|
|
onComplete(medias, nil)
|
|
}
|
|
}
|
|
|
|
static func saveMedia(name: String, data: Data, type: AttachmentType) -> URL? {
|
|
do {
|
|
let path = FileManager.default.temporaryDirectory.appendingPathComponent((name.addingPercentEncoding(withAllowedCharacters: .urlHostAllowed) ?? ""))
|
|
|
|
let decodedData: () = try data.write(to: path)
|
|
|
|
if type == .video {
|
|
let asset = AVURLAsset(url: path, options: nil)
|
|
let imgGenerator = AVAssetImageGenerator(asset: asset)
|
|
imgGenerator.appliesPreferredTrackTransform = true
|
|
let cgImage = try imgGenerator.copyCGImage(at: CMTimeMake(value: 0, timescale: 1), actualTime: nil)
|
|
let thumbnail = UIImage(cgImage: cgImage)
|
|
|
|
guard let data = thumbnail.jpegData(compressionQuality: 1) ?? thumbnail.pngData() else {
|
|
return nil
|
|
}
|
|
|
|
let urlName = FileManager.default.temporaryDirectory.appendingPathComponent("preview_" + (name.addingPercentEncoding(withAllowedCharacters: .urlHostAllowed) ?? "") + ".png")
|
|
|
|
let decodedData: () = try data.write(to: urlName)
|
|
}
|
|
|
|
return path
|
|
} catch let error {
|
|
print("*** Error generating thumbnail: \(error.localizedDescription)")
|
|
return nil
|
|
}
|
|
}
|
|
|
|
static func getURLThumbnail(name: String) -> URL {
|
|
return FileManager.default.temporaryDirectory.appendingPathComponent("preview_" + (name.addingPercentEncoding(withAllowedCharacters: .urlHostAllowed) ?? "") + ".png")
|
|
}
|
|
|
|
func updateUIViewController(_ uiViewController: PHPickerViewController, context: Context) {}
|
|
|
|
func makeCoordinator() -> Coordinator {
|
|
return Coordinator(self)
|
|
}
|
|
|
|
class Coordinator: PHPickerViewControllerDelegate {
|
|
|
|
private let parent: PhotoPicker
|
|
|
|
init(_ parent: PhotoPicker) {
|
|
self.parent = parent
|
|
}
|
|
|
|
func picker(_ picker: PHPickerViewController, didFinishPicking results: [PHPickerResult]) {
|
|
picker.dismiss(animated: true)
|
|
parent.onComplete(results)
|
|
}
|
|
}
|
|
}
|