mirror of
https://github.com/simplex-chat/simplex-chat.git
synced 2026-04-26 19:35:48 +00:00
4c8bc19182
* ios: send multiple images * multi-select works (TODO race conditions) * send multiple images, progress indicator in compose view * scroll between fullscreen images, scroll to quoted item * add swipe animation * fix model state when sending the image * fix sending multiple images * use MainActor * improve scrolling * faster scroll * improve scroll animation * fix model updates
161 lines
5.3 KiB
Swift
161 lines
5.3 KiB
Swift
//
|
|
// ImagePicker.swift
|
|
// SimpleX
|
|
//
|
|
// Created by Evgeny on 23/03/2022.
|
|
// Copyright © 2022 SimpleX Chat. All rights reserved.
|
|
//
|
|
|
|
import SwiftUI
|
|
import PhotosUI
|
|
|
|
struct LibraryImagePicker: View {
|
|
@Binding var image: UIImage?
|
|
var didFinishPicking: (_ didSelectItems: Bool) -> Void
|
|
@State var images: [UIImage] = []
|
|
|
|
var body: some View {
|
|
LibraryImageListPicker(images: $images, selectionLimit: 1, didFinishPicking: didFinishPicking)
|
|
.onChange(of: images) { image = $0.first }
|
|
}
|
|
}
|
|
|
|
struct LibraryImageListPicker: UIViewControllerRepresentable {
|
|
typealias UIViewControllerType = PHPickerViewController
|
|
@Binding var images: [UIImage]
|
|
var selectionLimit: Int
|
|
var didFinishPicking: (_ didSelectItems: Bool) -> Void
|
|
|
|
class Coordinator: PHPickerViewControllerDelegate {
|
|
let parent: LibraryImageListPicker
|
|
let dispatchQueue = DispatchQueue(label: "chat.simplex.app.LibraryImageListPicker")
|
|
var images: [UIImage] = []
|
|
var imageCount: Int = 0
|
|
|
|
init(_ parent: LibraryImageListPicker) {
|
|
self.parent = parent
|
|
}
|
|
|
|
func picker(_ picker: PHPickerViewController, didFinishPicking results: [PHPickerResult]) {
|
|
parent.didFinishPicking(!results.isEmpty)
|
|
guard !results.isEmpty else {
|
|
return
|
|
}
|
|
|
|
parent.images = []
|
|
images = []
|
|
imageCount = results.count
|
|
for result in results {
|
|
logger.log("LibraryImageListPicker result")
|
|
let p = result.itemProvider
|
|
if p.canLoadObject(ofClass: UIImage.self) {
|
|
p.loadObject(ofClass: UIImage.self) { image, error in
|
|
DispatchQueue.main.async {
|
|
self.loadImage(object: image, error: error)
|
|
}
|
|
}
|
|
} else {
|
|
dispatchQueue.sync { self.imageCount -= 1}
|
|
}
|
|
}
|
|
DispatchQueue.main.asyncAfter(deadline: .now() + 10) {
|
|
self.dispatchQueue.sync {
|
|
if self.parent.images.count == 0 {
|
|
logger.log("LibraryImageListPicker: added \(self.images.count) images out of \(results.count)")
|
|
self.parent.images = self.images
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func loadImage(object: Any?, error: Error? = nil) {
|
|
if let error = error {
|
|
logger.error("LibraryImageListPicker: couldn't load image with error: \(error.localizedDescription)")
|
|
} else if let image = object as? UIImage {
|
|
images.append(image)
|
|
logger.log("LibraryImageListPicker: added image")
|
|
}
|
|
dispatchQueue.sync {
|
|
self.imageCount -= 1
|
|
if self.imageCount == 0 && self.parent.images.count == 0 {
|
|
logger.log("LibraryImageListPicker: added all images")
|
|
self.parent.images = self.images
|
|
self.images = []
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func makeCoordinator() -> Coordinator {
|
|
Coordinator(self)
|
|
}
|
|
|
|
func makeUIViewController(context: Context) -> PHPickerViewController {
|
|
var config = PHPickerConfiguration()
|
|
config.filter = .images
|
|
config.selectionLimit = selectionLimit
|
|
let controller = PHPickerViewController(configuration: config)
|
|
controller.delegate = context.coordinator
|
|
return controller
|
|
}
|
|
|
|
func updateUIViewController(_ uiViewController: PHPickerViewController, context: Context) {
|
|
|
|
}
|
|
}
|
|
|
|
struct CameraImageListPicker: View {
|
|
@Binding var images: [UIImage]
|
|
@State var image: UIImage?
|
|
|
|
var body: some View {
|
|
CameraImagePicker(image: $image)
|
|
.onChange(of: image) { images = imageList($0) }
|
|
}
|
|
}
|
|
|
|
func imageList(_ img: UIImage?) -> [UIImage] {
|
|
if let img = img {
|
|
return [img]
|
|
} else {
|
|
return []
|
|
}
|
|
}
|
|
|
|
struct CameraImagePicker: UIViewControllerRepresentable {
|
|
@Environment(\.presentationMode) var presentationMode
|
|
@Binding var image: UIImage?
|
|
|
|
class Coordinator: NSObject, UINavigationControllerDelegate, UIImagePickerControllerDelegate {
|
|
let parent: CameraImagePicker
|
|
|
|
init(_ parent: CameraImagePicker) {
|
|
self.parent = parent
|
|
}
|
|
|
|
func imagePickerController(_ picker: UIImagePickerController,
|
|
didFinishPickingMediaWithInfo info: [UIImagePickerController.InfoKey: Any]) {
|
|
if let uiImage = info[.originalImage] as? UIImage {
|
|
parent.image = uiImage
|
|
}
|
|
parent.presentationMode.wrappedValue.dismiss()
|
|
}
|
|
}
|
|
|
|
func makeCoordinator() -> Coordinator {
|
|
Coordinator(self)
|
|
}
|
|
|
|
func makeUIViewController(context: UIViewControllerRepresentableContext<CameraImagePicker>) -> UIImagePickerController {
|
|
let picker = UIImagePickerController()
|
|
picker.sourceType = .camera
|
|
picker.allowsEditing = false
|
|
picker.delegate = context.coordinator
|
|
return picker
|
|
}
|
|
|
|
func updateUIViewController(_ uiViewController: UIImagePickerController, context: UIViewControllerRepresentableContext<CameraImagePicker>) {
|
|
|
|
}
|
|
}
|