人间清醒特约评论官 發表於 2022-11-19 00:16:00

【转】iOS开发-SiriKit应用

<p>原文网址:iOS开发-SiriKit应用 - 简书 (jianshu.com)</p>
<div>
<div>
<h1>关于SiriKit</h1>
<p>在6月14日凌晨的<code>WWDC2016</code>大会上,苹果提出<code>iOS10</code>是一次里程碑并且推出了十个新特性,大部分的特性是基于<code>iPhone</code>自身的原生应用的更新,具体的特性笔者不在这里再次叙述,请看客们移步WWDC2016下载自行观赏。要说里程碑在笔者看来有些夸大其实了,不过新增的通知中心联动<code>3D Touch</code>确实为人机交互带来新的发展,另外一个最大的亮点在于<code>Siri</code>的接口开放。在<code>iOS10</code>中提供了<code>SiriKit</code>框架在用户使用<code>Siri</code>的时候会生成<code>INExtension</code>对象来告知我们的应用,通过实现方法来让<code>Siri</code>获取应用想要展示给用户的内容<br>
</p>
<div class="image-package">
<div class="image-container">
<div class="image-container-fill">&nbsp;</div>
<div class="image-view" data-width="424" data-height="220"><img src="//upload-images.jianshu.io/upload_images/783864-6faafd363bba795c.png?imageMogr2/auto-orient/strip|imageView2/2/w/424/format/webp"></div>


</div>
<div class="image-caption">Siri服务</div>


</div>
<p>&nbsp;</p>
<p>在<code>iOS10</code>之后,苹果希望<code>Siri</code>能够给用户带来更多的功能体验,基于这个出发点,新增了<code>SiriKit</code>框架。<code>Siri</code>通过语言处理系统对用户发出的对话请求进行解析之后生成一个用来描述对话内容的<code>Intents</code>事件,然后通过<code>SiriKit</code>框架分发给集成框架的应用程序以此来获取应用的内容,比如完成类似<code>通过文字匹配查找应用聊天记录、聊天对象</code>的功能,此外它还支持<code>为用户使用苹果地图时提供应用内置服务</code>等功能。通过官方文档我们可以看到<code>SiriKit</code>框架支持的六类服务分别是:</p>
<ul>
<li>语音和视频通话</li>
<li>发送消息</li>
<li>收款或者付款</li>
<li>图片搜索</li>
<li>管理锻炼</li>
<li>行程预约</li>


</ul>
<p><code>Siri</code>和<code>Maps</code>通过<code>Intents extension</code>的扩展方式和我们的应用进行交互,其中,类型为<code>INExtension</code>的对象扮演着<code>Intents extension</code>扩展中直接协同<code>Siri</code>对象共同响应用户请求的关键角色。当我们实现了<code>Intents extension</code>扩展并产生了一个<code>Siri</code>请求事件时,一个典型的<code>Intent</code>事件的处理过程中总共有这三个步骤<code>Resolve</code>、<code>Confirm</code>和<code>Handle</code>:</p>
<ul>
<li>
<p><code>Resolve</code>阶段。在<code>Siri</code>获取到用户的语音输入之后,生成一个<code>INIntent</code>对象,将语音中的关键信息提取出来并且填充对应的属性。这个对象在稍后会传递给我们设置好的<code>INExtension</code>子类对象进行处理,根据子类遵循的不同服务<code>protocol</code>来选择不同的解决方案</p>

</li>
<li>
<p><code>Confirm</code>阶段。在上一个阶段通过<code>handler(for intent:)</code>返回了处理<code>intent</code>的对象,此阶段会依次调用<code>confirm</code>打头的实例方法来判断<code>Siri</code>填充的信息是否完成。匹配的判断结果包括<code>Exactly one match</code>、<code>Two or more matches</code>以及<code>No match</code>三种情况。这个过程中可以让<code>Siri</code>向用户征求更具体的参数信息</p>

