Skip to content

Instantly share code, notes, and snippets.

@db0company
Created January 19, 2015 14:28
Show Gist options
  • Star 55 You must be signed in to star a gist
  • Fork 3 You must be signed in to fork a gist
  • Save db0company/369bfa43cb84b145dfd8 to your computer and use it in GitHub Desktop.
Save db0company/369bfa43cb84b145dfd8 to your computer and use it in GitHub Desktop.
Get the top most viewController anywhere in the app (typically from the appDelegate). Get the current visible viewController.
extension UIViewController {
func topMostViewController() -> UIViewController {
if self.presentedViewController == nil {
return self
}
if let navigation = self.presentedViewController as? UINavigationController {
return navigation.visibleViewController.topMostViewController()
}
if let tab = self.presentedViewController as? UITabBarController {
if let selectedTab = tab.selectedViewController {
return selectedTab.topMostViewController()
}
return tab.topMostViewController()
}
return self.presentedViewController!.topMostViewController()
}
}
extension UIApplication {
func topMostViewController() -> UIViewController? {
return self.keyWindow?.rootViewController?.topMostViewController()
}
}
@dawand
Copy link

dawand commented Jan 24, 2018

Thank you for this gist. Just a small note that visibleViewController is optional so it would be safer to do this:

 if let navigation = self.presentedViewController as? UINavigationController {
        if let visibleController = navigation.visibleViewController {
             return visibleController.topMostViewController()
        }
}

@maryamaffinityclick
Copy link

above one would not work cause you have to return something, you can use this instead on line 7 :
return navigation.visibleViewController?.topMostViewController() ?? navigation

@bartleby
Copy link

bartleby commented Jul 8, 2018

extension UIViewController {
    func topMostViewController() -> UIViewController {
        
        if let presented = self.presentedViewController {
            return presented.topMostViewController()
        }
        
        if let navigation = self as? UINavigationController {
            return navigation.visibleViewController?.topMostViewController() ?? navigation
        }
        
        if let tab = self as? UITabBarController {
            return tab.selectedViewController?.topMostViewController() ?? tab
        }
        
        return self
    }
}

@OscarGorog
Copy link

This is how to use this code:

let topMostViewController = UIApplication.shared.topMostViewController()

@bj97301
Copy link

bj97301 commented Aug 28, 2018

Thanks bud

@anaidfazlic
Copy link

anaidfazlic commented Nov 8, 2018

extension UIViewController {
    func topMostViewController() -> UIViewController {
        
        if let presented = self.presentedViewController {
            return presented.topMostViewController()
        }
        
        if let navigation = self as? UINavigationController {
            return navigation.visibleViewController?.topMostViewController() ?? navigation
        }
        
        if let tab = self as? UITabBarController {
            return tab.selectedViewController?.topMostViewController() ?? tab
        }
        
        return self
    }
}

@bartleby Thank you sir.

@patelgaurav4u
Copy link

i think need to add for pageviewcontroller related code

@showlllw
Copy link

showlllw commented Jan 6, 2020

I'm curious where viewController's children has gone?

@JaeHwanWO
Copy link

Since self.keywindow is deprecated in iOS 13.0, I changed extension like this.

extension UIApplication {
  func topMostViewController() -> UIViewController? {
    return UIApplication.shared.windows.filter { $0.isKeyWindow }.first?.rootViewController?.topMostViewController()
  }
}

@TuenTuenna
Copy link

thanks!

@ielyas
Copy link

ielyas commented May 18, 2022

'windows' was deprecated in iOS 15.0, updated extension:

func topMostViewController() -> UIViewController? {
        
        let vc = UIApplication.shared.connectedScenes.filter {
            $0.activationState == .foregroundActive
        }.first(where: { $0 is UIWindowScene })
            .flatMap( { $0 as? UIWindowScene })?.windows
            .first(where: \.isKeyWindow)?
            .rootViewController?
            .topMostViewController()
        
        return vc
    }
extension UIViewController {
    
    func topMostViewController() -> UIViewController {
        if self.presentedViewController == nil {
            return self
        }
        
        if let navigation = self.presentedViewController as? UINavigationController {
            return navigation.visibleViewController!.topMostViewController()
        }
        
        if let tab = self.presentedViewController as? UITabBarController {
            if let selectedTab = tab.selectedViewController {
                return selectedTab.topMostViewController()
            }
            return tab.topMostViewController()
        }
        
        return self.presentedViewController!.topMostViewController()
    }
    
}

@santespatel
Copy link

Tested with ios 15.4.1

@available(iOS 15.0, *)
var keyWindow:UIWindow? {
    let scene = UIApplication.shared.connectedScenes.filter({$0.activationState == .foregroundActive}).first as? UIWindowScene
    let keyWindow = scene?.keyWindow
    return keyWindow
}

@AMBIENTE1
Copy link

@ielyas please, your code work with iPhone fine, but on iPad it returns vc : Fatal error: Unexpectedly found nil while unwrapping an Optional value
Any idea what's different for the iPad?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment