7

我正在尝试构建一个使用 CoreSpotlight 使内容在 iOS Spotlight 中可搜索的测试应用程序。deleteAllSearchableItems目前,我将其设置为每当用户点击刷新按钮时,它就会发出服务器请求indexSearchableItems(可能不是最好的方法,但目前最简单,因为我仍处于弄清楚这一点的开始阶段)。

下面是我的代码。

var searchableItems: [CSSearchableItem] = []
for i in 0 ..< self._ids.count {
    let attributeSet = CSSearchableItemAttributeSet(itemContentType: kUTTypeItem as String)

    attributeSet.title = self._titles[i]
    attributeSet.contentDescription = self._descriptions[i]

    let searchableItem = CSSearchableItem(uniqueIdentifier: self._ids[i], domainIdentifier: "com.domain.appname", attributeSet: attributeSet)

    searchableItems.append(searchableItem)
}

CSSearchableIndex.default().deleteAllSearchableItems(completionHandler: { (error) -> Void in
    CSSearchableIndex.default().indexSearchableItems(searchableItems) { (error) -> Void in
        if error != nil {
            print(error?.localizedDescription ?? "Error")
        }
    }
})

我要做的下一件事是我有一组名为self._images. 该数组内部是一个空字符串 ( "") 或一个 URL 字符串 ( "https://mywebsite.com/image.png")。

目前我可以使用下面的代码将 UIImageView 的图像设置为该 URL 处的图像,使用AlamofireImage

if (self._images[0] != "") {
    myImageView.af_setImage(withURL: URL(string: self._images[0])!)
}

如何将相同的图像放入 CoreSpotlight 属性集中?

我假设attributeSet.thumbnailData.af_setImage(withURL: imagedownloadURL)不会工作。特别是因为它是异步的。

整合它的最简单方法是什么?

我已经尝试了一些东西,例如 for 循环等,但最大的问题是异步,我似乎无法找到解决方案,因为它是异步的,而且我有多个项目。

非常感谢您!

4

2 回答 2

6

最简单的解决方案,但不是最好的...

DispatchQueue.global(qos: .userInitiated).async {
    var searchableItems: [CSSearchableItem] = []
    for i in 0 ..< self._ids.count {
        let attributeSet = CSSearchableItemAttributeSet(itemContentType: kUTTypeItem as String)
        attributeSet.title = self._titles[i]
        attributeSet.contentDescription = self._descriptions[i]
        let searchableItem = CSSearchableItem(uniqueIdentifier: self._ids[i], domainIdentifier: "com.domain.appname", attributeSet: attributeSet)

        if let imageUrl = URL(string: self._images[i]) {
            let urlRequest = URLRequest.init(url: imageUrl)
            if let cashedImage = UIImageView.af_sharedImageDownloader.imageCache?.image(for: urlRequest, withIdentifier: nil) {
                if let data = UIImageJPEGRepresentation(cashedImage, 1.0) {
                    attributeSet.thumbnailData = data
                }
            } else {
                if let data = NSData.init(contentsOf: imageUrl) {
                    attributeSet.thumbnailData = data as Data
                    if let image = UIImage.init(data: data as Data) {
                        let urlRequest = URLRequest.init(url: imageUrl)
                        UIImageView.af_sharedImageDownloader.imageCache?.add(image, for: urlRequest, withIdentifier: nil)
                    }
                }
            }
        }
        searchableItems.append(searchableItem)
    }

    CSSearchableIndex.default().deleteAllSearchableItems(completionHandler: { (error) -> Void in
        CSSearchableIndex.default().indexSearchableItems(searchableItems) { (error) -> Void in
            if error != nil {
                print(error?.localizedDescription ?? "Error")
            }
        }
    })
}

已编辑

为什么不是好的解决方案?为了避免异步问题,我只是将所有内容放在一个异步闭包中并进行同步图像下载。

更新

添加了默认的 Alamofire 兑现,在下载之前首先查看已兑现的图像。

于 2017-05-26T06:51:56.523 回答
5

这是使用 thumbnailURL 或 thumbnailData 的更好解决方案,因为您有一个缓存的图像抛出无论 alamofire、SDWebImage ...

简单而完美的表现

对于 thumbnailURL它只接受本地路径:

  1. 获取缓存图像的文件路径
  2. 从带有“file://”的文件路径前缀创建一个 URL 对象

“文件://”。表示它的本地路径

    if let path = SDImageCache.shared().defaultCachePath(forKey: item.url_thumbnail), let url = URL(string: "file://\(path)") {
        attributeSet.thumbnailURL = url
    }

对于 thumbnailData

只需分配本地 url 路径

attributeSet.thumbnailData = try? Data(contentsOf: url)

调试并了解正在发生的事情

 do {
     attributeSet.thumbnailData = try Data(contentsOf: url)
 }catch (let error as NSError){
      print(error)
 }

请注意我正在使用 SDWebImage 来获取缓存图像的路径,因为我正在使用它来缓存照片

于 2018-02-06T07:59:50.493 回答