</li>
<li>
<p>在<code>confirm</code>方法执行完成之后,<code>Siri</code>进行最后的处理阶段,生成答复对象,并且向此<code>intent</code>对象确认处理结果然后执显示结果给用户看</p>

</li>


</ul>
<p>&nbsp;</p>
<div class="image-package">
<div class="image-container">
<div class="image-container-fill">&nbsp;</div>
<div class="image-view" data-width="1299" data-height="843"><img src="//upload-images.jianshu.io/upload_images/783864-fdab2099f6e155bf.png?imageMogr2/auto-orient/strip|imageView2/2/w/1200/format/webp"></div>


</div>
<div class="image-caption">&nbsp;</div>


</div>

<br>
具体的执行过程请参考文档和讲解视频
<p>&nbsp;</p>
<h1>创建Intents Extension</h1>
<p><code>SiriKit</code>通过添加<code>App Extension</code>的方式来完成集成,这是一种独立于应用本身运行的代码结构,作为应用的扩展功能,只有在需要的时候系统会唤醒这些<code>Extension</code>代码来执行任务,然后在执行完毕之后将其杀死。另一方面,这些<code>Extension</code>在运行过程中的可占用内存是较少的,并且由于调用时机的限制,我们也无法在运行期间做一些<code>坏事</code><br>
</p>
<div class="image-package">
<div class="image-container">
<div class="image-container-fill">&nbsp;</div>
<div class="image-view" data-width="780" data-height="342"><img src="//upload-images.jianshu.io/upload_images/783864-403304791676e9e8.png?imageMogr2/auto-orient/strip|imageView2/2/w/780/format/webp"></div>


</div>
<div class="image-caption">&nbsp;</div>


</div>

<br>
现阶段集成<code>SiriKit</code>的条件是需要将开发工具升级到<code>Xcode8</code>,需要使用开发者账号到官方网站去下载<code>Xcode8_beta</code>版,并且需要将一台测试设备升级到<code>iOS10</code>系统。选中我们的应用,进入项目总览界面,新增一个<code>TARGET</code><br>
<div class="image-package">
<div class="image-container">
<div class="image-container-fill">&nbsp;</div>
<div class="image-view" data-width="824" data-height="1418"><img src="//upload-images.jianshu.io/upload_images/783864-417b31c0dcca0ac6.png?imageMogr2/auto-orient/strip|imageView2/2/w/824/format/webp"></div>


</div>
<div class="image-caption">&nbsp;</div>


</div>

<br>
<div class="image-package">
<div class="image-container">
<div class="image-container-fill">&nbsp;</div>
<div class="image-view" data-width="1460" data-height="1036"><img src="//upload-images.jianshu.io/upload_images/783864-f7ff8f29e73ea706.png?imageMogr2/auto-orient/strip|imageView2/2/w/1200/format/webp"></div>


</div>
<div class="image-caption">&nbsp;</div>


</div>

<br>
如上图所示,我创建的<code>Intents Extension</code>被我命名为<code>LXDSiriExtension</code>。记住在创建好一个<code>Extension</code>的时候,会询问你是否激活这个扩展,勾选是。另外还会提示你是否连同<code>Intents UI Extension</code>一并创建了,我们同样选是。这样我们在项目下面总共创建了<code>LXDSiriExtension</code>和<code>LXDSiriExtensionUI</code>两个<code>TARGET</code>,这两个文件目录下面分别存在着一个新的<code>info.plist</code>文件,这个文件用来设置<code>intent</code>事件发生时我们设置的处理类。这里借用<code>WWDC</code>在讲解时的一张<code>ppt</code>来了解:<br>
<div class="image-package">
<div class="image-container">
<div class="image-container-fill">&nbsp;</div>
<div class="image-view" data-width="1666" data-height="736"><img src="//upload-images.jianshu.io/upload_images/783864-c3cb94b6d018a498.png?imageMogr2/auto-orient/strip|imageView2/2/w/1200/format/webp"></div>


