首页 > 解决方案 > iOS NSCoreDataCoreSpotlightDelegate,如何重新索引所有项目

问题描述

我已经实现了 NSCoreDataCoreSpotlightDelegate,我错过了默认过期日期为 1 个月的部分,现在我在 3 个月前添加的项目没有显示在搜索索引中。

如何让 NSCoreDataCoreSpotlightDelegate 重新索引所有项目?

我以前可以这样称呼:

let container = NSPersistentContainer(name: "MyApp")
let psd = NSPersistentStoreDescription(url:  FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: "group.com.kiwlm.MyApp")!.appendingPathComponent("MyApp.sqlite"))

let mcdcsd = MyCoreDataCoreSpotlightDelegate(forStoreWith:psd, model: container.managedObjectModel)

container.persistentStoreDescriptions = [psd]

psd.setOption(mcdcsd, forKey:NSCoreDataCoreSpotlightExporter)

// uncomment the following to reindex all items
// mcdcsd.searchableIndex(CSSearchableIndex.default(), reindexAllSearchableItemsWithAcknowledgementHandler: {})

第一次,它将重新索引所有项目,但如果我重新运行应用程序并再次取消注释上述行,它将不会重新索引。

如果我卸载该应用程序,重新安装,然后取消注释上述行,那么它将第一次再次索引所有。

如何让它重新索引所有内容?

标签: iosswiftcore-datacorespotlight

解决方案


这是因为 NSCoreDataCoreSpotlightDelegate 的实现使用 beginIndexBatch 和 endIndexBatchWithClientState 来重新索引AllSearchableItems。所以它正在保存索引状态。当您调用 reindexAllSearchableItems 时,Spotlight 将检查状态并仅重新索引新项目。当您删除应用程序时,您也会删除索引状态。在第一次启动状态是空的和 NSCoreDataCoreSpotlightDelegate 索引所有项目。

我认为为了您的目的,最好设置 expireDate 而不是重新索引所有项目。您可以通过覆盖属性集在 MyCoreDataCoreSpotlightDelegate 中执行此操作:

override func attributeSet(for object: NSManagedObject) -> CSSearchableItemAttributeSet? {
    guard let attributeSet = super.attributeSet(for: object) else { return nil }
        if object.entity.name == "YourEntityName"  {
            // Setup necessary attributes
            attributeSet.setValue(Date.distantFuture, forKey: "expirationDate")
        }
    return attributeSet
}

要检查可搜索项目的过期日期,您可以使用 CSSearchQuery:

class QueryExample {
    var query : CSSearchQuery? = nil
    func startQuery(withTitle title : String) {
        var allItems = [CSSearchableItem]()
        // All the items that start with the specified title property using case insensitive comparison
        let queryString = "title == \"*\(title)*\"c"
        let attributes = ["title", "displayName", "keywords",
                      "contentType"]
        self.query = CSSearchQuery(queryString : queryString,
                               attributes : attributes)
        self.query?.foundItemsHandler = { (items : [CSSearchableItem])
        -> Void in
            allItems.append(contentsOf: items)
        }
        self.query?.completionHandler = { (error ) -> Void in
            for item in allItems {
                print("expirationDate:\(item.expirationDate)")
            }
        }
        self.query?.start()
    }
}

像这样调用查询:

   override func viewDidLoad() {
       let query = QueryExample()
       query.startQuery(withTitle: "S")
   }

你应该在控制台看到 expireDate:

到期日期:4001-01-01 00:00:00 +0000


推荐阅读