scala.collection.mutable.Publisher.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of scala-library Show documentation
Show all versions of scala-library Show documentation
Standard library for the Scala Programming Language
/* __ *\
** ________ ___ / / ___ Scala API **
** / __/ __// _ | / / / _ | (c) 2003-2013, LAMP/EPFL **
** __\ \/ /__/ __ |/ /__/ __ | http://scala-lang.org/ **
** /____/\___/_/ |_/____/_/ | | **
** |/ **
\* */
package scala.collection
package mutable
/** `Publisher[A,This]` objects publish events of type `A`
* to all registered subscribers. When subscribing, a subscriber may specify
* a filter which can be used to constrain the number of events sent to the
* subscriber. Subscribers may suspend their subscription, or reactivate a
* suspended subscription. Class `Publisher` is typically used
* as a mixin. The abstract type `Pub` models the type of the publisher itself.
*
* @tparam Evt type of the published event.
*
* @author Matthias Zenger
* @author Martin Odersky
* @version 2.8
* @since 1
*/
trait Publisher[Evt] {
type Pub <: Publisher[Evt]
type Sub = Subscriber[Evt, Pub]
type Filter = Evt => Boolean
/** The publisher itself of type `Pub`. Implemented by a cast from `this` here.
* Needs to be overridden if the actual publisher is different from `this`.
*/
protected val self: Pub = this.asInstanceOf[Pub]
private val filters = new HashMap[Sub, Set[Filter]] with MultiMap[Sub, Filter]
private val suspended = new HashSet[Sub]
def subscribe(sub: Sub) { subscribe(sub, event => true) }
def subscribe(sub: Sub, filter: Filter) { filters.addBinding(sub, filter) }
def suspendSubscription(sub: Sub) { suspended += sub }
def activateSubscription(sub: Sub) { suspended -= sub }
def removeSubscription(sub: Sub) { filters -= sub }
def removeSubscriptions() { filters.clear }
protected def publish(event: Evt) {
filters.keys.foreach(sub =>
if (!suspended.contains(sub) &&
filters.entryExists(sub, p => p(event)))
sub.notify(self, event)
)
}
/** Checks if two publishers are structurally identical.
*
* @return true, iff both publishers contain the same sequence of elements.
*/
override def equals(obj: Any): Boolean = obj match {
case that: Publisher[_] => filters == that.filters && suspended == that.suspended
case _ => false
}
}