</div>
<div class="image-caption">&nbsp;</div>


</div>

<br>
按图中的层次展开,<code>IntentsSupported</code>和<code>IntentsRestrictedWhileLocked</code>分别是两个字符串数组,每一个字符串表示的是应用扩展处理的<code>intent</code>事件的类名。前者表示支持的事件类型,后者表示在非锁屏状态下执行的事件类型。文件默认是<code>workout</code>类型的事件,在这里笔者改成了发送消息<code>INSendMessageIntent</code>。除此之外,<code>NSExtensionPrincipalClass</code>对应的是<code>INExtension</code>子类类名,这个类用来获取处理<code>intent</code>事件的类。<br>
<div class="image-package">
<div class="image-container">
<div class="image-container-fill">&nbsp;</div>
<div class="image-view" data-width="1684" data-height="302"><img src="//upload-images.jianshu.io/upload_images/783864-9f6751495a63cf7a.png?imageMogr2/auto-orient/strip|imageView2/2/w/1200/format/webp"></div>


</div>
<div class="image-caption">plist设置</div>


</div>

<br>
另外,官方讲解中提到了<code>Embedded frameworks</code>,在<code>session</code>中苹果开发人员通过一个消息聊天应用来示例集成<code>SiriKit</code>。由于应用扩展自身的运行机制和应用本身的运行机制不同,彼此之间创建的类是不能访问使用的。因此把我们需要的类开发成<code>frameworks</code>的方式导入我们的应用之后就能够在两种之中都使用到这些类。本文未使用<code>frameworks</code>导入功能,而是模拟了一个类用来管理事件处理过程中的部分逻辑,但是<code>Embedded frameworks</code>这个使用的准则需要记住。这个模拟类的具体代码如下:
<p>&nbsp;</p>
<div class="_2Uzcx_"><button class="VJbwyy" type="button"></button>
<pre class="line-numberslanguage-swift"><code class="language-swift"><span class="token keyword">import <span class="token builtin">Intents

<span class="token keyword">class <span class="token class-name">LXDMatch <span class="token punctuation">{
    <span class="token keyword">var handle<span class="token punctuation">: <span class="token builtin">String<span class="token operator">?
    <span class="token keyword">var displayName<span class="token punctuation">: <span class="token builtin">String<span class="token operator">?
    <span class="token keyword">var contactIdentifier<span class="token punctuation">: <span class="token builtin">String<span class="token operator">?

    <span class="token keyword">convenience <span class="token keyword">init<span class="token punctuation">(handle<span class="token punctuation">: <span class="token builtin">String<span class="token punctuation">, <span class="token number">_ displayName<span class="token punctuation">: <span class="token builtin">String<span class="token punctuation">, <span class="token number">_ contactIdentifier<span class="token punctuation">: <span class="token builtin">String<span class="token punctuation">) <span class="token punctuation">{
      <span class="token keyword">self<span class="token punctuation">.<span class="token keyword">init<span class="token punctuation">(<span class="token punctuation">)
      <span class="token keyword">self<span class="token punctuation">.handle <span class="token operator">= handle
      <span class="token keyword">self<span class="token punctuation">.displayName <span class="token operator">= displayName
      <span class="token keyword">self<span class="token punctuation">.contactIdentifier <span class="token operator">= contactIdentifier
    <span class="token punctuation">}

    <span class="token keyword">func <span class="token function">inPerson<span class="token punctuation">(<span class="token punctuation">) <span class="token operator">-<span class="token operator">&gt; <span class="token builtin">INPerson <span class="token punctuation">{
      <span class="token keyword">return <span class="token function">INPerson<span class="token punctuation">(handle<span class="token punctuation">: handle<span class="token operator">!<span class="token punctuation">, displayName<span class="token punctuation">: displayName<span class="token punctuation">, contactIdentifier<span class="token punctuation">: contactIdentifier<span class="token punctuation">)
    <span class="token punctuation">}
<span class="token punctuation">}

