de.sciss.synth.message.ServerMessages.scala Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of scalacollider_3 Show documentation
Show all versions of scalacollider_3 Show documentation
A sound synthesis library for the SuperCollider server
The newest version!
/*
* ServerMessages.scala
* (ScalaCollider)
*
* Copyright (c) 2008-2021 Hanns Holger Rutz. All rights reserved.
*
* This software is published under the GNU Affero General Public License v3+
*
*
* For further information, please contact Hanns Holger Rutz at
* [email protected]
*/
package de.sciss.synth
package message
import de.sciss.audiofile.{AudioFileType, SampleFormat}
import de.sciss.osc
import de.sciss.osc.{Message, Packet}
import de.sciss.synth.Import._
import java.nio.ByteBuffer
import scala.collection.{IndexedSeq => SIndexedSeq}
import scala.language.implicitConversions
/** Identifies messages received or sent by the SuperCollider server. */
sealed trait ServerMessage
/** Identifies messages sent to the SuperCollider server. */
sealed trait Send extends ServerMessage {
def isSynchronous: Boolean
}
/** Identifies messages sent to the server which are executed synchronously. */
sealed trait SyncSend extends Send {
final def isSynchronous = true
}
/** Identifies command messages sent to the server which are executed synchronously and do not return a message. */
trait SyncCmd extends SyncSend
/** Identifies query messages sent to the server which are executed synchronously and produce a reply message. */
trait SyncQuery extends SyncSend
/** Identifies messages sent to the server which are executed asynchronously and reply with a form of done-message. */
sealed trait AsyncSend extends Send {
final def isSynchronous = false
}
/** Identifies messages returned by SuperCollider server. */
trait Receive extends ServerMessage
/** Represents a `/synced` message, a reply from the server acknowledging that
* all asynchronous operations up to the corresponding `/sync` message (i.e. with
* the same id) have been completed
*/
final case class Synced(id: Int) extends Message("/synced", id) with Receive
/** Represents a `/sync` message, which is queued with the asynchronous messages
* on the server, and which, when executed, triggers a corresponding `/synced` reply
* message (i.e. with the same id)
*
* @param id an arbitrary identifier which can be used to match the corresponding
* reply message. typically the id is incremented by 1 for each
* `/sync` message sent out.
*/
final case class Sync(id: Int) extends Message("/sync", id) with AsyncSend {
def reply: Synced = Synced(id)
}
final case class StatusReply(numUGens: Int, numSynths: Int, numGroups: Int, numDefs: Int, avgCPU: Float,
peakCPU: Float, sampleRate: Double, actualSampleRate: Double)
extends Message("/status.reply", 1, numUGens, numSynths, numGroups, numDefs, avgCPU, peakCPU,
sampleRate, actualSampleRate)
with Receive {
override def toString(): String = {
val s1 = f"numUGens = $numUGens, numSynths = $numSynths, numGroups = $numGroups"
val s2 = f"cpu = $avgCPU%1.1f%%/$peakCPU%1.1f%%, sampleRate = $sampleRate%1.0f/$actualSampleRate%1.2f"
s"$productPrefix($s1, $s2)"
}
}
/** The `/status` message that queries the current statistics from the server. */
case object Status extends Message("/status") with SyncQuery
/** The `/dumpOSC` message that selects how the server reports incoming OSC packets.
*
* __Note:__ The OSC dump behavior of scsynth has long time been broken. It is recommended
* to use client-side only reporting, provided by the `dumpOSC` method of the `Server` class.
*
* @see [[Server#dumpOSC]]
*/
final case class DumpOSC(mode: osc.Dump) extends Message("/dumpOSC", mode.id) with SyncCmd
case object ClearSched extends Message("/clearSched") with SyncCmd
object Error {
val Off : Error = apply(0)
val On : Error = apply(1)
val BundleOff: Error = apply(-1)
val BundleOn : Error = apply(-2)
}
/** Produces an `/error` message that selects how the server will report errors to the console. */
final case class Error(mode: Int) extends Message("/error", mode) with SyncCmd
sealed trait NodeChange extends Receive {
def nodeId: Int
def info: NodeInfo.Data
}
private[synth] sealed trait NodeMessageFactory {
def apply(nodeId: Int, info: NodeInfo.Data): Message
}
/** The `/n_go` message is received from the server when a node has been newly created.
*
* @see [[SynthNew]]
* @see [[GroupNew]]
*/
object NodeGo extends NodeMessageFactory
final case class NodeGo(nodeId: Int, info: NodeInfo.Data)
extends Message("/n_go", info.toList(nodeId): _*) with NodeChange
/** The `/n_end` message is received from the server when a node has been freed.
*
* @see [[NodeFree]]
*/
object NodeEnd extends NodeMessageFactory
final case class NodeEnd(nodeId: Int, info: NodeInfo.Data)
extends Message("/n_end", info.toList(nodeId): _*) with NodeChange
/** The `/n_on` message is received from the server when a node has resumed.
*
* @see [[NodeRun]]
*/
object NodeOn extends NodeMessageFactory
final case class NodeOn(nodeId: Int, info: NodeInfo.Data)
extends Message("/n_on", info.toList(nodeId): _*) with NodeChange
/** The `/n_off` message is received from the server when a node has been paused.
*
* @see [[NodeRun]]
*/
object NodeOff extends NodeMessageFactory
final case class NodeOff(nodeId: Int, info: NodeInfo.Data)
extends Message("/n_off", info.toList(nodeId): _*) with NodeChange
/** The `/n_move` message is received from the server when a node has changed its position in the tree.
*
* @see [[NodeBefore]]
* @see [[NodeAfter]]
* @see [[GroupHead]]
* @see [[GroupTail]]
*/
object NodeMove extends NodeMessageFactory
final case class NodeMove(nodeId: Int, info: NodeInfo.Data)
extends Message("/n_move", info.toList(nodeId): _*) with NodeChange
object NodeInfo extends NodeMessageFactory {
/** @see [[NodeInfo]]
* @see [[NodeGo]]
* @see [[NodeEnd]]
* @see [[NodeOn]]
* @see [[NodeOff]]
* @see [[NodeMove]]
*/
abstract sealed class Data {
/** The identifier of the node's parent group. */
def parentId: Int
/** The identifier of the node preceding this node within the same group,
* or `-1` if there is no predecessor.
*/
def predId: Int
/** The identifier of the node following this node within the same group,
* or `-1` if there is no successor.
*/
def succId: Int
/** The serial presentation of the information within an OSC message.
* This method is used internally and probably not useful in other ways.
*/
def toList(nodeId: Int): List[Any]
}
/** Information about a `Synth` node. */
final case class SynthData(parentId: Int, predId: Int, succId: Int) extends Data {
def toList(nodeId: Int): List[Any] = nodeId :: parentId :: predId :: succId :: 0 :: Nil
}
/** Information about a `Group` node. */
final case class GroupData(parentId: Int, predId: Int, succId: Int, headId: Int, tailId: Int) extends Data {
def toList(nodeId: Int): List[Any] = nodeId :: parentId :: predId :: succId :: 1 :: headId :: tailId :: Nil
}
}
/** An `/n_info` message is received as a reply to an `/n_query` message.
*
* @param nodeId the identifier of the node for which information has been received
* @param info the information object describing the topological position of the node
*
* @see [[NodeQuery]]
*/
final case class NodeInfo(nodeId: Int, info: NodeInfo.Data)
extends Message("/n_info", info.toList(nodeId): _*) with NodeChange
// we need List[Any] as scala would otherwise expand to List[Float]!
object BufferInfo {
/** @see [[BufferInfo]] */
final case class Data(bufId: Int, numFrames: Int, numChannels: Int, sampleRate: Float) {
def asBuffer(server: Server): Buffer = Buffer(server, bufId).init(this)
}
}
/** A `/b_info` message is received in reply to a `/b_query` message.
*
* @see [[BufferQuery]]
*/
final case class BufferInfo(data: BufferInfo.Data*)
extends Message("/b_info", data.flatMap(info =>
List[Any](info.bufId, info.numFrames, info.numChannels, info.sampleRate)): _*)
with Receive
// ---- messages to the server ----
/** The `/notify` messages registers or de-registers a client with respect
* to receiving reply messages from the server.
*
* Booting or connecting a server through the regular API automatically handles and does
* not require the explicit use of this message.
*
* @param on if `true`, the client is registered, if `false` it is de-registered.
*/
final case class ServerNotify(on: Boolean)
extends Message("/notify", on)
with AsyncSend
/** The `/quit` message tells the server to shut down.
*
* @see [[Server#quit]]
*/
case object ServerQuit extends Message("/quit") with AsyncSend
object HasCompletion {
def unapply(h: HasCompletion): Option[Option[Packet]] = Some(h.completion)
}
sealed trait HasCompletion extends AsyncSend {
def completion: Option[Packet]
def updateCompletion(completion: Option[Packet]): Message with AsyncSend with HasCompletion
}
/** The `/b_query` messages requests a `/b_info` reply message from the server, providing
* information about the size and sample-rate of the specified buffers.
*
* @param ids a sequence of buffer identifiers to query
*
* @see [[Buffer#queryMsg]]
*/
final case class BufferQuery(ids: Int*) extends Message("/b_query", ids: _*) with SyncQuery
/** The `/b_free` message frees a buffer on the server side. The client side
* typically maintains a logical list of allocated buffer identifiers as well,
* so one should normally rely on the specific client side API to correctly
* free a buffer.
*
* @see [[Buffer#freeMsg]]
* @see [[BufferClose]]
* @see [[BufferAlloc]]
*/
final case class BufferFree(id: Int, completion: Option[Packet])
extends Message("/b_free", (id: Any) :: (completion.toList: List[Any]): _*)
with HasCompletion {
def updateCompletion(completion: Option[Packet]): BufferFree = copy(completion = completion)
}
/** The `/b_close` message ensures that a buffer closes an associated audio-file.
* This is a no-op if the buffer is not associated with an audio-file or if that
* file is already closed.
*
* @see [[Buffer#closeMsg]]
* @see [[BufferFree]]
* @see [[BufferAlloc]]
*/
final case class BufferClose(id: Int, completion: Option[Packet])
extends Message("/b_close", (id: Any) :: (completion.toList: List[Any]): _*)
with HasCompletion {
def updateCompletion(completion: Option[Packet]): BufferClose = copy(completion = completion)
}
/** The `/b_alloc` message tells the server to allocate memory for a buffer associated
* with its logical identifier.
*
* @see [[Buffer#allocMsg]]
* @see [[BufferFree]]
* @see [[BufferAllocRead]]
* @see [[BufferAllocReadChannel]]
*/
final case class BufferAlloc(id: Int, numFrames: Int, numChannels: Int, completion: Option[Packet])
extends Message("/b_alloc", id :: numFrames :: numChannels :: completion.toList: _*)
with HasCompletion {
def updateCompletion(completion: Option[Packet]): BufferAlloc = copy(completion = completion)
}
/** The `/b_allocRead` message tells the server to allocate memory for a buffer and read
* in a portion of an audio-file. The number of channels
* and the sample-rate of the buffer are determined by that audio-file.
*
* @param id the identifier to use for the buffer. It must denote a currently un-allocated buffer
* and be greater than or equal to zero and less than the maximum number of buffers.
* @param path the path of the audio-file to read. Since the server is an independent process, this must
* resolve with respect to the server's current working directory. If the server is running on
* a remote node, the path will be resolved in the server's local file system.
* @param startFrame the offset in frames into the audio-file to begin reading from
* @param numFrames the number of frames to read which will be the size of the allocated buffer. The special
* value less than or equal to zero denotes that the number of frames available in the file
* from the given offset is used (the entire file will be read).
*
* @see [[Buffer#allocReadMsg]]
* @see [[BufferFree]]
* @see [[BufferAlloc]]
* @see [[BufferAllocReadChannel]]
* @see [[BufferRead]]
*/
final case class BufferAllocRead(id: Int, path: String, startFrame: Int, numFrames: Int, completion: Option[Packet])
extends Message("/b_allocRead", id :: path :: startFrame :: numFrames :: completion.toList: _*)
with HasCompletion {
def updateCompletion(completion: Option[Packet]): BufferAllocRead = copy(completion = completion)
}
/** The `/b_allocReadChannel` message tells the server to allocate memory for a buffer and read
* in a portion of an audio-file, selecting a subset of its channels. The number of channels
* is given by the size of the `channels` argument, and the sample-rate of the buffer is determined
* by the audio-file.
*
* @param id the identifier to use for the buffer. It must denote a currently un-allocated buffer
* and be greater than or equal to zero and less than the maximum number of buffers.
* @param path the path of the audio-file to read. Since the server is an independent process, this must
* resolve with respect to the server's current working directory. If the server is running on
* a remote node, the path will be resolved in the server's local file system.
* @param startFrame the offset in frames into the audio-file to begin reading from
* @param numFrames the number of frames to read which will be the size of the allocated buffer. The special
* value of `-1` denotes that the number of frames available in the file
* from the given offset is used (the entire file will be read).
* @param channels a sequence of channel indices to read. Zero corresponds to the first channel of the file.
*
* @see [[Buffer#allocReadChannelMsg]]
* @see [[BufferFree]]
* @see [[BufferAlloc]]
* @see [[BufferAllocRead]]
* @see [[BufferReadChannel]]
*/
final case class BufferAllocReadChannel(id: Int, path: String, startFrame: Int, numFrames: Int,
channels: List[Int], completion: Option[Packet])
extends Message("/b_allocReadChannel", id :: path :: startFrame :: numFrames :: channels ::: completion.toList: _*)
with HasCompletion {
def updateCompletion(completion: Option[Packet]): BufferAllocReadChannel = copy(completion = completion)
}
/** The `/b_read` message tells the server to read a portion of an audio-file into an existing buffer.
*
* @param id the identifier of the buffer to read into.
* @param path the path of the audio-file to read.
* @param fileStartFrame the offset in frames into the audio-file to begin reading from
* @param numFrames the number of frames to read which will be the size of the allocated buffer. The special
* value of `-1` denotes that as many frames are read as are available in the file or
* fit into the buffer.
* @param bufStartFrame the frame offset in the buffer to begin writing to.
* @param leaveOpen if `true`, leaves the file open for streaming with the
* [[de.sciss.synth.ugen.DiskIn DiskIn]] UGen.
*
* @see [[Buffer#readMsg]]
* @see [[BufferAllocRead]]
* @see [[BufferReadChannel]]
*/
final case class BufferRead(id: Int, path: String, fileStartFrame: Int, numFrames: Int, bufStartFrame: Int,
leaveOpen: Boolean, completion: Option[Packet])
extends Message("/b_read",
id :: path :: fileStartFrame :: numFrames :: bufStartFrame :: leaveOpen :: completion.toList: _*)
with HasCompletion {
def updateCompletion(completion: Option[Packet]): BufferRead = copy(completion = completion)
}
/** The `/b_readChannel` message tells the server to read a portion of an audio-file into an existing buffer,
* selecting a subset of the file's channels.
*
* @param id the identifier of the buffer to read into.
* @param path the path of the audio-file to read.
* @param fileStartFrame the offset in frames into the audio-file to begin reading from
* @param numFrames the number of frames to read which will be the size of the allocated buffer. The special
* value of `-1` denotes that as many frames are read as are available in the file or
* fit into the buffer.
* @param bufStartFrame the frame offset in the buffer to begin writing to.
* @param leaveOpen if `true`, leaves the file open for streaming with the
* [[de.sciss.synth.ugen.DiskIn DiskIn]] UGen.
* @param channels a sequence of channel indices to read. Zero corresponds to the first channel of the file.
*
* @see [[Buffer#readChannelMsg]]
* @see [[BufferAllocReadChannel]]
* @see [[BufferRead]]
*/
final case class BufferReadChannel(id: Int, path: String, fileStartFrame: Int, numFrames: Int,
bufStartFrame: Int, leaveOpen: Boolean, channels: List[Int],
completion: Option[Packet])
extends Message("/b_readChannel",
id :: path :: fileStartFrame :: numFrames :: bufStartFrame :: leaveOpen :: channels ::: completion.toList: _*)
with HasCompletion {
def updateCompletion(completion: Option[Packet]): BufferReadChannel = copy(completion = completion)
}
/** The `/b_zero` message clears the contents of a buffer (all samples will be zero).
*
* @see [[Buffer#zeroMsg]]
* @see [[BufferFill]]
* @see [[BufferSet]]
* @see [[BufferSetn]]
* @see [[BufferGen]]
*/
final case class BufferZero(id: Int, completion: Option[Packet])
extends Message("/b_zero", (id: Any) :: (completion.toList: List[Any]): _*)
with HasCompletion {
def updateCompletion(completion: Option[Packet]): BufferZero = copy(completion = completion)
}
/** The `/b_write` message writes a portion of the buffer contents to an audio-file.
*
* @param id the identifier of the buffer whose contents to write.
* @param path the path of the audio-file to write to.
* @param fileType the header format of the audio-file
* @param sampleFormat the sample resolution of the audio-file
* @param numFrames the number of frames to write. The special
* value of `-1` denotes that the whole buffer content (or the remainder
* after the `startFrame`) is written out.
* @param startFrame the frame offset in the buffer to begin reading from
* @param leaveOpen if `true`, leaves the file open for streaming with
* the [[de.sciss.synth.ugen.DiskOut DiskOut]] UGen.
*
* @see [[Buffer#writeMsg]]
* @see [[BufferRead]]
*/
final case class BufferWrite(id: Int, path: String, fileType: AudioFileType, sampleFormat: SampleFormat,
numFrames: Int, startFrame: Int, leaveOpen: Boolean,
completion: Option[Packet])
extends Message("/b_write",
id :: path :: fileType.id :: sampleFormat.id :: numFrames :: startFrame :: leaveOpen :: completion.toList: _*)
with HasCompletion {
def updateCompletion(completion: Option[Packet]): BufferWrite = copy(completion = completion)
}
/** The `/b_set` message sets individual samples of the buffer to given values.
*
* @param id the identifier of the buffer whose contents to write.
* @param pairs pairs of sample offsets and sample values. The offsets are de-interleaved samples,
* so for multi-channel buffers, to address a particular frame, the frame index must
* be multiplied by the number of channels and offset by the channel to write into.
*
* @see [[Buffer#setMsg]]
* @see [[BufferSetn]]
* @see [[BufferFill]]
* @see [[BufferZero]]
* @see [[BufferGen]]
*/
final case class BufferSet(id: Int, pairs: FillValue*)
extends Message("/b_set", id +: pairs.flatMap(_.toList): _*)
with SyncCmd {
def copy(id: Int = id, pairs: Seq[FillValue] = pairs): BufferSet =
BufferSet(id = id, pairs = pairs: _*)
}
/** The `/b_setn` message sets individual ranges of samples of the buffer to given values.
*
* @param id the identifier of the buffer whose contents to write.
* @param indicesAndValues pairs of sample offsets and sequences of sample values.
* The offsets are de-interleaved samples,
* so for multi-channel buffers, to address a particular frame, the frame index must
* be multiplied by the number of channels and offset by the channel to write into.
*
* @see [[Buffer#setnMsg]]
* @see [[BufferSet]]
* @see [[BufferFill]]
* @see [[BufferZero]]
* @see [[BufferGen]]
*/
final case class BufferSetn(id: Int, indicesAndValues: (Int, SIndexedSeq[Float])*)
extends Message("/b_setn", id +: indicesAndValues.flatMap(iv => iv._1 +: iv._2.size +: iv._2): _*)
with SyncCmd {
def copy(id: Int = id, indicesAndValues: Seq[(Int, SIndexedSeq[Float])] = indicesAndValues): BufferSetn =
BufferSetn(id = id, indicesAndValues = indicesAndValues: _*)
}
/** The `/b_fill` message sets individual ranges of samples of the buffer to given values.
*
* @param id the identifier of the buffer whose contents to write.
* @param ranges tuples which specify the offset into the buffer, the number of samples to overwrite and the
* value with which to overwrite.
*
* @see [[Buffer#fillMsg]]
* @see [[BufferSet]]
* @see [[BufferSetn]]
* @see [[BufferZero]]
* @see [[BufferGen]]
*/
final case class BufferFill(id: Int, ranges: FillRange*)
extends Message("/b_fill", id +: ranges.flatMap(_.toList): _*)
with SyncCmd {
def copy(id: Int = id, ranges: Seq[FillRange] = ranges): BufferFill =
BufferFill(id = id, ranges = ranges: _*)
}
object BufferGen {
trait Command extends Product {
def name : String
def args : Seq[Any]
def readerKey : String
override final def productPrefix: String = readerKey
}
sealed trait WaveFill extends Command {
def normalize : Boolean
def wavetable : Boolean
def clear : Boolean
protected final def flags: Int = (if (normalize) 1 else 0) | (if (wavetable) 2 else 0) | (if (clear) 4 else 0)
}
object Sine1 {
final val readerKey = s"BufferGen$$Sine1"
}
/** OSC message for filling a buffer with a series of sine wave harmonics using specified amplitudes.
*
* @param partials amplitudes for the harmonics. The first value specifies the amplitude of the first
* partial, the second float value specifies the amplitude of the second partial, and so on.
* @param normalize if set, the peak amplitude of the generated waveform is normalized to `1.0`
* @param wavetable if set, the format of the waveform is chosen to be usable by interpolating
* oscillators such as [[de.sciss.synth.ugen.Osc Osc]] or [[de.sciss.synth.ugen.VOsc VOsc]]
* @param clear if set, the previous content is erased, otherwise the new waveform is added
* to the existing content
*/
final case class Sine1(partials: Seq[Float], normalize: Boolean, wavetable: Boolean, clear: Boolean)
extends WaveFill {
override def readerKey: String = Sine1.readerKey
def name = "sine1"
def args: Seq[Any] = flags +: partials
}
object Sine2 {
final val readerKey = s"BufferGen$$Sine2"
}
/** OSC message for filling a buffer with a series of sine waves using specified frequencies and amplitudes.
*
* @param partials pairs of frequencies and amplitudes for the partials.
* Frequencies are given as in cycles per buffer.
* @param normalize if set, the peak amplitude of the generated waveform is normalized to `1.0`
* @param wavetable if set, the format of the waveform is chosen to be usable by interpolating
* oscillators such as [[de.sciss.synth.ugen.Osc Osc]] or [[de.sciss.synth.ugen.VOsc VOsc]]
* @param clear if set, the previous content is erased, otherwise the new waveform is added
* to the existing content
*/
final case class Sine2(partials: Seq[(Float, Float)], normalize: Boolean, wavetable: Boolean, clear: Boolean)
extends WaveFill {
override def readerKey: String = Sine2.readerKey
def name = "sine2"
def args: Seq[Any] = flags +: partials.flatMap(tup => tup._1 :: tup._2 :: Nil)
}
object Sine3 {
final case class Data(freq: Float, amp: Float, phase: Float)
implicit def data(tup: (Float, Float, Float)): Data = Data(tup._1, tup._2, tup._3)
final val readerKey = s"BufferGen$$Sine3"
}
/** OSC message for filling a buffer with a series of sine waves using specified frequencies, amplitudes,
* and phases.
*
* @param partials triplets of frequencies, amplitudes and initial phases for the partials.
* Frequencies are given as in cycles per buffer. Phases are given in radians.
* @param normalize if set, the peak amplitude of the generated waveform is normalized to `1.0`
* @param wavetable if set, the format of the waveform is chosen to be usable by interpolating
* oscillators such as [[de.sciss.synth.ugen.Osc Osc]] or [[de.sciss.synth.ugen.VOsc VOsc]]
* @param clear if set, the previous content is erased, otherwise the new waveform is added
* to the existing content
*/
final case class Sine3(partials: Seq[(Float, Float, Float)], normalize: Boolean, wavetable: Boolean, clear: Boolean)
extends WaveFill {
override def readerKey: String = Sine3.readerKey
def name = "sine3"
def args: Seq[Any] = flags +: partials.flatMap(d => d._1 :: d._2 :: d._3 :: Nil)
}
object Cheby {
final val readerKey = s"BufferGen$$Cheby"
}
/** OSC message for filling a buffer with a series of Chebyshev polynomials.
* The formula of these polynomials is
* {{{
* cheby(n) = amplitude * cos(n * acos(x))
* }}}
* To eliminate a DC offset when used as a wave-shaper, the wavetable is offset so that the center value is zero.
*
* @param amps amplitudes for the harmonics. amplitudes for the harmonics. The first value specifies
* the amplitude for n = 1, the second float value specifies the amplitude for n = 2, and so on.
* @param normalize if set, the peak amplitude of the generated waveform is normalized to `1.0`
* @param wavetable if set, the format of the waveform is chosen to be usable by specific UGens
* such as such as [[de.sciss.synth.ugen.Shaper Shaper]] or
* [[de.sciss.synth.ugen.Osc Osc]]
* @param clear if set, the previous content is erased, otherwise the new waveform is added
* to the existing content
*/
final case class Cheby(amps: Seq[Float], normalize: Boolean, wavetable: Boolean, clear: Boolean)
extends WaveFill {
override def readerKey: String = Cheby.readerKey
def name = "cheby"
def args: Seq[Any] = flags +: amps
}
object Copy {
final val readerKey = s"BufferGen$$Copy"
}
/** Copies samples from the source buffer to the destination buffer specified in the `b_gen` message.
* If the number of samples to copy is negative, the maximum number of samples possible is copied.
*/
final case class Copy(targetOffset: Int, source: Int, sourceOffset: Int, num: Int)
extends Command {
override def readerKey: String = Copy.readerKey
def name = "copy"
def args: Seq[Any] = List(targetOffset, source, sourceOffset, num)
// XXX TODO remove in major version
@deprecated("Will be removed", since = "2.7.3")
def isSynchronous = false
}
object PreparePartConv {
final val readerKey = s"BufferGen$$PreparePartConv"
}
/** OSC message for preparing a buffer with an impulse response for the `PartConv` UGen.
*
* @param bufIn buffer id of the existing time domain signal content to prepare
* @param fftSize the partitioned convolution's fft size
*/
final case class PreparePartConv(bufIn: Int, fftSize: Int)
extends Command {
override def readerKey: String = PreparePartConv.readerKey
def name = "PreparePartConv"
def args: Seq[Any] = bufIn :: fftSize :: Nil
}
}
/** The `/b_gen` message uses a dedicated command to generate or manipulate the buffer content.
*
* @param id the identifier of the buffer whose contents to write.
* @param command the operation to carry out on the buffer, such as generating a waveform or copying the content
*
* @see [[Buffer#genMsg]]
* @see [[BufferSet]]
* @see [[BufferSetn]]
* @see [[BufferZero]]
* @see [[BufferFill]]
*/
final case class BufferGen(id: Int, command: BufferGen.Command)
extends Message("/b_gen", id +: command.name +: command.args: _*)
with AsyncSend {
// def isSynchronous: Boolean = command.isSynchronous
}
/** The `/b_get` message.
*
* @see [[Buffer#getMsg]]
* @see [[BufferGetn]]
* @see [[BufferSet]]
*/
final case class BufferGet(id: Int, index: Int*) // `indices` is taken by SeqLike
extends Message("/b_get", id +: index: _*)
with SyncQuery {
def copy(id: Int = id, index: Seq[Int] = index): BufferGet =
BufferGet(id = id, index = index: _*)
}
/** The `/b_getn` message.
*
* @param id the buffer identifier
* @param ranges the ranges of sample indices to obtain. For multi-channel
* buffers, this means if one wants to specify a particular range
* of frames, the frame indices must be multiplied by the number of channels
*
* @see [[Buffer#getnMsg]]
* @see [[BufferGet]]
* @see [[BufferSetn]]
*/
final case class BufferGetn(id: Int, ranges: Range*)
extends Message("/b_getn", id +: ranges.flatMap(_.toGetnSeq): _*)
with SyncQuery {
def copy(id: Int = id, ranges: Seq[Range] = ranges): BufferGetn =
BufferGetn(id = id, ranges = ranges: _*)
}
/** The `/c_set` message.
*
* @see [[ControlBus#setMsg]]
* @see [[ControlBusSetn]]
* @see [[ControlBusGet]]
* @see [[ControlBusFill]]
*/
final case class ControlBusSet(pairs: FillValue*)
extends Message("/c_set", pairs.flatMap(_.toList): _*)
with SyncCmd
/** The `/c_setn` message.
*
* @see [[ControlBus#setnMsg]]
* @see [[ControlBusSet]]
* @see [[ControlBusGetn]]
* @see [[ControlBusFill]]
*/
final case class ControlBusSetn(indicesAndValues: (Int, SIndexedSeq[Float])*)
extends Message("/c_setn", indicesAndValues.flatMap(iv => iv._1 +: iv._2.size +: iv._2): _*)
with SyncCmd
/** The `/c_get` message.
*
* @see [[ControlBus#getMsg]]
* @see [[ControlBusGetn]]
* @see [[ControlBusSet]]
*/
final case class ControlBusGet(index: Int*) // `indices` is taken by SeqLike
extends Message("/c_get", index: _*)
with SyncQuery
/** The `/c_getn` message.
*
* @see [[ControlBus#getnMsg]]
* @see [[ControlBusGet]]
* @see [[ControlBusSetn]]
*/
final case class ControlBusGetn(ranges: Range*)
extends Message("/c_getn", ranges.flatMap(_.toGetnSeq): _*)
with SyncQuery
/** The `/c_fill` message.
*
* @see [[ControlBus#fillMsg]]
* @see [[ControlBusSet]]
* @see [[ControlBusSetn]]
*/
final case class ControlBusFill(ranges: FillRange*)
extends Message("/c_fill", ranges.flatMap(_.toList): _*)
with SyncCmd
object GroupNew {
final case class Data(groupId: Int, addAction: Int, targetId: Int) {
def toList: List[Int] = groupId :: addAction :: targetId :: Nil
}
}
/** The `/g_new` message. */
final case class GroupNew(groups: GroupNew.Data*)
extends Message("/g_new", groups.flatMap(_.toList): _*)
with SyncCmd
/** The `/g_dumpTree` message. */
final case class GroupDumpTree(groups: (Int, Boolean)*)
extends Message("/g_dumpTree", groups.flatMap(g => (g._1: Any) :: (g._2: Any) :: Nil): _*)
with SyncCmd
/** The `/g_queryTree` message. */
final case class GroupQueryTree(groups: (Int, Boolean)*)
extends Message("/g_queryTree", groups.flatMap(g => (g._1: Any) :: (g._2: Any) :: Nil): _*)
with SyncQuery
/** The `/g_head` message pair-wise places nodes at the head of groups.
* {{{
* /g_head
* [
* Int - the id of the group at which head a node is to be placed (B)
* int - the id of the node to place (A)
* ] * N
* }}}
* So that for each pair, node A is moved to the head of group B.
*/
final case class GroupHead(groups: (Int, Int)*)
extends Message("/g_head", groups.flatMap(g => g._1 :: g._2 :: Nil): _*)
with SyncCmd
/** The `/g_tail` message pair-wise places nodes at the tail of groups.
* {{{
* /g_tail
* [
* Int - the id of the group at which tail a node is to be placed (B)
* int - the id of the node to place (A)
* ] * N
* }}}
* So that for each pair, node A is moved to the tail of group B.
*/
final case class GroupTail(groups: (Int, Int)*)
extends Message("/g_tail", groups.flatMap(g => g._1 :: g._2 :: Nil): _*)
with SyncCmd
/** The `/g_freeAll` message. */
final case class GroupFreeAll(ids: Int*)
extends Message("/g_freeAll", ids: _*)
with SyncCmd
/** The `/g_deepFree` message. */
final case class GroupDeepFree(ids: Int*)
extends Message("/g_deepFree", ids: _*)
with SyncCmd
/** The `/p_new` message. */
final case class ParGroupNew(groups: GroupNew.Data*)
extends Message("/p_new", groups.flatMap(_.toList): _*)
with SyncCmd
object HasControlSet {
def unapply(h: HasControlSet): Option[Seq[ControlSet]] = Some(h.controls)
}
sealed trait HasControlSet {
def controls: Seq[ControlSet]
def updateControls(controls: Seq[ControlSet]): Message with SyncCmd with HasControlSet
}
/** The `/s_new` message. */
final case class SynthNew(defName: String, id: Int, addAction: Int, targetId: Int, controls: ControlSet*)
extends Message("/s_new",
defName +: id +: addAction +: targetId +: controls.flatMap(_.toSetSeq): _*)
with SyncCmd with HasControlSet {
def copy(defName: String = defName, id: Int = id, addAction: Int = addAction,
targetId: Int = targetId, controls: Seq[ControlSet] = controls): SynthNew =
SynthNew(defName = defName, id = id, addAction = addAction, targetId = targetId, controls = controls: _*)
override def updateControls(controls: Seq[ControlSet]): SynthNew =
copy(controls = controls)
}
/** The `/s_get` message. */
final case class SynthGet(id: Int, controls: Any*)
extends Message("/s_get", id +: controls: _*)
with SyncQuery {
def copy(id: Int = id, controls: Seq[Any] = controls): SynthGet =
SynthGet(id = id, controls = controls: _*)
}
/** The `/s_getn` message. */
final case class SynthGetn(id: Int, controls: (Any, Int)*)
extends Message("/s_getn", id +: controls.flatMap(tup => tup._1 :: tup._2 :: Nil): _*)
with SyncQuery {
def copy(id: Int = id, controls: Seq[(Any, Int)] = controls): SynthGetn =
SynthGetn(id = id, controls = controls: _*)
}
/** The `/n_run` message. */
final case class NodeRun(nodes: (Int, Boolean)*)
extends Message("/n_run", nodes.flatMap(n => (n._1: Any) :: (n._2: Any) :: Nil): _*)
with SyncCmd
/** The `/n_set` message. */
final case class NodeSet(id: Int, pairs: ControlSet*)
extends Message("/n_set", id +: pairs.flatMap(_.toSetSeq): _*)
with SyncCmd with HasControlSet {
def copy(id: Int = id, pairs: Seq[ControlSet] = pairs): NodeSet =
NodeSet(id = id, pairs = pairs: _*)
def controls: Seq[ControlSet] = pairs
override def updateControls(controls: Seq[ControlSet]): NodeSet =
copy(pairs = controls)
}
/** The `/n_setn` message. */
final case class NodeSetn(id: Int, pairs: ControlSet*)
extends Message("/n_setn", id +: pairs.flatMap(_.toSetnSeq): _*)
with SyncCmd with HasControlSet {
def copy(id: Int = id, pairs: Seq[ControlSet] = pairs): NodeSetn =
NodeSetn(id = id, pairs = pairs: _*)
def controls: Seq[ControlSet] = pairs
override def updateControls(controls: Seq[ControlSet]): NodeSetn =
copy(pairs = controls)
}
/** The `/n_trace` message. */
final case class NodeTrace(ids: Int*)
extends Message("/n_trace", ids: _*)
with SyncCmd
/** The `/n_noid` message. */
final case class NodeNoId(ids: Int*)
extends Message("/n_noid", ids: _*)
with SyncCmd
/** The `/n_free` message. */
final case class NodeFree(ids: Int*)
extends Message("/n_free", ids: _*)
with SyncCmd
/** The `/n_map` message. */
final case class NodeMap(id: Int, mappings: ControlKBusMap.Single*)
extends Message("/n_map", id +: mappings.flatMap(_.toMapSeq): _*)
with SyncCmd {
def copy(id: Int = id, mappings: Seq[ControlKBusMap.Single] = mappings): NodeMap =
NodeMap(id = id, mappings = mappings: _*)
}
/** The `/n_mapn` message. */
final case class NodeMapn(id: Int, mappings: ControlKBusMap*)
extends Message("/n_mapn", id +: mappings.flatMap(_.toMapnSeq): _*)
with SyncCmd {
def copy(id: Int = id, mappings: Seq[ControlKBusMap] = mappings): NodeMapn =
NodeMapn(id = id, mappings = mappings: _*)
}
/** The `/n_mapa` message. */
final case class NodeMapa(id: Int, mappings: ControlABusMap.Single*)
extends Message("/n_mapa", id +: mappings.flatMap(_.toMapaSeq): _*)
with SyncCmd {
def copy(id: Int = id, mappings: Seq[ControlABusMap.Single] = mappings): NodeMapa =
NodeMapa(id = id, mappings = mappings: _*)
}
/** The `/n_mapan` message. */
final case class NodeMapan(id: Int, mappings: ControlABusMap*)
extends Message("/n_mapan", id +: mappings.flatMap(_.toMapanSeq): _*)
with SyncCmd {
def copy(id: Int = id, mappings: Seq[ControlABusMap] = mappings): NodeMapan =
NodeMapan(id = id, mappings = mappings: _*)
}
/** The `/n_fill` message. */
final case class NodeFill(id: Int, data: ControlFillRange*)
extends Message("/n_fill", id +: data.flatMap(_.toList): _*)
with SyncCmd {
def copy(id: Int = id, data: Seq[ControlFillRange] = data): NodeFill =
NodeFill(id = id, data = data: _*)
}
/** The `/n_before` message pair-wise places nodes before other nodes.
* {{{
* /n_before
* [
* Int - the id of the node to place (A)
* int - the id of the node before which the above is placed (B)
* ] * N
* }}}
* So that for each pair, node A in the same group as node B, to execute immediately before node B.
*/
final case class NodeBefore(groups: (Int, Int)*)
extends Message("/n_before", groups.flatMap(g => g._1 :: g._2 :: Nil): _*)
with SyncCmd
/** The `/n_after` message pair-wise places nodes after other nodes.
* {{{
* /n_after
* [
* Int - the id of the node to place (A)
* int - the id of the node after which the above is placed (B)
* ] * N
* }}}
* So that for each pair, node A in the same group as node B, to execute immediately after node B.
*/
final case class NodeAfter(groups: (Int, Int)*)
extends Message("/n_after", groups.flatMap(g => g._1 :: g._2 :: Nil): _*)
with SyncCmd
/** The `/n_query` message. */
final case class NodeQuery(ids: Int*) extends Message("/n_query", ids: _*) with SyncQuery
/** The `/n_order` message. */
final case class NodeOrder(addAction: Int, targetId: Int, ids: Int*)
extends Message("/n_order", addAction +: targetId +: ids: _*)
with SyncCmd {
def copy(addAction: Int = addAction, targetId: Int = targetId, ids: Seq[Int] = ids): NodeOrder =
NodeOrder(addAction = addAction, targetId = targetId, ids = ids: _*)
}
/** The `/d_recv` message. */
final case class SynthDefRecv(bytes: ByteBuffer, completion: Option[Packet])
extends Message("/d_recv", bytes :: completion.toList: _*)
with HasCompletion {
def updateCompletion(completion: Option[Packet]): SynthDefRecv = copy(completion = completion)
}
/** The `/d_free` message. */
final case class SynthDefFree(names: String*)
extends Message("/d_free", names: _*)
with SyncCmd
/** The `/d_load` message.
*
* @param path the path to the file that stores the definition. This can be a pattern
* like `"synthdefs/perc-*"`
*
* @see [[SynthDef.loadMsg]]
*/
final case class SynthDefLoad(path: String, completion: Option[Packet])
extends Message("/d_load", path :: completion.toList: _*)
with HasCompletion {
def updateCompletion(completion: Option[Packet]): SynthDefLoad = copy(completion = completion)
}
/** The `/d_loadDir` message tells the server to load all synth definitions within a directory.
*
* @see [[SynthDef.loadDirMsg]]
*/
final case class SynthDefLoadDir(path: String, completion: Option[Packet])
extends Message("/d_loadDir", path :: completion.toList: _*)
with HasCompletion {
def updateCompletion(completion: Option[Packet]): SynthDefLoadDir = copy(completion = completion)
}
/** The `/u_cmd` message allows one to send UGen specific commands. */
final case class UGenCommand(nodeId: Int, ugenIdx: Int, command: String, rest: Any*)
extends Message("/u_cmd", nodeId +: ugenIdx +: command +: rest)
with SyncCmd {
def copy(nodeId: Int = nodeId, ugenIdx: Int = ugenIdx, command: String = command, rest: Seq[Any] = rest): UGenCommand =
UGenCommand(nodeId = nodeId, ugenIdx = ugenIdx, command = command, rest = rest: _*)
}
/** The `/tr` message send from a [[de.sciss.synth.ugen.SendTrig SendTrig]] UGen. */
final case class Trigger(nodeId: Int, trig: Int, value: Float)
extends Message("/tr", nodeId, trig, value) with Receive
© 2015 - 2025 Weber Informatics LLC | Privacy Policy