LazyVGrid, List, LazyStacks не освобождают представления из памяти?
Я играю с новым средством выбора фотографий в SwiftUI 2, и я сделал простое приложение для отображения импортированных изображений в LazyVGrid, но при прокрутке вниз, если я импортировал около 150 изображений, приложение завершило всю память и вылетело (прекращено из-за к вопросу памяти).
Я пробовал то же самое с LazyVStack и List, но у них та же проблема, я ожидал, что ленивые элементы освободят все ячейки, которые выходят за пределы экрана, из памяти, но похоже, что это не работает.
Это ошибка или я что-то не так делаю?
Вот мой код:
import SwiftUI
struct Media: Identifiable {
var id = UUID()
var image: Image
}
struct ContentView: View {
@State var itemProviders: [NSItemProvider] = []
@State private var showingPhotoPicker: Bool = false
let columns = [
GridItem(.adaptive(minimum: 100, maximum: 100), spacing: 8)
]
@State var medias: [Media] = []
var body: some View {
NavigationView {
ScrollView {
LazyVGrid(columns: columns, spacing: 8) {
ForEach(medias) { media in
media.image
.resizable()
.scaledToFill()
.frame(width: 100, height: 100, alignment: .center)
.clipped()
}
}
}
.navigationBarTitle("Images \(medias.count)")
.navigationBarItems(leading: Button(action: {
loadImages()
}, label: {
Text("Import \(itemProviders.count) images")
}), trailing: Button(action: {
showingPhotoPicker.toggle()
}, label: {
Image(systemName: "photo.on.rectangle.angled")
}))
.sheet(isPresented: $showingPhotoPicker) {
MultiPHPickerView(itemProviders: $itemProviders)
}
}
}
func loadImages() {
for item in itemProviders {
if item.canLoadObject(ofClass: UIImage.self) {
item.loadObject(ofClass: UIImage.self) { image, error in
DispatchQueue.main.async {
guard let image = image as? UIImage else {
return
}
medias.append(Media(image: Image(uiImage: image)))
}
}
}
}
}
}
И PhotoPickerView:
import SwiftUI
import PhotosUI
struct MultiPHPickerView: UIViewControllerRepresentable {
@Environment(\.presentationMode) private var presentationMode
@Binding var itemProviders: [NSItemProvider]
func makeCoordinator() -> Coordinator {
Coordinator(self)
}
func makeUIViewController(context: Context) -> PHPickerViewController {
var configuration = PHPickerConfiguration()
configuration.filter = .images
configuration.selectionLimit = 0
let controller = PHPickerViewController(configuration: configuration)
controller.delegate = context.coordinator
return controller
}
func updateUIViewController( _ uiViewController: PHPickerViewController, context: Context) {}
class Coordinator: NSObject, PHPickerViewControllerDelegate {
@Environment(\.presentationMode) private var presentationMode
var parent: MultiPHPickerView
init( _ parent: MultiPHPickerView ) {
self.parent = parent
}
func picker(_ picker: PHPickerViewController, didFinishPicking results: [PHPickerResult]) {
picker.dismiss( animated: true )
self.parent.itemProviders = results.map(\.itemProvider)
}
}
}
1 ответ
Сами представления почти ничего не потребляют, а ленивые вещи работают так, как должны... вся ваша память занята загруженными и сохраненными изображениями
struct Media: Identifiable {
var id = UUID()
var image: Image // << here !!
}
struct ContentView: View {
// .. other code
@State var medias: [Media] = [] // << here !!