<span class="token keyword">class <span class="token class-name">LXDAccount <span class="token punctuation">{
    <span class="token keyword">private <span class="token keyword">static <span class="token keyword">let instance <span class="token operator">= <span class="token function">LXDAccount<span class="token punctuation">(<span class="token punctuation">)

    <span class="token keyword">private <span class="token keyword">init<span class="token punctuation">(<span class="token punctuation">) <span class="token punctuation">{
      <span class="token function">print<span class="token punctuation">(<span class="token string">"only call share() to get an instance of LXDAccount"<span class="token punctuation">)
    <span class="token punctuation">}

    <span class="token keyword">class <span class="token class-name">func <span class="token function">share<span class="token punctuation">(<span class="token punctuation">) <span class="token operator">-<span class="token operator">&gt; <span class="token builtin">LXDAccount <span class="token punctuation">{
      <span class="token keyword">return <span class="token builtin">LXDAccount<span class="token punctuation">.instance
    <span class="token punctuation">}

    <span class="token keyword">func <span class="token function">contact<span class="token punctuation">(matchingName<span class="token punctuation">: <span class="token builtin">String<span class="token punctuation">) <span class="token operator">-<span class="token operator">&gt; <span class="token punctuation">[<span class="token builtin">LXDMatch<span class="token punctuation">] <span class="token punctuation">{
      <span class="token keyword">return <span class="token punctuation">[<span class="token function">LXDMatch<span class="token punctuation">(handle<span class="token punctuation">: <span class="token function">NSStringFromClass<span class="token punctuation">(<span class="token builtin">LXDSendMessageIntentHandler<span class="token punctuation">.<span class="token function">classForCoder<span class="token punctuation">(<span class="token punctuation">)<span class="token punctuation">)<span class="token punctuation">, matchingName<span class="token punctuation">, matchingName<span class="token punctuation">)<span class="token punctuation">]
    <span class="token punctuation">}

    <span class="token keyword">func <span class="token function">send<span class="token punctuation">(message<span class="token punctuation">: <span class="token builtin">String<span class="token punctuation">, to recipients<span class="token punctuation">: <span class="token punctuation">[<span class="token builtin">INPerson<span class="token punctuation">]<span class="token punctuation">) <span class="token operator">-<span class="token operator">&gt; <span class="token builtin">INSendMessageIntentResponseCode <span class="token punctuation">{
      <span class="token function">print<span class="token punctuation">(<span class="token string">"Send a message: \"<span class="token interpolation"><span class="token delimiter variable">\(message<span class="token delimiter variable">)\" to <span class="token interpolation"><span class="token delimiter variable">\(recipients<span class="token delimiter variable">)"<span class="token punctuation">)
      <span class="token keyword">return <span class="token punctuation">.success
    <span class="token punctuation">}
<span class="token punctuation">}
</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
</div>
<p>在完成这些需要的工作之后,我们还需要对应用本身的<code>Info.plist</code>配置文件进行设置,新增一个关键字为<code>NSSiriUsageDescription</code>的字符串对象,对应填写的字符串将在我们征询用户<code>Siri</code>权限的时候显示给用户看。比如<code>Siri想要访问您的应用信息</code>之类的提示语。然后通过<code>INPreferences</code>类方法向用户请求<code>Siri</code>访问权限</p>
<div class="_2Uzcx_"><button class="VJbwyy" type="button"></button>
<pre class="line-numberslanguage-swift"><code class="language-swift"><span class="token keyword">import <span class="token builtin">Intents

