简体   繁体   中英

dismiss current view controller AFTER presenting new view controller - swift

I'm trying to dismiss a VC and present a new VC. but I don't want old VC to exist anymore. I use the code below to dismiss current VC and present new one. but this way, there's a time interval between dismiss and present. I don't want the user to notice this. so, I want to present new VC first and then dismiss previous one. Is there any way to do this?

let storyboard = UIStoryboard(name: "Main", bundle: nil)
let destinationController = self.storyboard?.instantiateViewController(withIdentifier: "login") as! Login
let presentingVC = self.presentingViewController
self.dismiss(animated: false, completion: { () -> Void   in
    presentingVC!.present(destinationController, animated: true, completion: nil)
})

Disclaimer

So unless you absolutely need to modally present your new VC, then I recommend just performing a segue between the two VCs. It seems that you are only presenting it modally because you want to manually dismiss it later from the original VC. Not only is using a segue this easier in my opinion, but it will also allow you to use the method I've outlined below.

Solution

This likely isn't the most elegant method, but you could pass the instance of the old VC through prepareForSegue to the next VC, and then dismiss it in the new VC's viewDidLoad .

For example, in your new VC you could have something like this:

class NewVC: UIViewController {

    ...
    var prevVC: PrevVC!

    override func viewDidLoad() {
        super.viewDidLoad()

        prevVC.dismiss(animated: false, completion: nil)
    }

}

So when your newVC loads, it dismisses the previous VC. All you would need to do in your prevVC class is pass on the instance in prepareForSegue like so.

class PrevVC: UIViewController {
    ...
    override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
        if let destinationVC = segue.destination as? NewVC {

            destinationVC.prevVC = self

        }
    }

}

Then of course you would just have to present the newVC when you want and then everything else would be taken care of.

I think the best way will be to dismiss the current VC when you actually go back. That means, present your destinationController using the current VC and then when you go back, dismiss both VC's

If you want to present VC B from VC A and want to dismiss VC A while Presenting, you can use this code:

You write this code in your VC A from which Button you want to present the VC B.

let parentVC = presentingViewController
dismiss(animated: true) {
    let vc = self.storyboard!.instantiateViewController(withIdentifier...)
    parentVC.present(vc, animated: true)`enter code here`
}

The way I do it is to use both

view.dismiss(animated: true, completion: nil)

and

presentingViewController?.presentingViewController?.dismiss(animated: true, completion: nil)

depend on how many view controllers before. Because they are all optional, Xcode will not complain if you don't have them when you are trying to dismiss. If that is the case, the single dismiss() function will work just fine.

I guess it is because I didn't have a good design in the first place, it works but I don't think you will run into this if you have a good structure in the first place.

So I just tried this in Xcode13. VC_A does a segue to VC_B. I tried to dismiss VC_A after the segue as well as passing the VC_A to VC_B and dismissing VC_A in the viewDidLoad of VC_B. When I dismiss VC_B it goes back to VC_A (also did a print of the deinit and it never gets called).

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM