de.sciss.fscape.Lazy.scala Maven / Gradle / Ivy
/*
* Lazy.scala
* (FScape)
*
* Copyright (c) 2001-2016 Hanns Holger Rutz. All rights reserved.
*
* This software is published under the GNU General Public License v2+
*
*
* For further information, please contact Hanns Holger Rutz at
* [email protected]
*/
package de.sciss.fscape
object Lazy {
/** A convenient implementation of the `Lazy` trait for elements which typically expand
* to ugens. This will be typically used for elements which do not directly need to
* generate ugens but rather spawn more graph elements. For the direct generation of
* `UGen`s, use a subtype of `UGenSource`.
*
* The constructor body of this trait will call `Graph.builder.addLazy` to automatically
* register this element when instantiated.
*
* @tparam U the type to which this element expands, e.g. `Unit` or `UGenInLike`
* @see [[de.sciss.fscape.UGenSource]]
*/
trait Expander[+U] extends Lazy {
// this acts now as a fast unique reference
@transient final private[this] lazy val ref = new AnyRef
// ---- constructor ----
Graph.builder.addLazy(this)
/** A final implementation of this method which calls `visit` on the builder,
* checking if this element has already been visited, and if not, will invoke
* the `expand` method. Therefore it is guaranteed, that the expansion to
* ugens is performed no more than once in the graph expansion.
*/
final private[fscape] def force(b: UGenGraph.Builder): Unit = expand(b)
/** A final implementation of this method which looks up the current ugen graph
* builder and then performs the expansion just as `force`, returning the
* expanded object
*
* @return the expanded object (e.g. `Unit` for a ugen with no outputs,
* or a single ugen, or a group of ugens)
*/
final private[fscape] def expand(implicit b: UGenGraph.Builder): U = b.visit(ref, makeUGens)
/** Abstract method which must be implemented by creating the actual `UGen`s
* during expansion. This method is at most called once during graph
* expansion
*
* @return the expanded object (depending on the type parameter `U`)
*/
protected def makeUGens(implicit b: UGenGraph.Builder): U
}
}
/** Elements implementing the `Lazy` trait may participate in the building of a
* `Graph` body. They can be added to the current graph by calling
* `Graph.builder.addLazy`. Then, when the graph is expanded, the
* `force` method is called on those registered elements, allowing them
* to either spawn new graph elements or actually expand to `UGen`s which
* can be added to the ugen graph builder argument.
*
* In most cases, lazy elements will expanded to ugens, and thus the subtype
* `Lazy.Expander` is the most convenient way to implement this trait, as it already
* does most of the logic, and provides for `GE`s `expand` method.
*
* @see [[de.sciss.fscape.Lazy.Expander]]
*/
trait Lazy extends Product {
/** This method is invoked by the `UGenGraph.Builder` instance when a `Graph`
* is expanded.
*
* @param b the ugen graph builder to which expanded `UGen`s or control proxies
* may be added.
*/
private[fscape] def force(b: UGenGraph.Builder): Unit
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy