如何在Siri Intent Extension中使用NSUseractivity打开SwiftUI应用并执行功能?

mra*_*hay 5 ios siri swift sirishortcuts swiftui

我的问题

我的应用程序使用Intents Extensions创建了自己的快捷方式操作。他们完美地执行了后台操作。

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

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

我尝试过的

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

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

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

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

我正在运行:iOS(iPhone XS,Beta 6)macOS 10.15 Catalina(Beta 5)Xcode 11.0(Beta 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
      }
Run Code Online (Sandbox Code Playgroud)

这是我的意图扩展内:

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."))
        }

Run Code Online (Sandbox Code Playgroud)

在info.plist中

<key>NSUserActivityTypes</key>
    <array>
        <string>com.me.project.activityName</string>
    </array>
Run Code Online (Sandbox Code Playgroud)

我在项目的一个快速文件中声明了这个(尽管我认为我不需要它):

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
}()
Run Code Online (Sandbox Code Playgroud)

意外的结果

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

当前,在“快捷方式”中运行操作后,我的应用程序启动到主屏幕,没有其他反应。在我的appDelegate中似乎没有命中任何断点。

如果是因为我使用NSUserActivity错误而导致我无法解决,无论是因为它是SwiftUI,还是因为它无法在意图内正常工作。

提前非常感谢您的帮助!

L. *_*han 11

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

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

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

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”)
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

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

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

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


use*_*357 7

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

ContentView
   .onContinueUserActivity("com.company.app.activityIdentifier") { userActivity in 
      handleShortcut(with: userActivity)
   }
Run Code Online (Sandbox Code Playgroud)

在哪里handleShortcut从给定的用户信息中获取用户信息NSUserActivity并从打开应用程序时启动要完成的事情

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

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