All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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