import { AbstractParam } from "../context/AbstractParam"; import { dbToGain, gainToDb } from "../type/Conversions"; import { AudioRange, Decibels, Frequency, NormalRange, Positive, Time, Unit, UnitName } from "../type/Units"; import { isAudioParam } from "../util/AdvancedTypeCheck"; import { optionsFromArguments } from "../util/Defaults"; import { Timeline } from "../util/Timeline"; import { isDefined } from "../util/TypeCheck"; import { ToneWithContext, ToneWithContextOptions } from "./ToneWithContext"; export interface ParamOptions extends ToneWithContextOptions { units: UnitName; value?: Type; param: AudioParam | Param; convert: boolean; } /** * the possible automation types */ type AutomationType = "linearRampToValueAtTime" | "exponentialRampToValueAtTime" | "setValueAtTime" | "setTargetAtTime" | "cancelScheduledValues"; interface TargetAutomationEvent { type: "setTargetAtTime"; time: number; value: number; constant: number; } interface NormalAutomationEvent { type: Exclude; time: number; value: number; } /** * The events on the automation */ export type AutomationEvent = NormalAutomationEvent | TargetAutomationEvent; /** * Param wraps the native Web Audio's AudioParam to provide * additional unit conversion functionality. It also * serves as a base-class for classes which have a single, * automatable parameter. */ export class Param extends ToneWithContext> implements AbstractParam { readonly name: string = "Param"; static getDefaults(): ParamOptions { return Object.assign(ToneWithContext.getDefaults(), { convert: true, units: "number" as UnitName, } as ParamOptions); } /** * The input connection */ readonly input: AudioParam; readonly units: UnitName; convert: boolean; overridden: boolean = false; /** * The timeline which tracks all of the automations. */ protected _events: Timeline; /** * The native parameter to control */ protected _param: AudioParam; /** * The default value before anything is assigned */ protected _initialValue: number; /** * The minimum output value */ private _minOutput = 1e-7; /** * @param param The AudioParam to wrap * @param units The unit name * @param convert Whether or not to convert the value to the target units */ constructor(param: AudioParam, units?: Unit, convert?: boolean); constructor(options: Partial>); constructor() { super(optionsFromArguments(Param.getDefaults(), arguments, ["param", "units", "convert"])); const options = optionsFromArguments(Param.getDefaults(), arguments, ["param", "units", "convert"]); this.assert(isDefined(options.param) && (isAudioParam(options.param) || options.param instanceof Param), "param must be an AudioParam"); while (!isAudioParam(options.param)) { options.param = options.param._param; } // initialize this._param = this.input = options.param; this._events = new Timeline(1000); this._initialValue = this._param.defaultValue; this.units = options.units; this.convert = options.convert; // if the value is defined, set it immediately if (isDefined(options.value) && options.value !== this._toType(this._initialValue)) { this.setValueAtTime(options.value, 0); } } get value(): Type { const now = this.now(); return this.getValueAtTime(now); } set value(value: Type) { this._initialValue = this._fromType(value); this.cancelScheduledValues(this.now()); this.setValueAtTime(value, this.now()); } get minValue(): number { if (this.units === "time" || this.units === "frequency" || this.units === "normalRange" || this.units === "positive" || this.units === "transportTime" || this.units === "ticks" || this.units === "bpm" || this.units === "hertz" || this.units === "samples") { return 0; } else if (this.units === "audioRange") { return -1; } else if (this.units === "decibels") { return -Infinity; } else { return this._param.minValue; } } get maxValue(): number { if (this.units === "normalRange" || this.units === "audioRange") { return 1; } else { return this._param.maxValue; } } /** * Type guard based on the unit name */ private _is(arg: any, type: UnitName): arg is T { return this.units === type; } /** * Convert the given value from the type specified by Param.units * into the destination value (such as Gain or Frequency). */ protected _fromType(val: Type): number { if (this.convert && !this.overridden) { if (this._is