<span class="token builtin">INPreferences<span class="token punctuation">.requestSiriAuthorization <span class="token punctuation">{
    <span class="token keyword">switch $<span class="token number">0 <span class="token punctuation">{
    <span class="token keyword">case <span class="token punctuation">.authorized<span class="token punctuation">:
      <span class="token function">print<span class="token punctuation">(<span class="token string">"用户已授权"<span class="token punctuation">)
      <span class="token keyword">break
            
    <span class="token keyword">case <span class="token punctuation">.notDetermined<span class="token punctuation">:
      <span class="token function">print<span class="token punctuation">(<span class="token string">"未决定"<span class="token punctuation">)
      <span class="token keyword">break

    <span class="token keyword">case <span class="token punctuation">.restricted<span class="token punctuation">:
      <span class="token function">print<span class="token punctuation">(<span class="token string">"权限受限制"<span class="token punctuation">)
      <span class="token keyword">break
            
    <span class="token keyword">case <span class="token punctuation">.denied<span class="token punctuation">:
      <span class="token function">print<span class="token punctuation">(<span class="token string">"拒绝授权"<span class="token punctuation">)
      <span class="token keyword">break
    <span class="token punctuation">}
<span class="token punctuation">}
</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
</div>
<h1>代码实现</h1>
<p>首先我们需要一个<code>INExtension</code>的子类,你也可以在默认创建的子类中实现代码。在方法中,我们通过判断<code>intent</code>的类型来创建对应的处理者实例,然后返回。在本文的示例中,假设我们对<code>Siri</code>说出这么一句话 <code>Siri,在微信上告诉我的家人们今天我不回去吃饭了</code>:</p>
<div class="_2Uzcx_"><button class="VJbwyy" type="button"></button>
<pre class="line-numberslanguage-swift"><code class="language-swift"><span class="token keyword">class <span class="token class-name">LXDIntentHandler<span class="token punctuation">: <span class="token builtin">INExtension <span class="token punctuation">{
    <span class="token keyword">override <span class="token keyword">func <span class="token function">handler<span class="token punctuation">(<span class="token keyword">for intent<span class="token punctuation">: <span class="token builtin">INIntent<span class="token punctuation">) <span class="token operator">-<span class="token operator">&gt; <span class="token builtin">AnyObject<span class="token operator">? <span class="token punctuation">{
   
      <span class="token keyword">if intent <span class="token keyword">is <span class="token builtin">INSendMessageIntent <span class="token punctuation">{
            <span class="token keyword">return <span class="token function">LXDSendMessageIntentHandler<span class="token punctuation">(<span class="token punctuation">)
      <span class="token punctuation">}
      <span class="token comment">//这里可以判断更多类型来返回
      <span class="token keyword">return <span class="token constant">nil
    <span class="token punctuation">}
<span class="token punctuation">}
</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
</div>
<p>通过判断<code>intent</code>事件是发送消息的聊天事件后,笔者创建了一个用来处理事件的<code>LXDSendMessageIntentHandler</code>类对象,并且返回。在对象创建完成之后需要完成<code>Resolve</code>、<code>Confirm</code>和<code>Handle</code>三个步骤,具体操作需要子类遵循实现<code>INSendMessageIntentHandling</code>协议来完成:</p>
<ul>
<li>
<p>Resolve阶段<br>
这个阶段需要我们找到消息的具体接收者。在这个过程中,可能会出现三种情况:<code>Exactly one match</code>、<code>Two or more matches</code>以及<code>No matches</code>,对于这三种情况的处理分别如下:</p>
<div class="_2Uzcx_"><button class="VJbwyy" type="button"></button>
<pre class="line-numberslanguage-swift"><code class="language-swift"><span class="token keyword">func <span class="token function">resolveRecipients<span class="token punctuation">(forSendMessage intent<span class="token punctuation">: <span class="token builtin">INSendMessageIntent<span class="token punctuation">, with completion<span class="token punctuation">: <span class="token punctuation">(<span class="token punctuation">[<span class="token builtin">INPersonResolutionResult<span class="token punctuation">]<span class="token punctuation">) <span class="token operator">-<span class="token operator">&gt; <span class="token builtin">Void<span class="token punctuation">) <span class="token punctuation">{
      <span class="token keyword">if <span class="token keyword">let recipients <span class="token operator">= intent<span class="token punctuation">.recipients <span class="token punctuation">{
          <span class="token keyword">var resolutionResults <span class="token operator">= <span class="token punctuation">[<span class="token builtin">INPersonResolutionResult<span class="token punctuation">]<span class="token punctuation">(<span class="token punctuation">)
          <span class="token keyword">forrecipient <span class="token keyword">in recipients <span class="token punctuation">{
            <span class="token keyword">let matches <span class="token operator">= <span class="token builtin">LXDAccount<span class="token punctuation">.<span class="token function">share<span class="token punctuation">(<span class="token punctuation">)<span class="token punctuation">.<span class="token function">contact<span class="token punctuation">(matchingName<span class="token punctuation">: recipient<span class="token punctuation">.displayName<span class="token punctuation">)
            <span class="token keyword">switch matches<span class="token punctuation">.<span class="token builtin">count <span class="token punctuation">{
            <span class="token keyword">case <span class="token number">2<span class="token punctuation">.<span class="token punctuation">.<span class="token punctuation">.<span class="token builtin">Int<span class="token punctuation">.<span class="token builtin">max<span class="token punctuation">:    <span class="token comment">//两个或更多匹配结果
                  <span class="token keyword">let disambiguations <span class="token operator">= matches<span class="token punctuation">.<span class="token builtin">map <span class="token punctuation">{ $<span class="token number">0<span class="token punctuation">.<span class="token function">inPerson<span class="token punctuation">(<span class="token punctuation">) <span class="token punctuation">}
                  resolutionResults<span class="token punctuation">.<span class="token function">append<span class="token punctuation">(<span class="token builtin">INPersonResolutionResult<span class="token punctuation">.<span class="token function">disambiguation<span class="token punctuation">(with<span class="token punctuation">: disambiguations<span class="token punctuation">)<span class="token punctuation">)
            <span class="token keyword">break
         
            <span class="token keyword">case <span class="token number">1<span class="token punctuation">:<span class="token comment">//一个匹配结果
                  <span class="token keyword">let recipient <span class="token operator">= matches<span class="token punctuation">[<span class="token number">0<span class="token punctuation">]<span class="token punctuation">.<span class="token function">inPerson<span class="token punctuation">(<span class="token punctuation">)
            resolutionResults<span class="token punctuation">.<span class="token function">append<span class="token punctuation">(<span class="token builtin">INPersonResolutionResult<span class="token punctuation">.<span class="token function">success<span class="token punctuation">(with<span class="token punctuation">: recipient<span class="token punctuation">)<span class="token punctuation">)
                  <span class="token keyword">break
            
            <span class="token keyword">case <span class="token number">0<span class="token punctuation">:<span class="token comment">//无匹配结果
                  resolutionResults<span class="token punctuation">.<span class="token function">append<span class="token punctuation">(<span class="token builtin">INPersonResolutionResult<span class="token punctuation">.<span class="token function">unsupported<span class="token punctuation">(with<span class="token punctuation">: <span class="token punctuation">.<span class="token keyword">none<span class="token punctuation">)<span class="token punctuation">)
                  <span class="token keyword">break

            <span class="token keyword">default<span class="token punctuation">:
                  <span class="token keyword">break
            <span class="token punctuation">}
          <span class="token punctuation">}
          <span class="token function">completion<span class="token punctuation">(resolutionResults<span class="token punctuation">)
      <span class="token punctuation">} <span class="token keyword">else <span class="token punctuation">{
          <span class="token comment">//未从用户语音中提取到信息,需要向用户征询更多关键信息
          <span class="token function">completion<span class="token punctuation">(<span class="token punctuation">[<span class="token builtin">INPersonResolutionResult<span class="token punctuation">.<span class="token function">needsValue<span class="token punctuation">(<span class="token punctuation">)<span class="token punctuation">]<span class="token punctuation">)
      <span class="token punctuation">}
<span class="token punctuation">}
</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
</div>
<p>上面的代码用来确认出消息中的<code>我的家人们</code>指代的是哪些人,其中每个联系人最终用一个<code>INPerson</code>的对象来表示。接着我们需要匹配消息的内容:<br>
func resolveContent(forSendMessage intent: INSendMessageIntent, with completion: (INStringResolutionResult) -&gt; Void) {<br>
if let text = intent.content where !text.isEmpty {<br>
completion(INStringResolutionResult.success(with: text))<br>
} else {<br>
//向用户征询发送的消息内容<br>
completion(INStringResolutionResult.needsValue())<br>
}<br>
}<br>
在匹配完消息接收者跟消息内容之后,对于<code>intent</code>事件的处理就会进入第二阶段<code>Confirm</code>确认值是否正确</p>


