Bond is a Swift binding framework that takes binding concept to a whole new level. It's simple, powerful, type-safe and multi-paradigm - just like Swift.
Bond was created with two goals in mind: simple to use and simple to understand. One might argue whether the former implies the latter, but Bond will save you some thinking because both are true in this case. Its foundation are few simple classes - everything else are extensions and syntactic sugars.
Note: This document describes Bond v4. If you are using a previous version of the framework, check out the Migration to Bond v4 section. Bond v4 is the only officially supported version for Swift 2.0.
Say you'd like to act on a text change event of a UITextField. Well, you could setup 'target-action' mechanism between your object and go through all that target-action selector registration pain, or you could simply use Bond and do this:
textField.bnd_text
.observe { text in
print(text)
}
Now, instead of printing what user has typed, you could even bind it to a UILabel:
textField.bnd_text
.bindTo(label.bnd_text)
That one line establishes a binding between text field's text property and label's text property. In effect, whenever user makes a change to the text field, that change will be automatically propagated to the label.
More often than not, direct binding is not enough. Usually you need to transform input is some way, like prepending a greeting to a name. Of course, Bond has full confidence in functional paradigm.
textField.bnd_text
.map { "Hi " + $0 }
.bindTo(label.bnd_text)
Whenever a change occurs in the text field, new value will be transformed by the closure and propagated to the label.
Notice how we've used bnd_text
property of the UITextField. It's an observable representation of the text
property provided by Bond framework. There are many other extensions like that one for various UIKit components. Just start typing .bnd on any UIKit object and you'll get the list of available extensions.
In addition to map
, another important functional construct is filter
function. It's useful when we are interested only in some values of a domain. For example, when observing events of a button, we might be interested only in TouchUpInside
event so we can perform certain action when user taps the button:
button.bnd_controlEvent
.filter { $0 == UIControlEvents.TouchUpInside }
.observe { e in
print("Button tapped.")
}
Handling TouchUpInside
event is used so frequently that Bond comes with the extension just for that event:
button.bnd_tap
.observe {
print("Button tapped.")
}
Bond can also combine multiple inputs into a single output. Following snippet depicts how values of two text fields can be reduced to a boolean value and applied to button's enabled property.
combineLatest(emailField.bnd_text, passField.bnd_text)
.map { email, pass in
return email.length > 0 && pass.length > 0
}
.bindTo(button.bnd_enabled)
Whenever user types something into any of these text fields, expression will be evaluated and button state updated.
Bond's power is not, however, in coupling various UI components, but in the binding of a Model (or a ViewModel) to a View and vice-versa. It's great for MVVM paradigm. Here is how one could bind user's number of followers property of the model to the label.
viewModel.numberOfFollowers
.map { "\($0)" }
.bindTo(label.bnd_text)
Point here is not in the simplicity of value assignment to text property of a label, but in the creation of a binding which automatically updates label text property whenever number of followers change.
Bond also supports two way bindings. Here is an example of how you could keep username text field and username property of your view model in sync (whenever any of them change, other one will be updated too):
viewModel.username.bidirectionalBindTo(usernameTextField.bnd_text)
Bond is also great for observing various different events and asynchronous tasks. For example, you could observe a notification just like this:
NSNotificationCenter.defaultCenter().bnd_notification("MyNotification")
.observe { notification in
print("Got \(notification)")
}
.disposeIn(bnd_bag)
Let me give you one last example. Say you have an array of repositories you would like to display in a collection view. For each repository you have a name and its owner's profile photo. Of course, photo is not immediately available as it has to be downloaded, but once you get it, you want it to appear in collection view's cell. Additionally, when user does 'pull down to refresh' and your array gets new repositories, you want those in collection view too.
So how do you proceed? Well, instead of implementing a data source object, observing photo downloads with KVO and manually updating the collection view with new items, with Bond you can do all that in just few lines:
repositories.bindTo(collectionView) { indexPath, array, collectionView in
let cell = collectionView.dequeueReusableCellWithReuseIdentifier("Cell", forIndexPath: indexPath) as! RepositoryCell
let repository = array[indexPath.section][indexPath.item]
repository.name
.bindTo(cell.nameLabel.bnd_text)
.disposeIn(cell.onReuseBag)
repository.photo
.bindTo(cell.avatarImageView.bnd_image)
.disposeIn(cell.onReuseBag)
return cell
})
Yes, that's right!
At the core of the framework is the class EventProducer
. It represents an abstract event generator that provides the mechanisms that enable interested parties, called observers, to observe generated events. For example, it can be used to represent a subject with a mutable state, like a variable or an array, and then inform observers of the state change whenever it happens. On the other hand it can represent an action, something without a state, and generate an event whenever the action occurs.
The most common use of the event producer is through its subclass Observable
that can mimic a variable or a property and enable observation of its change. The Observable is a generic type generalized over the wrapped value type. As the EventProducer is also a generic type, generalized over its event type, it is only natural to specialize such event producer to the type of the values it can encapsulate. To create the observable just initialize it with a value:
let captain = Observable(“Jim”)
Swift automatically infers the type of the observable from the passed value. In our example the type of the variable captain
is Observable<String>
. To change its value afterwards, you can use the method next
:
captain.next(“Spock”)
The value is accessible through the property value
:
print(captain.value) // prints: Spock
The property is both a getter that returns the observable’s value and a setter that updates the observable with a new value just like the method next
.
Now comes the interesting part. In order to make the observable useful it should be observed. Observing the observable means observing the events it generates, that is, in our case, the values that are being set. To observe the observable we register a closure of the type EventType -> ()
to it with the method observe, where EventType is the event (value) type:
captain.observe { name in
print(“Now the captain is \(name).”)
}
// prints: Now the captain is Spock.
The closure will be called at the time of the registration with the value currently set to the observable. If you are not interested in the current value, but only in the new ones, you can use the method observeNew
instead.
Now, whenever the value is changed, the observer closure will be called and side effects performed:
captain.next(“Scotty” ) // prints: Now the captain is Scotty.
which is same as:
captain.value = “Scotty” // prints: Now the captain is Scotty.
Using the observable that acts as a variable or a property that can be observed is just a specific usage of the EventProducer. As was already said, the event producer represents an abstract event generator. To create such event generator you can use the following designated initializer on EventProducer:
init(replayLength: Int, @noescape producer: (EventType -> ()) -> DisposableType?)
Parameter replayLength
defines how many events should be replayed to each new observer. It represents the memory of the event producer. Event producers don't have to have memory so zero is a valid value for this parameter. Event producers without a memory are used to represent actions, something without a state, like button taps.
Parameter producer
is a closure that actually generates events. The closure accepts a sink (another closure) through which it sends events and optionally returns a disposable that should be disposed when the created event producer is disposed.
An event producer (and so observable) can be observed by any number of observers. A new observer is registered with the already mentioned observe
method. Here is its signature:
func observe(observer: EventType -> ()) -> DisposableType
We've already talked about the closure parameter observer
, but it is also important to understand what the method returns. An observer stays registered until it’s unregistered or until the event producer is destroyed. To unregistered the observer manually we use a disposable object returned by the method observe
. Think of it as a subscription that can be cancelled. To cancel it simply use the method dispose
.
let subscription = captain.observe { name in … }
...
subscription.dispose()
The event producers are much more useful when they can be transformed and combined into another event producers. Bond comes with a number of methods that can transform an event producer into another event producer. Note that transforming an observable does not create another observable, but the event producer that has not concept of 'current value'.
func map<T>(transform: EventType -> T) -> EventProducer<T>
Creates an event producer that transforms each event from the receiver by the given transform closure.
func filter(includeEvent: EventType -> Bool) -> EventProducer<EventType>
Creates an event producer that forwards only events from the receiver that pass the given includeEvent
closure.
func deliverOn(queue: Queue) -> EventProducer<EventType>
Creates an event producer that forwards events from the receiver to the given Queue
.
func throttle(seconds: Queue.TimeInterval, queue: Queue) -> EventProducer<EventType>
Creates an event producer that forwards no more than one event in the given number of seconds.
func skip(var count: Int) -> EventProducer<EventType>
Creates an event producer that ignores first count events from the receiver but forwards any subsequent.
func startWith(event: EventType) -> EventProducer<EventType>
Creates an event producer that sends the given event and then continues by forwarding events from the receiver.
func combineLatestWith<U: EventProducerType>(other: U) -> EventProducer<(EventType, U.EventType)>
Creates an event producer that combines the latest value of the receiver with the latest value from the given event producer. Will not generate an event until both event producers have generated at least one event.
func switchToLatest() -> EventProducer<EventType.EventType>
Applicable only to the event producers whose events are also event producer. Creates an event producer that forwards events from the latest inner event producer.
func merge() -> EventProducer<EventType.EventType>
Applicable only to the event producers whose events are also event producer. Creates an event producer that forwards events from all received inner event producers.
func ignoreNil() -> EventProducer<EventType.SomeType>
Applicable only to the event producers whose events are optionals. Creates an event producer that forwards only events that are not nil values.
func distinct() -> EventProducer<EventType>
Applicable only to the event producers whose events conform to the protocol Equatable
. Creates an event producer that forwards only distinct events, i.e. no two equal events will be sent one after another.
Binding is a very simple concept. It's a way to propagate change. Change of a subject, like an observable, to an object, like a UI element or another observable. Let's say we need to update the observable that represents text of a label. Here is what we can do:
let captainName: Observable<String>
let nameLabelText: Observable<String>
captainName.observe { name in
nameLabelText.next(name)
}
That well make the label text update whenever the captain changes.
captainName.next(“Janeway”)
print(nameLabelText.value) // prints: Janeway
Bindings are at the core of Bond and there ought to be even simpler way to establish them. And, as you've seen it in the introduction, there is:
captainName.bindTo(nameLabelText)
Event producers and obsevables can be bound to any object that conforms to BindableType
protocol. Event producers themselves conform to that protocol, but you can make any type conform to it.
Method bindTo
returns a disposable that can cancel the binding. You usually don't need to worry about that because binding will be automatically canceled when either the event producer or target are deallocated.
UIKit and AppKit elements, of course, do not provide properties that are observable. UIKit and AppKit are also not KVO-compliant. Bond, therefore, provides its own extensions of the UIKit and AppKit elements in order to make bindings and property observations a piece of cake. For example, Bond provides its own variant of text
property to the UITextField
called bnd_text
. It's an observable of Observable<String?>
type that you can observe or make a binding to or from it.
let searchTextField = UITextField()
...
searchTextField.bnd_text.observeNew { text in
print("Searching for \(text).")
...
}
To learn about available extensions just start typing .bnd
on any UIKit or AppKit object or consult the Extensions section of the code reference. Extensions usually correspond to their respective UIKit or AppKit property names, just prefixed with bnd_
, so it shouldn't be hard to find them.
You rarely have to worry about disposing the observations, but when you do, Bond tries to be helpful. If you need to dispose an observation when your object (like view controller) is deallocated, you can use bnd_bag
extension provided on NSObject (and thus on all its subclasses). It's a dispose bag, a collection of disposables, that will dispose all added disposables when your object is deallocated.
For example, if your view model outlives the view controller that uses it you must manually dispose any observation you've made. The simplest way to do that is to add the disposable returned by the observe
method to the provided dispose bag:
class MyViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
viewModel.name
.observe { name in
print(name)
}
.disposeIn(bnd_bag)
}
}
Note that it's not necessary to dispose bindings. When the binding target is deallocated, the binding will be automatically disposed. That means that the following code is valid even if the view model outlives the view controller:
class MyViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
viewModel.name.bindTo(nameLabel.bnd_text)
}
}
You can use Bond to observe notifications from NSNotificationCenter. To do that use the following extension:
extension NSNotificationCenter {
public func bnd_notification(name: String, object: AnyObject?) -> EventProducer<NSNotification>
}
All you need to provide is a notification name you want to observe. Additionally, you can provide an object whose notifications you want to receive. Note that you should always manually dispose the observation when you no longer need it, preferably by putting the disposable in the dispose bag:
NSNotificationCenter.defaultCenter().bnd_notification("MyNotification", object: nil)
.observe { notification in
print("Received \(notification).")
}
.disposeIn(bnd_bag)
}
When working with arrays, it's usually not enough to know only that the array has changed, but how exactly did it change. New elements could have been inserted into the array and old ones deleted or updated. Bond provides mechanisms for observing such fine-grained changes.
Creating an Observable with an array would enable observation of change of the array as whole, but to observe fine-grained changes you have to use ObservableArray
type. Just like the Observable, it is a subclass of the EventProducer
class, but instead of sending events that match the wrapped value type, it sends events of the ObservableArrayEvent
type. Such event contains both the new state of the array (array itself) and the operation that was just applied to the array (like element insertion or deletion). Operation is an enum type that describes the change.
Let's go through an example. Say that we need an array of names that we would like to observe. We would define it like this:
let names = ObservableArray(["Jim", "Spock"])
ObservableArray
type mimics Array
type so you can do same operations on it that you can do on the Array
type. It also conforms to CollectionType
and SequenceType
protocols.
Observation is done in the same way as the observation of the Observable or EventProducer. Main point to learn is that events it generates are of ObservableArrayEvent
type. For example:
names.observe { event in
print("Array is now: \(event.sequence)")
switch event.operation {
case .Insert(let elements, let fromIndex):
print("Inserted \(elements) from index \(fromIndex)")
case .Remove(let range):
print("Removed elements in range \(range)")
case .Update(let elements, let fromIndex):
print("Updated \(elements) from index \(fromIndex)")
case .Reset(let array):
print("Array was reset to \(array)")
case .Batch(let operations):
print("Operations \(operations) were perform on the array")
}
}
Our observer will then be called whenever an operation is applied to the array. It will also be called initially, at the time of the registration, with the last operation that was applied to the array. In our case that would be .Reset
operation because it represents setting the array - something that the constructor does. Following will be printed:
$ Array is now: ["Jim", "Spock"]
$ Array was reset to ["Jim", "Spock"]
When we then change the array, our observer will be called. Appending new item
names.append("Scotty")
will result in
$ Array is now: ["Jim", "Spock", "Scotty"]
$ Inserted ["Scotty"] from index 2
Updating first element
names[0] = "Uhura"
will then result in
$ Array is now: ["Uhura", "Spock", "Scotty"]
$ Updated ["Uhura"] from index 0
Removing first element afterwards
names.removeAtIndex(0)
will result in
$ Array is now: ["Spock", "Scotty"]
$ Removed elements in range 0..<1
Sometimes it is necessary to batch operations to the single event. It can be done like this:
names.performBatchUpdates { names in
names.insert("Jim", atIndex: 0)
names.removeLast()
}
will result in
$ Array is now: ["Jim", "Spock"]
$ Operations [.Insert(elements: ["Jim"], fromIndex: 0), .Remove(range: 2..<3)] were perform on the array.
Observable arrays can bound to UITableViews and UICollectionViews, leveraging that mechanisms of fine-grained change events. To bind them to those views, use following methods:
public func bindTo(tableView: UITableView, proxyDataSource: BNDTableViewProxyDataSource? = nil, createCell: (NSIndexPath, ObservableArray<ObservableArray<ElementType>>, UITableView) -> UITableViewCell) -> DisposableType {
public func bindTo(collectionView: UICollectionView, proxyDataSource: BNDCollectionViewProxyDataSource? = nil, createCell: (NSIndexPath, ObservableArray<ObservableArray<ElementType>>, UICollectionView) -> UICollectionViewCell) -> DisposableType {
Don't let the long method signature scare you. Methods accept three arguments: a table or collection view, optional proxy data source if you need to provide other data then the cells (like section names) and a createCell
closure that will be used to create cells. Closure must accept three arguments that you use to create a cell and it must return a cell. Arguments it must accept are index path of the needed cell, a 2D observable array and a table or a collection view to dequeue cells from.
Such bindTo
methods are provided only on two-dimensional observable arrays. The outer one represents sections and each inner one represents rows or items of the respective section. If your data is not arranged in sections so you have a one-dimension array, you can simply use lift
method to wrap it into another array.
For example, let say we have two groups of names we would like to display in two sections of a table view:
let captains = ObservableArray(["Archer", "Kirk", "Picard"])
let firstOfficers = ObservableArray(["T'Pol", "Spock", "Riker"])
let dataSource = ObservableArray([captains, firstOfficers])
let tableView = UITableView()
dataSource.bindTo(tableView) { indexPath, dataSource, tableView in
let cell = tableView.dequeueReusableCellWithIdentifier("Cell", forIndexPath: indexPath)
let name = dataSource[indexPath.section][indexPath.row]
cell.textLabel.text = name
return cell
}
If, on the other hand, we want to display only captains - a one-dimensional array - we could do this:
captains.lift().bindTo(tableView) { indexPath, dataSource, tableView in
let cell = tableView.dequeueReusableCellWithIdentifier("Cell", forIndexPath: indexPath)
let name = dataSource[indexPath.section][indexPath.row]
cell.textLabel.text = name
return cell
}
Using the Bond framework can simplify interaction with KVO properties. You can make an observable representation of them using the following constructor:
extension Observable {
public convenience init(object: NSObject, keyPath: String)
}
You need to provide the object
whose keyPath
you want observed. Note that you also need to manually specialize the observable because Swift cannot infer the type of the property given only the key path.
let name = Observable<NSString?>(object: self.viewModel, keyPath: "name")
Be aware that the observable strongly references the given object. You never want to observe self
!
- Add the following to your Cartfile:
github "SwiftBond/Bond" ~> 4.0
- Run
carthage update
- Add the framework as described in Carthage Readme
- Add the following to your Podfile:
pod 'Bond', '~> 4.0'
- Run
pod install
with CocoaPods 0.36 or newer.
- Clone Bond as a submodule into the directory of your choice
git submodule add git@github.com:SwiftBond/Bond.git
git submodule update --init
- Drag Bond.xcodeproj into your project tree as a subproject
- Under your project's Build Phases, expand Target Dependencies
- Click the + and add Bond
- Expand the Link Binary With Libraries phase
- Click the + and add Bond
- Click the + at the top left corner to add a Copy Files build phase
- Set the directory to Frameworks
- Click the + and add Bond
Just get .swift files from Bond/ Directory and add them to your project.
Bond v4 represents a major evolution of the framework. It's core has been rewritten from scratch and, while concepts are still pretty much the same, some things have changed from the outside to. In order to successfully upgrade your project to Bond v4, it is recommended to re-read this document. After that, you can proceed with the conversion:
Convert objects of Dynamic
type to Observable
type. Simple renaming should do the trick.
Convert objects of DynamicArray
type to ObservableArray
type. Simple renaming should do the trick.
Bonds were used to observe changes of a Dynamic. With Bond v4, observing changes is much simpler. Instead of creating a new object, you can now use observe
method on any Observable type. In other words, code like
let myBond = Bond<Int>() { value in
print("Number of followers changed to \(value).")
}
numberOfFollowers.bindTo(myBond)
becomes
numberOfFollowers.observe { value in
println("Number of followers changed to \(value).")
}
To cancel observing in v3 you have used unbindAll
method on the Bond. In v4, cancelling the observation or unbinding the object is done with a disposable. Methods observe
and bindTo
return an object of a Disposable type. You can use that object to cancel observing, like this:
let disposable = numberOfFollowers.observe { value in
println("Number of followers changed to \(value).")
}
// ... and if you wish to cancel observing later, just call:
disposable.dispose()
Observing ObservableArrays is similar. Instead of calling various closures like DynamicArray did in v3, ObservableArray in v4 is an Observable that sends events that describe operation that was just applied to the ObservableArray. You can observe those in a following way:
array.observe { event in
switch event.operation {
case .Insert(let elements, let fromIndex):
// Did insert elements
case .Update(let elements, let fromIndex):
// Did update elements
case .Remove(let range):
// Did remove elements
case .Reset(let array):
// Did replace whole array with the another array
case .Batch(let operations):
// Did perform batch updates
}
}
In Bond v3, extensions were prefixed with dyn
, like in textField.dynText
. As Dynamics are now gone it makes no sense to keep that prefix. In v4 all extensions provided by Bond framework are prefixed with bnd_
.
While in Bond v3 you were able to bind, for example, a boolean Dynamic to a button,
canLogin.bindTo(loginButton)
it would not be clear from that line of code where you were really binding the value to. In Bond v4 you have to be specific and always provide a bindable destination, like:
canLogin.bindTo(loginButton.bnd_enabled)
Hope is that this will reduce any confusion and improve the code readability.
Talking about the code readability, another important decision has been made. In order to clearly express action behind a line of the code, method bindTo
has become a preferred way to bind objects. Operators ->>
and ->><
are still available, but their usage is not recommended.
You can use method observeNew
to observe only events that happen after the binding took place. Alternatively, you can use skip
method to skip event replaying:
name.skip(name.replayLength).bindTo(nameLabel.bnd_text)
What method reduce
did in v3 can now be achieved with a combination of combineLatest
and map
methods.
combineLatest(userLabel.bnd_text, passLabel.bnd_text).map { user, pass in
// do something
}
https://github.com/SwiftBond/Bond/releases
The MIT License (MIT)
Copyright (c) 2015 Srdan Rasic (@srdanrasic)
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.