使用 Swift 在 iOS 10 中集成 Siri —— SiriKit 教程

简介: 使用 Swift 在 iOS 10 中集成 Siri —— SiriKit 教程 转载地址:http://swift.gg/2016/06/28/adding-siri-to-ios-10-apps-in-swift-tutorial/ 下载 Xcode 8,配置 iOS 10 和 Swift 3 (可选)通过命令行编译 除非你想使用命令行编译,使用 Swift 3.0 的工具链并不需要对项目做任何改变。

使用 Swift 在 iOS 10 中集成 Siri —— SiriKit 教程

转载地址:http://swift.gg/2016/06/28/adding-siri-to-ios-10-apps-in-swift-tutorial/

下载 Xcode 8,配置 iOS 10 和 Swift 3

(可选)通过命令行编译

除非你想使用命令行编译,使用 Swift 3.0 的工具链并不需要对项目做任何改变。如果你想的话,打开 Xcode-beta,然后从顶部菜单栏中选择 Xcode > Preferences,接着选择 Location,在页面的底部,你会看到「Command Line Tool」这行设置,请在这里选择 Xcode 8.0。

现在,在 Terminal 使用命令行找到工程所在的文件夹,调用 xcodebuild 命令就可以编译工程了。

(可选)移植现有的 Swift 2 应用

如果你想对一个已使用 Swift 2.0 开发的工程引入 Siri 功能,需要点击工程,选择 Build Settings,在 Swift Compiler - Version 下面,找到 Use Legacy Swift Language Version 选项,设置成 No。这会造成编译器报错,然后你可以根据这些报错信息来修改代码,推荐你使用这个设置来更新代码,以适应 Swift 不断进化的语义。

开始使用 SiriKit

首先,在你的 App(或者是新建一个单视图的 Swift 模板工程),点击顶部的工程,然后点击左侧下方的 + 按钮,在这里(译者注:我在这里添加了一张图片,能够说的更明白)点击。

弹出的窗口中,选择 iOS > Application Extension,接着选择 Intents Extension。

这样就给工程添加了一个新的 intent,用于监听 Siri 的命令。其中的 Product Name 应该和你的工程文件名字相似,比如,你的 App 名为 MusicMatcher,你可以把这个 intent 的名字命名为 MusicMatcherSiriIntent。一定要选中 Include UI Extension 选项,我们之后会用到,这也是添加额外扩展的最简单的方法。

我刚刚创建的两个新 target 可以从项目的文件层级上找到。找到 Intent 文件夹下的 IntentHandler.swift 文件,看一下这里面的样本代码。默认会提供一些示例代码,允许用户说一下诸如「用 MusicMatcher 开始锻炼」的命令,MusicMatcher 是 App 的名字。

像这样运行示例应用

这个时候最好编译一下代码,然后在 iOS 真机上试一下命令。继续,编译应用的 target,从 Scheme 下拉菜单里选择 MusicMatcher,然后选择真机,点击 Run。

你看你会看到一个空白的应用出现,你使用的扩展这时会在后台加载到设备的系统文件里,现在点击 Stop 按钮来关闭应用。

接下来,找到你的 scheme,选择 Intent target,点击 Run。

这时会出现一个弹出框,问你需要连接哪个应用,选择你刚刚运行的应用:MusicMatcher。这会让真机上再次出现这个应用(还是一个空白的应用),不过这次调试台(debugger)中会出现连接的 Intent 扩展。

现在点击 home 按钮回到首屏,或者应用可能自己就退出了,因为你正在运行的是 Intent,不是应用本身(这不是崩溃!!!)。

启用扩展

扩展都已安装就位了,但是作为一个 iOS 用户,仍然需要进行 Siri 设置才能使用扩展。点击测试设备里的 Settings,选择 Siri 菜单,你会看到 MusicMatcher 出现在清单里,激活允许使用 Siri。

测试我们第一个 Siri 命令

尝试一下 Siri 命令,长按 Home 键或者说出「Hey Siri」来激活 Siri(当然需要你已经激活「Hey Siri」功能)。

试一下命令,比如「使用 MusicMatcher 开始锻炼」。

「对不起,你需要在应用里继续。」

如果你像我一样遇到了这样的错误信息:「Sorry, you’ll need to continue in the app.」(不知道什么原因,偶尔会出现这么一个问题,什么鬼?)

