iOS 今天扩展核心数据
iOS today extension with core data
我正在尝试为我的 ios 应用程序添加今日扩展。
今天的扩展将根据与核心数据一起保存的数据显示 'next' 课程。
我一直在做一些研究,我知道我必须与 appGroup 共享我的 persistentContainer。
所以我做了:
- 为 ios 应用和今天的扩展目标启用了应用组。
- 编写一个函数来分享它:
public extension NSPersistentContainer {
func addToAppGroup(id: String) {
guard let fileContainer = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: id) else {
fatalError("Shared file container could not be created.")
}
let storeURL = fileContainer.appendingPathComponent("\(self.name).sqlite")
let storeDescription = NSPersistentStoreDescription(url: storeURL)
self.persistentStoreDescriptions.append(storeDescription)
}
}
然后在我的核心数据堆栈中:
internal class CoreDataContainer {
static var persistentContainer: NSPersistentContainer = {
let container = NSPersistentCloudKitContainer(name: "SchoolCompanion")
container.addToAppGroup(id: "group.com.Ce-dricLoneux.School-Companion")
container.loadPersistentStores(completionHandler: { (_, error) in
if let error = error as NSError? {
fatalError("Unresolved error \(error), \(error.userInfo)")
}
})
return container
}()
// MARK: - Core Data Saving support
func saveContext () {
let context = CoreDataContainer.persistentContainer.viewContext
if context.hasChanges {
do {
try context.save()
} catch {
let nserror = error as NSError
fatalError("Unresolved error \(nserror), \(nserror.userInfo)")
}
}
}
}
文件和 xcdatamodel 在两个目标之间共享。
在这一点上,我认为我可以从我的扩展程序访问我的核心数据,但是当我执行提取请求时,我没有得到任何结果。永远不会执行 controllerDidChangeContent。
class TodayViewController: UIViewController, NCWidgetProviding {
private func configureFetchedResultsController() {
let request: NSFetchRequest<Course> = Course.fetchRequest()
request.sortDescriptors = []
self.fetchedResultsController = NSFetchedResultsController<Course>(fetchRequest: request, managedObjectContext: CoreDataContainer.persistentContainer.viewContext, sectionNameKeyPath: nil, cacheName: nil)
self.fetchedResultsController.delegate = self
do {
try self.fetchedResultsController.performFetch()
} catch {
print(error.localizedDescription)
}
}
override func viewDidLoad() {
super.viewDidLoad()
self.extensionContext?.widgetLargestAvailableDisplayMode = .compact
self.configureFetchedResultsController()
}
func widgetPerformUpdate(completionHandler: (@escaping (NCUpdateResult) -> Void)) {
// Perform any setup necessary in order to update the view.
// If an error is encountered, use NCUpdateResult.Failed
// If there's no update required, use NCUpdateResult.NoData
// If there's an update, use NCUpdateResult.NewData
completionHandler(NCUpdateResult.newData)
}
}
// MARK: - FetchedResultsController Delegate
extension TodayViewController: NSFetchedResultsControllerDelegate {
func controllerDidChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>) {
print("content did change")
}
}
我也尝试直接发出获取请求,但结果得到一个空数组。
为什么我没有得到任何结果?
另一件事:我使用 nsfetchedResult 控制器以便在每次从 ios 应用程序更新数据时刷新视图数据,这样可以吗?还是我应该使用 widgetPerformUpdate 方法?在那种情况下,我们不知道 today 扩展何时会刷新,它可能会显示过时的数据。
使用的主要文档:https://www.avanderlee.com/swift/core-data-app-extension-data-sharing/
您的应用组的 url 在这里:
let url = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: "SchoolCompanion)
sqlite 文件在那里。您不需要云容器。
您应该像下面那样子类化 NSPersistentCloudKitContainer,为 defaultDirectoryURL()
返回应用程序组 URL。然后在你的 CoreDataStack 中,使用 let container = GroupedPersistentCloudKitContainer(name: "SchoolCompanion")
。同时删除您对 addToAppGroup(...)
的调用。您需要在应用程序和扩展中实例化 GroupedPersistentCloudKitContainer,您还需要确保 GroupedPersistentCloudKitContainer 链接到两个目标。
class GroupedPersistentCloudKitContainer: NSPersistentCloudKitContainer {
enum URLStrings: String {
case group = "group.com.yourCompany.yourApp"
}
override class func defaultDirectoryURL() -> URL {
let url = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: URLStrings.group.rawValue)
if !FileManager.default.fileExists(atPath: url!.path) {
try? FileManager.default.createDirectory(at: url!, withIntermediateDirectories: true, attributes: nil)
}
return url!
}
...
}
我正在尝试为我的 ios 应用程序添加今日扩展。 今天的扩展将根据与核心数据一起保存的数据显示 'next' 课程。 我一直在做一些研究,我知道我必须与 appGroup 共享我的 persistentContainer。 所以我做了:
- 为 ios 应用和今天的扩展目标启用了应用组。
- 编写一个函数来分享它:
public extension NSPersistentContainer {
func addToAppGroup(id: String) {
guard let fileContainer = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: id) else {
fatalError("Shared file container could not be created.")
}
let storeURL = fileContainer.appendingPathComponent("\(self.name).sqlite")
let storeDescription = NSPersistentStoreDescription(url: storeURL)
self.persistentStoreDescriptions.append(storeDescription)
}
}
然后在我的核心数据堆栈中:
internal class CoreDataContainer {
static var persistentContainer: NSPersistentContainer = {
let container = NSPersistentCloudKitContainer(name: "SchoolCompanion")
container.addToAppGroup(id: "group.com.Ce-dricLoneux.School-Companion")
container.loadPersistentStores(completionHandler: { (_, error) in
if let error = error as NSError? {
fatalError("Unresolved error \(error), \(error.userInfo)")
}
})
return container
}()
// MARK: - Core Data Saving support
func saveContext () {
let context = CoreDataContainer.persistentContainer.viewContext
if context.hasChanges {
do {
try context.save()
} catch {
let nserror = error as NSError
fatalError("Unresolved error \(nserror), \(nserror.userInfo)")
}
}
}
}
文件和 xcdatamodel 在两个目标之间共享。 在这一点上,我认为我可以从我的扩展程序访问我的核心数据,但是当我执行提取请求时,我没有得到任何结果。永远不会执行 controllerDidChangeContent。
class TodayViewController: UIViewController, NCWidgetProviding {
private func configureFetchedResultsController() {
let request: NSFetchRequest<Course> = Course.fetchRequest()
request.sortDescriptors = []
self.fetchedResultsController = NSFetchedResultsController<Course>(fetchRequest: request, managedObjectContext: CoreDataContainer.persistentContainer.viewContext, sectionNameKeyPath: nil, cacheName: nil)
self.fetchedResultsController.delegate = self
do {
try self.fetchedResultsController.performFetch()
} catch {
print(error.localizedDescription)
}
}
override func viewDidLoad() {
super.viewDidLoad()
self.extensionContext?.widgetLargestAvailableDisplayMode = .compact
self.configureFetchedResultsController()
}
func widgetPerformUpdate(completionHandler: (@escaping (NCUpdateResult) -> Void)) {
// Perform any setup necessary in order to update the view.
// If an error is encountered, use NCUpdateResult.Failed
// If there's no update required, use NCUpdateResult.NoData
// If there's an update, use NCUpdateResult.NewData
completionHandler(NCUpdateResult.newData)
}
}
// MARK: - FetchedResultsController Delegate
extension TodayViewController: NSFetchedResultsControllerDelegate {
func controllerDidChangeContent(_ controller: NSFetchedResultsController<NSFetchRequestResult>) {
print("content did change")
}
}
我也尝试直接发出获取请求,但结果得到一个空数组。 为什么我没有得到任何结果?
另一件事:我使用 nsfetchedResult 控制器以便在每次从 ios 应用程序更新数据时刷新视图数据,这样可以吗?还是我应该使用 widgetPerformUpdate 方法?在那种情况下,我们不知道 today 扩展何时会刷新,它可能会显示过时的数据。
使用的主要文档:https://www.avanderlee.com/swift/core-data-app-extension-data-sharing/
您的应用组的 url 在这里:
let url = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: "SchoolCompanion)
sqlite 文件在那里。您不需要云容器。
您应该像下面那样子类化 NSPersistentCloudKitContainer,为 defaultDirectoryURL()
返回应用程序组 URL。然后在你的 CoreDataStack 中,使用 let container = GroupedPersistentCloudKitContainer(name: "SchoolCompanion")
。同时删除您对 addToAppGroup(...)
的调用。您需要在应用程序和扩展中实例化 GroupedPersistentCloudKitContainer,您还需要确保 GroupedPersistentCloudKitContainer 链接到两个目标。
class GroupedPersistentCloudKitContainer: NSPersistentCloudKitContainer {
enum URLStrings: String {
case group = "group.com.yourCompany.yourApp"
}
override class func defaultDirectoryURL() -> URL {
let url = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: URLStrings.group.rawValue)
if !FileManager.default.fileExists(atPath: url!.path) {
try? FileManager.default.createDirectory(at: url!, withIntermediateDirectories: true, attributes: nil)
}
return url!
}
...
}