Lightweight coordinator navigation library written for iOS π¬
The easiest way to get started is by installing via Xcode. Just add Director as a Swift package & choose the modules you want.
If you're adding Director as a dependency of your own Swift package, just add a package entry to your dependencies.
.package(
name: "Director",
url: "https://github.com/mitchtreece/Director",
.upToNextMajor(from: .init(1, 0, 0))
)
Director is broken down into several modules making it quick & easy to pick and choose exactly what you need.
Director
: Core classes, extensions, & dependenciesDirectorDI
: Swinject classes, extensions, & dependencies
As of Director 1.1.0
, CocoaPods support has been dropped in favor of SPM. If you're depending on a Director version prior to 1.1.0
, you can still integrate using CocoaPods.
pod 'Director', '~> 1.0'
Coordinators helps consolidate and manage view display state. They are used to define an application's path in chunks, while keeping navigation logic separate from app & view logic. Coordinators manage the navigation of only what they're concerned with. Multiple coordinators can live in a single navigation stack, each managing a slice of the stack's views. Likewise, a coordinator can be associated with as many (or as few) views as needed.
Director is a lightweight coordinator implementation that helps keep your application's navigation paths simple & reusable. There are three main components:
SceneDirector
SceneCoordinator
ViewCoordinator
A scene-level class that manages a scene's window & initial view coordinator presentation. Depending on your target iOS version, it's initialized & started on application launch or scene connection.
class SceneDelegate: UIResponder, UIWindowSceneDelegate {
var window: UIWindow?
private var director: SceneDirector!
func scene(_ scene: UIScene,
willConnectTo session: UISceneSession,
options connectionOptions: UIScene.ConnectionOptions) {
guard let windowScene = scene as? UIWindowScene else { return }
self.window = UIWindow(windowScene: windowScene)
self.director = SceneDirector(
MySceneCoordinator(),
window: self.window!
)
.start()
}
}
You might have noticed we initialized our window manually in the above example. Because our SceneDirector
manages our window and initial
view presentation, we do not use a main storyboard. You can simply remove the storyboard file added to your project by default, and clear the "Main Interface" field in your target & scene-configuration settings. You can continue using storyboards for your other view elements if desired. Simply load them directly from
your view coordinator subclasses.
A root-level coordinator class that manages the initial view coordinator displayed by an application. This should be subclassed to return an appropriate ViewCoordinator
instance on application launch or scene connection.
class MySceneCoordinator: SceneCoordinator {
override func build() -> ViewCoordinator {
return HomeCoordinator()
}
}
A class that manages the display state, navigation path, and presentation / dismissal logic for a set of related views.
class HomeCoordinator: ViewCoordinator {
override func build() -> UIViewController {
let vc = HomeViewController()
vc.delegate = self
return vc
}
}
We want to abstract away navigation logic from the view. Instead, we let our coordinator decide how to navigate when a user triggers actions on our view. To further enforce this separation of concerns, our view shouldn't know about it's parent coordinator. This is important. For example, what if we wanted a generic view to be used in many different coordinators? A protocol / delegate approach allows us to convey our user-actions to the coordinator while keeping it generic enough for reuse.
extension HomeCoordinator: HomeViewControllerDelegate {
func homeViewControllerDidTapDetail(_ viewController: HomeViewController) {
push(DetailViewController())
}
func homeViewControllerDidTapHelp(_ viewController: HomeViewController) {
modal(HelpViewController())
}
}
A lot of the time our navigation path can be complicated and tedious when presenting the same views from different places around our application. Instead of simply pushing or presenting our views modally, we can group navigation paths unrelated, or distinctly different from the current path in other coordinators.
For example, if our HomeCoordinator
from above needed to display a profile view with it's own navigation paths, we could group it into a ProfileCoordinator
and start it as a child of our HomeCoordinator
.
extension HomeCoordinator: HomeViewControllerDelegate {
...
func homeViewControllerDidTapProfile(_ viewController: HomeViewController) {
start(child: ProfileCoordinator())
}
}
The way a child coordinator is presented depends on what is returned from a view coordinator's build()
function. If a UIViewController
instance is returned, the parent coordinator will start & push onto it's existing
navigation stack. However, if a UINavigationController
instance is returned, the parent will treat this coordinator as being self-managed, and present the coordinator modally.
There are scenarios where a view coordinator might need to manage child views and their presentation manually. A tab-bar interface is a good example of this. Our root view (the tab bar) contains several child views (its tabs) each with their own navigation paths. To achieve this, you can use embedded view coordinators.
class TabCoordinator: ViewCoordinator {
private var tabBarController: UITabBarController!
private var redCoordinator = RedCoordinator()
private var greenCoordinator = GreenCoordinator()
private var blueCoordinator = BlueCoordinator()
override func build() -> UIViewController {
self.tabBarController = UITabBarController()
self.tabBarController.viewControllers = buildChildren()
return self.tabBarController
}
private func buildChildren() -> [UIViewController] {
startEmbedded(children: [
self.redCoordinator,
self.greenCoordinator,
self.blueCoordinator
])
return [
self.redCoordinator.rootViewController,
self.greenCoordinator.rootViewController,
self.blueCoordinator.rootViewController
]
}
}
By calling startEmbedded(child:)
or startEmbedded(children:)
, the child view coordinators are setup without any automatic presentation logic. This is ideal when building custom multi-view interfaces.
Director fully supports SwiftUI. Simply return a UIHostingController
when building your view coordinator.
class SwiftUICoordinator: ViewCoordinator {
override func build() -> UIViewController {
return UIHostingController(rootView: MyView())
}
}
Pull-requests are more than welcome. Bug fix? Feature? Open a PR and we'll get it merged in!