在控制台中你可能会看到类似的信息:

 

[objc]  view plain  copy
 在CODE上查看代码片派生到我的代码片
  1. dyld: Library not loaded: @rpath/libswiftCoreLocation.dylib  
  2.   Referenced from: /private/var/containers/Bundle/Application/CC815FA3-EB04-4322-B2BB-8E3F960681A0/LockScreenWidgets.app/PlugIns/JQIntentWithUI.appex/JQIntentWithUI  
  3.   Reason: image not found  
  4. Program ended with exit code: 1  

 

再次选择工程根目录,选择 MusicMatcher target。在 General 底下找到 Linked Frameworks and Libraries。点击 + 按钮,添加 CoreLocation.framework。现在可以再次编译在真机上运行,接着照着上面相同的步骤再次编译运行 intent target。我们还需要在工程里添加 CoreLocation 库,确保能添加到我们编译过的 Swift 工程中。

最后,从手机桌面激活 Siri。

「Hey Siri!」
「Start my workout using MusicMatcher(使用 MusicMatcher 开始锻炼)」

Siri 这时候应该会回应:「OK. exercise started on MusicMatcher(OK,开始用 MusicMatcher 锻炼身体)」,然后会出现一个 UI 界面写着「Workout Started(锻炼开始)」。

它是如何工作的呢?

模板中的 IntentHandler 类使用了一长串的协议:

首先最主要的就是 INExtension,允许我们一开始就把类当作一个 intent extension 来用。剩下的协议都是 intent handler 类型,在类里能够回调:

 

[objc]  view plain  copy
 在CODE上查看代码片派生到我的代码片
  1. INStartWorkoutIntentHandling  
  2. INPauseWorkoutIntentHandling  
  3. INResumeWorkoutIntentHandling  
  4. INCancelWorkoutIntentHandling  
  5. INEndWorkoutIntentHandling  

按住 Command 键点击这些协议的名字,会看到苹果提供的文档:第一个就是我们刚刚测试过的,INStartWorkoutIntentHandling

[objc]  view plain  copy
 在CODE上查看代码片派生到我的代码片
  1. /*! 
  2.  @brief Protocol to declare support for handling an INStartWorkoutIntent  
  3.  @abstract By implementing this protocol, a class can provide logic for resolving, confirming and handling the intent. 
  4.  @discussion The minimum requirement for an implementing class is that it should be able to handle the intent. The resolution and confirmation methods are optional. The handling method is always called last, after resolving and confirming the intent. 
  5.  */  

 

这会根据用户使用语言的不同而不同,不过最终的目的都是开始一次锻炼。INStartWorkoutIntentHandling 协议调用的几个方法都在示例代码里实现了。如果你想创建一个锻炼应用,你可以自行了解其他的内容。不过在这篇教程的剩余部分,我会添加一个新的 intent handler,来处理发送消息。换句话说,这协议告诉 SiriKit 我们准备处理英文句子「Start my workout with AppName Here.」

添加一个新的消息 Intent

确认应用可以完美运行后,让我们继续,添加一个新的 intent 类型,用于发送消息,这里的文档说明了下列信息:

 

[objc]  view plain  copy
 在CODE上查看代码片派生到我的代码片
  1. Send a message  
  2. Handler:INSendMessageIntentHandling protocol  
  3. Intent:INSendMessageIntent  
  4. Response:INSendMessageIntentResponse  

 

在类里添加 INSendMessageIntentHandling 协议。首先要明确,我们把它添加到类协议清单里,也就是在 IntentHandler.swift 文件里。由于实际上我不想使用这些 intent,所以我会删除它们,只留下这一个:

 

