module Athena::EventDispatcher::EventDispatcherInterface
#
An event dispatcher is the primary type of Athena::EventDispatcher
.
It maintains a registry of listeners, with events also being dispatched via this type.
When dispatched, the dispatcher notifies all listeners registered with that event.
Usage#
Listeners can be added in a few ways, with the simplest being registering a block directly on the dispatcher instance.
class MyEvent < AED::Event; end
dispatcher.listener MyEvent do |event, dispatcher|
# Do something with the event, and/or dispatcher
end
Another way involves passing an AED::Callable
instance, created manually or via the AED::Event.callable
method.
Lastly, a type that has one or more AEDA::AsEventListener
annotated methods may also be passed.
Once all listeners are registered, you can begin to dispatch events.
Dispatching an event is simply calling the #dispatch
method with an AED::Event
subclass instance as an argument.
Listener Priority#
As you may have noticed, each way of registering a listener has an optional priority parameter.
This value can be a positive or negative integer, with a default of 0
that controls the order in which each listener is executed.
The higher the value, the sooner that listener would be executed.
If two listeners have the same priority, they are executed in the order in which they were registered with the dispatcher.
class MyEvent < AED::Event; end
dispatcher = AED::EventDispatcher.new
dispatcher.listener(MyEvent, priority: -10) { pp "callback1" }
dispatcher.listener(MyEvent, priority: 10) { pp "callback2" }
dispatcher.listener(MyEvent) { pp "callback3" }
dispatcher.listener(MyEvent, priority: 20) { pp "callback4" }
dispatcher.listener(MyEvent) { pp "callback5" }
dispatcher.dispatch MyEvent.new
# =>
# "callback4"
# "callback2"
# "callback3"
# "callback5"
# "callback1"
Note
While the priority can be any Int32
, best practices suggest keeping it in the -255..255
range.
Direct including types
Athena::EventDispatcher::EventDispatcher
Methods#
abstract #dispatch(event : AED::Event) : AED::Event
#
Dispatches the provided event to all listeners listening on that event. Listeners are executed in priority order, highest first.
abstract #has_listeners?(event_class : AED::Event.class) : Bool
#
Returns true
if this dispatcher has any listeners on the provided event_class.
abstract #has_listeners? : Bool
#
Returns true
if there are any listeners on any event.
abstract #listener(callable : AED::Callable) : AED::Callable
#
Registers the provided callable listener to this dispatcher.
#listener(listener : T) : Nil forall T
#
Registers the provided listener instance to this dispatcher.
T
is any type that has methods annotated with AEDA::AsEventListener
.
abstract #listener(callable : AED::Callable, *, priority : Int32) : AED::Callable
#
Registers the provided callable listener to this dispatcher, overriding its priority with that of the provided priority.
abstract #listener(event_class : E.class, *, priority : Int32 = 0, name : String | Nil = nil, &block : E, AED::EventDispatcherInterface -> Nil) : AED::Callable forall E
#
Registers the block as an AED::Callable
on the provided event_class, optionally with the provided priority and/or name.
abstract #listeners(for event_class : AED::Event.class) : Array(AED::Callable)
#
Returns an Array(AED::Callable)
for all listeners on the provided event_class.
abstract #listeners : Hash(AED::Event.class, Array(AED::Callable))
#
Returns a hash of all registered listeners as a Hash(AED::Event.class, Array(AED::Callable))
.
abstract #remove_listener(callable : AED::Callable) : Nil
#
Deregisters the provided callable from this dispatcher.
Tip
The callable may be one retrieved via either #listeners
method.
#remove_listener(listener : T) : Nil forall T
#
Deregisters listeners based on the provided listener from this dispatcher.
T
is any type that has methods annotated with AEDA::AsEventListener
.