我的问题

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

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

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

我尝试过的

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

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

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

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

我在跑:
iOS(iPhone XS,Beta 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>

我在项目的一个快速文件中声明了这个(尽管我认为我不需要它):
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,还是因为某种意图而无法正常工作,我就无法解决。

非常感谢您的帮助!

最佳答案

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

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

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

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 Intent。

对于Handoff,尽管不应按照文档所述使用.userActivities,但是将调用关联的委托(delegate)方法(scene(_:continue:)):

07-27 18:28