5

目前,我已经在我的 AppDelegate 中实现了这两个方法

func application(application: UIApplication, continueUserActivity userActivity: NSUserActivity, restorationHandler: ([AnyObject]?) -> Void) -> Bool

func application(app: UIApplication, openURL url: NSURL, options: [String : AnyObject]) -> Bool

如果用户使用 Spotlight 的搜索结果打开我的应用程序,将调用第一个,如果从 Apple Maps 打开我的应用程序(因为它是一个路由应用程序),则会调用第二个。

我的问题是,从 APPDELEGATE 转到特定内容的最佳UIViewController方法是什么(独立于用户在什么视图中)?

我问的原因是因为目前我正试图根据用户的位置手动导航到它。例如,它们可能位于UIViewController模态显示的 a 中(然后需要将其关闭),或者它们可能位于 a 的深处,UINavigationController应用程序将需要在其中调用popToRootViewController.

这样做,代码变得毛茸茸的,似乎不能正常工作。这样做似乎也不正确,因为它非常脆弱。

4

2 回答 2

0

我认为重新导航的好地方是UIApplicationDidBecomeActiveNotification根视图控制器中的订阅事件(无论您使用什么)。

当您执行代码时AppDelegate- 只需安排一个操作:将您的包裹与抽象数量的属性、数据参数等组合起来。将它存储在某个地方(NSUserDefails - 是个好地方,但它甚至可能是SqlCipher实例)。并保持休息,希望能关注后续事件。

何时UIApplicationDidBecomeActiveNotification触发 - 唤醒,捕获存储的 Action 包裹,并根据 Action 属性执行重新导航。

关于您的模态视图控制器。他们(确切地说——谁显示的控制器)应该准备好在重新导航事件到达时关闭他们以模态方式显示的所有 VC。

于 2020-05-12T09:55:31.983 回答
0

我只是想弄清楚如何实施您提到的第一种方法,并从https://www.hackingwithswift.com/read/32/4/how-to-add-core-spotlight-to-找到了一个有用的开始index-your-app-content 他的代码是:

func application(application: UIApplication, continueUserActivity userActivity: NSUserActivity, restorationHandler: ([AnyObject]?) -> Void) -> Bool {
    if userActivity.activityType == CSSearchableItemActionType {
        if let uniqueIdentifier = userActivity.userInfo?[CSSearchableItemActivityIdentifier] as? String {
            let splitViewController = self.window!.rootViewController as! UISplitViewController
            let navigationController = splitViewController.viewControllers[splitViewController.viewControllers.count-1] as! UINavigationController

            if let masterVC = navigationController.topViewController as? MasterViewController {
                masterVC.showTutorial(Int(uniqueIdentifier)!)
            }
        }
    }

    return true
}

我发现self.window?.rootViewController 为?yourRootViewControllerClass是您问题的一个很好的跳板。

我的代码非常基本,如下所示:

// create a storyBoard item to instantiate a viewController from. If you have multiple storyboards, use the appropriate one. I just have one so it is "Main"
let sb = UIStoryboard(name: "Main", bundle: nil)

// get the navigation controller from the window and instantiate the viewcontroller I need.
if let viewController = sb.instantiateViewControllerWithIdentifier("DetailViewController") as? ViewController,
let nav = window?.rootViewController as? UINavigationController {

    viewController.setupController(bookName)// setup the viewController however you need to. I have a method that I use (I grabbed the bookName variable from the userActivity)
    nav.pushViewController(viewController, animated: false)//use the navigation controller to present this view.

} 

这对我有用,但我必须警告一下。我的应用程序只有一个带有三个视图控制器(NavController->tableViewController->ViewController)的故事板。我不确定这个逻辑将如何在更复杂的应用程序上工作。

另一个很好的参考是: http: //www.appcoda.com/core-spotlight-framework/

func application(application: UIApplication, continueUserActivity userActivity: NSUserActivity, restorationHandler: ([AnyObject]?) -> Void) -> Bool {
    let viewController = (window?.rootViewController as! UINavigationController).viewControllers[0] as! ViewController
viewController.restoreUserActivityState(userActivity)

    return true
}

那个 viewController 有这个方法:

override func restoreUserActivityState(activity: NSUserActivity) {
    if activity.activityType == CSSearchableItemActionType {
        if let userInfo = activity.userInfo {
            let selectedMovie = userInfo[CSSearchableItemActivityIdentifier] as! String
            selectedMovieIndex = Int(selectedMovie.componentsSeparatedByString(".").last!)
            performSegueWithIdentifier("idSegueShowMovieDetails", sender: self)
        }
    }
}
于 2016-03-04T23:31:31.493 回答