我的问题
我的应用程序使用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:)
):