web-dev-qa-db-ja.com

リモート通知からViewControllerを開きます

アプリがリモート通知をキャッチしたときに、特定のViewControllerを開こうとしました。

私のプロジェクトのアーキテクチャを示しましょう。ここに私のストーリーボード: enter image description here

通知を受け取ったら、「SimplePostViewController」を開きたいので、これが私のappDelegateです。

var window: UIWindow?
var navigationVC: UINavigationController?

func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {
    let notificationTypes: UIUserNotificationType = [UIUserNotificationType.Alert, UIUserNotificationType.Badge, UIUserNotificationType.Sound]
    let pushNotificationSettings = UIUserNotificationSettings(forTypes: notificationTypes, categories: nil)
    let storyboard = UIStoryboard(name: "Main", bundle: nil)

    self.navigationVC = storyboard.instantiateViewControllerWithIdentifier("LastestPostsNavigationController") as? UINavigationController
    application.registerUserNotificationSettings(pushNotificationSettings)
    application.registerForRemoteNotifications()
    return true
}

func application(application: UIApplication, didReceiveRemoteNotification userInfo: [NSObject : AnyObject]) {
    if let postId = userInfo["postId"] as? String {
        print(postId)

        let api = EVWordPressAPI(wordpressOauth2Settings: Wordpress.wordpressOauth2Settings, site: Wordpress.siteName)

        api.postById(postId) { post in
            if (post != nil) {
                self.navigationVC!.pushViewController(SimplePostViewController(), animated: true)
            } else {
                print("An error occurred")
            }
        }

    }
}

アプリの起動時にUINavigationViewControllerを保存し、通知を受け取ったときに新しいSimplePostViewControllerをプッシュしようとします。しかし、何も起こりません。ブレークポイントを配置し、pushViewControllerメソッドに到達したが、SimplePostViewControllerのViewWillAppearには到達していないことを確認しました。

また、「whats new」ビューを使用して、セグエを実行しましたが、何も起こりません。

解決策:

for child in (self.rootViewController?.childViewControllers)! {
    if child.restorationIdentifier == "LastestPostsNavigationController" {
       let lastestPostsTableViewController = (child.childViewControllers[0]) as! LastestPostsTableViewController
       let simplePostVC = (self.storyboard?.instantiateViewControllerWithIdentifier("PostViewController"))! as! PostViewController

       simplePostVC.post = post
       lastestPostsTableViewController.navigationController?.pushViewController(simplePostVC, animated: true)
    }
 }

私が使う :

child.childViewControllers[0]

私の例では子供が1人しかいないからです。

9
Lilrom

機能をわかりやすくするために、リモート通知ではなくローカル通知を使用してサンプルプロジェクトを作成しましたが、アプリデリゲートのdidreceiveremote通知でウィンドウのルートビューコントローラーを設定するのと同じくらい簡単なはずです。

func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: [NSObject: AnyObject]?) -> Bool {

    // Subscribe for notifications - assume the user chose yes for now
    application.registerUserNotificationSettings(UIUserNotificationSettings(forTypes: [.Alert, .Badge, .Sound], categories: nil))

    return true
}


func applicationDidEnterBackground(application: UIApplication) {
    //Crete a local notification
    let notification = UILocalNotification()
    notification.alertBody = "This is a fake notification"
    notification.fireDate  = NSDate(timeIntervalSinceNow: 2)
    UIApplication.sharedApplication().scheduleLocalNotification(notification)
}

func application(application: UIApplication, didReceiveLocalNotification notification: UILocalNotification) {
    let sb = UIStoryboard(name: "Main", bundle: nil)
    let otherVC = sb.instantiateViewControllerWithIdentifier("otherVC") as! OtherViewController
    window?.rootViewController = otherVC;
}

func application(application: UIApplication, didReceiveRemoteNotification userInfo: [NSObject : AnyObject]) {
    //Your code here
}

`ビュー階層を管理し、通知ユーザーデータから送信する必要があるものをビュー階層に送信することについて心配する必要があります。

私の例では、ビュー秒後に起動するアプリを閉じると、ローカル通知を作成します。その後、通知からアプリを起動すると、「その他のビューコントローラー」が開きます。これは、この場合は「SimplePostViewController」になります。

また、didFinishLaunchWithOptionsでリモート通知に登録していることを確認してください。

Githubの非常に単純なサンプル: https://github.com/spt131/exampleNotificationResponse

10
Sam T