【问题标题】:How can I open my SwiftUI app and perform a function using NSUseractivity from within Siri Intent Extension?如何从 Siri Intent Extension 中打开我的 SwiftUI 应用程序并使用 NSUseractivity 执行功能?
【发布时间】:2019-12-22 06:25:21
【问题描述】:

我的问题

我的应用有自己的使用 Intents Extensions 创建的 Shortcuts 操作。他们完美地执行后台操作。

对于某些操作,我试图让意图扩展在快捷方式中运行时打开主(容器)应用程序并执行功能。

我在使用 NSUserActivity 时遇到问题,我不确定这是因为它是一个 SwiftUI 项目还是我实现它的方式(或两者兼而有之)。

我尝试过的

我已将我的 NSUserActivity 名称注册为我的 info.plist(“com.me.project.activityName”)中的 NSUserActivityType。

我已将以下代码添加到我的 AppDelegate。

我在 Intent 扩展中初始化了一个新的 NSUserActivity,其类型与 info.plist 中声明的类型相同。

我也尝试过在应用内声明活动(我认为我不需要这样做?)

我正在跑步: iOS(iPhone XS,测试版 6) macOS 10.15 Catalina(测试版 5) Xcode 11.0(测试版 5)

到目前为止我的代码

我的 AppDelegate 中有这个:

func application(_ application: UIApplication, continue userActivity: NSUserActivity, restorationHandler: @escaping ([UIUserActivityRestoring]?) -> Void) -> Bool {

          if userActivity.activityType == "com.me.project.activityName" {
              if let url = URL(string: "https://www.google.com") {
                  UIApplication.shared.open(url)
              }
              return true
          }
          return false
      }

这是我的意图扩展:

let openApp = intent.launchApp?.boolValue ?? false
        if openApp {

            let UA = NSUserActivity(activityType: "com.me.project.activityName")
            UA.title = "Dummy title"

            completion(UserActivityTestIntentResponse(code: .continueInApp, userActivity: UA))

        } else {
            completion(UserActivityTestIntentResponse.success(result: "You chose not to open the app with a user activity."))
        }

在 info.plist 中

<key>NSUserActivityTypes</key>
    <array>
        <string>com.me.project.activityName</string>
    </array>

我在我的项目的一个 swift 文件中声明了这个(虽然我认为我不需要它):

let openURLActivityType = "com.me.project.activityName"

let viewPageActivity: NSUserActivity = {
    let userActivity = NSUserActivity(activityType: openURLActivityType)
    userActivity.title = "Dummy Title"
    userActivity.suggestedInvocationPhrase = "Dummy phrase"
    userActivity.isEligibleForSearch = false
    userActivity.isEligibleForPrediction = false
    return userActivity
}()

意外结果

我希望在快捷方式中运行该操作时,我的应用会打开并且网站“https://www.google.com”。

目前,在快捷方式中运行该操作后,我的应用程序会启动到主屏幕,并且没有其他任何反应。我的 appDelegate 中似乎没有遇到断点。

我不知道是因为我使用了错误的 NSUserActivity,是因为它是 SwiftUI,还是因为它无法从 Intent 内部工作。

非常感谢您的帮助!

【问题讨论】:

  • 我遇到了同样的问题。对我来说,虽然我只是使用NSUserActivity 并尝试实现切换和 Spotlight 搜索功能 (isEligibleForSearch)。对我来说,我的appDelegate 似乎也没有遇到断点。我正在使用运行 Xcode 11 beta 7 的 macOS Catalina 10.15 beta 7、iOS 13 beta 8 和 iPhone X
  • 我大部分时间都在工作,将其添加到我的场景委托中:func scene(_ scene: UIScene, continue userActivity: NSUserActivity) { if let intent = userActivity.interaction?.intent as? YourIntent { print(“do something”) } } 但是,如果应用程序一开始就关闭,它不会触发,所以我仍在尝试让它运行应用程序委托.
  • 谢谢,这对我帮助很大。我想出了如何在应用程序关闭时继续 UserActivity。在下面查看我的答案

标签: ios swift swiftui siri sirishortcuts


【解决方案1】:

就像您已经评论过的那样,当应用程序仍在后台时,您可以使用func scene(_ scene: UIScene, continue userActivity: NSUserActivity) 继续NSUserActivity

但是,当应用程序关闭时,您可以在 SceneDelegate 内的 scene(_:willConnectTo:options:) 方法内实现 let userActvity = connectionOptions.userActivities.first 以访问用户想要继续的活动。

只需将这段代码添加到此方法的标准实现中,如下所示:

func scene(_ scene: UIScene, willConnectTo session: UISceneSession, options connectionOptions: UIScene.ConnectionOptions) {
    // Use this method to optionally configure and attach the UIWindow `window` to the provided UIWindowScene `scene`.
    // If using a storyboard, the `window` property will automatically be initialized and attached to the scene.
    // This delegate does not imply the connecting scene or session are new (see `application:configurationForConnectingSceneSession` instead).

    // Use a UIHostingController as window root view controller
    if let windowScene = scene as? UIWindowScene {
        let window = UIWindow(windowScene: windowScene)
        window.rootViewController = UIHostingController(rootView: ContentView())
        self.window = window
        window.makeKeyAndVisible()

        //Standard implementation until here
        if let userActvity = connectionOptions.userActivities.first {
            if let intent = userActivity.interaction?.intent as? YourIntent {
                print(“do something”)
            }
        }
    }
}

这适用于带有 Swift 5 Xcode 11 的 Siri Intents。

对于 Handoff,尽管 .userActivities 不应像文档中所说的那样使用,而是将调用相关的委托方法 (scene(_:continue:)):

此属性不包含与相关的用户活动对象 不可触摸。在连接时,UIKit 只提供 Handoff 的类型 handoffUserActivityType 属性中的交互。后来,它调用 委托的附加方法来传递 NSUserActivity 对象本身。

【讨论】:

  • 有趣,我今天下午也想通了。感谢您的帮助!
  • 不太适合我。最后一段应该是if let _ = connectionOptions.userActivities.first?.interaction?.intent as? YourIntent { //your code }
  • 你在一个 if 语句中建议做的事情在我的回答中被分成两部分
【解决方案2】:

如果您的应用程序是 SwiftUI 应用程序,也许您应该考虑添加如下内容:

ContentView
   .onContinueUserActivity("com.company.app.activityIdentifier") { userActivity in 
      handleShortcut(with: userActivity)
   }

handleShortcut 从给定的NSUserActivity 获取 userInfos 并在打开您的应用程序时启动要完成的事情

这将替换在您的 SwiftUI 应用程序中不再调用的 func application(_ application: UIApplication, continue userActivity: NSUserActivity, restorationHandler: @escaping ([UIUserActivityRestoring]?) -&gt; Void) -&gt; Bool 委托方法中所做的任何实现,除非您编写更多(而不是微不足道的)代码行来调用它

您可以在以下位置找到更多详细信息:

【讨论】:

  • 感谢您的回答。最初的问题是在宣布 SwiftUI 的 App 协议之前,但这是我从那以后一直这样做的方式。
猜你喜欢
  • 1970-01-01
  • 2019-01-29
  • 2019-07-16
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2020-01-28
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多