</li>
<li>
<p>Confirm阶段<br>
在这个阶段<code>intent</code>对象本身的信息预计是已经完成填充的,我们通过获取这些填充值来判断是否符合我们的要求。同时在这个阶段,<code>Siri</code>会尝试唤醒应用来准备完成最后的处理操作。前面说了为了保证在应用和应用拓展之间能够进行通信,最好使用<code>frameworks</code>的方式来标记应用是否被启动,再进行相应操作。<br>
func confirm(sendMessage intent: INSendMessageIntent, completion: (INSendMessageIntentResponse) -&gt; Void) {<br>
/// let content = intent.content<br>
/// let recipients = intent.recipients<br>
/// do or judge in content &amp; recipients<br>
completion(INSendMessageIntentResponse(code: .success, userActivity: nil))<br>
/// Launch your app to do something like store message record<br>
/// Use a singleton in frameworks to remark if the app has launched<br>
/// if not launched, use the code following<br>
/// completion(INSendMessageIntentResponse(code: .failureRequiringAppLaunch, userActivity: nil))<br>
}<br>
<code>Confirm</code>阶段是我们最后可以尝试修改<code>intent</code>事件中传递的数值的时候。要记住一点,完全精确的内容固然是最好的答案,但是过多的让<code>Siri</code>询问用户参数的详细信息也会导致用户的抵触</p>