[objc]  view plain  copy
 在CODE上查看代码片派生到我的代码片
  1. class IntentHandler: INExtension, INSendMessageIntentHandling {  
  2.     ...  

 

另外,如果你需要核对具体是哪些方法,只需要按住 Command 键然后鼠标点击 INSendMessageIntentHandling,然后看一下哪些方法前面没有 optional关键词即可。如果这时候编译,是不会通过编译的,因为我们还需要实现一些遵守 INSendMessageIntentHandling 协议所必需的方法。

在这里,我们发现只有一个必须实现的方法:

[objc]  view plain  copy
 在CODE上查看代码片派生到我的代码片
  1. /*! 
  2.  @brief handling method 
  3.   
  4.  @abstract Execute the task represented by the INSendMessageIntent that's passed in 
  5.  @discussion This method is called to actually execute the intent. The app must return a response for this intent. 
  6.   
  7.  @param  sendMessageIntent The input intent 
  8.  @param  completion The response handling block takes a INSendMessageIntentResponse containing the details of the result of having executed the intent 
  9.   
  10.  @see  INSendMessageIntentResponse 
  11.  */  
  12. public func handle(sendMessage intent: INSendMessageIntent, completion: (INSendMessageIntentResponse) -> Swift.Void)  

回到 IntentHandler.swift 文件,添加一行分隔符(借助 jump bar,在导航查找代码时这个分隔符会非常有用)遵守新消息意图协议

// MARK: - INSendMessageIntentHandling

在 MARK 底下,我们来实现方法。我发现 Xcode 8 非常有用,通过敲击方法名字的开始部分,剩下的都能交给自动补全来完成了,然后选择对应的方法。

在 handler 里,我们需要创建一个 INSendMessageIntentResponse,来回调闭包。先假设所有的信息发送都很成功,在 INSendMessageIntentResponse 里返回一个用户活动的成功值,和默认模板中的实现非常类似。还需要添加一个 print 方法,当 handler 方法被 Siri 事件触发后我们就能知晓啦:

[objc]  view plain  copy
 在CODE上查看代码片派生到我的代码片
  1. func handle(sendMessage intent: INSendMessageIntent, completion: (INSendMessageIntentResponse) -> Void) {  
  2.     print("Message intent is being handled.")  
  3.     let userActivity = NSUserActivity(activityType: NSStringFromClass(INSendMessageIntent))  
  4.     let response = INSendMessageIntentResponse(code: .success, userActivity: userActivity)  
  5.     completion(response)  
  6. }  

把这个 intent 类型添加到 Info.plist

在具备处理 INSendMessageIntent 方法之前,我们需要在 Info.plist 文件里添加一些值,就当作是应用的授权吧。

在 intent 的 Info.plist 文件里,找到并点开 NSExtension 键。接着点开 NSExtensionAttributes,然后是 IntentsSupported,我们需要给INSendMessageIntent 新添加一行,允许应用处理信息 intents。

测试新的 intent

现在我们已经设置好了新的 intent,来测试一下。记住,你必须先编译 App,在真机上运行,接着运行扩展进行调试,如果你不这样做,扩展要么不会工作,要么不会在 Xcode 的控制台中打印日志。

调用 Siri 的 intent,你现在可以看到会出现一个新的信息窗口,这个窗口目前还是空的,毕竟我们还没有给应用编写什么逻辑,我们需要实现剩下的调用,还要添加一些信息的逻辑,实现更好的用户体验。我们会在 已经发布的 Part 2 里解决这些事情。

相关文章
|
7月前
|
定位技术 开发工具 iOS开发
百宝箱开放平台 ✖️ iOS 集成说明
本文介绍百宝箱智能体与友盟+ iOS 应用集成方法,涵盖手动集成、权限配置、SDK 初始化、日志查看及效果验证。提供依赖库添加、Info.plist 权限声明、channel 规范、demo 工程下载与运行说明,助您快速完成接入。
363 1
|
运维 监控 安全
Cisco ISR 4000 Series IOS XE 17.18.1a ED 发布 - 思科 4000 系列集成服务路由器 IOS XE 系统软件
Cisco ISR 4000 Series IOS XE 17.18.1a ED - 思科 4000 系列集成服务路由器 IOS XE 系统软件
281 0
|
12月前
|
人工智能 安全 Shell
Jupyter MCP服务器部署实战:AI模型与Python环境无缝集成教程
Jupyter MCP服务器基于模型上下文协议(MCP),实现大型语言模型与Jupyter环境的无缝集成。它通过标准化接口,让AI模型安全访问和操作Jupyter核心组件,如内核、文件系统和终端。本文深入解析其技术架构、功能特性及部署方法。MCP服务器解决了传统AI模型缺乏实时上下文感知的问题,支持代码执行、变量状态获取、文件管理等功能,提升编程效率。同时,严格的权限控制确保了安全性。作为智能化交互工具,Jupyter MCP为动态计算环境与AI模型之间搭建了高效桥梁。
747 2
Jupyter MCP服务器部署实战:AI模型与Python环境无缝集成教程
|
缓存 前端开发 API
(网页系统集成CAD功能)在线CAD中配置属性的使用教程
本文介绍了Mxcad SDK在线预览和编辑CAD图纸的功能及配置方法。通过Vite、CDN或Webpack实现集成,用户可自定义设置以满足项目需求。主要内容包括:1)`createMxCad()`方法的初始属性配置,如画布ID、WASM文件路径、字体加载路径等;2)`MxFun.setIniset()`方法提供的更多CAD初始配置;3)`McObject`对象API用于动态调整视图背景色、浏览模式等。此外,还提供了在线Demo(https://demo2.mxdraw3d.com:3000/mxcad/)供用户测试实时效果。
|
安全 Java API
【三方服务集成】最新版 | 阿里云短信服务SMS使用教程(包含支持单双参数模板的工具类,拿来即用!)
阿里云短信服务提供API/SDK和控制台调用方式,支持验证码、通知、推广等短信类型。需先注册阿里云账号并实名认证,然后在短信服务控制台申请资质、签名和模板,并创建AccessKey。最后通过Maven引入依赖,使用工具类发送短信验证码。
7689 3
【三方服务集成】最新版 | 阿里云短信服务SMS使用教程(包含支持单双参数模板的工具类,拿来即用!)
|
存储 Java 开发工具
【三方服务集成】最新版 | 阿里云OSS对象存储服务使用教程(包含OSS工具类优化、自定义阿里云OSS服务starter)
阿里云OSS(Object Storage Service)是一种安全、可靠且成本低廉的云存储服务,支持海量数据存储。用户可通过网络轻松存储和访问各类文件,如文本、图片、音频和视频等。使用OSS后,项目中的文件上传业务无需在服务器本地磁盘存储文件,而是直接上传至OSS,由其管理和保障数据安全。此外,介绍了OSS服务的开通流程、Bucket创建、AccessKey配置及环境变量设置,并提供了Java SDK示例代码,帮助用户快速上手。最后,展示了如何通过自定义starter简化工具类集成,实现便捷的文件上传功能。
5520 7
【三方服务集成】最新版 | 阿里云OSS对象存储服务使用教程(包含OSS工具类优化、自定义阿里云OSS服务starter)
|
安全 数据处理 Swift
深入探索iOS开发中的Swift语言特性
本文旨在为开发者提供对Swift语言在iOS平台开发的深度理解,涵盖从基础语法到高级特性的全面分析。通过具体案例和代码示例,揭示Swift如何简化编程过程、提高代码效率,并促进iOS应用的创新。文章不仅适合初学者作为入门指南,也适合有经验的开发者深化对Swift语言的认识。
468 9
|
安全 Swift iOS开发
Swift 与 UIKit 在 iOS 应用界面开发中的关键技术和实践方法
本文深入探讨了 Swift 与 UIKit 在 iOS 应用界面开发中的关键技术和实践方法。Swift 以其简洁、高效和类型安全的特点,结合 UIKit 丰富的组件和功能,为开发者提供了强大的工具。文章从 Swift 的语法优势、类型安全、编程模型以及与 UIKit 的集成,到 UIKit 的主要组件和功能,再到构建界面的实践技巧和实际案例分析,全面介绍了如何利用这些技术创建高质量的用户界面。
491 2
|
Swift iOS开发 UED
如何使用Swift和UIKit在iOS应用中实现自定义按钮动画
【10月更文挑战第18天】本文通过一个具体案例,介绍如何使用Swift和UIKit在iOS应用中实现自定义按钮动画。当用户按下按钮时,按钮将从圆形变为椭圆形并从蓝色渐变为绿色;释放按钮时,动画恢复原状。通过UIView的动画方法和弹簧动画效果,实现平滑自然的动画过渡。
313 5
|
Swift iOS开发 UED
如何使用Swift和UIKit在iOS应用中实现自定义按钮动画
本文通过一个具体案例,介绍如何使用Swift和UIKit在iOS应用中实现自定义按钮动画。当用户点击按钮时,按钮将从圆形变为椭圆形,颜色从蓝色渐变到绿色;释放按钮时,动画以相反方式恢复。通过UIView的动画方法和弹簧动画效果,实现平滑自然的过渡。
366 1