Skip to content

adamcumiskey/Vine

Repository files navigation

Vine

CI Status Version License Platform

Introduction

A Vine is a child that weakly retains and controls its Root.

The Root type needs to

  1. Strongly retain the Vine
  2. Assign vine.root = self in the initializer
  3. Call vine.start() after the Root has finished initializing

When creating a Vine, the init(start: StartFunction?) can be used to pass in a closure that will get executed when start() is called. This should be sufficient for most cases where only some initial bootstrapping logic needs to occur. If you need a more advanced Vine

Vines are especially useful as an alternative to the Coordinator pattern for managing navigation. Vine Example

Motivation

Removing navigation logic from view controllers is a good way to separate concerns and increase testability. The Coordinator pattern has been written about extensively, and is widely accepted. However this pattern causes significant memory management overhead and makes it dangerous to use UIKit navigation methods directly.

Coordinators store a reference to a root UIKit object and an array of child coordinators. When adding a new view controller with a child coordinator, the parent must hold a strong reference to the child to prevent it from getting deallocated. When the child view controller is removed, the child coordinator reference also needs to be removed from the parent. But what happens when a someone doesn't feel like delegating a dismiss command through 4 layers of coordinators and just calls dismiss(animated:completion:) from a random view controller? If that view controller is managed by a coordinator, the coordinator won't get removed until its parent is removed. If that coordinator is strongly retaining view controllers (quite common from what I've seen) they will also stay stuck in memory even though they are no longer on screen.

For example, a coordinator managing a UINavigationController stack receives a message to present a modal. Uncoordinated 1 The coordinator creates the modal view controller along with a coordinator to manage it. The view controller is presented from the topViewController and the child coordinator is attached to the parent. Uncoordinated 2 The modal view controller calls dismiss(animated:completion:) without telling the parent Coordinator Uncoordinated 3 The modal view controller is dismissed, but the Child Coordinator still holds a reference to it. Uncoordinated 4 Both the child coordinator and any view controllers strongly referenced are now leaked.

Creating a separate object graph for navigation logic introduces the overhead of maintaining parity with the UI tree, and failure to do so results in memory leaks. This bookkeeping is often tedious, error prone, and results in a web of dependencies. Vines allow you to safely call any UIKit navigation method from any Vine without causing a memory leak. For example, if a Vine was powering a model UINavigationController with a few view controllers on the stack, it is safe to call self.dismiss(animated:completion:) from any of the view controllers. For this reason Vine takes the opinion that it's better to rely on view lifecycle for memory management in UI applications.

Installation

Vine is available through CocoaPods. To install it, simply add the following line to your Podfile:

pod 'Vine'

Author

Adam Cumiskey, [email protected]

License

Vine is available under the MIT license. See the LICENSE file for more info.

About

A library for growing iOS applications

Resources

License

Stars

Watchers

Forks

Packages

No packages published