</li>
<li>
<p>Handle阶段<br>
<code>Handle</code>阶段不需要做太多额外的工作,判断一下消息接收者或消息内容是否存在,如果存在,执行类似<code>保存/发送</code>的工作,然后完成。否则告诉<code>Siri</code>本次的<code>intent</code>事件处理处理失败,我们还可以通过配置<code>NSUserActivity</code>对象来告诉<code>Siri</code>失败的原因<br>
func handle(sendMessage intent: INSendMessageIntent, completion: (INSendMessageIntentResponse) -&gt; Void) {<br>
if intent.recipients != nil &amp;&amp; intent.content != nil {<br>
/// do some thing success send message<br>
let success = LXDAccount.share().send(message: intent.content!, to: intent.recipients!)<br>
completion(INSendMessageIntentResponse(code: success, userActivity: nil))<br>
} else {<br>
let userActivity = NSUserActivity(activityType: String(INSendMessageIntent))<br>
userActivity.userInfo = <br>
completion(INSendMessageIntentResponse(code: .failure, userActivity: userActivity))<br>
}<br>
}</p>

</li>


</ul>
<h1>事件UI</h1>
<p>可以看到上面的代码主要集中在事件处理的逻辑上,那么在和<code>Siri</code>交互的过程中,我们同样可以让<code>Siri</code>展示响应的自定义界面:<br>
</p>
<div class="image-package">
<div class="image-container">
<div class="image-container-fill">&nbsp;</div>
<div class="image-view" data-width="398" data-height="724"><img src="//upload-images.jianshu.io/upload_images/783864-4c2af1608a13c186.png?imageMogr2/auto-orient/strip|imageView2/2/w/398/format/webp"></div>


