iOS9 App Search之NSUserActivity与Core Spotlight

最近因为项目要求,开始研究起iOS9推出的feature--App Search APIs。何为App Search?在iOS9之前,用户通过spotlight只能搜索app的名字,或者苹果自带app的一些内容,比如搜索通讯录、备忘录中的内容。iOS9开始,用户可以通过spotlight搜索网页内容或者任何app内的数据。

1、search APIs主要包括三部分

1.1、NSUserActivity

NSUserActivity在iOS8中就出现了,当时用于Handoff。
在iOS9中,我们可以把想要在spotlight中搜出来的内容放到NSUserActivity中。userActivityUIResponser的属性,通常会在用户访问某一个页面时,对UIViewControlleruserActivity属性赋值。
NSUserActivity可以用来对历史访问记录

1.2、Core Spotlight

通过该技术,用户可以通过spotlight搜到app中曾经出现过或者现有的所有内容。

1.3、Web Markup

当你web页面添加了标记语言,苹果的爬虫会根据用户在spotlight的输入,去抓取你网站的数据,如果有找到合适的结果,你的网站就可能会(苹果有自己的权重计算方法)显示在spotlight的结果列表中。这时候如果安装了相对应的app并且支持deep link,用户就可以打开对应的app内容页面。
对于经常使用spotlight的用户来说,这会很有助于提升app的曝光度。
苹果官方推荐Web Markup结合smart app banneruniversal links使用。
这次花了很多时间在搞的这个东西,不难搞,但是需要web端和服务端配合。重要的是,你的web站点要支持https访问,我们公司的还不支持,纠结再三之后,暂时放弃这个功能。

在本文会简单记录第一、二部分内容的实现。至于第三部分内容,虽然因为https不支持,无法成功实现,但是还是会在下文中做一个学习整理。

2、NSUserActivity

2.1、创建NSUSerActivity实例

在viewModel层中创建的NSUSerActivity实例

- (NSUserActivity *)serviceProjectUserActivity
{
    if (!_serviceProjectUserActivity) {
        _serviceProjectUserActivity = [[NSUserActivity alloc] initWithActivityType:@"com.xxxx.appIdentifier.serviceProject"];
        _serviceProjectUserActivity.title = self.projectEntity.projectTitle ?: @"";
        _serviceProjectUserActivity.userInfo = @{@"id" : self.projectEntity.projectId ?: @""};
        _serviceProjectUserActivity.keywords = [NSSet setWithObjects:self.projectEntity.destination ?: @"", nil];
        // 好像并没有什么卵用,不知道哪里用错了
        _serviceProjectUserActivity.expirationDate = [NSDate dateWithTimeIntervalSinceNow:60 * 60 * 24 *31];
        // 如果没有显示设置为yes,则不可搜索到
        _serviceProjectUserActivity.eligibleForSearch = YES;
        //
        CSSearchableItemAttributeSet *attributeSet = [[CSSearchableItemAttributeSet alloc] initWithItemContentType:CFBridgingRelease(kUTTypeContact)];
        attributeSet.contentDescription = self.projectEntity.features ?: @"";
        // ??????网络路径不可显示
        attributeSet.thumbnailURL = [NSURL URLWithString:self.projectEntity.backImgUrl ?: @""];
        // 防止NSUserActivity和Core Spotlight可能重复索引,这里设置为nil
        attributeSet.relatedUniqueIdentifier = nil;
        _serviceProjectUserActivity.contentAttributeSet = attributeSet;
    }
    return _serviceProjectUserActivity;
}

2.2、赋值给UIViewController的userActivity属性

在viewController层监测,如果页面数据请求完成,给userActivity属性赋值

 __weak typeof(self)weakSelf = self;
    [RACObserve(self.viewModel, projectEntity) subscribeNext:^(id x) {
        if (x) {
            __strong __typeof(weakSelf)strongSelf = weakSelf;
            strongSelf.userActivity = strongSelf.viewModel.serviceProjectUserActivity;
        }
    }];

2.3、在Appdelegate中处理页面跳转

- (BOOL)application:(UIApplication *)application continueUserActivity:(NSUserActivity *)userActivity restorationHandler:(void (^)(NSArray * _Nullable))restorationHandler
{
    if ([userActivity.activityType isEqualToString:@"com.xxxx.appIdentifier.serviceProject"]) {
        NSString *projectID = userActivity.userInfo[@"id"];
        if (projectID.length > 0) {
            // 处理具体页面跳转
            return YES;
        }
        return NO;
    }
    return NO;
}

3、Core Spotlight

3.1、创建要搜索的项,并将所有的项加入默认索引空间

- (void)indexAllDomesticCityForAppSearch
{
    // 把国内一级城市从数据库中拿出来
    NSString *sql = [NSString stringWithFormat:@"select cn, zone_id from %@ where rank == '1' and zone_id <= 900000", self.tableName];
    
    [self loadDestinationWithSQL:sql andResultSetHandler:^(FMResultSet *result) {
        NSMutableArray *searchableItems = [[NSMutableArray alloc] init];
        while ([result next]) {
            NSString *zoneName = [result stringForColumnIndex:0];
            NSString *zoneID = [result stringForColumnIndex:1];
           // 创建对应的CSSearchableItem
            CSSearchableItemAttributeSet *attributeSet = [[CSSearchableItemAttributeSet alloc] initWithItemContentType:kUTTypeContent];
            attributeSet.title = zoneName;
            attributeSet.contentDescription = @"专业旅游";
            // 这个属性主要是将NSUserActivity与Core Spotlight indexed object进行一个关联,防止出现重复的内容(如果出现重复内容,是因为开始的时候测试NSUserActivity的时候没有设置id,还原一下模拟器就好了)
            attributeSet.relatedUniqueIdentifier = zoneID;
            CSSearchableItem *item = [[CSSearchableItem alloc] initWithUniqueIdentifier:zoneID domainIdentifier:@"com.xxxx.appIndetifier.destinations" attributeSet:attributeSet];
            [searchableItems addObject:item];
        }
         // 所有的items加入索引
        CSSearchableIndex *defaultSearchableIndex = [CSSearchableIndex defaultSearchableIndex];
        [defaultSearchableIndex indexSearchableItems:[searchableItems copy] completionHandler:^(NSError * _Nullable error) {
            
        }];
    }];
}

3.2、在Appdelegate中处理从spotlight打开app

- (BOOL)application:(UIApplication *)application continueUserActivity:(NSUserActivity *)userActivity restorationHandler:(void (^)(NSArray * _Nullable))restorationHandler
{
    if ([userActivity.activityType isEqualToString:@"com.xxxx.appIndetifier.destinations"]) {
        NSString *zoneID = userActivity.userInfo[CSSearchableItemActivityIdentifier];
        if (zoneID.length > 0) {
            // 处理具体页面跳转
            return YES;
        }
        return NO;
    }
    return NO;
}

  • iOS9 by Tutorials 学习笔记二:App Search

你可能感兴趣的:(iOS9 App Search之NSUserActivity与Core Spotlight)