de.sciss.synth.Curve.scala Maven / Gradle / Ivy
/*
* Curve.scala
* (ScalaCollider)
*
* Copyright (c) 2008-2016 Hanns Holger Rutz. All rights reserved.
*
* This software is published under the GNU Lesser General Public License v2.1+
*
*
* For further information, please contact Hanns Holger Rutz at
* [email protected]
*/
package de.sciss.synth
import de.sciss.serial.{DataInput, DataOutput, ImmutableSerializer}
import scala.annotation.switch
import scala.language.implicitConversions
import scala.math.{Pi, abs, cos, max, pow, sin, sqrt}
object Curve {
case object step extends Curve {
final val id = 0
override def productPrefix = "Curve$step$" // compatible with SoundProcesses serialization
override def toString = "step"
def levelAt(pos: Float, y1: Float, y2: Float) = if (pos < 1f) y1 else y2
}
case object linear extends Curve {
final val id = 1
override def productPrefix = "Curve$linear$"
override def toString = "linear"
def levelAt(pos: Float, y1: Float, y2: Float) = pos * (y2 - y1) + y1
}
val lin = linear
case object exponential extends Curve {
final val id = 2
override def productPrefix = "Curve$exponential$"
override def toString = "exponential"
def levelAt(pos: Float, y1: Float, y2: Float) = {
val y1Lim = max(0.0001f, y1)
(y1Lim * pow(y2 / y1Lim, pos)).toFloat
}
}
val exp = exponential
case object sine extends Curve {
final val id = 3
override def productPrefix = "Curve$sine$"
override def toString = "sine"
def levelAt(pos: Float, y1: Float, y2: Float) =
(y1 + (y2 - y1) * (-cos(Pi * pos) * 0.5 + 0.5)).toFloat
}
case object welch extends Curve {
final val id = 4
override def productPrefix = "Curve$welch$"
override def toString = "welch"
def levelAt(pos: Float, y1: Float, y2: Float) = if (y1 < y2) {
(y1 + (y2 - y1) * sin(Pi * 0.5 * pos)).toFloat
} else {
(y2 - (y2 - y1) * sin(Pi * 0.5 * (1 - pos))).toFloat
}
}
implicit def fromDouble(d: Double): Curve = parametric(d.toFloat)
object parametric {
final val id = 5
}
final case class parametric(/*override val */ curvature: Float) extends Curve {
def id = parametric.id
override def productPrefix = s"Curve$$parametric"
override def toString = s"parametric($curvature)"
def levelAt(pos: Float, y1: Float, y2: Float) = if (abs(curvature) < 0.0001f) {
pos * (y2 - y1) + y1
} else {
val denominator = 1.0 - math.exp(curvature)
val numerator = 1.0 - math.exp(pos * curvature)
(y1 + (y2 - y1) * (numerator / denominator)).toFloat
}
}
case object squared extends Curve {
final val id = 6
override def productPrefix = "Curve$squared$"
override def toString = "squared"
def levelAt(pos: Float, y1: Float, y2: Float) = {
val y1Pow2 = sqrt(y1)
val y2Pow2 = sqrt(y2)
val yPow2 = pos * (y2Pow2 - y1Pow2) + y1Pow2
(yPow2 * yPow2).toFloat
}
}
case object cubed extends Curve {
final val id = 7
override def productPrefix = "Curve$cubed$"
override def toString = "cubed"
def levelAt(pos: Float, y1: Float, y2: Float) = {
val y1Pow3 = pow(y1, 0.3333333)
val y2Pow3 = pow(y2, 0.3333333)
val yPow3 = pos * (y2Pow3 - y1Pow3) + y1Pow3
(yPow3 * yPow3 * yPow3).toFloat
}
}
implicit object serializer extends ImmutableSerializer[Curve] {
def write(shape: Curve, out: DataOutput): Unit = {
out.writeInt(shape.id)
shape match {
case parametric(c) => out.writeFloat(c)
case _ =>
}
}
def read(in: DataInput): Curve =
(in.readInt(): @switch) match {
case step .id => step
case linear .id => linear
case exponential.id => exponential
case sine .id => sine
case welch .id => welch
case parametric .id => parametric(in.readFloat())
case squared .id => squared
case cubed .id => cubed
case other => sys.error(s"Unexpected envelope shape ID $other")
}
}
}
sealed trait Curve {
def id: Int
def levelAt(pos: Float, y1: Float, y2: Float): Float
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy