Как использовать данные выборки в Intent Handler для редактирования виджета iOS 14?

В настоящее время я разрабатываю приложение с использованием SwiftUI.

Я пытаюсь сделать так, чтобы widget пользователь мог редактировать некоторые данные с помощью IntentConfiguration

Я хочу использовать некоторые данные выборки из CoreData в классе IntentHandler для редактирования виджета, как на изображении ниже.

введите описание изображения здесь

Я пытался сделать какие-то коды, но они не работают ...

Как я мог решить свои коды?


Вот коды:

IntentHandler.swift

import WidgetKit
import SwiftUI
import CoreData
import Intents

class IntentHandler: INExtension, ConfigurationIntentHandling {
    
    var moc = PersistenceController.shared.managedObjectContext

    var timerEntity_0:TimerEntity?
    var timerEntity_1:TimerEntity?
    var timerEntity_2:TimerEntity?

    init(context : NSManagedObjectContext) {
        self.moc = context

        let request = NSFetchRequest<TimerEntity>(entityName: "TimerEntity")

        do{
            let result = try moc.fetch(request)
            timerEntity_0 = result[0]
            timerEntity_1 = result[1]
            timerEntity_2 = result[2]
        }
        catch let error as NSError{
            print("Could not fetch.\(error.userInfo)")
        }
    }
   
    func provideNameOptionsCollection(for intent: ConfigurationIntent, searchTerm: String?, with completion: @escaping (INObjectCollection<NSString>?, Error?) -> Void) {
        
        let nameIdentifiers:[NSString] = [
            NSString(string: timerEntity_0?.task ?? "default"),
            NSString(string: timerEntity_1?.task ?? "default"),
            NSString(string: timerEntity_2?.task ?? "default")
            // "meeting",
            // "cooking",
            // "shoping"
        ]
        let allNameIdentifiers = INObjectCollection(items: nameIdentifiers)
        
        completion(allNameIdentifiers,nil)
    }
    
    override func handler(for intent: INIntent) -> Any {
        
        return self
    }
}

Widget.swift

import WidgetKit
import SwiftUI
import Intents

struct Provider: IntentTimelineProvider {
    
    typealias Intent = ConfigurationIntent
    
    func placeholder(in context: Context) -> SimpleEntry {
        SimpleEntry(date: Date(), configuration: ConfigurationIntent(), name: "")
    }

    func getSnapshot(for configuration: ConfigurationIntent, in context: Context, completion: @escaping (SimpleEntry) -> ()) {
        let entry = SimpleEntry(date: Date(), configuration: configuration, name: "")
        completion(entry)
    }

    func getTimeline(for configuration: ConfigurationIntent, in context: Context, completion: @escaping (Timeline<Entry>) -> ()) {
        var entries: [SimpleEntry] = []

        // Generate a timeline consisting of five entries an hour apart, starting from the current date.
        let currentDate = Date()
        for hourOffset in 0 ..< 5 {
            let entryDate = Calendar.current.date(byAdding: .hour, value: hourOffset, to: currentDate)!
            let entry = SimpleEntry(date: entryDate, configuration: configuration, name: configuration.Name ?? "")
            entries.append(entry)
        }

        let timeline = Timeline(entries: entries, policy: .atEnd)
        completion(timeline)
    }
}

struct SimpleEntry: TimelineEntry {
    let date: Date
    let configuration: ConfigurationIntent
    var name:String
}

struct TimerIntentWidgetEntryView : View {
    var entry: Provider.Entry

    var body: some View {
        Text(entry.name)
            .font(.title)
        Text(entry.date, style: .time)
    }
}

@main
struct TimerIntentWidget: Widget {
    let kind: String = "TimerIntentWidget"

    var body: some WidgetConfiguration {
        IntentConfiguration(kind: kind, intent: ConfigurationIntent.self, provider: Provider()) { entry in
            TimerIntentWidgetEntryView(entry: entry)
        }
        .configurationDisplayName("My Widget")
        .description("This is an example widget.")
    }
}

tWidget.intentdefinition  введите описание изображения здесь


Xcode: версия 12.0.1

iOS: 14.0

Жизненный цикл: приложение SwiftUI


person Tio    schedule 07.01.2021    source источник


Ответы (1)


Я мог бы отобразить список в виджете, используя выборку данных из CoreData, как в приведенном ниже коде:


IntentHandler.swift

import WidgetKit
import SwiftUI
import CoreData
import Intents

class IntentHandler: INExtension, ConfigurationIntentHandling {
    
    var moc = PersistenceController.shared.managedObjectContext

    var timerEntity_0:TimerEntity?
    var timerEntity_1:TimerEntity?
    var timerEntity_2:TimerEntity?

    func provideNameOptionsCollection(for intent: ConfigurationIntent, searchTerm: String?, with completion: @escaping (INObjectCollection<NSString>?, Error?) -> Void) {
        
        let request = NSFetchRequest<TimerEntity>(entityName: "TimerEntity")

        do{
            let result = try moc.fetch(request)
            timerEntity_0 = result[0]
            timerEntity_1 = result[1]
            timerEntity_2 = result[2]
        }
        catch let error as NSError{
            print("Could not fetch.\(error.userInfo)")
        }

        let nameIdentifiers:[NSString] = [
            NSString(string: timerEntity_0?.task ?? "default"),
            NSString(string: timerEntity_1?.task ?? "default"),
            NSString(string: timerEntity_2?.task ?? "default")
            // "meeting",
            // "cooking",
            // "shoping"
        ]
        let allNameIdentifiers = INObjectCollection(items: nameIdentifiers)
        
        completion(allNameIdentifiers,nil)
    }
    
    override func handler(for intent: INIntent) -> Any {
        
        return self
    }
}
person Tio    schedule 07.01.2021
comment
Мне что-то не хватает, я пробовал разные вещи и у меня не работает загрузка, в лучшем случае у меня просто долго крутится колесо. Мне нужно было создать массив и сохранить его в общей папке группы с обновлением, когда приложение открыто и закрыто, что мы использовали для списка динамического выбора. - person Kurt Lane; 17.03.2021