Created
February 11, 2015 09:15
-
-
Save ColinEberhardt/05fafaca143ac78dbe09 to your computer and use it in GitHub Desktop.
An eventing mechanism for Swift
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/// An object that has some tear-down logic | |
public protocol Disposable { | |
func dispose() | |
} | |
/// An event provides a mechanism for raising notifications, together with some | |
/// associated data. Multiple function handlers can be added, with each being invoked, | |
/// with the event data, when the event is raised. | |
public class Event<T> { | |
public typealias EventHandler = T -> () | |
private var eventHandlers = [Invocable]() | |
public init() { | |
} | |
/// Raises the event, invoking all handlers | |
public func raise(data: T) { | |
for handler in self.eventHandlers { | |
handler.invoke(data) | |
} | |
} | |
/// Adds the given handler | |
public func addHandler<U: AnyObject>(target: U, handler: (U) -> EventHandler) -> Disposable { | |
let wrapper = EventHandlerWrapper(target: target, handler: handler, event: self) | |
eventHandlers.append(wrapper) | |
return wrapper | |
} | |
} | |
// MARK:- Private | |
// A protocol for a type that can be invoked | |
private protocol Invocable: class { | |
func invoke(data: Any) | |
} | |
// takes a reference to a handler, as a class method, allowing | |
// a weak reference to the owning type. | |
// see: http://oleb.net/blog/2014/07/swift-instance-methods-curried-functions/ | |
private class EventHandlerWrapper<T: AnyObject, U> : Invocable, Disposable { | |
weak var target: T? | |
let handler: T -> U -> () | |
let event: Event<U> | |
init(target: T?, handler: T -> U -> (), event: Event<U>){ | |
self.target = target | |
self.handler = handler | |
self.event = event; | |
} | |
func invoke(data: Any) -> () { | |
if let t = target { | |
handler(t)(data as U) | |
} | |
} | |
func dispose() { | |
event.eventHandlers = event.eventHandlers.filter { $0 !== self } | |
} | |
} |
it seems that this method only works on simple types such as int / string etc..
how about Arrays and Dictionaries?
It is great to start writing KVO. But how to remove the observer?
I am getting a build error on the eventHandlers.
in the dispose function it is saying that eventHandlers is inaccessible due to the private protection level, but you can't remove private because of the [Invocables] ...
Ideas how to fix?
Make if file private:
fileprivate var eventHandlers = [Invocable]()
Ideas how to fix?
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
How about automatically disposing the
EventHandlerWrapper
when the target reference isnil
within theinvoke
method?