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 !!
Другие вопросы по тегам