</div>
<div class="image-caption">&nbsp;</div>


</div>

<br>
在我们创建<code>Intents Extension</code>的时候,同时<code>Xcode</code>也询问我们是否创建<code>Intents UI Extension</code>。在后者的文件目录下也有一个<code>Info.plist</code>,有着跟前面类似的键值对,差别在于后者只有一个状态的设置。<br>
<div class="image-package">
<div class="image-container">
<div class="image-container-fill">&nbsp;</div>
<div class="image-view" data-width="1276" data-height="558"><img src="//upload-images.jianshu.io/upload_images/783864-2525371722ef807f.png?imageMogr2/auto-orient/strip|imageView2/2/w/1200/format/webp"></div>


</div>
<div class="image-caption">&nbsp;</div>


</div>

<br>
在这个文件目录下存在一个故事板<code>MainInterface</code>,这个故事板就是<code>Siri</code>和应用交互时展示给用户看的界面。通过修改这个故事板的界面元素,就可以实现上图中的效果了。此外,在这个界面将要展示之前,我们可以修改类文件中的代码完成界面信息填充的操作:
<p>&nbsp;</p>
<div class="_2Uzcx_"><button class="VJbwyy" type="button"></button>
<pre class="line-numberslanguage-swift"><code class="language-swift"><span class="token keyword">func <span class="token function">configure<span class="token punctuation">(with interaction<span class="token punctuation">: <span class="token builtin">INInteraction<span class="token operator">!<span class="token punctuation">, context<span class="token punctuation">: <span class="token builtin">INUIHostedViewContext<span class="token punctuation">, completion<span class="token punctuation">: <span class="token punctuation">(<span class="token punctuation">(<span class="token builtin">CGSize<span class="token punctuation">) <span class="token operator">-<span class="token operator">&gt; <span class="token builtin">Void<span class="token punctuation">)<span class="token operator">!<span class="token punctuation">) <span class="token punctuation">{
    <span class="token comment">//这里执行界面设置的代码,完成之后执行completion代码就会让界面展示出来
    <span class="token keyword">if <span class="token keyword">let completion <span class="token operator">= completion <span class="token punctuation">{
      <span class="token function">completion<span class="token punctuation">(<span class="token keyword">self<span class="token punctuation">.desiredSize<span class="token punctuation">)
    <span class="token punctuation">}
<span class="token punctuation">}

<span class="token keyword">var desiredSize<span class="token punctuation">: <span class="token builtin">CGSize <span class="token punctuation">{
    <span class="token keyword">return <span class="token keyword">self<span class="token punctuation">.extensionContext<span class="token operator">!<span class="token punctuation">.hostedViewMaximumAllowedSize
<span class="token punctuation">}
</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
</div>
<h1>尾言</h1>
<p>在观看<code>WWDC2016</code>的新特性的时候,最开始给<code>Siri</code>和应用的交互惊艳到了。但是后来阅读文档发现这种交互仍然存在着过多的限制,整体而言并没有对<code>Siri</code>的使用带来更明显的提升。但是毫无疑问,这种交互如果苹果能继续对其进行补充发展,可以给我们的应用带来更多的新活力。<br>
文集:iOS开发</p>

</div>

<br><br>作者:sindri的小巢<br>链接:https://www.jianshu.com/p/0881bb0ff538<br>来源:简书<br>著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。</div>

</div>
<div id="MySignature" role="contentinfo">
    Stay hungry, stay foolish!<br><br>
来源:https://www.cnblogs.com/wi100sh/p/16905299.html
頁: [1]
查看完整版本: 【转】iOS开发-SiriKit应用